Encrypt files

This commit is contained in:
Adria Navarro 2023-06-12 11:49:38 +01:00
parent d7f64fe6a4
commit 1f4cdf348f
3 changed files with 29 additions and 14 deletions

View File

@ -2,6 +2,7 @@ import crypto from "crypto"
import fs from "fs" import fs from "fs"
import zlib from "zlib" import zlib from "zlib"
import env from "../environment" import env from "../environment"
import { join } from "path"
const ALGO = "aes-256-ctr" const ALGO = "aes-256-ctr"
const SEPARATOR = "-" const SEPARATOR = "-"
@ -63,11 +64,15 @@ export function decrypt(
return Buffer.concat([base, final]).toString() return Buffer.concat([base, final]).toString()
} }
export async function encryptFile(filePath: string, secret: string) { export async function encryptFile(
const outputFilePath = `${filePath}.enc` { dir, filename }: { dir: string; filename: string },
secret: string
) {
const outputFileName = `${filename}.enc`
const filePath = join(dir, filename)
const inputFile = fs.createReadStream(filePath) const inputFile = fs.createReadStream(filePath)
const outputFile = fs.createWriteStream(outputFilePath) const outputFile = fs.createWriteStream(join(dir, outputFileName))
const salt = crypto.randomBytes(RANDOM_BYTES) const salt = crypto.randomBytes(RANDOM_BYTES)
const stretched = stretchString(secret, salt) const stretched = stretchString(secret, salt)
@ -77,9 +82,12 @@ export async function encryptFile(filePath: string, secret: string) {
encrypted.pipe(outputFile) encrypted.pipe(outputFile)
return new Promise<string>(r => { return new Promise<{ filename: string; dir: string }>(r => {
outputFile.on("finish", () => { outputFile.on("finish", () => {
r(outputFilePath) r({
filename: outputFileName,
dir,
})
}) })
}) })
} }

View File

@ -9,7 +9,7 @@ export async function exportAppDump(ctx: any) {
ctx.req.setTimeout(0) ctx.req.setTimeout(0)
const appName = decodeURI(ctx.query.appname) const appName = decodeURI(ctx.query.appname)
excludeRows = isQsTrue(excludeRows) excludeRows = isQsTrue(excludeRows)
const extension = encryptPassword ? "data" : "tar.gz" const extension = encryptPassword ? "enc.tar.gz" : "tar.gz"
const backupIdentifier = `${appName}-export-${new Date().getTime()}.${extension}` const backupIdentifier = `${appName}-export-${new Date().getTime()}.${extension}`
ctx.attachment(backupIdentifier) ctx.attachment(backupIdentifier)
ctx.body = await sdk.backups.streamExportApp({ ctx.body = await sdk.backups.streamExportApp({

View File

@ -145,21 +145,28 @@ export async function exportApp(appId: string, config?: ExportOpts) {
filter: defineFilter(config?.excludeRows, config?.excludeLogs), filter: defineFilter(config?.excludeRows, config?.excludeLogs),
exportPath: dbPath, exportPath: dbPath,
}) })
if (config?.encryptPassword) {
for (let file of fs.readdirSync(tmpPath)) {
const path = join(tmpPath, file)
await encryption.encryptFile(
{ dir: tmpPath, filename: file },
config.encryptPassword
)
fs.rmSync(path)
}
}
// if tar requested, return where the tarball is // if tar requested, return where the tarball is
if (config?.tar) { if (config?.tar) {
// now the tmpPath contains both the DB export and attachments, tar this // now the tmpPath contains both the DB export and attachments, tar this
const tarPath = tarFilesToTmp(tmpPath, fs.readdirSync(tmpPath)) const tarPath = tarFilesToTmp(tmpPath, fs.readdirSync(tmpPath))
// cleanup the tmp export files as tarball returned // cleanup the tmp export files as tarball returned
fs.rmSync(tmpPath, { recursive: true, force: true }) fs.rmSync(tmpPath, { recursive: true, force: true })
if (!config.encryptPassword) {
return tarPath
}
const encryptedTarPath = await encryption.encryptFile( return tarPath
tarPath,
config.encryptPassword
)
return encryptedTarPath
} }
// tar not requested, turn the directory where export is // tar not requested, turn the directory where export is
else { else {