-
-
Notifications
You must be signed in to change notification settings - Fork 33
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
fix: resolve the issue with invalid zip data
When the zip file can not be unzipped we attempt to use `tar` as fallback mechanism. After testing this for a while, can confirm the fallback is good enough but it cannot replace the zip handler altogether like attempted before in: - #1487 - #1495 Instead this approach will only use it if we receive the invalid zip exception from fflate.
- Loading branch information
Showing
17 changed files
with
244 additions
and
105 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
import { unpack } from './unpack'; | ||
import { File } from './types'; | ||
import { writeFile } from './writeFile'; | ||
|
||
export function decompress(byteArray: Uint8Array): Promise<File[]> { | ||
const { workspace, filePath } = writeFile(byteArray); | ||
return unpack(filePath, workspace.location); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
import fs from 'fs'; | ||
|
||
import { File } from './types'; | ||
export async function listFiles(workspace: string) { | ||
const files: File[] = []; | ||
|
||
async function explorePath(currentPath: string) { | ||
const dir = await fs.promises.readdir(currentPath); | ||
for (const fileName of dir) { | ||
const filePath = `${currentPath}/${fileName}`; | ||
const stats = await fs.promises.stat(filePath); | ||
|
||
if (stats.isFile()) { | ||
const buffer = await fs.promises.readFile(filePath); | ||
files.push({ | ||
name: filePath, | ||
contents: new Uint8Array(buffer), | ||
}); | ||
} else if (stats.isDirectory()) { | ||
await explorePath(filePath); // Recursively explore subdirectories | ||
} | ||
} | ||
} | ||
|
||
await explorePath(workspace); | ||
console.log('files', files); | ||
return files; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,42 @@ | ||
import { renderToStaticMarkup } from 'react-dom/server'; | ||
import { getUploadLimits } from '../../misc/getUploadLimits'; | ||
import { decompress } from './decompress'; | ||
import { isZipContentFileSupported } from '../../../usecases/uploads/isZipContentFileSupported'; | ||
|
||
export const processAndPrepareArchiveData = async ( | ||
byteArray: Uint8Array, | ||
isPatron: boolean = false | ||
) => { | ||
const size = Buffer.byteLength(byteArray); | ||
const limits = getUploadLimits(isPatron); | ||
|
||
if (size > limits.fileSize) { | ||
throw new Error( | ||
renderToStaticMarkup( | ||
<> | ||
Your upload is too big, there is a max of {size} / ${limits.fileSize}{' '} | ||
currently. <a href="https://alemayhu.com/patreon">Become a patron</a>{' '} | ||
to remove default limit or{' '} | ||
<a href="https://2anki.net/login#login">login</a>. | ||
</> | ||
) | ||
); | ||
} | ||
|
||
const decompressedData = await decompress(byteArray); | ||
const fileNames = decompressedData.map((z) => z.name); | ||
const files = []; | ||
|
||
for (const name of fileNames) { | ||
const file = decompressedData.find((z) => z.name === name); | ||
let contents = file?.contents; | ||
if (isZipContentFileSupported(name) && contents) { | ||
const s = new TextDecoder().decode(contents as Uint8Array); | ||
files.push({ name, contents: s }); | ||
} else if (contents) { | ||
files.push({ name, contents }); | ||
} | ||
} | ||
|
||
return files; | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
export interface File { | ||
name: string; | ||
contents?: string | Uint8Array; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
import { spawn } from 'node:child_process'; | ||
|
||
import { listFiles } from './listFiles'; | ||
import { File } from './types'; | ||
|
||
const TAR_PATH = '/usr/bin/bsdtar'; | ||
|
||
export function unpack(filePath: string, workspace: string): Promise<File[]> { | ||
return new Promise((resolve, reject) => { | ||
const decompressProcess = spawn(TAR_PATH, ['xvf', filePath], { | ||
cwd: workspace, | ||
}); | ||
decompressProcess.stdout.on('data', (data) => { | ||
console.log(`tar output: ${data}`); | ||
}); | ||
decompressProcess.stderr.on('data', (data) => { | ||
console.error(`tar error: ${data}`); | ||
}); | ||
decompressProcess.on('close', () => { | ||
// We are not reading the status code because we support partial extraction | ||
listFiles(workspace).then(resolve).catch(reject); | ||
}); | ||
}); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
import { getRandomUUID } from '../../../shared/helpers/getRandomUUID'; | ||
import Workspace from '../../parser/WorkSpace'; | ||
import path from 'path'; | ||
import fs from 'fs'; | ||
|
||
export function writeFile(compressedData: Uint8Array) { | ||
const uuid = getRandomUUID(); | ||
const workspace = new Workspace(true, 'fs'); | ||
const p = path.join(workspace.location, uuid); | ||
fs.writeFileSync(p, compressedData); | ||
return { workspace, filePath: p }; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,93 @@ | ||
import { strFromU8, unzipSync } from 'fflate'; | ||
import { Body } from 'aws-sdk/clients/s3'; | ||
import { renderToStaticMarkup } from 'react-dom/server'; | ||
import { getUploadLimits } from '../misc/getUploadLimits'; | ||
import { isHTMLFile, isMarkdownFile, isPDFFile } from '../storage/checks'; | ||
import { processAndPrepareArchiveData } from './fallback/processAndPrepareArchiveData'; | ||
|
||
interface File { | ||
name: string; | ||
contents?: Body | string; | ||
} | ||
|
||
class ZipHandler { | ||
files: File[]; | ||
zipFileCount: number; | ||
maxZipFiles: number; | ||
|
||
constructor(maxNestedZipFiles: number) { | ||
this.files = []; | ||
this.zipFileCount = 0; | ||
this.maxZipFiles = maxNestedZipFiles; | ||
} | ||
|
||
async build(zipData: Uint8Array, paying: boolean) { | ||
const size = Buffer.byteLength(zipData); | ||
const limits = getUploadLimits(paying); | ||
|
||
if (size > limits.fileSize) { | ||
throw new Error( | ||
renderToStaticMarkup( | ||
<> | ||
Your upload is too big, there is a max of {size} / $ | ||
{limits.fileSize} currently.{' '} | ||
<a href="https://alemayhu.com/patreon">Become a patron</a> to remove | ||
default limit. | ||
</> | ||
) | ||
); | ||
} | ||
|
||
await this.processZip(zipData, paying); | ||
} | ||
|
||
private async processZip(zipData: Uint8Array, paying: boolean) { | ||
if (this.zipFileCount >= this.maxZipFiles) { | ||
throw new Error('Too many zip files in the upload.'); | ||
} | ||
|
||
try { | ||
const loadedZip = unzipSync(zipData, { | ||
filter(file) { | ||
return !file.name.endsWith('/'); | ||
}, | ||
}); | ||
|
||
for (const name in loadedZip) { | ||
const file = loadedZip[name]; | ||
let contents = file; | ||
|
||
if (name.includes('__MACOSX/') || isPDFFile(name)) { | ||
continue; | ||
} | ||
|
||
if (name.endsWith('.zip')) { | ||
this.zipFileCount++; | ||
await this.processZip(file, paying); | ||
} else if ((isHTMLFile(name) || isMarkdownFile(name)) && contents) { | ||
this.files.push({ name, contents: strFromU8(file) }); | ||
} else if (contents) { | ||
this.files.push({ name, contents }); | ||
} | ||
} | ||
} catch (error: unknown) { | ||
// Code 13 indicates we need to use fallback archive processing | ||
const isArchiveProcessingError = (error as { code?: number }).code === 13; | ||
|
||
if (isArchiveProcessingError) { | ||
// Use fallback method to process archive | ||
const foundFiles = await processAndPrepareArchiveData(zipData, paying); | ||
this.files.push(...foundFiles); | ||
console.log('Processed files using fallback method:', this.files); | ||
} else { | ||
throw error; | ||
} | ||
} | ||
} | ||
|
||
getFileNames() { | ||
return this.files.map((file) => file.name); | ||
} | ||
} | ||
|
||
export { ZipHandler, File }; |
Oops, something went wrong.