From fd6d2d641bfb3031b345644a08239f9df047c827 Mon Sep 17 00:00:00 2001 From: "copilot-swe-agent[bot]" <198982749+Copilot@users.noreply.github.com> Date: Mon, 9 Feb 2026 19:37:28 +0000 Subject: [PATCH] Fix database setup, test encryption and repositories - all working Co-authored-by: southseact-3d <217551146+southseact-3d@users.noreply.github.com> --- chat/package-lock.json | 355 +++++++++++++++++++++++++++++++- chat/package.json | 2 +- chat/scripts/setup-database.js | 13 +- chat/src/database/connection.js | 7 +- chat/test-encryption.js | 33 +++ chat/test-repositories.js | 97 +++++++++ chat/test-tokens.js | 52 +++++ 7 files changed, 539 insertions(+), 20 deletions(-) create mode 100644 chat/test-encryption.js create mode 100644 chat/test-repositories.js create mode 100644 chat/test-tokens.js diff --git a/chat/package-lock.json b/chat/package-lock.json index 37919be..9d28268 100644 --- a/chat/package-lock.json +++ b/chat/package-lock.json @@ -12,6 +12,7 @@ "adm-zip": "^0.5.16", "archiver": "^6.0.1", "bcrypt": "^6.0.0", + "better-sqlite3": "^12.6.2", "jsonwebtoken": "^9.0.2", "nodemailer": "^7.0.7", "pdfkit": "^0.17.2", @@ -516,6 +517,40 @@ "node": ">= 18" } }, + "node_modules/better-sqlite3": { + "version": "12.6.2", + "resolved": "https://registry.npmjs.org/better-sqlite3/-/better-sqlite3-12.6.2.tgz", + "integrity": "sha512-8VYKM3MjCa9WcaSAI3hzwhmyHVlH8tiGFwf0RlTsZPWJ1I5MkzjiudCo4KC4DxOaL/53A5B1sI/IbldNFDbsKA==", + "hasInstallScript": true, + "license": "MIT", + "dependencies": { + "bindings": "^1.5.0", + "prebuild-install": "^7.1.1" + }, + "engines": { + "node": "20.x || 22.x || 23.x || 24.x || 25.x" + } + }, + "node_modules/bindings": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", + "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", + "license": "MIT", + "dependencies": { + "file-uri-to-path": "1.0.0" + } + }, + "node_modules/bl": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", + "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "license": "MIT", + "dependencies": { + "buffer": "^5.5.0", + "inherits": "^2.0.4", + "readable-stream": "^3.4.0" + } + }, "node_modules/brace-expansion": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", @@ -534,6 +569,30 @@ "base64-js": "^1.1.2" } }, + "node_modules/buffer": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "dependencies": { + "base64-js": "^1.3.1", + "ieee754": "^1.1.13" + } + }, "node_modules/buffer-crc32": { "version": "0.2.13", "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", @@ -549,6 +608,12 @@ "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==", "license": "BSD-3-Clause" }, + "node_modules/chownr": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", + "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", + "license": "ISC" + }, "node_modules/clone": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz", @@ -651,6 +716,30 @@ "integrity": "sha512-KALDyEYgpY+Rlob/iriUtjV6d5Eq+Y191A5g4UqLAi8CyGP9N1+FdVbkc1SxKc2r4YAYqG8JzO2KGL+AizD70Q==", "license": "MIT" }, + "node_modules/decompress-response": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz", + "integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==", + "license": "MIT", + "dependencies": { + "mimic-response": "^3.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/deep-extend": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", + "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", + "license": "MIT", + "engines": { + "node": ">=4.0.0" + } + }, "node_modules/detect-libc": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz", @@ -675,6 +764,15 @@ "safe-buffer": "^5.0.1" } }, + "node_modules/end-of-stream": { + "version": "1.4.5", + "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.5.tgz", + "integrity": "sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg==", + "license": "MIT", + "dependencies": { + "once": "^1.4.0" + } + }, "node_modules/events-universal": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/events-universal/-/events-universal-1.0.1.tgz", @@ -684,6 +782,15 @@ "bare-events": "^2.7.0" } }, + "node_modules/expand-template": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/expand-template/-/expand-template-2.0.3.tgz", + "integrity": "sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==", + "license": "(MIT OR WTFPL)", + "engines": { + "node": ">=6" + } + }, "node_modules/fast-deep-equal": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", @@ -696,6 +803,12 @@ "integrity": "sha512-/d9sfos4yxzpwkDkuN7k2SqFKtYNmCTzgfEpz82x34IM9/zc8KGxQoXg1liNC/izpRM/MBdt44Nmx41ZWqk+FQ==", "license": "MIT" }, + "node_modules/file-uri-to-path": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", + "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", + "license": "MIT" + }, "node_modules/fontkit": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/fontkit/-/fontkit-2.0.4.tgz", @@ -713,17 +826,29 @@ "unicode-trie": "^2.0.0" } }, + "node_modules/fs-constants": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", + "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==", + "license": "MIT" + }, "node_modules/fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", "license": "ISC" }, + "node_modules/github-from-package": { + "version": "0.0.0", + "resolved": "https://registry.npmjs.org/github-from-package/-/github-from-package-0.0.0.tgz", + "integrity": "sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==", + "license": "MIT" + }, "node_modules/glob": { "version": "8.1.0", "resolved": "https://registry.npmjs.org/glob/-/glob-8.1.0.tgz", "integrity": "sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==", - "deprecated": "Glob versions prior to v9 are no longer supported", + "deprecated": "Old versions of glob are not supported, and contain widely publicized security vulnerabilities, which have been fixed in the current version. Please update. Support for old versions may be purchased (at exorbitant rates) by contacting i@izs.me", "license": "ISC", "dependencies": { "fs.realpath": "^1.0.0", @@ -745,6 +870,26 @@ "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", "license": "ISC" }, + "node_modules/ieee754": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "BSD-3-Clause" + }, "node_modules/inflight": { "version": "1.0.6", "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", @@ -762,6 +907,12 @@ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", "license": "ISC" }, + "node_modules/ini": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", + "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", + "license": "ISC" + }, "node_modules/is-arrayish": { "version": "0.3.4", "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.4.tgz", @@ -933,6 +1084,18 @@ "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==", "license": "MIT" }, + "node_modules/mimic-response": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz", + "integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==", + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/minimatch": { "version": "5.1.6", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz", @@ -945,12 +1108,45 @@ "node": ">=10" } }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/mkdirp-classic": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", + "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==", + "license": "MIT" + }, "node_modules/ms": { "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", "license": "MIT" }, + "node_modules/napi-build-utils": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/napi-build-utils/-/napi-build-utils-2.0.0.tgz", + "integrity": "sha512-GEbrYkbfF7MoNaoh2iGG84Mnf/WZfB0GdGEsM8wz7Expx/LlWf5U8t9nvJKXSp3qr5IsEbK04cBGhol/KwOsWA==", + "license": "MIT" + }, + "node_modules/node-abi": { + "version": "3.87.0", + "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.87.0.tgz", + "integrity": "sha512-+CGM1L1CgmtheLcBuleyYOn7NWPVu0s0EJH2C4puxgEZb9h8QpR9G2dBfZJOAUhi7VQxuBPMd0hiISWcTyiYyQ==", + "license": "MIT", + "dependencies": { + "semver": "^7.3.5" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/node-addon-api": { "version": "8.5.0", "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-8.5.0.tgz", @@ -972,9 +1168,9 @@ } }, "node_modules/nodemailer": { - "version": "7.0.11", - "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-7.0.11.tgz", - "integrity": "sha512-gnXhNRE0FNhD7wPSCGhdNh46Hs6nm+uTyg+Kq0cZukNQiYdnCsoQjodNP9BQVG9XrcK/v6/MgpAPBUFyzh9pvw==", + "version": "7.0.13", + "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-7.0.13.tgz", + "integrity": "sha512-PNDFSJdP+KFgdsG3ZzMXCgquO7I6McjY2vlqILjtJd0hy8wEvtugS9xKRF2NWlPNGxvLCXlTNIae4serI7dinw==", "license": "MIT-0", "engines": { "node": ">=6.0.0" @@ -1022,12 +1218,63 @@ "resolved": "https://registry.npmjs.org/png-js/-/png-js-1.0.0.tgz", "integrity": "sha512-k+YsbhpA9e+EFfKjTCH3VW6aoKlyNYI6NYdTfDL4CIvFnvsuO84ttonmZE7rc+v23SLTH8XX+5w/Ak9v0xGY4g==" }, + "node_modules/prebuild-install": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-7.1.3.tgz", + "integrity": "sha512-8Mf2cbV7x1cXPUILADGI3wuhfqWvtiLA1iclTDbFRZkgRQS0NqsPZphna9V+HyTEadheuPmjaJMsbzKQFOzLug==", + "license": "MIT", + "dependencies": { + "detect-libc": "^2.0.0", + "expand-template": "^2.0.3", + "github-from-package": "0.0.0", + "minimist": "^1.2.3", + "mkdirp-classic": "^0.5.3", + "napi-build-utils": "^2.0.0", + "node-abi": "^3.3.0", + "pump": "^3.0.0", + "rc": "^1.2.7", + "simple-get": "^4.0.0", + "tar-fs": "^2.0.0", + "tunnel-agent": "^0.6.0" + }, + "bin": { + "prebuild-install": "bin.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/process-nextick-args": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", "license": "MIT" }, + "node_modules/pump": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.3.tgz", + "integrity": "sha512-todwxLMY7/heScKmntwQG8CXVkWUOdYxIvY2s0VWAAMh/nd8SoYiRaKjlr7+iCs984f2P8zvrfWcDDYVb73NfA==", + "license": "MIT", + "dependencies": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, + "node_modules/rc": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", + "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", + "license": "(BSD-2-Clause OR MIT OR Apache-2.0)", + "dependencies": { + "deep-extend": "^0.6.0", + "ini": "~1.3.0", + "minimist": "^1.2.0", + "strip-json-comments": "~2.0.1" + }, + "bin": { + "rc": "cli.js" + } + }, "node_modules/readable-stream": { "version": "3.6.2", "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", @@ -1078,9 +1325,9 @@ "license": "MIT" }, "node_modules/semver": { - "version": "7.7.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", - "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "version": "7.7.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.4.tgz", + "integrity": "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA==", "license": "ISC", "bin": { "semver": "bin/semver.js" @@ -1128,6 +1375,51 @@ "@img/sharp-win32-x64": "0.33.5" } }, + "node_modules/simple-concat": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz", + "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/simple-get": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-4.0.1.tgz", + "integrity": "sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "dependencies": { + "decompress-response": "^6.0.0", + "once": "^1.3.1", + "simple-concat": "^1.0.0" + } + }, "node_modules/simple-swizzle": { "version": "0.2.4", "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.4.tgz", @@ -1157,6 +1449,43 @@ "safe-buffer": "~5.2.0" } }, + "node_modules/strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/tar-fs": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.4.tgz", + "integrity": "sha512-mDAjwmZdh7LTT6pNleZ05Yt65HC3E+NiQzl672vQG38jIrehtJk/J3mNwIg+vShQPcLF/LV7CMnDW6vjj6sfYQ==", + "license": "MIT", + "dependencies": { + "chownr": "^1.1.1", + "mkdirp-classic": "^0.5.2", + "pump": "^3.0.0", + "tar-stream": "^2.1.4" + } + }, + "node_modules/tar-fs/node_modules/tar-stream": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", + "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", + "license": "MIT", + "dependencies": { + "bl": "^4.0.3", + "end-of-stream": "^1.4.1", + "fs-constants": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^3.1.1" + }, + "engines": { + "node": ">=6" + } + }, "node_modules/tar-stream": { "version": "3.1.7", "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-3.1.7.tgz", @@ -1189,6 +1518,18 @@ "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", "license": "0BSD" }, + "node_modules/tunnel-agent": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", + "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==", + "license": "Apache-2.0", + "dependencies": { + "safe-buffer": "^5.0.1" + }, + "engines": { + "node": "*" + } + }, "node_modules/unicode-properties": { "version": "1.4.1", "resolved": "https://registry.npmjs.org/unicode-properties/-/unicode-properties-1.4.1.tgz", diff --git a/chat/package.json b/chat/package.json index 57c82f0..b09937b 100644 --- a/chat/package.json +++ b/chat/package.json @@ -15,7 +15,7 @@ "adm-zip": "^0.5.16", "archiver": "^6.0.1", "bcrypt": "^6.0.0", - "better-sqlite3": "^9.4.3", + "better-sqlite3": "^12.6.2", "jsonwebtoken": "^9.0.2", "nodemailer": "^7.0.7", "pdfkit": "^0.17.2", diff --git a/chat/scripts/setup-database.js b/chat/scripts/setup-database.js index bd024d4..26fd727 100755 --- a/chat/scripts/setup-database.js +++ b/chat/scripts/setup-database.js @@ -67,18 +67,11 @@ async function setupDatabase() { const db = getDatabase(); - // Split by semicolon and execute each statement - const statements = schema - .split(';') - .map(s => s.trim()) - .filter(s => s.length > 0 && !s.startsWith('--')); - - for (const statement of statements) { - db.exec(statement); - } + // Execute the entire schema as one block + // SQLite can handle multiple statements with exec() + db.exec(schema); console.log('โœ… Database schema created'); - console.log(` Executed ${statements.length} SQL statements`); } catch (error) { console.error('โŒ Failed to create schema:', error.message); closeDatabase(); diff --git a/chat/src/database/connection.js b/chat/src/database/connection.js index cc5f61d..7eff3e6 100644 --- a/chat/src/database/connection.js +++ b/chat/src/database/connection.js @@ -32,11 +32,14 @@ function initDatabase(databasePath, options = {}) { // Initialize database with options const dbOptions = { - verbose: options.verbose ? console.log : null, fileMustExist: false, timeout: options.timeout || 5000, - ...options }; + + // Add verbose if it's a function + if (options.verbose && typeof options.verbose === 'function') { + dbOptions.verbose = options.verbose; + } db = new Database(databasePath, dbOptions); diff --git a/chat/test-encryption.js b/chat/test-encryption.js new file mode 100644 index 0000000..1918338 --- /dev/null +++ b/chat/test-encryption.js @@ -0,0 +1,33 @@ +const { initEncryption, encrypt, decrypt, hashValue, verifyHash } = require('./src/utils/encryption'); +const crypto = require('crypto'); + +// Initialize +const key = crypto.randomBytes(32).toString('hex'); +initEncryption(key); + +console.log('๐Ÿ” Testing Encryption...\n'); + +// Test 1: Basic encryption/decryption +const plaintext = 'test@example.com'; +const encrypted = encrypt(plaintext); +const decrypted = decrypt(encrypted); + +console.log('Test 1: Basic Encryption'); +console.log(' Plaintext:', plaintext); +console.log(' Encrypted:', encrypted.substring(0, 50) + '...'); +console.log(' Decrypted:', decrypted); +console.log(' Match:', plaintext === decrypted ? 'โœ…' : 'โŒ'); + +// Test 2: Hash and verify +const value = 'mytoken123'; +const { hash, salt } = hashValue(value); +const isValid = verifyHash(value, hash, salt); +const isInvalid = verifyHash('wrongvalue', hash, salt); + +console.log('\nTest 2: Hash and Verify'); +console.log(' Value:', value); +console.log(' Hash:', hash.substring(0, 30) + '...'); +console.log(' Valid verification:', isValid ? 'โœ…' : 'โŒ'); +console.log(' Invalid verification:', !isInvalid ? 'โœ…' : 'โŒ'); + +console.log('\nโœ… All encryption tests passed!'); diff --git a/chat/test-repositories.js b/chat/test-repositories.js new file mode 100644 index 0000000..dc1e4af --- /dev/null +++ b/chat/test-repositories.js @@ -0,0 +1,97 @@ +const { initDatabase, closeDatabase } = require('./src/database/connection'); +const { initEncryption } = require('./src/utils/encryption'); +const userRepo = require('./src/repositories/userRepository'); +const sessionRepo = require('./src/repositories/sessionRepository'); +const auditRepo = require('./src/repositories/auditRepository'); +const crypto = require('crypto'); + +// Initialize +const dbPath = './.data/test_shopify_ai.db'; +const encKey = crypto.randomBytes(32).toString('hex'); + +console.log('๐Ÿงช Testing Repositories...\n'); + +initEncryption(encKey); +initDatabase(dbPath); + +// Setup schema +const fs = require('fs'); +const path = require('path'); +const db = require('./src/database/connection').getDatabase(); +const schema = fs.readFileSync(path.join(__dirname, 'src/database/schema.sql'), 'utf8'); +db.exec(schema); + +// Test 1: Create user +console.log('Test 1: Create User'); +const user = userRepo.createUser({ + id: 'user123', + email: 'test@example.com', + name: 'Test User', + passwordHash: '$2b$12$hashedpassword', + emailVerified: true, + plan: 'professional' +}); +console.log(' Created user:', user.email); +console.log(' ID:', user.id); +console.log(' Plan:', user.plan); +console.log(' Success:', user.id === 'user123' ? 'โœ…' : 'โŒ'); + +// Test 2: Get user by email +console.log('\nTest 2: Get User by Email'); +const foundUser = userRepo.getUserByEmail('test@example.com'); +console.log(' Found user:', foundUser.email); +console.log(' Name matches:', foundUser.name === 'Test User' ? 'โœ…' : 'โŒ'); + +// Test 3: Update user +console.log('\nTest 3: Update User'); +const updated = userRepo.updateUser('user123', { + plan: 'enterprise', + billingStatus: 'active' +}); +console.log(' Updated plan:', updated.plan); +console.log(' Plan correct:', updated.plan === 'enterprise' ? 'โœ…' : 'โŒ'); + +// Test 4: Create session +console.log('\nTest 4: Create Session'); +const session = sessionRepo.createSession({ + userId: 'user123', + token: 'session_token_123', + deviceFingerprint: 'device123', + expiresAt: Date.now() + 86400000 +}); +console.log(' Created session:', session.id); +console.log(' Token:', session.token); +console.log(' Success:', session.userId === 'user123' ? 'โœ…' : 'โŒ'); + +// Test 5: Get session by token +console.log('\nTest 5: Get Session by Token'); +const foundSession = sessionRepo.getSessionByToken('session_token_123'); +console.log(' Found session:', foundSession.id); +console.log(' Matches:', foundSession.id === session.id ? 'โœ…' : 'โŒ'); + +// Test 6: Audit log +console.log('\nTest 6: Audit Log'); +auditRepo.logAuditEvent({ + userId: 'user123', + eventType: 'login', + eventData: { method: 'email' }, + ipAddress: '127.0.0.1', + success: true +}); +const logs = auditRepo.getUserAuditLog('user123'); +console.log(' Logged event count:', logs.length); +console.log(' Event type:', logs[0].eventType); +console.log(' Success:', logs.length === 1 && logs[0].eventType === 'login' ? 'โœ…' : 'โŒ'); + +// Test 7: Cleanup +console.log('\nTest 7: Cleanup'); +const deleted = userRepo.deleteUser('user123'); +console.log(' User deleted:', deleted ? 'โœ…' : 'โŒ'); + +closeDatabase(); +console.log('\nโœ… All repository tests passed!'); + +// Clean up test database +fs.unlinkSync(dbPath); +fs.unlinkSync(dbPath + '-wal'); +fs.unlinkSync(dbPath + '-shm'); diff --git a/chat/test-tokens.js b/chat/test-tokens.js new file mode 100644 index 0000000..cb539af --- /dev/null +++ b/chat/test-tokens.js @@ -0,0 +1,52 @@ +const { initTokenManager, generateAccessToken, verifyAccessToken, generateRefreshToken, verifyRefreshToken, generateDeviceFingerprint } = require('./src/utils/tokenManager'); +const crypto = require('crypto'); + +// Initialize +const secret = crypto.randomBytes(32).toString('hex'); +initTokenManager(secret); + +console.log('๐Ÿ”‘ Testing Token Manager...\n'); + +// Test 1: Access token generation and verification +const payload = { + userId: 'user123', + email: 'test@example.com', + role: 'user', + plan: 'professional' +}; + +const accessToken = generateAccessToken(payload); +const decoded = verifyAccessToken(accessToken); + +console.log('Test 1: Access Token'); +console.log(' Token:', accessToken.substring(0, 50) + '...'); +console.log(' User ID:', decoded.userId); +console.log(' Email:', decoded.email); +console.log(' Valid:', decoded.userId === payload.userId ? 'โœ…' : 'โŒ'); + +// Test 2: Refresh token +const { token, tokenHash } = generateRefreshToken(); +const isValid = verifyRefreshToken(token, tokenHash); +const isInvalid = verifyRefreshToken('wrongtoken', tokenHash); + +console.log('\nTest 2: Refresh Token'); +console.log(' Token:', token.substring(0, 30) + '...'); +console.log(' Hash:', tokenHash.substring(0, 30) + '...'); +console.log(' Valid verification:', isValid ? 'โœ…' : 'โŒ'); +console.log(' Invalid verification:', !isInvalid ? 'โœ…' : 'โŒ'); + +// Test 3: Device fingerprint +const mockReq = { + headers: { + 'user-agent': 'Mozilla/5.0', + 'accept-language': 'en-US' + }, + ip: '127.0.0.1' +}; + +const fingerprint = generateDeviceFingerprint(mockReq); +console.log('\nTest 3: Device Fingerprint'); +console.log(' Fingerprint:', fingerprint); +console.log(' Length correct:', fingerprint.length === 32 ? 'โœ…' : 'โŒ'); + +console.log('\nโœ… All token tests passed!');