From 34454abec0a136005f035341ee605ccbb2ba3cad Mon Sep 17 00:00:00 2001 From: Brandon Liu Date: Mon, 6 Jan 2025 12:32:42 +0800 Subject: [PATCH 1/2] Improve verify * properly exit with status != 0 when invalid archive detected * detect missing magic number instead of crashing * detect mismatch between header data length and actual file length * detect bounds area <= 0 --- pmtiles/verify.go | 29 +++++++++++++++++++++++------ 1 file changed, 23 insertions(+), 6 deletions(-) diff --git a/pmtiles/verify.go b/pmtiles/verify.go index 96bfe64..99bd1ed 100644 --- a/pmtiles/verify.go +++ b/pmtiles/verify.go @@ -8,6 +8,7 @@ import ( "io" "log" "math" + "os" "time" ) @@ -43,6 +44,18 @@ func Verify(_ *log.Logger, file string) error { header, err := deserializeHeader(b[0:HeaderV3LenBytes]) + if err != nil { + return fmt.Errorf("Failed to read %s, %w", key, err) + } + + fileInfo, _ := os.Stat(file) + + lengthFromHeader := int64(HeaderV3LenBytes + header.RootLength + header.MetadataLength + header.LeafDirectoryLength + header.TileDataLength) + + if fileInfo.Size() != lengthFromHeader { + return fmt.Errorf("Total length of archive %v does not match header %v:", fileInfo.Size(), lengthFromHeader) + } + var CollectEntries func(uint64, uint64, func(EntryV3)) CollectEntries = func(dir_offset uint64, dir_length uint64, f func(EntryV3)) { @@ -102,27 +115,31 @@ func Verify(_ *log.Logger, file string) error { }) if uint64(addressedTiles) != header.AddressedTilesCount { - fmt.Printf("Invalid: header AddressedTilesCount=%v but %v tiles addressed.", header.AddressedTilesCount, addressedTiles) + return fmt.Errorf("Invalid: header AddressedTilesCount=%v but %v tiles addressed.", header.AddressedTilesCount, addressedTiles) } if uint64(tileEntries) != header.TileEntriesCount { - fmt.Printf("Invalid: header TileEntriesCount=%v but %v tile entries.", header.TileEntriesCount, tileEntries) + return fmt.Errorf("Invalid: header TileEntriesCount=%v but %v tile entries.", header.TileEntriesCount, tileEntries) } if offsets.GetCardinality() != header.TileContentsCount { - fmt.Printf("Invalid: header TileContentsCount=%v but %v tile contents.", header.TileContentsCount, offsets.GetCardinality()) + return fmt.Errorf("Invalid: header TileContentsCount=%v but %v tile contents.", header.TileContentsCount, offsets.GetCardinality()) } if z, _, _ := IDToZxy(minTileID); z != header.MinZoom { - fmt.Printf("Invalid: header MinZoom=%v does not match min tile z %v", header.MinZoom, z) + return fmt.Errorf("Invalid: header MinZoom=%v does not match min tile z %v", header.MinZoom, z) } if z, _, _ := IDToZxy(maxTileID); z != header.MaxZoom { - fmt.Printf("Invalid: header MaxZoom=%v does not match max tile z %v", header.MaxZoom, z) + return fmt.Errorf("Invalid: header MaxZoom=%v does not match max tile z %v", header.MaxZoom, z) } if !(header.CenterZoom >= header.MinZoom && header.CenterZoom <= header.MaxZoom) { - fmt.Printf("Invalid: header CenterZoom=%v not within MinZoom/MaxZoom.", header.CenterZoom) + return fmt.Errorf("Invalid: header CenterZoom=%v not within MinZoom/MaxZoom.", header.CenterZoom) + } + + if header.MinLonE7 >= header.MaxLonE7 || header.MinLatE7 >= header.MaxLatE7 { + return fmt.Errorf("Invalid: bounds has area <= 0: clients may not display tiles correctly.") } fmt.Printf("Completed verify in %v.\n", time.Since(start)) From 9a1abad2b2fd3ec87080ac82d7c6ca6349037a1b Mon Sep 17 00:00:00 2001 From: Brandon Liu Date: Mon, 6 Jan 2025 12:46:19 +0800 Subject: [PATCH 2/2] formatting --- pmtiles/verify.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pmtiles/verify.go b/pmtiles/verify.go index 99bd1ed..20dd514 100644 --- a/pmtiles/verify.go +++ b/pmtiles/verify.go @@ -53,7 +53,7 @@ func Verify(_ *log.Logger, file string) error { lengthFromHeader := int64(HeaderV3LenBytes + header.RootLength + header.MetadataLength + header.LeafDirectoryLength + header.TileDataLength) if fileInfo.Size() != lengthFromHeader { - return fmt.Errorf("Total length of archive %v does not match header %v:", fileInfo.Size(), lengthFromHeader) + return fmt.Errorf("Total length of archive %v does not match header %v", fileInfo.Size(), lengthFromHeader) } var CollectEntries func(uint64, uint64, func(EntryV3))