luau-unzip/tests/extract.luau
Erica Marigold 6182e65756
feat: simplify path construction logic
Simplies logic to just use nearest parent as the base instead of
traversing downwards. This reduces some redundancy, however fixes a bug
with repeated components in the constructed path and also improves
performance by skipping a loop altogether.

Also fixes recursive extraction tests failing `pandoc_soft_links.zip`.
2025-02-13 16:54:29 +00:00

75 lines
2.5 KiB
Text

local fs = require("@lune/fs")
local process = require("@lune/process")
local frktest = require("../lune_packages/frktest")
local check = frktest.assert.check
local ZipReader = require("../lib")
-- Reuse the same ZIP files from metadata tests
local ZIPS = fs.readDir("tests/data")
local FALLIBLES = {
"invalid_cde_number_of_files_allocation_greater_offset.zip",
"invalid_cde_number_of_files_allocation_smaller_offset.zip",
"invalid_offset.zip",
"invalid_offset2.zip",
"chinese.zip", -- Contains non local specific encoding which can't be parsed without OS APIs
"non_utf8.zip", -- FIXME: Lune breaks for non utf8 data in process stdout
}
return function(test: typeof(frktest.test))
test.suite("ZIP extraction tests", function()
for _, file in ZIPS do
if not string.match(file, "%.zip$") then
continue
end
local checkErr: ((...any) -> any?) -> nil = if table.find(FALLIBLES, file)
then check.should_error
else check.should_not_error
test.case(`Extracts files correctly - {file}`, function()
checkErr(function()
local zipPath = "tests/data/" .. file
local data = fs.readFile(zipPath)
local zip = ZipReader.load(buffer.fromstring(data))
-- Test both string and buffer extraction
local stringOptions = { type = "text" :: "text", decompress = true }
local bufferOptions = { type = "binary" :: "binary", decompress = true }
-- Extract and verify each file
zip:walk(function(entry)
if entry.isDirectory then
return
end
-- Extract with unzip for comparison
local unzipOutput = process.spawn(`unzip`, { "-p", zipPath, entry:getPath() })
-- NOTE: We use assert since we don't know whether to expect true or false
assert(unzipOutput.ok)
-- Test string extraction
local contentString = zip:extract(entry, stringOptions) :: string
check.equal(#contentString, entry.size)
check.equal(contentString, unzipOutput.stdout)
-- Test buffer extraction
local contentBuffer = zip:extract(entry, bufferOptions) :: buffer
check.equal(buffer.len(contentBuffer), entry.size)
check.equal(buffer.tostring(contentBuffer), unzipOutput.stdout)
-- Test directory extraction
local parentPath = entry:getPath():match("(.+)/[^/]*$") or "/"
local dirContents = zip:extractDirectory(parentPath, stringOptions)
check.not_nil(dirContents[entry:getPath()])
check.equal(dirContents[entry:getPath()], unzipOutput.stdout)
end)
return
end)
end)
end
end)
end