feat: Add metadata namespace and fix stats calculation

This is a major release with breaking changes to metadata format.

BREAKING CHANGES:
- All metadata keys now use 'dg-' namespace prefix (becomes 'x-amz-meta-dg-*' in S3)
- Old metadata format is not supported - all files must be re-uploaded
- Stats behavior changed: quick mode no longer shows misleading warnings

Features:
- Metadata now uses real package version (dg-tool: deltaglider/VERSION)
- All metadata keys properly namespaced with 'dg-' prefix
- Clean stats output in quick mode (no per-file warning spam)
- Fixed nonsensical negative compression ratios in quick mode

Fixes:
- Stats now correctly handles delta files without metadata
- Space saved shows 0 instead of negative numbers when metadata unavailable
- Removed misleading warnings in quick mode (metadata not fetched is expected)
- Fixed metadata keys to use hyphens instead of underscores

Documentation:
- Added comprehensive metadata documentation
- Added stats calculation behavior guide
- Added real version tracking documentation

Tests:
- Updated all tests to use new dg- prefixed metadata keys
- All 73 unit tests passing
- All quality checks passing (ruff, mypy)

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
Simone Scarduzio
2025-10-15 11:19:10 +02:00
parent e1259b7ea8
commit db7d14f8a8
7 changed files with 197 additions and 102 deletions

View File

