feat: add built-in backup decryption

This commit is contained in:
Samuel 2024-12-31 16:16:18 +01:00
parent ad9933903c
commit 58a803b0ce
17 changed files with 641 additions and 437 deletions

3
.gitmodules vendored Normal file
View file

@ -0,0 +1,3 @@
[submodule "signal-decrypt-backup-wasm"]
path = signal-decrypt-backup-wasm
url = https://git.duskflower.dev/duskflower/signal-decrypt-backup-wasm

View file

@ -1,7 +1 @@
# pnpm
export PNPM_HOME="/home/samuel/.local/share/pnpm"
case ":$PATH:" in
*":$PNPM_HOME:"*) ;;
*) export PATH="$PNPM_HOME:$PATH" ;;
esac
pnpm dlx lint-staged pnpm dlx lint-staged

View file

@ -108,7 +108,13 @@
} }
} }
}, },
"ignore": ["dist/**/*.ts", "dist/**", "**/*.mjs", "eslint.config.js", "**/*.js"] "ignore": [
"dist/**/*.ts",
"dist/**",
"**/*.mjs",
"eslint.config.js",
"**/*.js"
]
}, },
"javascript": { "javascript": {
"formatter": { "formatter": {

View file

@ -8,8 +8,6 @@
"dev": "vite", "dev": "vite",
"build": "vite build", "build": "vite build",
"serve": "vite preview", "serve": "vite preview",
"postinstall": "cp ./node_modules/sql.js/dist/sql-wasm.wasm ./src/assets/sql-wasm.wasm",
"generate-db-types": "kysely-codegen --dialect=sqlite --url=./src/assets/database.sqlite",
"prepare": "husky" "prepare": "husky"
}, },
"license": "MIT", "license": "MIT",
@ -27,16 +25,20 @@
"postcss": "^8.4.49", "postcss": "^8.4.49",
"tailwindcss": "^3.4.17", "tailwindcss": "^3.4.17",
"typescript": "^5.7.2", "typescript": "^5.7.2",
"vite": "^6.0.4", "vite": "^6.0.6",
"vite-plugin-solid": "^2.11.0" "vite-plugin-solid": "^2.11.0",
"vite-plugin-wasm": "^3.4.1"
}, },
"dependencies": { "dependencies": {
"@duskflower/signal-decrypt-backup-wasm": "^0.2.0",
"@kobalte/core": "^0.13.7", "@kobalte/core": "^0.13.7",
"@kobalte/tailwindcss": "^0.9.0", "@kobalte/tailwindcss": "^0.9.0",
"@solid-primitives/refs": "^1.0.8", "@solid-primitives/refs": "^1.0.8",
"@solid-primitives/storage": "^4.2.1",
"@solid-primitives/workers": "^0.3.0", "@solid-primitives/workers": "^0.3.0",
"@solidjs/meta": "^0.29.4", "@solidjs/meta": "^0.29.4",
"@solidjs/router": "^0.15.2", "@solidjs/router": "^0.15.2",
"@sqlite.org/sqlite-wasm": "3.47.2-build1",
"@tanstack/solid-table": "^8.20.5", "@tanstack/solid-table": "^8.20.5",
"chart.js": "^4.4.7", "chart.js": "^4.4.7",
"chartjs-chart-wordcloud": "^4.4.4", "chartjs-chart-wordcloud": "^4.4.4",
@ -47,11 +49,10 @@
"date-fns": "^4.1.0", "date-fns": "^4.1.0",
"kysely": "^0.27.5", "kysely": "^0.27.5",
"kysely-wasm": "^0.7.0", "kysely-wasm": "^0.7.0",
"lucide-solid": "^0.468.0", "lucide-solid": "^0.469.0",
"seroval": "^1.1.1", "seroval": "^1.1.1",
"solid-js": "^1.9.3", "solid-js": "^1.9.3",
"sql.js": "^1.12.0", "tailwind-merge": "^2.6.0",
"tailwind-merge": "^2.5.5",
"tailwindcss-animate": "^1.0.7" "tailwindcss-animate": "^1.0.7"
}, },
"lint-staged": { "lint-staged": {

503
pnpm-lock.yaml generated
View file

@ -8,6 +8,9 @@ importers:
.: .:
dependencies: dependencies:
'@duskflower/signal-decrypt-backup-wasm':
specifier: ^0.2.0
version: 0.2.0
'@kobalte/core': '@kobalte/core':
specifier: ^0.13.7 specifier: ^0.13.7
version: 0.13.7(solid-js@1.9.3) version: 0.13.7(solid-js@1.9.3)
@ -17,6 +20,9 @@ importers:
'@solid-primitives/refs': '@solid-primitives/refs':
specifier: ^1.0.8 specifier: ^1.0.8
version: 1.0.8(solid-js@1.9.3) version: 1.0.8(solid-js@1.9.3)
'@solid-primitives/storage':
specifier: ^4.2.1
version: 4.2.1(solid-js@1.9.3)
'@solid-primitives/workers': '@solid-primitives/workers':
specifier: ^0.3.0 specifier: ^0.3.0
version: 0.3.0(solid-js@1.9.3) version: 0.3.0(solid-js@1.9.3)
@ -26,6 +32,9 @@ importers:
'@solidjs/router': '@solidjs/router':
specifier: ^0.15.2 specifier: ^0.15.2
version: 0.15.2(solid-js@1.9.3) version: 0.15.2(solid-js@1.9.3)
'@sqlite.org/sqlite-wasm':
specifier: 3.47.2-build1
version: 3.47.2-build1
'@tanstack/solid-table': '@tanstack/solid-table':
specifier: ^8.20.5 specifier: ^8.20.5
version: 8.20.5(solid-js@1.9.3) version: 8.20.5(solid-js@1.9.3)
@ -57,20 +66,17 @@ importers:
specifier: ^0.7.0 specifier: ^0.7.0
version: 0.7.0(kysely@0.27.5) version: 0.7.0(kysely@0.27.5)
lucide-solid: lucide-solid:
specifier: ^0.468.0 specifier: ^0.469.0
version: 0.468.0(solid-js@1.9.3) version: 0.469.0(solid-js@1.9.3)
seroval: seroval:
specifier: ^1.1.1 specifier: ^1.1.1
version: 1.1.1 version: 1.1.1
solid-js: solid-js:
specifier: ^1.9.3 specifier: ^1.9.3
version: 1.9.3 version: 1.9.3
sql.js:
specifier: ^1.12.0
version: 1.12.0
tailwind-merge: tailwind-merge:
specifier: ^2.5.5 specifier: ^2.6.0
version: 2.5.5 version: 2.6.0
tailwindcss-animate: tailwindcss-animate:
specifier: ^1.0.7 specifier: ^1.0.7
version: 1.0.7(tailwindcss@3.4.17) version: 1.0.7(tailwindcss@3.4.17)
@ -115,11 +121,14 @@ importers:
specifier: ^5.7.2 specifier: ^5.7.2
version: 5.7.2 version: 5.7.2
vite: vite:
specifier: ^6.0.4 specifier: ^6.0.6
version: 6.0.4(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1) version: 6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)
vite-plugin-solid: vite-plugin-solid:
specifier: ^2.11.0 specifier: ^2.11.0
version: 2.11.0(solid-js@1.9.3)(vite@6.0.4(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)) version: 2.11.0(solid-js@1.9.3)(vite@6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1))
vite-plugin-wasm:
specifier: ^3.4.1
version: 3.4.1(vite@6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1))
packages: packages:
@ -335,146 +344,155 @@ packages:
peerDependencies: peerDependencies:
solid-js: ^1.8 solid-js: ^1.8
'@esbuild/aix-ppc64@0.24.0': '@duskflower/signal-decrypt-backup-wasm@0.2.0':
resolution: {integrity: sha512-WtKdFM7ls47zkKHFVzMz8opM7LkcsIp9amDUBIAWirg70RM71WRSjdILPsY5Uv1D42ZpUfaPILDlfactHgsRkw==} resolution: {integrity: sha512-vVDdJZhVFSfEYTsKQsHO85p/zyx9TuVTj6jaPvH1MRYtvL5812VTSQ1MBeERP1XWyySrZhNyNtvy2dV1A0Z0kQ==, tarball: https://git.duskflower.dev/api/packages/duskflower/npm/%40duskflower%2Fsignal-decrypt-backup-wasm/-/0.2.0/signal-decrypt-backup-wasm-0.2.0.tgz}
'@esbuild/aix-ppc64@0.24.2':
resolution: {integrity: sha512-thpVCb/rhxE/BnMLQ7GReQLLN8q9qbHmI55F4489/ByVg2aQaQ6kbcLb6FHkocZzQhxc4gx0sCk0tJkKBFzDhA==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [ppc64] cpu: [ppc64]
os: [aix] os: [aix]
'@esbuild/android-arm64@0.24.0': '@esbuild/android-arm64@0.24.2':
resolution: {integrity: sha512-Vsm497xFM7tTIPYK9bNTYJyF/lsP590Qc1WxJdlB6ljCbdZKU9SY8i7+Iin4kyhV/KV5J2rOKsBQbB77Ab7L/w==} resolution: {integrity: sha512-cNLgeqCqV8WxfcTIOeL4OAtSmL8JjcN6m09XIgro1Wi7cF4t/THaWEa7eL5CMoMBdjoHOTh/vwTO/o2TRXIyzg==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [arm64] cpu: [arm64]
os: [android] os: [android]
'@esbuild/android-arm@0.24.0': '@esbuild/android-arm@0.24.2':
resolution: {integrity: sha512-arAtTPo76fJ/ICkXWetLCc9EwEHKaeya4vMrReVlEIUCAUncH7M4bhMQ+M9Vf+FFOZJdTNMXNBrWwW+OXWpSew==} resolution: {integrity: sha512-tmwl4hJkCfNHwFB3nBa8z1Uy3ypZpxqxfTQOcHX+xRByyYgunVbZ9MzUUfb0RxaHIMnbHagwAxuTL+tnNM+1/Q==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [arm] cpu: [arm]
os: [android] os: [android]
'@esbuild/android-x64@0.24.0': '@esbuild/android-x64@0.24.2':
resolution: {integrity: sha512-t8GrvnFkiIY7pa7mMgJd7p8p8qqYIz1NYiAoKc75Zyv73L3DZW++oYMSHPRarcotTKuSs6m3hTOa5CKHaS02TQ==} resolution: {integrity: sha512-B6Q0YQDqMx9D7rvIcsXfmJfvUYLoP722bgfBlO5cGvNVb5V/+Y7nhBE3mHV9OpxBf4eAS2S68KZztiPaWq4XYw==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [x64] cpu: [x64]
os: [android] os: [android]
'@esbuild/darwin-arm64@0.24.0': '@esbuild/darwin-arm64@0.24.2':
resolution: {integrity: sha512-CKyDpRbK1hXwv79soeTJNHb5EiG6ct3efd/FTPdzOWdbZZfGhpbcqIpiD0+vwmpu0wTIL97ZRPZu8vUt46nBSw==} resolution: {integrity: sha512-kj3AnYWc+CekmZnS5IPu9D+HWtUI49hbnyqk0FLEJDbzCIQt7hg7ucF1SQAilhtYpIujfaHr6O0UHlzzSPdOeA==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [arm64] cpu: [arm64]
os: [darwin] os: [darwin]
'@esbuild/darwin-x64@0.24.0': '@esbuild/darwin-x64@0.24.2':
resolution: {integrity: sha512-rgtz6flkVkh58od4PwTRqxbKH9cOjaXCMZgWD905JOzjFKW+7EiUObfd/Kav+A6Gyud6WZk9w+xu6QLytdi2OA==} resolution: {integrity: sha512-WeSrmwwHaPkNR5H3yYfowhZcbriGqooyu3zI/3GGpF8AyUdsrrP0X6KumITGA9WOyiJavnGZUwPGvxvwfWPHIA==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [x64] cpu: [x64]
os: [darwin] os: [darwin]
'@esbuild/freebsd-arm64@0.24.0': '@esbuild/freebsd-arm64@0.24.2':
resolution: {integrity: sha512-6Mtdq5nHggwfDNLAHkPlyLBpE5L6hwsuXZX8XNmHno9JuL2+bg2BX5tRkwjyfn6sKbxZTq68suOjgWqCicvPXA==} resolution: {integrity: sha512-UN8HXjtJ0k/Mj6a9+5u6+2eZ2ERD7Edt1Q9IZiB5UZAIdPnVKDoG7mdTVGhHJIeEml60JteamR3qhsr1r8gXvg==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [arm64] cpu: [arm64]
os: [freebsd] os: [freebsd]
'@esbuild/freebsd-x64@0.24.0': '@esbuild/freebsd-x64@0.24.2':
resolution: {integrity: sha512-D3H+xh3/zphoX8ck4S2RxKR6gHlHDXXzOf6f/9dbFt/NRBDIE33+cVa49Kil4WUjxMGW0ZIYBYtaGCa2+OsQwQ==} resolution: {integrity: sha512-TvW7wE/89PYW+IevEJXZ5sF6gJRDY/14hyIGFXdIucxCsbRmLUcjseQu1SyTko+2idmCw94TgyaEZi9HUSOe3Q==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [x64] cpu: [x64]
os: [freebsd] os: [freebsd]
'@esbuild/linux-arm64@0.24.0': '@esbuild/linux-arm64@0.24.2':
resolution: {integrity: sha512-TDijPXTOeE3eaMkRYpcy3LarIg13dS9wWHRdwYRnzlwlA370rNdZqbcp0WTyyV/k2zSxfko52+C7jU5F9Tfj1g==} resolution: {integrity: sha512-7HnAD6074BW43YvvUmE/35Id9/NB7BeX5EoNkK9obndmZBUk8xmJJeU7DwmUeN7tkysslb2eSl6CTrYz6oEMQg==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [arm64] cpu: [arm64]
os: [linux] os: [linux]
'@esbuild/linux-arm@0.24.0': '@esbuild/linux-arm@0.24.2':
resolution: {integrity: sha512-gJKIi2IjRo5G6Glxb8d3DzYXlxdEj2NlkixPsqePSZMhLudqPhtZ4BUrpIuTjJYXxvF9njql+vRjB2oaC9XpBw==} resolution: {integrity: sha512-n0WRM/gWIdU29J57hJyUdIsk0WarGd6To0s+Y+LwvlC55wt+GT/OgkwoXCXvIue1i1sSNWblHEig00GBWiJgfA==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [arm] cpu: [arm]
os: [linux] os: [linux]
'@esbuild/linux-ia32@0.24.0': '@esbuild/linux-ia32@0.24.2':
resolution: {integrity: sha512-K40ip1LAcA0byL05TbCQ4yJ4swvnbzHscRmUilrmP9Am7//0UjPreh4lpYzvThT2Quw66MhjG//20mrufm40mA==} resolution: {integrity: sha512-sfv0tGPQhcZOgTKO3oBE9xpHuUqguHvSo4jl+wjnKwFpapx+vUDcawbwPNuBIAYdRAvIDBfZVvXprIj3HA+Ugw==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [ia32] cpu: [ia32]
os: [linux] os: [linux]
'@esbuild/linux-loong64@0.24.0': '@esbuild/linux-loong64@0.24.2':
resolution: {integrity: sha512-0mswrYP/9ai+CU0BzBfPMZ8RVm3RGAN/lmOMgW4aFUSOQBjA31UP8Mr6DDhWSuMwj7jaWOT0p0WoZ6jeHhrD7g==} resolution: {integrity: sha512-CN9AZr8kEndGooS35ntToZLTQLHEjtVB5n7dl8ZcTZMonJ7CCfStrYhrzF97eAecqVbVJ7APOEe18RPI4KLhwQ==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [loong64] cpu: [loong64]
os: [linux] os: [linux]
'@esbuild/linux-mips64el@0.24.0': '@esbuild/linux-mips64el@0.24.2':
resolution: {integrity: sha512-hIKvXm0/3w/5+RDtCJeXqMZGkI2s4oMUGj3/jM0QzhgIASWrGO5/RlzAzm5nNh/awHE0A19h/CvHQe6FaBNrRA==} resolution: {integrity: sha512-iMkk7qr/wl3exJATwkISxI7kTcmHKE+BlymIAbHO8xanq/TjHaaVThFF6ipWzPHryoFsesNQJPE/3wFJw4+huw==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [mips64el] cpu: [mips64el]
os: [linux] os: [linux]
'@esbuild/linux-ppc64@0.24.0': '@esbuild/linux-ppc64@0.24.2':
resolution: {integrity: sha512-HcZh5BNq0aC52UoocJxaKORfFODWXZxtBaaZNuN3PUX3MoDsChsZqopzi5UupRhPHSEHotoiptqikjN/B77mYQ==} resolution: {integrity: sha512-shsVrgCZ57Vr2L8mm39kO5PPIb+843FStGt7sGGoqiiWYconSxwTiuswC1VJZLCjNiMLAMh34jg4VSEQb+iEbw==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [ppc64] cpu: [ppc64]
os: [linux] os: [linux]
'@esbuild/linux-riscv64@0.24.0': '@esbuild/linux-riscv64@0.24.2':
resolution: {integrity: sha512-bEh7dMn/h3QxeR2KTy1DUszQjUrIHPZKyO6aN1X4BCnhfYhuQqedHaa5MxSQA/06j3GpiIlFGSsy1c7Gf9padw==} resolution: {integrity: sha512-4eSFWnU9Hhd68fW16GD0TINewo1L6dRrB+oLNNbYyMUAeOD2yCK5KXGK1GH4qD/kT+bTEXjsyTCiJGHPZ3eM9Q==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [riscv64] cpu: [riscv64]
os: [linux] os: [linux]
'@esbuild/linux-s390x@0.24.0': '@esbuild/linux-s390x@0.24.2':
resolution: {integrity: sha512-ZcQ6+qRkw1UcZGPyrCiHHkmBaj9SiCD8Oqd556HldP+QlpUIe2Wgn3ehQGVoPOvZvtHm8HPx+bH20c9pvbkX3g==} resolution: {integrity: sha512-S0Bh0A53b0YHL2XEXC20bHLuGMOhFDO6GN4b3YjRLK//Ep3ql3erpNcPlEFed93hsQAjAQDNsvcK+hV90FubSw==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [s390x] cpu: [s390x]
os: [linux] os: [linux]
'@esbuild/linux-x64@0.24.0': '@esbuild/linux-x64@0.24.2':
resolution: {integrity: sha512-vbutsFqQ+foy3wSSbmjBXXIJ6PL3scghJoM8zCL142cGaZKAdCZHyf+Bpu/MmX9zT9Q0zFBVKb36Ma5Fzfa8xA==} resolution: {integrity: sha512-8Qi4nQcCTbLnK9WoMjdC9NiTG6/E38RNICU6sUNqK0QFxCYgoARqVqxdFmWkdonVsvGqWhmm7MO0jyTqLqwj0Q==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [x64] cpu: [x64]
os: [linux] os: [linux]
'@esbuild/netbsd-x64@0.24.0': '@esbuild/netbsd-arm64@0.24.2':
resolution: {integrity: sha512-hjQ0R/ulkO8fCYFsG0FZoH+pWgTTDreqpqY7UnQntnaKv95uP5iW3+dChxnx7C3trQQU40S+OgWhUVwCjVFLvg==} resolution: {integrity: sha512-wuLK/VztRRpMt9zyHSazyCVdCXlpHkKm34WUyinD2lzK07FAHTq0KQvZZlXikNWkDGoT6x3TD51jKQ7gMVpopw==}
engines: {node: '>=18'}
cpu: [arm64]
os: [netbsd]
'@esbuild/netbsd-x64@0.24.2':
resolution: {integrity: sha512-VefFaQUc4FMmJuAxmIHgUmfNiLXY438XrL4GDNV1Y1H/RW3qow68xTwjZKfj/+Plp9NANmzbH5R40Meudu8mmw==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [x64] cpu: [x64]
os: [netbsd] os: [netbsd]
'@esbuild/openbsd-arm64@0.24.0': '@esbuild/openbsd-arm64@0.24.2':
resolution: {integrity: sha512-MD9uzzkPQbYehwcN583yx3Tu5M8EIoTD+tUgKF982WYL9Pf5rKy9ltgD0eUgs8pvKnmizxjXZyLt0z6DC3rRXg==} resolution: {integrity: sha512-YQbi46SBct6iKnszhSvdluqDmxCJA+Pu280Av9WICNwQmMxV7nLRHZfjQzwbPs3jeWnuAhE9Jy0NrnJ12Oz+0A==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [arm64] cpu: [arm64]
os: [openbsd] os: [openbsd]
'@esbuild/openbsd-x64@0.24.0': '@esbuild/openbsd-x64@0.24.2':
resolution: {integrity: sha512-4ir0aY1NGUhIC1hdoCzr1+5b43mw99uNwVzhIq1OY3QcEwPDO3B7WNXBzaKY5Nsf1+N11i1eOfFcq+D/gOS15Q==} resolution: {integrity: sha512-+iDS6zpNM6EnJyWv0bMGLWSWeXGN/HTaF/LXHXHwejGsVi+ooqDfMCCTerNFxEkM3wYVcExkeGXNqshc9iMaOA==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [x64] cpu: [x64]
os: [openbsd] os: [openbsd]
'@esbuild/sunos-x64@0.24.0': '@esbuild/sunos-x64@0.24.2':
resolution: {integrity: sha512-jVzdzsbM5xrotH+W5f1s+JtUy1UWgjU0Cf4wMvffTB8m6wP5/kx0KiaLHlbJO+dMgtxKV8RQ/JvtlFcdZ1zCPA==} resolution: {integrity: sha512-hTdsW27jcktEvpwNHJU4ZwWFGkz2zRJUz8pvddmXPtXDzVKTTINmlmga3ZzwcuMpUvLw7JkLy9QLKyGpD2Yxig==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [x64] cpu: [x64]
os: [sunos] os: [sunos]
'@esbuild/win32-arm64@0.24.0': '@esbuild/win32-arm64@0.24.2':
resolution: {integrity: sha512-iKc8GAslzRpBytO2/aN3d2yb2z8XTVfNV0PjGlCxKo5SgWmNXx82I/Q3aG1tFfS+A2igVCY97TJ8tnYwpUWLCA==} resolution: {integrity: sha512-LihEQ2BBKVFLOC9ZItT9iFprsE9tqjDjnbulhHoFxYQtQfai7qfluVODIYxt1PgdoyQkz23+01rzwNwYfutxUQ==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [arm64] cpu: [arm64]
os: [win32] os: [win32]
'@esbuild/win32-ia32@0.24.0': '@esbuild/win32-ia32@0.24.2':
resolution: {integrity: sha512-vQW36KZolfIudCcTnaTpmLQ24Ha1RjygBo39/aLkM2kmjkWmZGEJ5Gn9l5/7tzXA42QGIoWbICfg6KLLkIw6yw==} resolution: {integrity: sha512-q+iGUwfs8tncmFC9pcnD5IvRHAzmbwQ3GPS5/ceCyHdjXubwQWI12MKWSNSMYLJMq23/IUCvJMS76PDqXe1fxA==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [ia32] cpu: [ia32]
os: [win32] os: [win32]
'@esbuild/win32-x64@0.24.0': '@esbuild/win32-x64@0.24.2':
resolution: {integrity: sha512-7IAFPrjSQIJrGsK6flwg7NFmwBoSTyF3rl7If0hNUFQU4ilTsEPL6GuMuU9BfIWVVGuRnuIidkSMC+c0Otu8IA==} resolution: {integrity: sha512-7VTgWzgMGvup6aSqDPLiW5zHaxYJGTO4OokMjIlrCtf+VpEL+cXKtCvg723iguPYI5oaUNdS+/V7OU2gvXVWEg==}
engines: {node: '>=18'} engines: {node: '>=18'}
cpu: [x64] cpu: [x64]
os: [win32] os: [win32]
@ -498,10 +516,6 @@ packages:
resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==}
engines: {node: '>=12'} engines: {node: '>=12'}
'@jridgewell/gen-mapping@0.3.5':
resolution: {integrity: sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==}
engines: {node: '>=6.0.0'}
'@jridgewell/gen-mapping@0.3.8': '@jridgewell/gen-mapping@0.3.8':
resolution: {integrity: sha512-imAbBGkb+ebQyxKgzv5Hu2nmROxoDOXHh80evxdoXNOrvAnVx7zimzc1Oo5h9RlfV4vPXaE2iM5pOFbvOCClWA==} resolution: {integrity: sha512-imAbBGkb+ebQyxKgzv5Hu2nmROxoDOXHh80evxdoXNOrvAnVx7zimzc1Oo5h9RlfV4vPXaE2iM5pOFbvOCClWA==}
engines: {node: '>=6.0.0'} engines: {node: '>=6.0.0'}
@ -554,98 +568,98 @@ packages:
resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==} resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==}
engines: {node: '>=14'} engines: {node: '>=14'}
'@rollup/rollup-android-arm-eabi@4.28.1': '@rollup/rollup-android-arm-eabi@4.29.1':
resolution: {integrity: sha512-2aZp8AES04KI2dy3Ss6/MDjXbwBzj+i0GqKtWXgw2/Ma6E4jJvujryO6gJAghIRVz7Vwr9Gtl/8na3nDUKpraQ==} resolution: {integrity: sha512-ssKhA8RNltTZLpG6/QNkCSge+7mBQGUqJRisZ2MDQcEGaK93QESEgWK2iOpIDZ7k9zPVkG5AS3ksvD5ZWxmItw==}
cpu: [arm] cpu: [arm]
os: [android] os: [android]
'@rollup/rollup-android-arm64@4.28.1': '@rollup/rollup-android-arm64@4.29.1':
resolution: {integrity: sha512-EbkK285O+1YMrg57xVA+Dp0tDBRB93/BZKph9XhMjezf6F4TpYjaUSuPt5J0fZXlSag0LmZAsTmdGGqPp4pQFA==} resolution: {integrity: sha512-CaRfrV0cd+NIIcVVN/jx+hVLN+VRqnuzLRmfmlzpOzB87ajixsN/+9L5xNmkaUUvEbI5BmIKS+XTwXsHEb65Ew==}
cpu: [arm64] cpu: [arm64]
os: [android] os: [android]
'@rollup/rollup-darwin-arm64@4.28.1': '@rollup/rollup-darwin-arm64@4.29.1':
resolution: {integrity: sha512-prduvrMKU6NzMq6nxzQw445zXgaDBbMQvmKSJaxpaZ5R1QDM8w+eGxo6Y/jhT/cLoCvnZI42oEqf9KQNYz1fqQ==} resolution: {integrity: sha512-2ORr7T31Y0Mnk6qNuwtyNmy14MunTAMx06VAPI6/Ju52W10zk1i7i5U3vlDRWjhOI5quBcrvhkCHyF76bI7kEw==}
cpu: [arm64] cpu: [arm64]
os: [darwin] os: [darwin]
'@rollup/rollup-darwin-x64@4.28.1': '@rollup/rollup-darwin-x64@4.29.1':
resolution: {integrity: sha512-WsvbOunsUk0wccO/TV4o7IKgloJ942hVFK1CLatwv6TJspcCZb9umQkPdvB7FihmdxgaKR5JyxDjWpCOp4uZlQ==} resolution: {integrity: sha512-j/Ej1oanzPjmN0tirRd5K2/nncAhS9W6ICzgxV+9Y5ZsP0hiGhHJXZ2JQ53iSSjj8m6cRY6oB1GMzNn2EUt6Ng==}
cpu: [x64] cpu: [x64]
os: [darwin] os: [darwin]
'@rollup/rollup-freebsd-arm64@4.28.1': '@rollup/rollup-freebsd-arm64@4.29.1':
resolution: {integrity: sha512-HTDPdY1caUcU4qK23FeeGxCdJF64cKkqajU0iBnTVxS8F7H/7BewvYoG+va1KPSL63kQ1PGNyiwKOfReavzvNA==} resolution: {integrity: sha512-91C//G6Dm/cv724tpt7nTyP+JdN12iqeXGFM1SqnljCmi5yTXriH7B1r8AD9dAZByHpKAumqP1Qy2vVNIdLZqw==}
cpu: [arm64] cpu: [arm64]
os: [freebsd] os: [freebsd]
'@rollup/rollup-freebsd-x64@4.28.1': '@rollup/rollup-freebsd-x64@4.29.1':
resolution: {integrity: sha512-m/uYasxkUevcFTeRSM9TeLyPe2QDuqtjkeoTpP9SW0XxUWfcYrGDMkO/m2tTw+4NMAF9P2fU3Mw4ahNvo7QmsQ==} resolution: {integrity: sha512-hEioiEQ9Dec2nIRoeHUP6hr1PSkXzQaCUyqBDQ9I9ik4gCXQZjJMIVzoNLBRGet+hIUb3CISMh9KXuCcWVW/8w==}
cpu: [x64] cpu: [x64]
os: [freebsd] os: [freebsd]
'@rollup/rollup-linux-arm-gnueabihf@4.28.1': '@rollup/rollup-linux-arm-gnueabihf@4.29.1':
resolution: {integrity: sha512-QAg11ZIt6mcmzpNE6JZBpKfJaKkqTm1A9+y9O+frdZJEuhQxiugM05gnCWiANHj4RmbgeVJpTdmKRmH/a+0QbA==} resolution: {integrity: sha512-Py5vFd5HWYN9zxBv3WMrLAXY3yYJ6Q/aVERoeUFwiDGiMOWsMs7FokXihSOaT/PMWUty/Pj60XDQndK3eAfE6A==}
cpu: [arm] cpu: [arm]
os: [linux] os: [linux]
'@rollup/rollup-linux-arm-musleabihf@4.28.1': '@rollup/rollup-linux-arm-musleabihf@4.29.1':
resolution: {integrity: sha512-dRP9PEBfolq1dmMcFqbEPSd9VlRuVWEGSmbxVEfiq2cs2jlZAl0YNxFzAQS2OrQmsLBLAATDMb3Z6MFv5vOcXg==} resolution: {integrity: sha512-RiWpGgbayf7LUcuSNIbahr0ys2YnEERD4gYdISA06wa0i8RALrnzflh9Wxii7zQJEB2/Eh74dX4y/sHKLWp5uQ==}
cpu: [arm] cpu: [arm]
os: [linux] os: [linux]
'@rollup/rollup-linux-arm64-gnu@4.28.1': '@rollup/rollup-linux-arm64-gnu@4.29.1':
resolution: {integrity: sha512-uGr8khxO+CKT4XU8ZUH1TTEUtlktK6Kgtv0+6bIFSeiSlnGJHG1tSFSjm41uQ9sAO/5ULx9mWOz70jYLyv1QkA==} resolution: {integrity: sha512-Z80O+taYxTQITWMjm/YqNoe9d10OX6kDh8X5/rFCMuPqsKsSyDilvfg+vd3iXIqtfmp+cnfL1UrYirkaF8SBZA==}
cpu: [arm64] cpu: [arm64]
os: [linux] os: [linux]
'@rollup/rollup-linux-arm64-musl@4.28.1': '@rollup/rollup-linux-arm64-musl@4.29.1':
resolution: {integrity: sha512-QF54q8MYGAqMLrX2t7tNpi01nvq5RI59UBNx+3+37zoKX5KViPo/gk2QLhsuqok05sSCRluj0D00LzCwBikb0A==} resolution: {integrity: sha512-fOHRtF9gahwJk3QVp01a/GqS4hBEZCV1oKglVVq13kcK3NeVlS4BwIFzOHDbmKzt3i0OuHG4zfRP0YoG5OF/rA==}
cpu: [arm64] cpu: [arm64]
os: [linux] os: [linux]
'@rollup/rollup-linux-loongarch64-gnu@4.28.1': '@rollup/rollup-linux-loongarch64-gnu@4.29.1':
resolution: {integrity: sha512-vPul4uodvWvLhRco2w0GcyZcdyBfpfDRgNKU+p35AWEbJ/HPs1tOUrkSueVbBS0RQHAf/A+nNtDpvw95PeVKOA==} resolution: {integrity: sha512-5a7q3tnlbcg0OodyxcAdrrCxFi0DgXJSoOuidFUzHZ2GixZXQs6Tc3CHmlvqKAmOs5eRde+JJxeIf9DonkmYkw==}
cpu: [loong64] cpu: [loong64]
os: [linux] os: [linux]
'@rollup/rollup-linux-powerpc64le-gnu@4.28.1': '@rollup/rollup-linux-powerpc64le-gnu@4.29.1':
resolution: {integrity: sha512-pTnTdBuC2+pt1Rmm2SV7JWRqzhYpEILML4PKODqLz+C7Ou2apEV52h19CR7es+u04KlqplggmN9sqZlekg3R1A==} resolution: {integrity: sha512-9b4Mg5Yfz6mRnlSPIdROcfw1BU22FQxmfjlp/CShWwO3LilKQuMISMTtAu/bxmmrE6A902W2cZJuzx8+gJ8e9w==}
cpu: [ppc64] cpu: [ppc64]
os: [linux] os: [linux]
'@rollup/rollup-linux-riscv64-gnu@4.28.1': '@rollup/rollup-linux-riscv64-gnu@4.29.1':
resolution: {integrity: sha512-vWXy1Nfg7TPBSuAncfInmAI/WZDd5vOklyLJDdIRKABcZWojNDY0NJwruY2AcnCLnRJKSaBgf/GiJfauu8cQZA==} resolution: {integrity: sha512-G5pn0NChlbRM8OJWpJFMX4/i8OEU538uiSv0P6roZcbpe/WfhEO+AT8SHVKfp8qhDQzaz7Q+1/ixMy7hBRidnQ==}
cpu: [riscv64] cpu: [riscv64]
os: [linux] os: [linux]
'@rollup/rollup-linux-s390x-gnu@4.28.1': '@rollup/rollup-linux-s390x-gnu@4.29.1':
resolution: {integrity: sha512-/yqC2Y53oZjb0yz8PVuGOQQNOTwxcizudunl/tFs1aLvObTclTwZ0JhXF2XcPT/zuaymemCDSuuUPXJJyqeDOg==} resolution: {integrity: sha512-WM9lIkNdkhVwiArmLxFXpWndFGuOka4oJOZh8EP3Vb8q5lzdSCBuhjavJsw68Q9AKDGeOOIHYzYm4ZFvmWez5g==}
cpu: [s390x] cpu: [s390x]
os: [linux] os: [linux]
'@rollup/rollup-linux-x64-gnu@4.28.1': '@rollup/rollup-linux-x64-gnu@4.29.1':
resolution: {integrity: sha512-fzgeABz7rrAlKYB0y2kSEiURrI0691CSL0+KXwKwhxvj92VULEDQLpBYLHpF49MSiPG4sq5CK3qHMnb9tlCjBw==} resolution: {integrity: sha512-87xYCwb0cPGZFoGiErT1eDcssByaLX4fc0z2nRM6eMtV9njAfEE6OW3UniAoDhX4Iq5xQVpE6qO9aJbCFumKYQ==}
cpu: [x64] cpu: [x64]
os: [linux] os: [linux]
'@rollup/rollup-linux-x64-musl@4.28.1': '@rollup/rollup-linux-x64-musl@4.29.1':
resolution: {integrity: sha512-xQTDVzSGiMlSshpJCtudbWyRfLaNiVPXt1WgdWTwWz9n0U12cI2ZVtWe/Jgwyv/6wjL7b66uu61Vg0POWVfz4g==} resolution: {integrity: sha512-xufkSNppNOdVRCEC4WKvlR1FBDyqCSCpQeMMgv9ZyXqqtKBfkw1yfGMTUTs9Qsl6WQbJnsGboWCp7pJGkeMhKA==}
cpu: [x64] cpu: [x64]
os: [linux] os: [linux]
'@rollup/rollup-win32-arm64-msvc@4.28.1': '@rollup/rollup-win32-arm64-msvc@4.29.1':
resolution: {integrity: sha512-wSXmDRVupJstFP7elGMgv+2HqXelQhuNf+IS4V+nUpNVi/GUiBgDmfwD0UGN3pcAnWsgKG3I52wMOBnk1VHr/A==} resolution: {integrity: sha512-F2OiJ42m77lSkizZQLuC+jiZ2cgueWQL5YC9tjo3AgaEw+KJmVxHGSyQfDUoYR9cci0lAywv2Clmckzulcq6ig==}
cpu: [arm64] cpu: [arm64]
os: [win32] os: [win32]
'@rollup/rollup-win32-ia32-msvc@4.28.1': '@rollup/rollup-win32-ia32-msvc@4.29.1':
resolution: {integrity: sha512-ZkyTJ/9vkgrE/Rk9vhMXhf8l9D+eAhbAVbsGsXKy2ohmJaWg0LPQLnIxRdRp/bKyr8tXuPlXhIoGlEB5XpJnGA==} resolution: {integrity: sha512-rYRe5S0FcjlOBZQHgbTKNrqxCBUmgDJem/VQTCcTnA2KCabYSWQDrytOzX7avb79cAAweNmMUb/Zw18RNd4mng==}
cpu: [ia32] cpu: [ia32]
os: [win32] os: [win32]
'@rollup/rollup-win32-x64-msvc@4.28.1': '@rollup/rollup-win32-x64-msvc@4.29.1':
resolution: {integrity: sha512-ZvK2jBafvttJjoIdKm/Q/Bh7IJ1Ose9IBOwpOXcOvW3ikGTQGmKDgxTC6oCAzW6PynbkKP8+um1du81XJHZ0JA==} resolution: {integrity: sha512-+10CMg9vt1MoHj6x1pxyjPSMjHTIlqs8/tBztXvPAx24SKs9jwVnKqHJumlH/IzhaPUaj3T6T6wfZr8okdXaIg==}
cpu: [x64] cpu: [x64]
os: [win32] os: [win32]
@ -694,6 +708,18 @@ packages:
peerDependencies: peerDependencies:
solid-js: ^1.6.12 solid-js: ^1.6.12
'@solid-primitives/storage@4.2.1':
resolution: {integrity: sha512-1XUJeaSlizH9Eam/+IbIpslHEnggJMNZXzfsr06AlbG6tJtQENMu0+94ZIvooxt4Cyw46wPzcnHYbSK7LzoQAA==}
peerDependencies:
'@tauri-apps/plugin-store': '*'
solid-js: ^1.6.12
solid-start: '*'
peerDependenciesMeta:
'@tauri-apps/plugin-store':
optional: true
solid-start:
optional: true
'@solid-primitives/trigger@1.1.0': '@solid-primitives/trigger@1.1.0':
resolution: {integrity: sha512-00BbAiXV66WwjHuKZc3wr0+GLb9C24mMUmi3JdTpNFgHBbrQGrIHubmZDg36c5/7wH+E0GQtOOanwQS063PO+A==} resolution: {integrity: sha512-00BbAiXV66WwjHuKZc3wr0+GLb9C24mMUmi3JdTpNFgHBbrQGrIHubmZDg36c5/7wH+E0GQtOOanwQS063PO+A==}
peerDependencies: peerDependencies:
@ -719,6 +745,10 @@ packages:
peerDependencies: peerDependencies:
solid-js: ^1.8.6 solid-js: ^1.8.6
'@sqlite.org/sqlite-wasm@3.47.2-build1':
resolution: {integrity: sha512-jDRWfcPYmOsJGN1GplXbw5ZlB+PSzihw1EDElzqkenSI+yK7QYOfwhzYnBR9H8Bv5MHi6EUMb4t3CnLfXCSA8A==}
hasBin: true
'@swc/helpers@0.5.15': '@swc/helpers@0.5.15':
resolution: {integrity: sha512-JQ5TuMi45Owi4/BIMAJBoSQoOJu12oOk/gADqlcUL9JEdHB8vyjUSsxqeNXnmXHjYKMi2WcYtezGEEhqUI/E2g==} resolution: {integrity: sha512-JQ5TuMi45Owi4/BIMAJBoSQoOJu12oOk/gADqlcUL9JEdHB8vyjUSsxqeNXnmXHjYKMi2WcYtezGEEhqUI/E2g==}
@ -1101,8 +1131,8 @@ packages:
error-ex@1.3.2: error-ex@1.3.2:
resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==} resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==}
esbuild@0.24.0: esbuild@0.24.2:
resolution: {integrity: sha512-FuLPevChGDshgSicjisSooU0cemp/sGXR841D5LHMB7mTVOmsEHcAxaH3irL53+8YDIeVNQEySh4DaYU/iuPqQ==} resolution: {integrity: sha512-+9egpBW8I3CD5XPe0n6BfT5fxLzxrlDzqydF3aviG+9ni1lDC/OvMHcxqEFV0+LANZG5R1bFMWfUrjVsdwxJvA==}
engines: {node: '>=18'} engines: {node: '>=18'}
hasBin: true hasBin: true
@ -1285,10 +1315,6 @@ packages:
resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==}
engines: {node: '>=8'} engines: {node: '>=8'}
is-core-module@2.15.1:
resolution: {integrity: sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==}
engines: {node: '>= 0.4'}
is-core-module@2.16.0: is-core-module@2.16.0:
resolution: {integrity: sha512-urTSINYfAYgcbLb0yDQ6egFm6h3Mo1DcF9EkyXSRjjzdHbsulg01qhwWuXdOoUBuTkbQ80KDboXa0vFJ+BDH+g==} resolution: {integrity: sha512-urTSINYfAYgcbLb0yDQ6egFm6h3Mo1DcF9EkyXSRjjzdHbsulg01qhwWuXdOoUBuTkbQ80KDboXa0vFJ+BDH+g==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@ -1478,8 +1504,8 @@ packages:
lru-cache@5.1.1: lru-cache@5.1.1:
resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==} resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==}
lucide-solid@0.468.0: lucide-solid@0.469.0:
resolution: {integrity: sha512-saTgTS9QvkDdWMl2mHxxubz1A8+3hfbdKeZ2KbluONqBhxwncFDyYQCqfsYMzyrGn3JPRy8O6ha0N9qM0TsSSA==} resolution: {integrity: sha512-kBZl5AFg02g/wcwaapTwOwjHw0VvyyFmZm3BE6McKjs0GjiauWtjYbrhf2bCtDpScEtcinhIG/LpRExBlIV3fA==}
peerDependencies: peerDependencies:
solid-js: ^1.4.7 solid-js: ^1.4.7
@ -1739,10 +1765,6 @@ packages:
resolution: {integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==} resolution: {integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==}
engines: {node: '>=8'} engines: {node: '>=8'}
resolve@1.22.8:
resolution: {integrity: sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==}
hasBin: true
resolve@1.22.9: resolve@1.22.9:
resolution: {integrity: sha512-QxrmX1DzraFIi9PxdG5VkRfRwIgjwyud+z/iBwfRRrVmHc+P9Q7u2lSSpQ6bjr2gy5lrqIiU9vb6iAeGf2400A==} resolution: {integrity: sha512-QxrmX1DzraFIi9PxdG5VkRfRwIgjwyud+z/iBwfRRrVmHc+P9Q7u2lSSpQ6bjr2gy5lrqIiU9vb6iAeGf2400A==}
hasBin: true hasBin: true
@ -1758,8 +1780,8 @@ packages:
rfdc@1.4.1: rfdc@1.4.1:
resolution: {integrity: sha512-q1b3N5QkRUWUl7iyylaaj3kOpIT0N2i9MqIEQXP73GVsN9cw3fdx8X63cEmWhJGi2PPCF23Ijp7ktmd39rawIA==} resolution: {integrity: sha512-q1b3N5QkRUWUl7iyylaaj3kOpIT0N2i9MqIEQXP73GVsN9cw3fdx8X63cEmWhJGi2PPCF23Ijp7ktmd39rawIA==}
rollup@4.28.1: rollup@4.29.1:
resolution: {integrity: sha512-61fXYl/qNVinKmGSTHAZ6Yy8I3YIJC/r2m9feHo6SwVAVcLT5MPwOUFe7EuURA/4m0NR8lXG4BBXuo/IZEsjMg==} resolution: {integrity: sha512-RaJ45M/kmJUzSWDs1Nnd5DdV4eerC98idtUOVr6FfKcgxqvjwHmxc5upLF9qZU9EpsVzzhleFahrT3shLuJzIw==}
engines: {node: '>=18.0.0', npm: '>=8.0.0'} engines: {node: '>=18.0.0', npm: '>=8.0.0'}
hasBin: true hasBin: true
@ -1849,9 +1871,6 @@ packages:
resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==}
engines: {node: '>= 10.x'} engines: {node: '>= 10.x'}
sql.js@1.12.0:
resolution: {integrity: sha512-Bi+43yMx/tUFZVYD4AUscmdL6NHn3gYQ+CM+YheFWLftOmrEC/Mz6Yh7E96Y2WDHYz3COSqT+LP6Z79zgrwJlA==}
string-argv@0.3.2: string-argv@0.3.2:
resolution: {integrity: sha512-aqD2Q0144Z+/RqG52NeHEkZauTAUWJO8c6yTftGJKO3Tja5tUgIfmIl6kExvhtxSDP7fXB6DvzkfMpCd/F3G+Q==} resolution: {integrity: sha512-aqD2Q0144Z+/RqG52NeHEkZauTAUWJO8c6yTftGJKO3Tja5tUgIfmIl6kExvhtxSDP7fXB6DvzkfMpCd/F3G+Q==}
engines: {node: '>=0.6.19'} engines: {node: '>=0.6.19'}
@ -1904,8 +1923,8 @@ packages:
resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
tailwind-merge@2.5.5: tailwind-merge@2.6.0:
resolution: {integrity: sha512-0LXunzzAZzo0tEPxV3I297ffKZPlKDrjj7NXphC8V5ak9yHC5zRmxnOe2m/Rd/7ivsOMJe3JZ2JVocoDdQTRBA==} resolution: {integrity: sha512-P+Vu1qXfzediirmHOC3xKGAYeZtPcV9g76X+xg2FD4tYgR71ewMA35Y3sCz3zhiN/dwefRpJX0yBcgwi1fXNQA==}
tailwindcss-animate@1.0.7: tailwindcss-animate@1.0.7:
resolution: {integrity: sha512-bl6mpH3T7I3UFxuvDEXLxy/VuFxBk5bbzplh7tXI68mwMokNYd1t9qPBHlnyTwfa4JGC4zP516I1hYYtQ/vspA==} resolution: {integrity: sha512-bl6mpH3T7I3UFxuvDEXLxy/VuFxBk5bbzplh7tXI68mwMokNYd1t9qPBHlnyTwfa4JGC4zP516I1hYYtQ/vspA==}
@ -1988,8 +2007,13 @@ packages:
'@testing-library/jest-dom': '@testing-library/jest-dom':
optional: true optional: true
vite@6.0.4: vite-plugin-wasm@3.4.1:
resolution: {integrity: sha512-zwlH6ar+6o6b4Wp+ydhtIKLrGM/LoqZzcdVmkGAFun0KHTzIzjh+h0kungEx7KJg/PYnC80I4TII9WkjciSR6Q==} resolution: {integrity: sha512-ja3nSo2UCkVeitltJGkS3pfQHAanHv/DqGatdI39ja6McgABlpsZ5hVgl6wuR8Qx5etY3T5qgDQhOWzc5RReZA==}
peerDependencies:
vite: ^2 || ^3 || ^4 || ^5 || ^6
vite@6.0.6:
resolution: {integrity: sha512-NSjmUuckPmDU18bHz7QZ+bTYhRR0iA72cs2QAxCqDpafJ0S6qetco0LB3WW2OxlMHS0JmAv+yZ/R3uPmMyGTjQ==}
engines: {node: ^18.0.0 || ^20.0.0 || >=22.0.0} engines: {node: ^18.0.0 || ^20.0.0 || >=22.0.0}
hasBin: true hasBin: true
peerDependencies: peerDependencies:
@ -2086,7 +2110,7 @@ snapshots:
'@ampproject/remapping@2.3.0': '@ampproject/remapping@2.3.0':
dependencies: dependencies:
'@jridgewell/gen-mapping': 0.3.5 '@jridgewell/gen-mapping': 0.3.8
'@jridgewell/trace-mapping': 0.3.25 '@jridgewell/trace-mapping': 0.3.25
'@babel/code-frame@7.26.2': '@babel/code-frame@7.26.2':
@ -2121,7 +2145,7 @@ snapshots:
dependencies: dependencies:
'@babel/parser': 7.26.3 '@babel/parser': 7.26.3
'@babel/types': 7.26.3 '@babel/types': 7.26.3
'@jridgewell/gen-mapping': 0.3.5 '@jridgewell/gen-mapping': 0.3.8
'@jridgewell/trace-mapping': 0.3.25 '@jridgewell/trace-mapping': 0.3.25
jsesc: 3.0.2 jsesc: 3.0.2
@ -2348,76 +2372,81 @@ snapshots:
'@floating-ui/dom': 1.6.12 '@floating-ui/dom': 1.6.12
solid-js: 1.9.3 solid-js: 1.9.3
'@esbuild/aix-ppc64@0.24.0': '@duskflower/signal-decrypt-backup-wasm@0.2.0': {}
'@esbuild/aix-ppc64@0.24.2':
optional: true optional: true
'@esbuild/android-arm64@0.24.0': '@esbuild/android-arm64@0.24.2':
optional: true optional: true
'@esbuild/android-arm@0.24.0': '@esbuild/android-arm@0.24.2':
optional: true optional: true
'@esbuild/android-x64@0.24.0': '@esbuild/android-x64@0.24.2':
optional: true optional: true
'@esbuild/darwin-arm64@0.24.0': '@esbuild/darwin-arm64@0.24.2':
optional: true optional: true
'@esbuild/darwin-x64@0.24.0': '@esbuild/darwin-x64@0.24.2':
optional: true optional: true
'@esbuild/freebsd-arm64@0.24.0': '@esbuild/freebsd-arm64@0.24.2':
optional: true optional: true
'@esbuild/freebsd-x64@0.24.0': '@esbuild/freebsd-x64@0.24.2':
optional: true optional: true
'@esbuild/linux-arm64@0.24.0': '@esbuild/linux-arm64@0.24.2':
optional: true optional: true
'@esbuild/linux-arm@0.24.0': '@esbuild/linux-arm@0.24.2':
optional: true optional: true
'@esbuild/linux-ia32@0.24.0': '@esbuild/linux-ia32@0.24.2':
optional: true optional: true
'@esbuild/linux-loong64@0.24.0': '@esbuild/linux-loong64@0.24.2':
optional: true optional: true
'@esbuild/linux-mips64el@0.24.0': '@esbuild/linux-mips64el@0.24.2':
optional: true optional: true
'@esbuild/linux-ppc64@0.24.0': '@esbuild/linux-ppc64@0.24.2':
optional: true optional: true
'@esbuild/linux-riscv64@0.24.0': '@esbuild/linux-riscv64@0.24.2':
optional: true optional: true
'@esbuild/linux-s390x@0.24.0': '@esbuild/linux-s390x@0.24.2':
optional: true optional: true
'@esbuild/linux-x64@0.24.0': '@esbuild/linux-x64@0.24.2':
optional: true optional: true
'@esbuild/netbsd-x64@0.24.0': '@esbuild/netbsd-arm64@0.24.2':
optional: true optional: true
'@esbuild/openbsd-arm64@0.24.0': '@esbuild/netbsd-x64@0.24.2':
optional: true optional: true
'@esbuild/openbsd-x64@0.24.0': '@esbuild/openbsd-arm64@0.24.2':
optional: true optional: true
'@esbuild/sunos-x64@0.24.0': '@esbuild/openbsd-x64@0.24.2':
optional: true optional: true
'@esbuild/win32-arm64@0.24.0': '@esbuild/sunos-x64@0.24.2':
optional: true optional: true
'@esbuild/win32-ia32@0.24.0': '@esbuild/win32-arm64@0.24.2':
optional: true optional: true
'@esbuild/win32-x64@0.24.0': '@esbuild/win32-ia32@0.24.2':
optional: true
'@esbuild/win32-x64@0.24.2':
optional: true optional: true
'@floating-ui/core@1.6.8': '@floating-ui/core@1.6.8':
@ -2448,12 +2477,6 @@ snapshots:
wrap-ansi: 8.1.0 wrap-ansi: 8.1.0
wrap-ansi-cjs: wrap-ansi@7.0.0 wrap-ansi-cjs: wrap-ansi@7.0.0
'@jridgewell/gen-mapping@0.3.5':
dependencies:
'@jridgewell/set-array': 1.2.1
'@jridgewell/sourcemap-codec': 1.5.0
'@jridgewell/trace-mapping': 0.3.25
'@jridgewell/gen-mapping@0.3.8': '@jridgewell/gen-mapping@0.3.8':
dependencies: dependencies:
'@jridgewell/set-array': 1.2.1 '@jridgewell/set-array': 1.2.1
@ -2515,61 +2538,61 @@ snapshots:
'@pkgjs/parseargs@0.11.0': '@pkgjs/parseargs@0.11.0':
optional: true optional: true
'@rollup/rollup-android-arm-eabi@4.28.1': '@rollup/rollup-android-arm-eabi@4.29.1':
optional: true optional: true
'@rollup/rollup-android-arm64@4.28.1': '@rollup/rollup-android-arm64@4.29.1':
optional: true optional: true
'@rollup/rollup-darwin-arm64@4.28.1': '@rollup/rollup-darwin-arm64@4.29.1':
optional: true optional: true
'@rollup/rollup-darwin-x64@4.28.1': '@rollup/rollup-darwin-x64@4.29.1':
optional: true optional: true
'@rollup/rollup-freebsd-arm64@4.28.1': '@rollup/rollup-freebsd-arm64@4.29.1':
optional: true optional: true
'@rollup/rollup-freebsd-x64@4.28.1': '@rollup/rollup-freebsd-x64@4.29.1':
optional: true optional: true
'@rollup/rollup-linux-arm-gnueabihf@4.28.1': '@rollup/rollup-linux-arm-gnueabihf@4.29.1':
optional: true optional: true
'@rollup/rollup-linux-arm-musleabihf@4.28.1': '@rollup/rollup-linux-arm-musleabihf@4.29.1':
optional: true optional: true
'@rollup/rollup-linux-arm64-gnu@4.28.1': '@rollup/rollup-linux-arm64-gnu@4.29.1':
optional: true optional: true
'@rollup/rollup-linux-arm64-musl@4.28.1': '@rollup/rollup-linux-arm64-musl@4.29.1':
optional: true optional: true
'@rollup/rollup-linux-loongarch64-gnu@4.28.1': '@rollup/rollup-linux-loongarch64-gnu@4.29.1':
optional: true optional: true
'@rollup/rollup-linux-powerpc64le-gnu@4.28.1': '@rollup/rollup-linux-powerpc64le-gnu@4.29.1':
optional: true optional: true
'@rollup/rollup-linux-riscv64-gnu@4.28.1': '@rollup/rollup-linux-riscv64-gnu@4.29.1':
optional: true optional: true
'@rollup/rollup-linux-s390x-gnu@4.28.1': '@rollup/rollup-linux-s390x-gnu@4.29.1':
optional: true optional: true
'@rollup/rollup-linux-x64-gnu@4.28.1': '@rollup/rollup-linux-x64-gnu@4.29.1':
optional: true optional: true
'@rollup/rollup-linux-x64-musl@4.28.1': '@rollup/rollup-linux-x64-musl@4.29.1':
optional: true optional: true
'@rollup/rollup-win32-arm64-msvc@4.28.1': '@rollup/rollup-win32-arm64-msvc@4.29.1':
optional: true optional: true
'@rollup/rollup-win32-ia32-msvc@4.28.1': '@rollup/rollup-win32-ia32-msvc@4.29.1':
optional: true optional: true
'@rollup/rollup-win32-x64-msvc@4.28.1': '@rollup/rollup-win32-x64-msvc@4.29.1':
optional: true optional: true
'@solid-primitives/event-listener@2.3.3(solid-js@1.9.3)': '@solid-primitives/event-listener@2.3.3(solid-js@1.9.3)':
@ -2622,6 +2645,11 @@ snapshots:
'@solid-primitives/utils': 6.2.3(solid-js@1.9.3) '@solid-primitives/utils': 6.2.3(solid-js@1.9.3)
solid-js: 1.9.3 solid-js: 1.9.3
'@solid-primitives/storage@4.2.1(solid-js@1.9.3)':
dependencies:
'@solid-primitives/utils': 6.2.3(solid-js@1.9.3)
solid-js: 1.9.3
'@solid-primitives/trigger@1.1.0(solid-js@1.9.3)': '@solid-primitives/trigger@1.1.0(solid-js@1.9.3)':
dependencies: dependencies:
'@solid-primitives/utils': 6.2.3(solid-js@1.9.3) '@solid-primitives/utils': 6.2.3(solid-js@1.9.3)
@ -2643,6 +2671,8 @@ snapshots:
dependencies: dependencies:
solid-js: 1.9.3 solid-js: 1.9.3
'@sqlite.org/sqlite-wasm@3.47.2-build1': {}
'@swc/helpers@0.5.15': '@swc/helpers@0.5.15':
dependencies: dependencies:
tslib: 2.8.1 tslib: 2.8.1
@ -3018,32 +3048,33 @@ snapshots:
dependencies: dependencies:
is-arrayish: 0.2.1 is-arrayish: 0.2.1
esbuild@0.24.0: esbuild@0.24.2:
optionalDependencies: optionalDependencies:
'@esbuild/aix-ppc64': 0.24.0 '@esbuild/aix-ppc64': 0.24.2
'@esbuild/android-arm': 0.24.0 '@esbuild/android-arm': 0.24.2
'@esbuild/android-arm64': 0.24.0 '@esbuild/android-arm64': 0.24.2
'@esbuild/android-x64': 0.24.0 '@esbuild/android-x64': 0.24.2
'@esbuild/darwin-arm64': 0.24.0 '@esbuild/darwin-arm64': 0.24.2
'@esbuild/darwin-x64': 0.24.0 '@esbuild/darwin-x64': 0.24.2
'@esbuild/freebsd-arm64': 0.24.0 '@esbuild/freebsd-arm64': 0.24.2
'@esbuild/freebsd-x64': 0.24.0 '@esbuild/freebsd-x64': 0.24.2
'@esbuild/linux-arm': 0.24.0 '@esbuild/linux-arm': 0.24.2
'@esbuild/linux-arm64': 0.24.0 '@esbuild/linux-arm64': 0.24.2
'@esbuild/linux-ia32': 0.24.0 '@esbuild/linux-ia32': 0.24.2
'@esbuild/linux-loong64': 0.24.0 '@esbuild/linux-loong64': 0.24.2
'@esbuild/linux-mips64el': 0.24.0 '@esbuild/linux-mips64el': 0.24.2
'@esbuild/linux-ppc64': 0.24.0 '@esbuild/linux-ppc64': 0.24.2
'@esbuild/linux-riscv64': 0.24.0 '@esbuild/linux-riscv64': 0.24.2
'@esbuild/linux-s390x': 0.24.0 '@esbuild/linux-s390x': 0.24.2
'@esbuild/linux-x64': 0.24.0 '@esbuild/linux-x64': 0.24.2
'@esbuild/netbsd-x64': 0.24.0 '@esbuild/netbsd-arm64': 0.24.2
'@esbuild/openbsd-arm64': 0.24.0 '@esbuild/netbsd-x64': 0.24.2
'@esbuild/openbsd-x64': 0.24.0 '@esbuild/openbsd-arm64': 0.24.2
'@esbuild/sunos-x64': 0.24.0 '@esbuild/openbsd-x64': 0.24.2
'@esbuild/win32-arm64': 0.24.0 '@esbuild/sunos-x64': 0.24.2
'@esbuild/win32-ia32': 0.24.0 '@esbuild/win32-arm64': 0.24.2
'@esbuild/win32-x64': 0.24.0 '@esbuild/win32-ia32': 0.24.2
'@esbuild/win32-x64': 0.24.2
escalade@3.2.0: {} escalade@3.2.0: {}
@ -3209,10 +3240,6 @@ snapshots:
dependencies: dependencies:
binary-extensions: 2.3.0 binary-extensions: 2.3.0
is-core-module@2.15.1:
dependencies:
hasown: 2.0.2
is-core-module@2.16.0: is-core-module@2.16.0:
dependencies: dependencies:
hasown: 2.0.2 hasown: 2.0.2
@ -3356,7 +3383,7 @@ snapshots:
dependencies: dependencies:
yallist: 3.1.1 yallist: 3.1.1
lucide-solid@0.468.0(solid-js@1.9.3): lucide-solid@0.469.0(solid-js@1.9.3):
dependencies: dependencies:
solid-js: 1.9.3 solid-js: 1.9.3
@ -3571,7 +3598,7 @@ snapshots:
rechoir@0.6.2: rechoir@0.6.2:
dependencies: dependencies:
resolve: 1.22.8 resolve: 1.22.9
require-directory@2.1.1: {} require-directory@2.1.1: {}
@ -3581,12 +3608,6 @@ snapshots:
resolve-from@5.0.0: {} resolve-from@5.0.0: {}
resolve@1.22.8:
dependencies:
is-core-module: 2.15.1
path-parse: 1.0.7
supports-preserve-symlinks-flag: 1.0.0
resolve@1.22.9: resolve@1.22.9:
dependencies: dependencies:
is-core-module: 2.16.0 is-core-module: 2.16.0
@ -3602,29 +3623,29 @@ snapshots:
rfdc@1.4.1: {} rfdc@1.4.1: {}
rollup@4.28.1: rollup@4.29.1:
dependencies: dependencies:
'@types/estree': 1.0.6 '@types/estree': 1.0.6
optionalDependencies: optionalDependencies:
'@rollup/rollup-android-arm-eabi': 4.28.1 '@rollup/rollup-android-arm-eabi': 4.29.1
'@rollup/rollup-android-arm64': 4.28.1 '@rollup/rollup-android-arm64': 4.29.1
'@rollup/rollup-darwin-arm64': 4.28.1 '@rollup/rollup-darwin-arm64': 4.29.1
'@rollup/rollup-darwin-x64': 4.28.1 '@rollup/rollup-darwin-x64': 4.29.1
'@rollup/rollup-freebsd-arm64': 4.28.1 '@rollup/rollup-freebsd-arm64': 4.29.1
'@rollup/rollup-freebsd-x64': 4.28.1 '@rollup/rollup-freebsd-x64': 4.29.1
'@rollup/rollup-linux-arm-gnueabihf': 4.28.1 '@rollup/rollup-linux-arm-gnueabihf': 4.29.1
'@rollup/rollup-linux-arm-musleabihf': 4.28.1 '@rollup/rollup-linux-arm-musleabihf': 4.29.1
'@rollup/rollup-linux-arm64-gnu': 4.28.1 '@rollup/rollup-linux-arm64-gnu': 4.29.1
'@rollup/rollup-linux-arm64-musl': 4.28.1 '@rollup/rollup-linux-arm64-musl': 4.29.1
'@rollup/rollup-linux-loongarch64-gnu': 4.28.1 '@rollup/rollup-linux-loongarch64-gnu': 4.29.1
'@rollup/rollup-linux-powerpc64le-gnu': 4.28.1 '@rollup/rollup-linux-powerpc64le-gnu': 4.29.1
'@rollup/rollup-linux-riscv64-gnu': 4.28.1 '@rollup/rollup-linux-riscv64-gnu': 4.29.1
'@rollup/rollup-linux-s390x-gnu': 4.28.1 '@rollup/rollup-linux-s390x-gnu': 4.29.1
'@rollup/rollup-linux-x64-gnu': 4.28.1 '@rollup/rollup-linux-x64-gnu': 4.29.1
'@rollup/rollup-linux-x64-musl': 4.28.1 '@rollup/rollup-linux-x64-musl': 4.29.1
'@rollup/rollup-win32-arm64-msvc': 4.28.1 '@rollup/rollup-win32-arm64-msvc': 4.29.1
'@rollup/rollup-win32-ia32-msvc': 4.28.1 '@rollup/rollup-win32-ia32-msvc': 4.29.1
'@rollup/rollup-win32-x64-msvc': 4.28.1 '@rollup/rollup-win32-x64-msvc': 4.29.1
fsevents: 2.3.3 fsevents: 2.3.3
run-parallel@1.2.0: run-parallel@1.2.0:
@ -3706,8 +3727,6 @@ snapshots:
split2@4.2.0: {} split2@4.2.0: {}
sql.js@1.12.0: {}
string-argv@0.3.2: {} string-argv@0.3.2: {}
string-width@4.2.3: string-width@4.2.3:
@ -3764,7 +3783,7 @@ snapshots:
supports-preserve-symlinks-flag@1.0.0: {} supports-preserve-symlinks-flag@1.0.0: {}
tailwind-merge@2.5.5: {} tailwind-merge@2.6.0: {}
tailwindcss-animate@1.0.7(tailwindcss@3.4.17): tailwindcss-animate@1.0.7(tailwindcss@3.4.17):
dependencies: dependencies:
@ -3854,7 +3873,7 @@ snapshots:
validate-html-nesting@1.2.2: {} validate-html-nesting@1.2.2: {}
vite-plugin-solid@2.11.0(solid-js@1.9.3)(vite@6.0.4(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)): vite-plugin-solid@2.11.0(solid-js@1.9.3)(vite@6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)):
dependencies: dependencies:
'@babel/core': 7.26.0 '@babel/core': 7.26.0
'@types/babel__core': 7.20.5 '@types/babel__core': 7.20.5
@ -3862,25 +3881,29 @@ snapshots:
merge-anything: 5.1.7 merge-anything: 5.1.7
solid-js: 1.9.3 solid-js: 1.9.3
solid-refresh: 0.6.3(solid-js@1.9.3) solid-refresh: 0.6.3(solid-js@1.9.3)
vite: 6.0.4(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1) vite: 6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)
vitefu: 1.0.4(vite@6.0.4(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)) vitefu: 1.0.4(vite@6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1))
transitivePeerDependencies: transitivePeerDependencies:
- supports-color - supports-color
vite@6.0.4(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1): vite-plugin-wasm@3.4.1(vite@6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)):
dependencies: dependencies:
esbuild: 0.24.0 vite: 6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)
vite@6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1):
dependencies:
esbuild: 0.24.2
postcss: 8.4.49 postcss: 8.4.49
rollup: 4.28.1 rollup: 4.29.1
optionalDependencies: optionalDependencies:
'@types/node': 22.10.2 '@types/node': 22.10.2
fsevents: 2.3.3 fsevents: 2.3.3
jiti: 2.4.2 jiti: 2.4.2
yaml: 2.6.1 yaml: 2.6.1
vitefu@1.0.4(vite@6.0.4(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)): vitefu@1.0.4(vite@6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)):
optionalDependencies: optionalDependencies:
vite: 6.0.4(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1) vite: 6.0.6(@types/node@22.10.2)(jiti@2.4.2)(yaml@2.6.1)
which@2.0.2: which@2.0.2:
dependencies: dependencies:

@ -0,0 +1 @@
Subproject commit 6559df90b63429fcd24cf42fb4ebd91ecdf4d38e

View file

@ -1,5 +1,5 @@
import { type Component } from "solid-js";
import { Route } from "@solidjs/router"; import { Route } from "@solidjs/router";
import { type Component } from "solid-js";
import { DmId, GroupId, Home, Overview, preloadDmId } from "./pages"; import { DmId, GroupId, Home, Overview, preloadDmId } from "./pages";
import "./app.css"; import "./app.css";

View file

@ -0,0 +1,34 @@
import type { Component, JSX, ValidComponent } from "solid-js"
import { splitProps } from "solid-js"
import type { PolymorphicProps } from "@kobalte/core/polymorphic"
import * as ProgressPrimitive from "@kobalte/core/progress"
import { Label } from "~/components/ui/label"
type ProgressRootProps<T extends ValidComponent = "div"> =
ProgressPrimitive.ProgressRootProps<T> & { children?: JSX.Element }
const Progress = <T extends ValidComponent = "div">(
props: PolymorphicProps<T, ProgressRootProps<T>>
) => {
const [local, others] = splitProps(props as ProgressRootProps, ["children"])
return (
<ProgressPrimitive.Root {...others}>
{local.children}
<ProgressPrimitive.Track class="relative h-2 w-full overflow-hidden rounded-full bg-secondary">
<ProgressPrimitive.Fill class="h-full w-[var(--kb-progress-fill-width)] flex-1 bg-primary transition-all" />
</ProgressPrimitive.Track>
</ProgressPrimitive.Root>
)
}
const ProgressLabel: Component<ProgressPrimitive.ProgressLabelProps> = (props) => {
return <ProgressPrimitive.Label as={Label} {...props} />
}
const ProgressValueLabel: Component<ProgressPrimitive.ProgressValueLabelProps> = (props) => {
return <ProgressPrimitive.ValueLabel as={Label} {...props} />
}
export { Progress, ProgressLabel, ProgressValueLabel }

View file

@ -1,17 +1,52 @@
import { sql, type NotNull } from "kysely"; import { sql, type NotNull } from "kysely";
import { db, kyselyDb, SELF_ID, setDbHash } from "./db";
import { cached } from "./lib/db-cache"; import { cached } from "./lib/db-cache";
import { kyselyDb, SELF_ID } from "./db"; import { hashString } from "./lib/hash";
export const loadDb = (
statements: string[],
progressCallback?: (percentage: number) => void,
) => {
const length = statements.length;
let percentage = 0;
for (let i = 0; i < length; i++) {
const statement = statements[i];
const newPercentage = Math.round((i / length) * 100);
try {
db.exec(statement);
if (newPercentage !== percentage) {
progressCallback?.(newPercentage);
percentage = newPercentage;
}
} catch (e) {
throw new Error(`statement failed: ${statement}`, {
cause: e,
});
}
}
setDbHash(hashString(statements.join()));
};
const allThreadsOverviewQueryRaw = () => const allThreadsOverviewQueryRaw = () =>
kyselyDb() kyselyDb
?.selectFrom("thread") .selectFrom("thread")
.innerJoin( .innerJoin(
(eb) => (eb) =>
eb eb
.selectFrom("message") .selectFrom("message")
.select((eb) => ["message.thread_id", eb.fn.countAll().as("message_count")]) .select((eb) => [
"message.thread_id",
eb.fn.countAll().as("message_count"),
])
.where((eb) => { .where((eb) => {
return eb.and([eb("message.body", "is not", null), eb("message.body", "is not", "")]); return eb.and([
eb("message.body", "is not", null),
eb("message.body", "is not", ""),
]);
}) })
.groupBy("message.thread_id") .groupBy("message.thread_id")
.as("message"), .as("message"),
@ -41,8 +76,8 @@ const allThreadsOverviewQueryRaw = () =>
export const allThreadsOverviewQuery = cached(allThreadsOverviewQueryRaw); export const allThreadsOverviewQuery = cached(allThreadsOverviewQueryRaw);
const overallSentMessagesQueryRaw = (recipientId: number) => const overallSentMessagesQueryRaw = (recipientId: number) =>
kyselyDb() kyselyDb
?.selectFrom("message") .selectFrom("message")
.select((eb) => eb.fn.countAll().as("messageCount")) .select((eb) => eb.fn.countAll().as("messageCount"))
.where((eb) => .where((eb) =>
eb.and([ eb.and([
@ -56,8 +91,8 @@ const overallSentMessagesQueryRaw = (recipientId: number) =>
export const overallSentMessagesQuery = cached(overallSentMessagesQueryRaw); export const overallSentMessagesQuery = cached(overallSentMessagesQueryRaw);
const dmPartnerRecipientQueryRaw = (dmId: number) => const dmPartnerRecipientQueryRaw = (dmId: number) =>
kyselyDb() kyselyDb
?.selectFrom("recipient") .selectFrom("recipient")
.select([ .select([
"recipient._id", "recipient._id",
"recipient.system_joined_name", "recipient.system_joined_name",
@ -65,7 +100,9 @@ const dmPartnerRecipientQueryRaw = (dmId: number) =>
"recipient.nickname_joined_name", "recipient.nickname_joined_name",
]) ])
.innerJoin("thread", "recipient._id", "thread.recipient_id") .innerJoin("thread", "recipient._id", "thread.recipient_id")
.where((eb) => eb.and([eb("thread._id", "=", dmId), eb("recipient._id", "!=", SELF_ID)])) .where((eb) =>
eb.and([eb("thread._id", "=", dmId), eb("recipient._id", "!=", SELF_ID)]),
)
.$narrowType<{ .$narrowType<{
_id: number; _id: number;
}>() }>()
@ -74,30 +111,47 @@ const dmPartnerRecipientQueryRaw = (dmId: number) =>
export const dmPartnerRecipientQuery = cached(dmPartnerRecipientQueryRaw); export const dmPartnerRecipientQuery = cached(dmPartnerRecipientQueryRaw);
const threadSentMessagesOverviewQueryRaw = (threadId: number) => const threadSentMessagesOverviewQueryRaw = (threadId: number) =>
kyselyDb() kyselyDb
?.selectFrom("message") .selectFrom("message")
.select(["from_recipient_id", sql<string>`datetime(date_sent / 1000, 'unixepoch')`.as("message_datetime")]) .select([
"from_recipient_id",
sql<string>`datetime(date_sent / 1000, 'unixepoch')`.as(
"message_datetime",
),
])
.orderBy(["message_datetime"]) .orderBy(["message_datetime"])
.where((eb) => eb.and([eb("body", "is not", null), eb("body", "!=", ""), eb("thread_id", "=", threadId)])) .where((eb) =>
eb.and([
eb("body", "is not", null),
eb("body", "!=", ""),
eb("thread_id", "=", threadId),
]),
)
.execute(); .execute();
export const threadSentMessagesOverviewQuery = cached(threadSentMessagesOverviewQueryRaw); export const threadSentMessagesOverviewQuery = cached(
threadSentMessagesOverviewQueryRaw,
);
const threadMostUsedWordsQueryRaw = (threadId: number, limit = 10) => const threadMostUsedWordsQueryRaw = (threadId: number, limit = 10) =>
kyselyDb() kyselyDb
?.withRecursive("words", (eb) => { .withRecursive("words", (eb) => {
return eb return eb
.selectFrom("message") .selectFrom("message")
.select([ .select([
sql`LOWER(substr(body, 1, instr(body || " ", " ") - 1))`.as("word"), sql`LOWER(substr(body, 1, instr(body || " ", " ") - 1))`.as("word"),
sql`(substr(body, instr(body || " ", " ") + 1))`.as("rest"), sql`(substr(body, instr(body || " ", " ") + 1))`.as("rest"),
]) ])
.where((eb) => eb.and([eb("body", "is not", null), eb("thread_id", "=", threadId)])) .where((eb) =>
eb.and([eb("body", "is not", null), eb("thread_id", "=", threadId)]),
)
.unionAll((ebInner) => { .unionAll((ebInner) => {
return ebInner return ebInner
.selectFrom("words") .selectFrom("words")
.select([ .select([
sql`LOWER(substr(rest, 1, instr(rest || " ", " ") - 1))`.as("word"), sql`LOWER(substr(rest, 1, instr(rest || " ", " ") - 1))`.as(
"word",
),
sql`(substr(rest, instr(rest || " ", " ") + 1))`.as("rest"), sql`(substr(rest, instr(rest || " ", " ") + 1))`.as("rest"),
]) ])
.where("rest", "<>", ""); .where("rest", "<>", "");

View file

@ -1,50 +1,30 @@
import { createEffect, createMemo, createRoot, createSignal } from "solid-js"; import { makePersisted } from "@solid-primitives/storage";
import sqlite3InitModule from "@sqlite.org/sqlite-wasm";
import { Kysely } from "kysely"; import { Kysely } from "kysely";
import type { DB } from "kysely-codegen"; import type { DB } from "kysely-codegen";
import { SqlJsDialect } from "kysely-wasm"; import { OfficialWasmDialect } from "kysely-wasm";
import initSqlJS, { type Database } from "sql.js"; import { createSignal } from "solid-js";
import workerUrl from "./lib/kysely-official-wasm-worker/worker?url";
import wasmURL from "./assets/sql-wasm.wasm?url";
export const SELF_ID = 2; export const SELF_ID = 2;
export const SQL = await initSqlJS({ const sqlite3 = await sqlite3InitModule({
locateFile: () => wasmURL, print: console.log,
printErr: console.error,
}); });
export const [db, setDb] = createSignal<Database | undefined>(); export const db = new sqlite3.oo1.DB("signal");
const sqlJsDialect = () => { export const worker = new Worker(workerUrl, {
const currentDb = db(); type: "module",
if (currentDb) {
return new SqlJsDialect({
database: currentDb,
});
}
};
export const kyselyDb = createRoot(() => {
createEffect(() => {
const currentDb = db();
if (currentDb) {
currentDb.create_function("is_not_empty", (str: string | null) => {
return str !== null && str !== "";
});
}
}); });
return createMemo(() => { const dialect = new OfficialWasmDialect({
const currentSqlJsDialect = sqlJsDialect(); database: db,
});
if (!currentSqlJsDialect) { export const kyselyDb = new Kysely<DB>({
return; dialect,
} });
return new Kysely<DB>({ export const [dbHash, setDbHash] = makePersisted(createSignal<number>());
dialect: currentSqlJsDialect,
});
});
});

View file

@ -1,11 +1,9 @@
/* @refresh reload */ /* @refresh reload */
import { render } from "solid-js/web";
import { Router, useNavigate } from "@solidjs/router";
import { MetaProvider } from "@solidjs/meta"; import { MetaProvider } from "@solidjs/meta";
import { Router } from "@solidjs/router";
import { render } from "solid-js/web";
import App from "./App"; import App from "./App";
import { createEffect } from "solid-js";
import { db } from "./db";
const root = document.getElementById("root"); const root = document.getElementById("root");
@ -21,18 +19,18 @@ if (root) {
<div class="mx-auto max-w-screen-2xl"> <div class="mx-auto max-w-screen-2xl">
<MetaProvider> <MetaProvider>
<Router <Router
root={(props) => { // root={(props) => {
const navigate = useNavigate(); // const navigate = useNavigate();
const { pathname } = props.location; // const { pathname } = props.location;
createEffect(() => { // createEffect(() => {
if (!db() && pathname !== "/") { // if (!db() && pathname !== "/") {
navigate("/"); // navigate("/");
} // }
}); // });
return props.children; // return props.children;
}} // }}
> >
<App /> <App />
</Router> </Router>

View file

@ -1,9 +1,9 @@
import { on, createSignal, createEffect, createRoot, createMemo } from "solid-js"; import { deserialize, serialize } from "seroval";
import { serialize, deserialize } from "seroval"; import { createEffect, createMemo, createRoot, on } from "solid-js";
import { createSignaledWorker } from "@solid-primitives/workers"; import { dbHash } from "~/db";
import { db } from "~/db"; import { hashString } from "./hash";
const DATABASE_HASH_PREFIX = "database"; export const DATABASE_HASH_PREFIX = "database";
// clear the cache on new session so that selecting a different database does not result in wrong cache entries // clear the cache on new session so that selecting a different database does not result in wrong cache entries
const clearDbCache = () => { const clearDbCache = () => {
@ -16,64 +16,32 @@ const clearDbCache = () => {
} }
}; };
// https://stackoverflow.com/a/7616484 let prevDbHash = dbHash();
const hashString = (str: string) => {
let hash = 0,
i,
chr;
if (str.length === 0) return hash;
for (i = 0; i < str.length; i++) {
chr = str.charCodeAt(i);
hash = (hash << 5) - hash + chr;
hash |= 0; // Convert to 32bit integer
}
return hash;
};
const HASH_STORE_KEY = `${DATABASE_HASH_PREFIX}_hash`;
createRoot(() => { createRoot(() => {
const [dbHash, setDbHash] = createSignal(localStorage.getItem(HASH_STORE_KEY)); createEffect(() => {
on(
// offloaded because this can take a long time (>1s easily) and would block the mainthread dbHash,
createSignaledWorker({ (currentDbHash) => {
input: db, if (currentDbHash && currentDbHash !== prevDbHash) {
output: setDbHash, prevDbHash = currentDbHash;
func: function hashDb(currentDb: ReturnType<typeof db>) { clearDbCache();
const hashString = (str: string) => {
let hash = 0,
i,
chr;
if (str.length === 0) return hash;
for (i = 0; i < str.length; i++) {
chr = str.charCodeAt(i);
hash = (hash << 5) - hash + chr;
hash |= 0; // Convert to 32bit integer
}
return hash;
};
if (currentDb?.export) {
return hashString(new TextDecoder().decode(currentDb.export())).toString();
} }
}, },
}); {
defer: true,
createEffect(() => { },
on(dbHash, (currentDbHash) => { );
if (currentDbHash) {
clearDbCache();
localStorage.setItem(HASH_STORE_KEY, currentDbHash);
}
});
}); });
}); });
class LocalStorageCacheAdapter { class LocalStorageCacheAdapter {
keys = new Set<string>(Object.keys(localStorage).filter((key) => key.startsWith(this.prefix))); keys = new Set<string>(
Object.keys(localStorage).filter((key) => key.startsWith(this.prefix)),
);
prefix = "database"; prefix = "database";
#dbLoaded = createMemo(() => !!db()); // TODO: real way of detecting if the db is loaded, on loading the db and opfs (if persisted db?)
#dbLoaded = createMemo(() => !!dbHash());
#createKey(cacheName: string, key: string): string { #createKey(cacheName: string, key: string): string {
return `${this.prefix}-${cacheName}-${key}`; return `${this.prefix}-${cacheName}-${key}`;
@ -86,7 +54,10 @@ class LocalStorageCacheAdapter {
try { try {
localStorage.setItem(fullKey, serialize(value)); localStorage.setItem(fullKey, serialize(value));
} catch (error: unknown) { } catch (error: unknown) {
if (error instanceof DOMException && error.name === "QUOTA_EXCEEDED_ERR") { if (
error instanceof DOMException &&
error.name === "QUOTA_EXCEEDED_ERR"
) {
console.error("Storage quota exceeded, not caching new function calls"); console.error("Storage quota exceeded, not caching new function calls");
} else { } else {
console.error(error); console.error(error);
@ -146,7 +117,10 @@ const createHashKey = (...args: unknown[]) => {
return hashString(stringToHash); return hashString(stringToHash);
}; };
export const cached = <T extends unknown[], R, TT>(fn: (...args: T) => R, self?: ThisType<TT>): ((...args: T) => R) => { export const cached = <T extends unknown[], R, TT>(
fn: (...args: T) => R,
self?: ThisType<TT>,
): ((...args: T) => R) => {
const cacheName = hashString(fn.toString()).toString(); const cacheName = hashString(fn.toString()).toString();
// important to return a promise on follow-up calls even if the data is immediately available // important to return a promise on follow-up calls even if the data is immediately available

47
src/lib/decryptor.ts Normal file
View file

@ -0,0 +1,47 @@
import {
BackupDecryptor,
type DecryptionResult,
} from "@duskflower/signal-decrypt-backup-wasm";
const CHUNK_SIZE = 1024 * 1024 * 40; // 40MB chunks
export async function decryptBackup(
file: File,
passphrase: string,
progressCallback: (progress: number) => void,
): Promise<DecryptionResult> {
const fileSize = file.size;
const decryptor = new BackupDecryptor();
decryptor.set_progress_callback(fileSize, progressCallback);
let offset = 0;
try {
while (offset < file.size) {
const chunk = file.slice(offset, offset + CHUNK_SIZE);
const arrayBuffer = await chunk.arrayBuffer();
const uint8Array = new Uint8Array(arrayBuffer);
decryptor.feed_data(uint8Array);
let done = false;
while (!done) {
try {
done = decryptor.process_chunk(passphrase);
} catch (e) {
console.error("Error processing chunk:", e);
throw e;
}
}
offset += CHUNK_SIZE;
}
const result = decryptor.finish();
return result;
} catch (e) {
console.error("Decryption failed:", e);
throw e;
}
}

13
src/lib/hash.ts Normal file
View file

@ -0,0 +1,13 @@
// https://stackoverflow.com/a/7616484
export const hashString = (str: string) => {
let hash = 0,
i,
chr;
if (str.length === 0) return hash;
for (i = 0; i < str.length; i++) {
chr = str.charCodeAt(i);
hash = (hash << 5) - hash + chr;
hash |= 0; // Convert to 32bit integer
}
return hash;
};

View file

@ -1,47 +1,90 @@
import { useNavigate, type RouteSectionProps } from "@solidjs/router";
import { createSignal, Show, type Component, type JSX } from "solid-js"; import { createSignal, Show, type Component, type JSX } from "solid-js";
import { type RouteSectionProps, useNavigate } from "@solidjs/router";
import { Title } from "@solidjs/meta";
import { Portal } from "solid-js/web"; import { Portal } from "solid-js/web";
import { Flex } from "~/components/ui/flex"; import { Flex } from "~/components/ui/flex";
import { Title } from "@solidjs/meta";
import { setDb, SQL } from "~/db"; import {
Progress,
ProgressLabel,
ProgressValueLabel,
} from "~/components/ui/progress";
// import { db } from "~/db";
import { loadDb } from "~/db-queries";
import { decryptBackup } from "~/lib/decryptor";
export const Home: Component<RouteSectionProps> = () => { export const Home: Component<RouteSectionProps> = () => {
const [isLoadingDb, setIsLoadingDb] = createSignal(false); const [decryptionProgress, setDecryptionProgress] = createSignal<number>();
const [isLoadingDatabase, setIsLoadingDatabase] = createSignal(false);
const [passphrase, setPassphrase] = createSignal("");
const navigate = useNavigate(); const navigate = useNavigate();
const onFileChange: JSX.ChangeEventHandler<HTMLInputElement, Event> = (event) => { const onFileChange: JSX.ChangeEventHandler<HTMLInputElement, Event> = (
event,
) => {
const file = event.currentTarget.files?.[0]; const file = event.currentTarget.files?.[0];
if (file) { const currentPassphrase = passphrase();
const reader = new FileReader();
reader.addEventListener("load", () => { if (file && currentPassphrase) {
setIsLoadingDb(true); decryptBackup(file, currentPassphrase, setDecryptionProgress)
.then((result) => {
setDecryptionProgress(undefined);
setIsLoadingDatabase(true);
setTimeout(() => { setTimeout(() => {
const Uints = new Uint8Array(reader.result as ArrayBuffer); loadDb(result.database_statements);
setDb(new SQL.Database(Uints));
setIsLoadingDb(false);
navigate("/overview");
}, 10);
});
reader.readAsArrayBuffer(file); setIsLoadingDatabase(false);
navigate("/overview");
}, 0);
})
.catch((error) => {
console.error("Decryption failed:", error);
});
} }
}; };
return ( return (
<> <>
<Portal> <Portal>
<Show when={isLoadingDb()}> <Flex
<Flex alignItems="center" justifyContent="center" class="fixed inset-0 backdrop-blur-lg backdrop-filter"> flexDirection="col"
<p class="font-bold text-2xl">Loading database</p> alignItems="center"
</Flex> justifyContent="center"
class="fixed inset-0 backdrop-blur-lg backdrop-filter gap-y-8"
classList={{
hidden: decryptionProgress() === undefined && !isLoadingDatabase(),
}}
>
<Show when={decryptionProgress() !== undefined}>
<p class="font-bold text-2xl">Decrypting database</p>
<Progress
value={decryptionProgress()}
minValue={0}
maxValue={100}
getValueLabel={({ value }) => `${value}%`}
class="w-[300px] space-y-1"
>
<div class="flex justify-between">
<ProgressLabel>Processing...</ProgressLabel>
<ProgressValueLabel />
</div>
</Progress>
</Show> </Show>
<Show when={isLoadingDatabase()}>
<p class="font-bold text-2xl">Loading database</p>
</Show>
</Flex>
</Portal> </Portal>
<Title>Signal stats</Title> <Title>Signal stats</Title>
<div> <div>
<input type="file" accept=".sqlite" onChange={onFileChange}></input> <input
type="password"
onChange={(event) => setPassphrase(event.currentTarget.value)}
/>
<input type="file" accept=".backup" onChange={onFileChange} />
</div> </div>
</> </>
); );

View file

@ -1,17 +1,25 @@
import { type Component, createResource, Show } from "solid-js";
import type { RouteSectionProps } from "@solidjs/router"; import type { RouteSectionProps } from "@solidjs/router";
import { type Component, createResource, Show } from "solid-js";
import { allThreadsOverviewQuery, overallSentMessagesQuery } from "~/db-queries"; import {
allThreadsOverviewQuery,
overallSentMessagesQuery,
} from "~/db-queries";
import { OverviewTable, type RoomOverview } from "./overview-table";
import { getNameFromRecipient } from "~/lib/get-name-from-recipient";
import { Title } from "@solidjs/meta"; import { Title } from "@solidjs/meta";
import { SELF_ID } from "~/db"; import { SELF_ID } from "~/db";
import { getNameFromRecipient } from "~/lib/get-name-from-recipient";
import { OverviewTable, type RoomOverview } from "./overview-table";
export const Overview: Component<RouteSectionProps> = () => { export const Overview: Component<RouteSectionProps> = () => {
const [allSelfSentMessagesCount] = createResource(() => overallSentMessagesQuery(SELF_ID)); console.log(overallSentMessagesQuery(SELF_ID));
const [roomOverview] = createResource<RoomOverview[] | undefined>(async () => { const [allSelfSentMessagesCount] = createResource(() =>
overallSentMessagesQuery(SELF_ID),
);
const [roomOverview] = createResource<RoomOverview[] | undefined>(
async () => {
return (await allThreadsOverviewQuery())?.map((row) => { return (await allThreadsOverviewQuery())?.map((row) => {
const isGroup = row.title !== null; const isGroup = row.title !== null;
@ -20,7 +28,11 @@ export const Overview: Component<RouteSectionProps> = () => {
if (row.title !== null) { if (row.title !== null) {
name = row.title; name = row.title;
} else { } else {
name = getNameFromRecipient(row.nickname_joined_name, row.system_joined_name, row.profile_joined_name); name = getNameFromRecipient(
row.nickname_joined_name,
row.system_joined_name,
row.profile_joined_name,
);
} }
return { return {
@ -28,21 +40,31 @@ export const Overview: Component<RouteSectionProps> = () => {
recipientId: row.recipient_id, recipientId: row.recipient_id,
archived: Boolean(row.archived), archived: Boolean(row.archived),
messageCount: row.message_count, messageCount: row.message_count,
lastMessageDate: row.last_message_date ? new Date(row.last_message_date) : undefined, lastMessageDate: row.last_message_date
? new Date(row.last_message_date)
: undefined,
name, name,
isGroup, isGroup,
}; };
}); });
}); },
);
return ( return (
<> <>
<Title>Signal statistics overview</Title> <Title>Signal statistics overview</Title>
<div> <div>
<p>All messages: {allSelfSentMessagesCount()?.messageCount as number}</p> <p>
<Show when={!roomOverview.loading && roomOverview()} fallback="Loading..."> All messages: {allSelfSentMessagesCount()?.messageCount as number}
{(currentRoomOverview) => <OverviewTable data={currentRoomOverview()} />} </p>
<Show
when={!roomOverview.loading && roomOverview()}
fallback="Loading..."
>
{(currentRoomOverview) => (
<OverviewTable data={currentRoomOverview()} />
)}
</Show> </Show>
</div> </div>
</> </>

View file

@ -1,12 +1,23 @@
import path from "path"; import path from "path";
import { defineConfig } from "vite"; import { defineConfig } from "vite";
import solidPlugin from "vite-plugin-solid"; import solidPlugin from "vite-plugin-solid";
import wasm from "vite-plugin-wasm";
export default defineConfig({ export default defineConfig({
plugins: [solidPlugin()], plugins: [solidPlugin(), wasm()],
server: { server: {
headers: {
"Cross-Origin-Opener-Policy": "same-origin",
"Cross-Origin-Embedder-Policy": "require-corp",
},
port: 3000, port: 3000,
}, },
optimizeDeps: {
exclude: [
"@duskflower/signal-decrypt-backup-wasm",
"@sqlite.org/sqlite-wasm",
],
},
build: { build: {
target: "esnext", target: "esnext",
}, },