@@ -50,10 +50,10 @@ class TestDeltaServicePut:
ref_sha = service.hasher.sha256(io.BytesIO(ref_content))
ref_metadata = {
"tool": "deltaglider/0.1.0",
"source_name": "original.zip",
"file_sha256": ref_sha,
"created_at": "2025-01-01T00:00:00Z",
"dg-tool": "deltaglider/0.1.0",
"dg-source-name": "original.zip",
"dg-file-sha256": ref_sha,
"dg-created-at": "2025-01-01T00:00:00Z",
}
mock_storage.head.return_value = ObjectHead(
key="test/prefix/reference.bin",
@@ -98,7 +98,7 @@ class TestDeltaServicePut:
ref_sha = service.hasher.sha256(io.BytesIO(ref_content))
ref_metadata = {
"file_sha256": ref_sha,
"dg-file-sha256": ref_sha,
}
mock_storage.head.return_value = ObjectHead(
key="test/prefix/reference.bin",
@@ -200,15 +200,15 @@ class TestDeltaServiceVerify:
ref_sha = service.hasher.sha256(io.BytesIO(ref_content))
delta_metadata = {
"tool": "deltaglider/0.1.0",
"original_name": "file.zip",
"file_sha256": test_sha,
"file_size": str(len(test_content)),
"created_at": "2025-01-01T00:00:00Z",
"ref_key": "test/reference.bin",
"ref_sha256": ref_sha,
"delta_size": "100",
"delta_cmd": "xdelta3 -e -9 -s reference.bin file.zip file.zip.delta",
"dg-tool": "deltaglider/0.1.0",
"dg-original-name": "file.zip",
"dg-file-sha256": test_sha,
"dg-file-size": str(len(test_content)),
"dg-created-at": "2025-01-01T00:00:00Z",
"dg-ref-key": "test/reference.bin",
"dg-ref-sha256": ref_sha,
"dg-delta-size": "100",
"dg-delta-cmd": "xdelta3 -e -9 -s reference.bin file.zip file.zip.delta",
}
mock_storage.head.return_value = ObjectHead(
key="test/file.zip.delta",

View File

@@ -81,11 +81,11 @@ class TestBucketStatsAlgorithm:
def mock_head(path):
if "file1.zip.delta" in path:
head = Mock()
head.metadata = {"file_size": "19500000", "compression_ratio": "0.997"}
head.metadata = {"dg-file-size": "19500000", "compression_ratio": "0.997"}
return head
elif "file2.zip.delta" in path:
head = Mock()
head.metadata = {"file_size": "19600000", "compression_ratio": "0.997"}
head.metadata = {"dg-file-size": "19600000", "compression_ratio": "0.997"}
return head
return None
@@ -153,11 +153,11 @@ class TestBucketStatsAlgorithm:
def mock_head(path):
if "v1.zip.delta" in path:
head = Mock()
head.metadata = {"file_size": "19500000"}
head.metadata = {"dg-file-size": "19500000"}
return head
elif "v2.zip.delta" in path:
head = Mock()
head.metadata = {"file_size": "19600000"}
head.metadata = {"dg-file-size": "19600000"}
return head
return None
@@ -218,11 +218,11 @@ class TestBucketStatsAlgorithm:
def mock_head(path):
if "pro/v1.zip.delta" in path:
head = Mock()
head.metadata = {"file_size": "19500000"}
head.metadata = {"dg-file-size": "19500000"}
return head
elif "enterprise/v1.zip.delta" in path:
head = Mock()
head.metadata = {"file_size": "24500000"}
head.metadata = {"dg-file-size": "24500000"}
return head
return None
@@ -273,7 +273,7 @@ class TestBucketStatsAlgorithm:
assert mock_client.service.storage.list_objects.call_count == 2
def test_delta_file_without_metadata(self, mock_client):
"""Test handling of delta files with missing metadata."""
"""Test handling of delta files with missing metadata in quick mode."""
# Setup: Delta file without metadata
mock_client.service.storage.list_objects.return_value = {
"objects": [
@@ -283,21 +283,22 @@ class TestBucketStatsAlgorithm:
"is_truncated": False,
}
# No metadata available
# No metadata available (quick mode doesn't fetch metadata)
mock_client.service.storage.head.return_value = None
# Execute
stats = get_bucket_stats(mock_client, "no-metadata-bucket")
# Execute in quick mode (default)
stats = get_bucket_stats(mock_client, "no-metadata-bucket", mode="quick")
# Verify - falls back to using delta size as original size
# Verify - without metadata, original size cannot be calculated
assert stats.object_count == 1
assert stats.total_size == 50000 # Falls back to delta size
assert stats.total_size == 0 # Cannot calculate without metadata
assert stats.compressed_size == 20050000 # reference + delta
assert stats.space_saved == 0 # Cannot calculate without metadata
assert stats.delta_objects == 1
# Verify warning was logged
# Verify warning was logged about incomplete stats in quick mode
warning_calls = mock_client.service.logger.warning.call_args_list
assert any("no original_size" in str(call) for call in warning_calls)
assert any("Quick mode cannot calculate" in str(call) for call in warning_calls)
def test_parallel_metadata_fetching(self, mock_client):
"""Test that metadata is fetched in parallel for performance."""
@@ -323,7 +324,7 @@ class TestBucketStatsAlgorithm:
# Mock metadata
def mock_head(path):
head = Mock()
head.metadata = {"file_size": "19500000"}
head.metadata = {"dg-file-size": "19500000"}
return head
mock_client.service.storage.head.side_effect = mock_head
@@ -337,7 +338,7 @@ class TestBucketStatsAlgorithm:
futures = []
for i in range(num_deltas):
future = Mock()
future.result.return_value = (f"file{i}.zip.delta", {"file_size": "19500000"})
future.result.return_value = (f"file{i}.zip.delta", {"dg-file-size": "19500000"})
futures.append(future)
mock_pool.submit.side_effect = futures
@@ -366,9 +367,9 @@ class TestBucketStatsAlgorithm:
}
metadata_by_key = {
"alpha/file1.zip.delta": {"file_size": "100", "compression_ratio": "0.9"},
"alpha/file2.zip.delta": {"file_size": "120", "compression_ratio": "0.88"},
"beta/file1.zip.delta": {"file_size": "210", "compression_ratio": "0.9"},
"alpha/file1.zip.delta": {"dg-file-size": "100", "compression_ratio": "0.9"},
"alpha/file2.zip.delta": {"dg-file-size": "120", "compression_ratio": "0.88"},
"beta/file1.zip.delta": {"dg-file-size": "210", "compression_ratio": "0.9"},
}
def mock_head(path: str):
@@ -417,7 +418,7 @@ class TestBucketStatsAlgorithm:
raise Exception("S3 error")
elif "file2.zip.delta" in path:
head = Mock()
head.metadata = {"file_size": "19600000"}
head.metadata = {"dg-file-size": "19600000"}
return head
return None
@@ -426,11 +427,16 @@ class TestBucketStatsAlgorithm:
# Execute - should handle error gracefully
stats = get_bucket_stats(mock_client, "error-bucket", mode="detailed")
# Verify - file1 uses fallback, file2 uses metadata
# Verify - file1 has no metadata (error), file2 uses metadata
assert stats.object_count == 2
assert stats.delta_objects == 2
# file1 falls back to delta size (50000), file2 uses metadata (19600000)
assert stats.total_size == 50000 + 19600000
# file1 has no metadata so not counted in original size, file2 uses metadata (19600000)
assert stats.total_size == 19600000
# Verify warning was logged for file1
warning_calls = mock_client.service.logger.warning.call_args_list
assert any("file1.zip.delta" in str(call) and "no original_size metadata" in str(call)
for call in warning_calls)
def test_multiple_orphaned_references(self, mock_client):
"""Test detection of multiple orphaned reference.bin files."""