diff --git a/.gitignore b/.gitignore index 4d04ce397f..1f2dea3a56 100644 --- a/.gitignore +++ b/.gitignore @@ -42,4 +42,4 @@ data-docs/backup data-docs/log data-docs/session data-docs/session_secret.txt -data-docs/document.* \ No newline at end of file +data-docs/document.* diff --git a/README.md b/README.md index 428e832fa5..265e105430 100644 --- a/README.md +++ b/README.md @@ -37,6 +37,7 @@ Feel free to join our official conversations. We would love to hear what feature * Fast and easy [navigation between notes](https://triliumnext.github.io/Docs/Wiki/note-navigation), full text search and [note hoisting](https://triliumnext.github.io/Docs/Wiki/note-hoisting) * Seamless [note versioning](https://triliumnext.github.io/Docs/Wiki/note-revisions) * Note [attributes](https://triliumnext.github.io/Docs/Wiki/attributes) can be used for note organization, querying and advanced [scripting](https://triliumnext.github.io/Docs/Wiki/scripts) +* Direct OpenID and TOTP integration for more secure login * [Synchronization](https://triliumnext.github.io/Docs/Wiki/synchronization) with self-hosted sync server * there's a [3rd party service for hosting synchronisation server](https://trilium.cc/paid-hosting) * [Sharing](https://triliumnext.github.io/Docs/Wiki/sharing) (publishing) notes to public internet diff --git a/config-sample.ini b/config-sample.ini index cde9ac7c46..b2dace5c88 100644 --- a/config-sample.ini +++ b/config-sample.ini @@ -43,4 +43,17 @@ cookieMaxAge=1814400 [Sync] #syncServerHost= #syncServerTimeout= -#syncServerProxy= \ No newline at end of file +#syncServerProxy= + +[MultiFactorAuthentication] +# Set the base URL for OAuth/OpenID authentication +# This is the URL of the service that will be used to verify the user's identity +oauthBaseUrl= + +# Set the client ID for OAuth/OpenID authentication +# This is the ID of the client that will be used to verify the user's identity +oauthClientId= + +# Set the client secret for OAuth/OpenID authentication +# This is the secret of the client that will be used to verify the user's identity +oauthClientSecret= diff --git a/db/migrations/0229__add_oauth_user_data_table.sql b/db/migrations/0229__add_oauth_user_data_table.sql new file mode 100644 index 0000000000..ea2db4ef9f --- /dev/null +++ b/db/migrations/0229__add_oauth_user_data_table.sql @@ -0,0 +1,14 @@ +-- Add the oauth user data table +CREATE TABLE IF NOT EXISTS "user_data" +( + tmpID INT, + username TEXT, + email TEXT, + userIDEncryptedDataKey TEXT, + userIDVerificationHash TEXT, + salt TEXT, + derivedKey TEXT, + isSetup TEXT DEFAULT "false", + UNIQUE (tmpID), + PRIMARY KEY (tmpID) +); \ No newline at end of file diff --git a/db/schema.sql b/db/schema.sql index 1b4c463211..8bf9db1e7b 100644 --- a/db/schema.sql +++ b/db/schema.sql @@ -126,6 +126,19 @@ CREATE TABLE IF NOT EXISTS "attachments" utcDateScheduledForErasureSince TEXT DEFAULT NULL, isDeleted INT not null, deleteId TEXT DEFAULT NULL); +CREATE TABLE IF NOT EXISTS "user_data" +( + tmpID INT, + username TEXT, + email TEXT, + userIDEncryptedDataKey TEXT, + userIDVerificationHash TEXT, + salt TEXT, + derivedKey TEXT, + isSetup TEXT DEFAULT "false", + UNIQUE (tmpID), + PRIMARY KEY (tmpID) +); CREATE INDEX IDX_attachments_ownerId_role on attachments (ownerId, role); diff --git a/images/google-logo.svg b/images/google-logo.svg new file mode 100644 index 0000000000..c69cd1e7f9 --- /dev/null +++ b/images/google-logo.svg @@ -0,0 +1,7 @@ + + + + + + + \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index 58b35d91ee..fc2f707ada 100644 --- a/package-lock.json +++ b/package-lock.json @@ -40,6 +40,7 @@ "escape-html": "1.0.3", "eslint-linter-browserify": "9.23.0", "express": "4.21.2", + "express-openid-connect": "^2.17.1", "express-rate-limit": "7.5.0", "express-session": "1.18.1", "force-graph": "1.49.5", @@ -84,6 +85,7 @@ "strip-bom": "5.0.0", "striptags": "3.2.0", "swagger-ui-express": "5.0.1", + "time2fa": "^1.3.0", "tmp": "0.2.3", "turndown": "7.2.0", "unescape": "1.0.1", @@ -1869,278 +1871,6 @@ "node": ">=14.14" } }, - "node_modules/@esbuild/aix-ppc64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.0.tgz", - "integrity": "sha512-O7vun9Sf8DFjH2UtqK8Ku3LkquL9SZL8OLY1T5NZkA34+wG3OQF7cl4Ql8vdNzM6fzBbYfLaiRLIOZ+2FOCgBQ==", - "cpu": [ - "ppc64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "aix" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/android-arm": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.0.tgz", - "integrity": "sha512-PTyWCYYiU0+1eJKmw21lWtC+d08JDZPQ5g+kFyxP0V+es6VPPSUhM6zk8iImp2jbV6GwjX4pap0JFbUQN65X1g==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/android-arm64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.0.tgz", - "integrity": "sha512-grvv8WncGjDSyUBjN9yHXNt+cq0snxXbDxy5pJtzMKGmmpPxeAmAhWxXI+01lU5rwZomDgD3kJwulEnhTRUd6g==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/android-x64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.0.tgz", - "integrity": "sha512-m/ix7SfKG5buCnxasr52+LI78SQ+wgdENi9CqyCXwjVR2X4Jkz+BpC3le3AoBPYTC9NHklwngVXvbJ9/Akhrfg==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/darwin-arm64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.0.tgz", - "integrity": "sha512-mVwdUb5SRkPayVadIOI78K7aAnPamoeFR2bT5nszFUZ9P8UpK4ratOdYbZZXYSqPKMHfS1wdHCJk1P1EZpRdvw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/darwin-x64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.0.tgz", - "integrity": "sha512-DgDaYsPWFTS4S3nWpFcMn/33ZZwAAeAFKNHNa1QN0rI4pUjgqf0f7ONmXf6d22tqTY+H9FNdgeaAa+YIFUn2Rg==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/freebsd-arm64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.0.tgz", - "integrity": "sha512-VN4ocxy6dxefN1MepBx/iD1dH5K8qNtNe227I0mnTRjry8tj5MRk4zprLEdG8WPyAPb93/e4pSgi1SoHdgOa4w==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/freebsd-x64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.0.tgz", - "integrity": "sha512-mrSgt7lCh07FY+hDD1TxiTyIHyttn6vnjesnPoVDNmDfOmggTLXRv8Id5fNZey1gl/V2dyVK1VXXqVsQIiAk+A==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-arm": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.0.tgz", - "integrity": "sha512-vkB3IYj2IDo3g9xX7HqhPYxVkNQe8qTK55fraQyTzTX/fxaDtXiEnavv9geOsonh2Fd2RMB+i5cbhu2zMNWJwg==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-arm64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.0.tgz", - "integrity": "sha512-9QAQjTWNDM/Vk2bgBl17yWuZxZNQIF0OUUuPZRKoDtqF2k4EtYbpyiG5/Dk7nqeK6kIJWPYldkOcBqjXjrUlmg==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-ia32": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.0.tgz", - "integrity": "sha512-43ET5bHbphBegyeqLb7I1eYn2P/JYGNmzzdidq/w0T8E2SsYL1U6un2NFROFRg1JZLTzdCoRomg8Rvf9M6W6Gg==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-loong64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.0.tgz", - "integrity": "sha512-fC95c/xyNFueMhClxJmeRIj2yrSMdDfmqJnyOY4ZqsALkDrrKJfIg5NTMSzVBr5YW1jf+l7/cndBfP3MSDpoHw==", - "cpu": [ - "loong64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-mips64el": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.0.tgz", - "integrity": "sha512-nkAMFju7KDW73T1DdH7glcyIptm95a7Le8irTQNO/qtkoyypZAnjchQgooFUDQhNAy4iu08N79W4T4pMBwhPwQ==", - "cpu": [ - "mips64el" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-ppc64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.0.tgz", - "integrity": "sha512-NhyOejdhRGS8Iwv+KKR2zTq2PpysF9XqY+Zk77vQHqNbo/PwZCzB5/h7VGuREZm1fixhs4Q/qWRSi5zmAiO4Fw==", - "cpu": [ - "ppc64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-riscv64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.0.tgz", - "integrity": "sha512-5S/rbP5OY+GHLC5qXp1y/Mx//e92L1YDqkiBbO9TQOvuFXM+iDqUNG5XopAnXoRH3FjIUDkeGcY1cgNvnXp/kA==", - "cpu": [ - "riscv64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-s390x": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.0.tgz", - "integrity": "sha512-XM2BFsEBz0Fw37V0zU4CXfcfuACMrppsMFKdYY2WuTS3yi8O1nFOhil/xhKTmE1nPmVyvQJjJivgDT+xh8pXJA==", - "cpu": [ - "s390x" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, "node_modules/@esbuild/linux-x64": { "version": "0.25.0", "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.0.tgz", @@ -2158,125 +1888,6 @@ "node": ">=18" } }, - "node_modules/@esbuild/netbsd-x64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.0.tgz", - "integrity": "sha512-jl+qisSB5jk01N5f7sPCsBENCOlPiS/xptD5yxOx2oqQfyourJwIKLRA2yqWdifj3owQZCL2sn6o08dBzZGQzA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "netbsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/openbsd-arm64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.0.tgz", - "integrity": "sha512-21sUNbq2r84YE+SJDfaQRvdgznTD8Xc0oc3p3iW/a1EVWeNj/SdUCbm5U0itZPQYRuRTW20fPMWMpcrciH2EJw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "openbsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/openbsd-x64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.0.tgz", - "integrity": "sha512-2gwwriSMPcCFRlPlKx3zLQhfN/2WjJ2NSlg5TKLQOJdV0mSxIcYNTMhk3H3ulL/cak+Xj0lY1Ym9ysDV1igceg==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "openbsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/sunos-x64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.0.tgz", - "integrity": "sha512-bxI7ThgLzPrPz484/S9jLlvUAHYMzy6I0XiU1ZMeAEOBcS0VePBFxh1JjTQt3Xiat5b6Oh4x7UC7IwKQKIJRIg==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "sunos" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/win32-arm64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.0.tgz", - "integrity": "sha512-ZUAc2YK6JW89xTbXvftxdnYy3m4iHIkDtK3CLce8wg8M2L+YZhIvO1DKpxrd0Yr59AeNNkTiic9YLf6FTtXWMw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/win32-ia32": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.0.tgz", - "integrity": "sha512-eSNxISBu8XweVEWG31/JzjkIGbGIJN/TrRoiSVZwZ6pkC6VX4Im/WV2cz559/TXLcYbcrDN8JtKgd9DJVIo8GA==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/win32-x64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.0.tgz", - "integrity": "sha512-ZENoHJBxA20C2zFzh6AI4fT6RraMzjYw4xKWemRTRmRVtN9c5DcH9r/f2ihEkMjOW5eGgrwCslG/+Y/3bL+DHQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=18" - } - }, "node_modules/@eslint-community/eslint-utils": { "version": "4.4.1", "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.1.tgz", @@ -3150,6 +2761,21 @@ "@shikijs/vscode-textmate": "^10.0.2" } }, + "node_modules/@hapi/hoek": { + "version": "9.3.0", + "resolved": "https://registry.npmjs.org/@hapi/hoek/-/hoek-9.3.0.tgz", + "integrity": "sha512-/c6rf4UJlmHlC9b5BaNvzAcFv7HZ2QHaV0D4/HNlBdvFnvQq8RI4kYdhyPCl7Xj+oWvTWQ8ujhqS53LIgAe6KQ==", + "license": "BSD-3-Clause" + }, + "node_modules/@hapi/topo": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/@hapi/topo/-/topo-5.1.0.tgz", + "integrity": "sha512-foQZKJig7Ob0BMAYBfcJk8d77QtOe7Wo4ox7ff1lQYoNNAb6jwcY1ncdoy2e9wQZzvNy7ODZCYJkK8kzmcAnAg==", + "license": "BSD-3-Clause", + "dependencies": { + "@hapi/hoek": "^9.0.0" + } + }, "node_modules/@highlightjs/cdn-assets": { "version": "11.11.1", "resolved": "https://registry.npmjs.org/@highlightjs/cdn-assets/-/cdn-assets-11.11.1.tgz", @@ -4255,299 +3881,77 @@ "version": "3.0.2", "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", - "deprecated": "Rimraf versions prior to v4 are no longer supported", - "dev": true, - "license": "ISC", - "dependencies": { - "glob": "^7.1.3" - }, - "bin": { - "rimraf": "bin.js" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/@parcel/watcher": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher/-/watcher-2.5.1.tgz", - "integrity": "sha512-dfUnCxiN9H4ap84DvD2ubjw+3vUNpstxa0TneY/Paat8a3R4uQZDLSvWjmznAY/DoahqTHl9V46HF/Zs3F29pg==", - "dev": true, - "hasInstallScript": true, - "license": "MIT", - "optional": true, - "dependencies": { - "detect-libc": "^1.0.3", - "is-glob": "^4.0.3", - "micromatch": "^4.0.5", - "node-addon-api": "^7.0.0" - }, - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - }, - "optionalDependencies": { - "@parcel/watcher-android-arm64": "2.5.1", - "@parcel/watcher-darwin-arm64": "2.5.1", - "@parcel/watcher-darwin-x64": "2.5.1", - "@parcel/watcher-freebsd-x64": "2.5.1", - "@parcel/watcher-linux-arm-glibc": "2.5.1", - "@parcel/watcher-linux-arm-musl": "2.5.1", - "@parcel/watcher-linux-arm64-glibc": "2.5.1", - "@parcel/watcher-linux-arm64-musl": "2.5.1", - "@parcel/watcher-linux-x64-glibc": "2.5.1", - "@parcel/watcher-linux-x64-musl": "2.5.1", - "@parcel/watcher-win32-arm64": "2.5.1", - "@parcel/watcher-win32-ia32": "2.5.1", - "@parcel/watcher-win32-x64": "2.5.1" - } - }, - "node_modules/@parcel/watcher-android-arm64": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-android-arm64/-/watcher-android-arm64-2.5.1.tgz", - "integrity": "sha512-KF8+j9nNbUN8vzOFDpRMsaKBHZ/mcjEjMToVMJOhTozkDonQFFrRcfdLWn6yWKCmJKmdVxSgHiYvTCef4/qcBA==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-darwin-arm64": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-darwin-arm64/-/watcher-darwin-arm64-2.5.1.tgz", - "integrity": "sha512-eAzPv5osDmZyBhou8PoF4i6RQXAfeKL9tjb3QzYuccXFMQU0ruIc/POh30ePnaOyD1UXdlKguHBmsTs53tVoPw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-darwin-x64": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-darwin-x64/-/watcher-darwin-x64-2.5.1.tgz", - "integrity": "sha512-1ZXDthrnNmwv10A0/3AJNZ9JGlzrF82i3gNQcWOzd7nJ8aj+ILyW1MTxVk35Db0u91oD5Nlk9MBiujMlwmeXZg==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-freebsd-x64": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-freebsd-x64/-/watcher-freebsd-x64-2.5.1.tgz", - "integrity": "sha512-SI4eljM7Flp9yPuKi8W0ird8TI/JK6CSxju3NojVI6BjHsTyK7zxA9urjVjEKJ5MBYC+bLmMcbAWlZ+rFkLpJQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-arm-glibc": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-arm-glibc/-/watcher-linux-arm-glibc-2.5.1.tgz", - "integrity": "sha512-RCdZlEyTs8geyBkkcnPWvtXLY44BCeZKmGYRtSgtwwnHR4dxfHRG3gR99XdMEdQ7KeiDdasJwwvNSF5jKtDwdA==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-arm-musl": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-arm-musl/-/watcher-linux-arm-musl-2.5.1.tgz", - "integrity": "sha512-6E+m/Mm1t1yhB8X412stiKFG3XykmgdIOqhjWj+VL8oHkKABfu/gjFj8DvLrYVHSBNC+/u5PeNrujiSQ1zwd1Q==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-arm64-glibc": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-arm64-glibc/-/watcher-linux-arm64-glibc-2.5.1.tgz", - "integrity": "sha512-LrGp+f02yU3BN9A+DGuY3v3bmnFUggAITBGriZHUREfNEzZh/GO06FF5u2kx8x+GBEUYfyTGamol4j3m9ANe8w==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-arm64-musl": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-arm64-musl/-/watcher-linux-arm64-musl-2.5.1.tgz", - "integrity": "sha512-cFOjABi92pMYRXS7AcQv9/M1YuKRw8SZniCDw0ssQb/noPkRzA+HBDkwmyOJYp5wXcsTrhxO0zq1U11cK9jsFg==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-x64-glibc": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-x64-glibc/-/watcher-linux-x64-glibc-2.5.1.tgz", - "integrity": "sha512-GcESn8NZySmfwlTsIur+49yDqSny2IhPeZfXunQi48DMugKeZ7uy1FX83pO0X22sHntJ4Ub+9k34XQCX+oHt2A==", - "cpu": [ - "x64" - ], + "deprecated": "Rimraf versions prior to v4 are no longer supported", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" + "license": "ISC", + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" }, "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" + "url": "https://github.com/sponsors/isaacs" } }, - "node_modules/@parcel/watcher-linux-x64-musl": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-x64-musl/-/watcher-linux-x64-musl-2.5.1.tgz", - "integrity": "sha512-n0E2EQbatQ3bXhcH2D1XIAANAcTZkQICBPVaxMeaCVBtOpBZpWJuf7LwyWPSBDITb7In8mqQgJ7gH8CILCURXg==", - "cpu": [ - "x64" - ], - "dev": true, + "node_modules/@panva/asn1.js": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@panva/asn1.js/-/asn1.js-1.0.0.tgz", + "integrity": "sha512-UdkG3mLEqXgnlKsWanWcgb6dOjUzJ+XC5f+aWw30qrtjxeNUSfKX1cd5FBzOaXQumoe9nIqeZUvrRJS03HCCtw==", "license": "MIT", - "optional": true, - "os": [ - "linux" - ], "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" + "node": ">=10.13.0" } }, - "node_modules/@parcel/watcher-win32-arm64": { + "node_modules/@parcel/watcher": { "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-win32-arm64/-/watcher-win32-arm64-2.5.1.tgz", - "integrity": "sha512-RFzklRvmc3PkjKjry3hLF9wD7ppR4AKcWNzH7kXR7GUe0Igb3Nz8fyPwtZCSquGrhU5HhUNDr/mKBqj7tqA2Vw==", - "cpu": [ - "arm64" - ], + "resolved": "https://registry.npmjs.org/@parcel/watcher/-/watcher-2.5.1.tgz", + "integrity": "sha512-dfUnCxiN9H4ap84DvD2ubjw+3vUNpstxa0TneY/Paat8a3R4uQZDLSvWjmznAY/DoahqTHl9V46HF/Zs3F29pg==", "dev": true, + "hasInstallScript": true, "license": "MIT", "optional": true, - "os": [ - "win32" - ], + "dependencies": { + "detect-libc": "^1.0.3", + "is-glob": "^4.0.3", + "micromatch": "^4.0.5", + "node-addon-api": "^7.0.0" + }, "engines": { "node": ">= 10.0.0" }, "funding": { "type": "opencollective", "url": "https://opencollective.com/parcel" + }, + "optionalDependencies": { + "@parcel/watcher-android-arm64": "2.5.1", + "@parcel/watcher-darwin-arm64": "2.5.1", + "@parcel/watcher-darwin-x64": "2.5.1", + "@parcel/watcher-freebsd-x64": "2.5.1", + "@parcel/watcher-linux-arm-glibc": "2.5.1", + "@parcel/watcher-linux-arm-musl": "2.5.1", + "@parcel/watcher-linux-arm64-glibc": "2.5.1", + "@parcel/watcher-linux-arm64-musl": "2.5.1", + "@parcel/watcher-linux-x64-glibc": "2.5.1", + "@parcel/watcher-linux-x64-musl": "2.5.1", + "@parcel/watcher-win32-arm64": "2.5.1", + "@parcel/watcher-win32-ia32": "2.5.1", + "@parcel/watcher-win32-x64": "2.5.1" } }, - "node_modules/@parcel/watcher-win32-ia32": { + "node_modules/@parcel/watcher-linux-x64-glibc": { "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-win32-ia32/-/watcher-win32-ia32-2.5.1.tgz", - "integrity": "sha512-c2KkcVN+NJmuA7CGlaGD1qJh1cLfDnQsHjE89E60vUEMlqduHGCdCLJCID5geFVM0dOtA3ZiIO8BoEQmzQVfpQ==", + "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-x64-glibc/-/watcher-linux-x64-glibc-2.5.1.tgz", + "integrity": "sha512-GcESn8NZySmfwlTsIur+49yDqSny2IhPeZfXunQi48DMugKeZ7uy1FX83pO0X22sHntJ4Ub+9k34XQCX+oHt2A==", "cpu": [ - "ia32" + "x64" ], "dev": true, "license": "MIT", "optional": true, "os": [ - "win32" + "linux" ], "engines": { "node": ">= 10.0.0" @@ -4557,10 +3961,10 @@ "url": "https://opencollective.com/parcel" } }, - "node_modules/@parcel/watcher-win32-x64": { + "node_modules/@parcel/watcher-linux-x64-musl": { "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@parcel/watcher-win32-x64/-/watcher-win32-x64-2.5.1.tgz", - "integrity": "sha512-9lHBdJITeNR++EvSQVUcaZoWupyHfXe1jZvGZ06O/5MflPcuPLtEphScIBL+AiCWBO46tDSHzWyD0uDmmZqsgA==", + "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-x64-musl/-/watcher-linux-x64-musl-2.5.1.tgz", + "integrity": "sha512-n0E2EQbatQ3bXhcH2D1XIAANAcTZkQICBPVaxMeaCVBtOpBZpWJuf7LwyWPSBDITb7In8mqQgJ7gH8CILCURXg==", "cpu": [ "x64" ], @@ -4568,7 +3972,7 @@ "license": "MIT", "optional": true, "os": [ - "win32" + "linux" ], "engines": { "node": ">= 10.0.0" @@ -4973,288 +4377,92 @@ "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - } - } - }, - "node_modules/@radix-ui/react-use-escape-keydown": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@radix-ui/react-use-escape-keydown/-/react-use-escape-keydown-1.1.0.tgz", - "integrity": "sha512-L7vwWlR1kTTQ3oh7g1O0CBF3YCyyTj8NmhLR+phShpyA50HCfBFKVJTpshm9PzLiKmehsrQzTYTpX9HvmC9rhw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@radix-ui/react-use-callback-ref": "1.1.0" - }, - "peerDependencies": { - "@types/react": "*", - "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - } - } - }, - "node_modules/@radix-ui/react-use-layout-effect": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@radix-ui/react-use-layout-effect/-/react-use-layout-effect-1.1.0.tgz", - "integrity": "sha512-+FPE0rOdziWSrH9athwI1R0HDVbWlEhd+FR+aSDk4uWGmSJ9Z54sdZVDQPZAinJhJXwfT+qnj969mCsT2gfm5w==", - "dev": true, - "license": "MIT", - "peerDependencies": { - "@types/react": "*", - "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - } - } - }, - "node_modules/@radix-ui/react-use-rect": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@radix-ui/react-use-rect/-/react-use-rect-1.1.0.tgz", - "integrity": "sha512-0Fmkebhr6PiseyZlYAOtLS+nb7jLmpqTrJyv61Pe68MKYW6OWdRE2kI70TaYY27u7H0lajqM3hSMMLFq18Z7nQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@radix-ui/rect": "1.1.0" - }, - "peerDependencies": { - "@types/react": "*", - "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - } - } - }, - "node_modules/@radix-ui/react-use-size": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@radix-ui/react-use-size/-/react-use-size-1.1.0.tgz", - "integrity": "sha512-XW3/vWuIXHa+2Uwcc2ABSfcCledmXhhQPlGbfcRXbiUQI5Icjcg19BGCZVKKInYbvUCut/ufbbLLPFC5cbb1hw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@radix-ui/react-use-layout-effect": "1.1.0" - }, - "peerDependencies": { - "@types/react": "*", - "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - } - } - }, - "node_modules/@radix-ui/rect": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@radix-ui/rect/-/rect-1.1.0.tgz", - "integrity": "sha512-A9+lCBZoaMJlVKcRBz2YByCG+Cp2t6nAnMnNba+XiWxnj6r4JUFqfsgwocMBZU9LPtdxC6wB56ySYpc7LQIoJg==", - "dev": true, - "license": "MIT" - }, - "node_modules/@rollup/rollup-android-arm-eabi": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.30.1.tgz", - "integrity": "sha512-pSWY+EVt3rJ9fQ3IqlrEUtXh3cGqGtPDH1FQlNZehO2yYxCHEX1SPsz1M//NXwYfbTlcKr9WObLnJX9FsS9K1Q==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ] - }, - "node_modules/@rollup/rollup-android-arm64": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.30.1.tgz", - "integrity": "sha512-/NA2qXxE3D/BRjOJM8wQblmArQq1YoBVJjrjoTSBS09jgUisq7bqxNHJ8kjCHeV21W/9WDGwJEWSN0KQ2mtD/w==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ] - }, - "node_modules/@rollup/rollup-darwin-arm64": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.30.1.tgz", - "integrity": "sha512-r7FQIXD7gB0WJ5mokTUgUWPl0eYIH0wnxqeSAhuIwvnnpjdVB8cRRClyKLQr7lgzjctkbp5KmswWszlwYln03Q==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ] - }, - "node_modules/@rollup/rollup-darwin-x64": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.30.1.tgz", - "integrity": "sha512-x78BavIwSH6sqfP2xeI1hd1GpHL8J4W2BXcVM/5KYKoAD3nNsfitQhvWSw+TFtQTLZ9OmlF+FEInEHyubut2OA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ] - }, - "node_modules/@rollup/rollup-freebsd-arm64": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.30.1.tgz", - "integrity": "sha512-HYTlUAjbO1z8ywxsDFWADfTRfTIIy/oUlfIDmlHYmjUP2QRDTzBuWXc9O4CXM+bo9qfiCclmHk1x4ogBjOUpUQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ] - }, - "node_modules/@rollup/rollup-freebsd-x64": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.30.1.tgz", - "integrity": "sha512-1MEdGqogQLccphhX5myCJqeGNYTNcmTyaic9S7CG3JhwuIByJ7J05vGbZxsizQthP1xpVx7kd3o31eOogfEirw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ] - }, - "node_modules/@rollup/rollup-linux-arm-gnueabihf": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.30.1.tgz", - "integrity": "sha512-PaMRNBSqCx7K3Wc9QZkFx5+CX27WFpAMxJNiYGAXfmMIKC7jstlr32UhTgK6T07OtqR+wYlWm9IxzennjnvdJg==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-arm-musleabihf": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.30.1.tgz", - "integrity": "sha512-B8Rcyj9AV7ZlEFqvB5BubG5iO6ANDsRKlhIxySXcF1axXYUyqwBok+XZPgIYGBgs7LDXfWfifxhw0Ik57T0Yug==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-arm64-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.30.1.tgz", - "integrity": "sha512-hqVyueGxAj3cBKrAI4aFHLV+h0Lv5VgWZs9CUGqr1z0fZtlADVV1YPOij6AhcK5An33EXaxnDLmJdQikcn5NEw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } }, - "node_modules/@rollup/rollup-linux-arm64-musl": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.30.1.tgz", - "integrity": "sha512-i4Ab2vnvS1AE1PyOIGp2kXni69gU2DAUVt6FSXeIqUCPIR3ZlheMW3oP2JkukDfu3PsexYRbOiJrY+yVNSk9oA==", - "cpu": [ - "arm64" - ], + "node_modules/@radix-ui/react-use-escape-keydown": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-escape-keydown/-/react-use-escape-keydown-1.1.0.tgz", + "integrity": "sha512-L7vwWlR1kTTQ3oh7g1O0CBF3YCyyTj8NmhLR+phShpyA50HCfBFKVJTpshm9PzLiKmehsrQzTYTpX9HvmC9rhw==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "linux" - ] + "dependencies": { + "@radix-ui/react-use-callback-ref": "1.1.0" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } }, - "node_modules/@rollup/rollup-linux-loongarch64-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.30.1.tgz", - "integrity": "sha512-fARcF5g296snX0oLGkVxPmysetwUk2zmHcca+e9ObOovBR++9ZPOhqFUM61UUZ2EYpXVPN1redgqVoBB34nTpQ==", - "cpu": [ - "loong64" - ], + "node_modules/@radix-ui/react-use-layout-effect": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-layout-effect/-/react-use-layout-effect-1.1.0.tgz", + "integrity": "sha512-+FPE0rOdziWSrH9athwI1R0HDVbWlEhd+FR+aSDk4uWGmSJ9Z54sdZVDQPZAinJhJXwfT+qnj969mCsT2gfm5w==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "linux" - ] + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } }, - "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.30.1.tgz", - "integrity": "sha512-GLrZraoO3wVT4uFXh67ElpwQY0DIygxdv0BNW9Hkm3X34wu+BkqrDrkcsIapAY+N2ATEbvak0XQ9gxZtCIA5Rw==", - "cpu": [ - "ppc64" - ], + "node_modules/@radix-ui/react-use-rect": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-rect/-/react-use-rect-1.1.0.tgz", + "integrity": "sha512-0Fmkebhr6PiseyZlYAOtLS+nb7jLmpqTrJyv61Pe68MKYW6OWdRE2kI70TaYY27u7H0lajqM3hSMMLFq18Z7nQ==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "linux" - ] + "dependencies": { + "@radix-ui/rect": "1.1.0" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } }, - "node_modules/@rollup/rollup-linux-riscv64-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.30.1.tgz", - "integrity": "sha512-0WKLaAUUHKBtll0wvOmh6yh3S0wSU9+yas923JIChfxOaaBarmb/lBKPF0w/+jTVozFnOXJeRGZ8NvOxvk/jcw==", - "cpu": [ - "riscv64" - ], + "node_modules/@radix-ui/react-use-size": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-size/-/react-use-size-1.1.0.tgz", + "integrity": "sha512-XW3/vWuIXHa+2Uwcc2ABSfcCledmXhhQPlGbfcRXbiUQI5Icjcg19BGCZVKKInYbvUCut/ufbbLLPFC5cbb1hw==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "linux" - ] + "dependencies": { + "@radix-ui/react-use-layout-effect": "1.1.0" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } }, - "node_modules/@rollup/rollup-linux-s390x-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.30.1.tgz", - "integrity": "sha512-GWFs97Ruxo5Bt+cvVTQkOJ6TIx0xJDD/bMAOXWJg8TCSTEK8RnFeOeiFTxKniTc4vMIaWvCplMAFBt9miGxgkA==", - "cpu": [ - "s390x" - ], + "node_modules/@radix-ui/rect": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@radix-ui/rect/-/rect-1.1.0.tgz", + "integrity": "sha512-A9+lCBZoaMJlVKcRBz2YByCG+Cp2t6nAnMnNba+XiWxnj6r4JUFqfsgwocMBZU9LPtdxC6wB56ySYpc7LQIoJg==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] + "license": "MIT" }, "node_modules/@rollup/rollup-linux-x64-gnu": { "version": "4.30.1", @@ -5284,48 +4492,6 @@ "linux" ] }, - "node_modules/@rollup/rollup-win32-arm64-msvc": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.30.1.tgz", - "integrity": "sha512-WabtHWiPaFF47W3PkHnjbmWawnX/aE57K47ZDT1BXTS5GgrBUEpvOzq0FI0V/UYzQJgdb8XlhVNH8/fwV8xDjw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@rollup/rollup-win32-ia32-msvc": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.30.1.tgz", - "integrity": "sha512-pxHAU+Zv39hLUTdQQHUVHf4P+0C47y/ZloorHpzs2SXMRqeAWmGghzAhfOlzFHHwjvgokdFAhC4V+6kC1lRRfw==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@rollup/rollup-win32-x64-msvc": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.30.1.tgz", - "integrity": "sha512-D6qjsXGcvhTjv0kI4fU8tUuBDF/Ueee4SVX79VfNDXZa64TfCW1Slkb6Z7O1p7vflqZjcmOVdZlqf8gvJxc6og==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, "node_modules/@scarf/scarf": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/@scarf/scarf/-/scarf-1.4.0.tgz", @@ -5361,6 +4527,27 @@ "dev": true, "license": "MIT" }, + "node_modules/@sideway/address": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/@sideway/address/-/address-4.1.5.tgz", + "integrity": "sha512-IqO/DUQHUkPeixNQ8n0JA6102hT9CmaljNTPmQ1u8MEhBo/R4Q8eKLN/vGZxuebwOroDB4cbpjheD4+/sKFK4Q==", + "license": "BSD-3-Clause", + "dependencies": { + "@hapi/hoek": "^9.0.0" + } + }, + "node_modules/@sideway/formula": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@sideway/formula/-/formula-3.0.1.tgz", + "integrity": "sha512-/poHZJJVjx3L+zVD6g9KgHfYnb443oi7wLu/XKojDviHy6HOEOA6z1Trk5aR1dGcmPenJEgb2sK2I80LeS3MIg==", + "license": "BSD-3-Clause" + }, + "node_modules/@sideway/pinpoint": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@sideway/pinpoint/-/pinpoint-2.0.0.tgz", + "integrity": "sha512-RNiOoTPkptFtSVzQevY/yWtZwf/RxyVnPy/OcA9HBM3MlGDnBEYL5B41H0MTn0Uec8Hi+2qUtTfG2WWZBmMejQ==", + "license": "BSD-3-Clause" + }, "node_modules/@sindresorhus/is": { "version": "4.6.0", "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-4.6.0.tgz", @@ -7129,7 +6316,6 @@ "version": "3.1.0", "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz", "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==", - "dev": true, "license": "MIT", "dependencies": { "clean-stack": "^2.0.0", @@ -7259,44 +6445,6 @@ "node": ">= 8" } }, - "node_modules/appdmg": { - "version": "0.6.6", - "resolved": "https://registry.npmjs.org/appdmg/-/appdmg-0.6.6.tgz", - "integrity": "sha512-GRmFKlCG+PWbcYF4LUNonTYmy0GjguDy6Jh9WP8mpd0T6j80XIJyXBiWlD0U+MLNhqV9Nhx49Gl9GpVToulpLg==", - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "dependencies": { - "async": "^1.4.2", - "ds-store": "^0.1.5", - "execa": "^1.0.0", - "fs-temp": "^1.0.0", - "fs-xattr": "^0.3.0", - "image-size": "^0.7.4", - "is-my-json-valid": "^2.20.0", - "minimist": "^1.1.3", - "parse-color": "^1.0.0", - "path-exists": "^4.0.0", - "repeat-string": "^1.5.4" - }, - "bin": { - "appdmg": "bin/appdmg.js" - }, - "engines": { - "node": ">=8.5" - } - }, - "node_modules/appdmg/node_modules/async": { - "version": "1.5.2", - "resolved": "https://registry.npmjs.org/async/-/async-1.5.2.tgz", - "integrity": "sha512-nSVgobk4rv61R9PUSDtYt7mPVB2olxNR5RWJcAsH676/ef11bUZwvu7+RGYrYauVdDPcO519v68wRhXQtxsV9w==", - "dev": true, - "license": "MIT", - "optional": true - }, "node_modules/append-field": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/append-field/-/append-field-1.0.0.tgz", @@ -7626,17 +6774,6 @@ "license": "Apache-2.0", "optional": true }, - "node_modules/base32-encode": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/base32-encode/-/base32-encode-1.2.0.tgz", - "integrity": "sha512-cHFU8XeRyx0GgmoWi5qHMCVRiqU6J3MHWxVgun7jggCBUpVzm1Ir7M9dYr2whjSNc3tFeXfQ/oZjQu/4u55h9A==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "to-data-view": "^1.1.0" - } - }, "node_modules/base64-js": { "version": "1.5.1", "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", @@ -7657,6 +6794,15 @@ ], "license": "MIT" }, + "node_modules/base64url": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/base64url/-/base64url-3.0.1.tgz", + "integrity": "sha512-ir1UPr3dkwexU7FdV8qBBbNDRUhMmIekYMFZfi+C/sLNnRESKPl23nB9b2pltqfOQNnGzsDdId90AEtG5tCx4A==", + "license": "MIT", + "engines": { + "node": ">=6.0.0" + } + }, "node_modules/better-sqlite3": { "version": "11.9.1", "resolved": "https://registry.npmjs.org/better-sqlite3/-/better-sqlite3-11.9.1.tgz", @@ -7817,15 +6963,6 @@ "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", "license": "MIT" }, - "node_modules/body-parser/node_modules/toidentifier": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", - "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", - "license": "MIT", - "engines": { - "node": ">=0.6" - } - }, "node_modules/boolbase": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", @@ -7913,17 +7050,6 @@ "object-assign": "^4.1.1" } }, - "node_modules/bplist-creator": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/bplist-creator/-/bplist-creator-0.0.8.tgz", - "integrity": "sha512-Za9JKzD6fjLC16oX2wsXfc+qBEhJBJB1YPInoAQpMLhDuj5aVOv1baGeIQSq1Fr3OCqzvsoQcSBSwGId/Ja2PA==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "stream-buffers": "~2.2.0" - } - }, "node_modules/brace-expansion": { "version": "1.1.11", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", @@ -8563,7 +7689,6 @@ "version": "2.2.0", "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", - "dev": true, "license": "MIT", "engines": { "node": ">=6" @@ -9290,15 +8415,6 @@ "node": ">= 0.8" } }, - "node_modules/csrf-csrf/node_modules/toidentifier": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", - "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", - "license": "MIT", - "engines": { - "node": ">=0.6" - } - }, "node_modules/css-loader": { "version": "7.1.2", "resolved": "https://registry.npmjs.org/css-loader/-/css-loader-7.1.2.tgz", @@ -10373,19 +9489,6 @@ "unidragger": "^3.0.0" } }, - "node_modules/ds-store": { - "version": "0.1.6", - "resolved": "https://registry.npmjs.org/ds-store/-/ds-store-0.1.6.tgz", - "integrity": "sha512-kY21M6Lz+76OS3bnCzjdsJSF7LBpLYGCVfavW8TgQD2XkcqIZ86W0y9qUDZu6fp7SIZzqosMDW2zi7zVFfv4hw==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "bplist-creator": "~0.0.3", - "macos-alias": "~0.2.5", - "tn1150": "^0.1.0" - } - }, "node_modules/dunder-proto": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", @@ -11202,14 +10305,6 @@ "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", "license": "MIT" }, - "node_modules/encode-utf8": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/encode-utf8/-/encode-utf8-1.0.3.tgz", - "integrity": "sha512-ucAnuBEhUK4boH2HjVYG5Q2mQyPorvv0u/ocS+zhdw0S8AlHYY+GOFhP1Gio5z4icpP2ivFSvhtFjQi8+T9ppw==", - "dev": true, - "license": "MIT", - "optional": true - }, "node_modules/encodeurl": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", @@ -11412,23 +10507,6 @@ "@esbuild/win32-x64": "0.25.0" } }, - "node_modules/esbuild/node_modules/@esbuild/netbsd-arm64": { - "version": "0.25.0", - "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.0.tgz", - "integrity": "sha512-RuG4PSMPFfrkH6UwCAqBzauBWTygTvb1nxWasEJooGSJ/NwRw7b2HOwyRTQIU97Hq37l3npXoZGYMy3b3xYvPw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "netbsd" - ], - "engines": { - "node": ">=18" - } - }, "node_modules/escalade": { "version": "3.2.0", "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", @@ -11917,6 +10995,40 @@ "url": "https://opencollective.com/express" } }, + "node_modules/express-openid-connect": { + "version": "2.18.0", + "resolved": "https://registry.npmjs.org/express-openid-connect/-/express-openid-connect-2.18.0.tgz", + "integrity": "sha512-UynJUKAn29jYtXGnjLqn22YES2GNn5GhT7iEiN3W7EaWMt/8dg39UJM9av4R44rPwEE4JNzIkd/Gg4InIiMQNQ==", + "license": "MIT", + "dependencies": { + "base64url": "^3.0.1", + "clone": "^2.1.2", + "cookie": "^0.7.1", + "debug": "^4.3.4", + "futoin-hkdf": "^1.5.1", + "http-errors": "^1.8.1", + "joi": "^17.7.0", + "jose": "^2.0.7", + "on-headers": "^1.0.2", + "openid-client": "^4.9.1", + "url-join": "^4.0.1" + }, + "engines": { + "node": "^10.19.0 || >=12.0.0 < 13 || >=13.7.0 < 14 || >= 14.2.0" + }, + "peerDependencies": { + "express": ">= 4.17.0" + } + }, + "node_modules/express-openid-connect/node_modules/clone": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz", + "integrity": "sha512-3Pe/CF1Nn94hyhIYpjtiLhdCoEoz0DqQ+988E9gmeEdQZlojxnOb74wctFyuwWQHzqyf9X7C7MG8juUpqBJT8w==", + "license": "MIT", + "engines": { + "node": ">=0.8" + } + }, "node_modules/express-rate-limit": { "version": "7.5.0", "resolved": "https://registry.npmjs.org/express-rate-limit/-/express-rate-limit-7.5.0.tgz", @@ -12010,16 +11122,7 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", - "license": "MIT" - }, - "node_modules/express/node_modules/toidentifier": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", - "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", - "license": "MIT", - "engines": { - "node": ">=0.6" - } + "license": "MIT" }, "node_modules/ext-list": { "version": "2.2.2", @@ -12478,17 +11581,6 @@ "graceful-fs": "^4.1.6" } }, - "node_modules/fmix": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/fmix/-/fmix-0.1.0.tgz", - "integrity": "sha512-Y6hyofImk9JdzU8k5INtTXX1cu8LDlePWDFU5sftm9H+zKCr5SGrVjdhkvsim646cw5zD0nADj8oHyXMZmCZ9w==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "imul": "^1.0.0" - } - }, "node_modules/follow-redirects": { "version": "1.15.9", "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", @@ -12700,32 +11792,6 @@ "node": ">= 8" } }, - "node_modules/fs-temp": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/fs-temp/-/fs-temp-1.2.1.tgz", - "integrity": "sha512-okTwLB7/Qsq82G6iN5zZJFsOfZtx2/pqrA7Hk/9fvy+c+eJS9CvgGXT2uNxwnI14BDY9L/jQPkaBgSvlKfSW9w==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "random-path": "^0.1.0" - } - }, - "node_modules/fs-xattr": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/fs-xattr/-/fs-xattr-0.3.1.tgz", - "integrity": "sha512-UVqkrEW0GfDabw4C3HOrFlxKfx0eeigfRne69FxSBdHIP8Qt5Sq6Pu3RM9KmMlkygtC4pPKkj5CiPO5USnj2GA==", - "dev": true, - "hasInstallScript": true, - "license": "MIT", - "optional": true, - "os": [ - "!win32" - ], - "engines": { - "node": ">=8.6.0" - } - }, "node_modules/fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", @@ -12757,6 +11823,15 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/futoin-hkdf": { + "version": "1.5.3", + "resolved": "https://registry.npmjs.org/futoin-hkdf/-/futoin-hkdf-1.5.3.tgz", + "integrity": "sha512-SewY5KdMpaoCeh7jachEWFsh1nNlaDjNHZXWqL5IGwtpEYHTgkr2+AMCgNwKWkcc0wpSYrZfR7he4WdmHFtDxQ==", + "license": "Apache-2.0", + "engines": { + "node": ">=8" + } + }, "node_modules/fuzzy": { "version": "0.1.3", "resolved": "https://registry.npmjs.org/fuzzy/-/fuzzy-0.1.3.tgz", @@ -12818,28 +11893,6 @@ "license": "MIT", "optional": true }, - "node_modules/generate-function": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/generate-function/-/generate-function-2.3.1.tgz", - "integrity": "sha512-eeB5GfMNeevm/GRYq20ShmsaGcmI81kIX2K9XQx5miC8KdHaC6Jm0qQ8ZNeGOi7wYB8OsdxKs+Y2oVuTFuVwKQ==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "is-property": "^1.0.2" - } - }, - "node_modules/generate-object-property": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/generate-object-property/-/generate-object-property-1.2.0.tgz", - "integrity": "sha512-TuOwZWgJ2VAMEGJvAyPWvpqxSANF0LDpmyHauMjFYzaACvn+QTT/AZomvPCzVBV7yDN3OmwHQ5OvHaeLKre3JQ==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "is-property": "^1.0.0" - } - }, "node_modules/get-caller-file": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", @@ -13490,6 +12543,40 @@ "integrity": "sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ==", "license": "BSD-2-Clause" }, + "node_modules/http-errors": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.8.1.tgz", + "integrity": "sha512-Kpk9Sm7NmI+RHhnj6OIWDI1d6fIoFAtFt9RLaTMRlg/8w49juAStsrBgp0Dp4OdxdVbRIeKhtCUvoi/RuAhO4g==", + "license": "MIT", + "dependencies": { + "depd": "~1.1.2", + "inherits": "2.0.4", + "setprototypeof": "1.2.0", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.1" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/http-errors/node_modules/depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/http-errors/node_modules/statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha512-OpZ3zP+jT1PI7I8nemJX4AKmAX070ZkYPVWV/AaKTJl+tXCTGyVdC1a4SL8RUQYEwk/f34ZX8UTykN68FwrqAA==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, "node_modules/http-proxy-agent": { "version": "7.0.2", "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-7.0.2.tgz", @@ -13683,20 +12770,6 @@ "integrity": "sha512-QpLcX9ZSsq3YYUUnD3nFDY8H7wctAhQj/TFKL8Ya8v5fMm3CFXxo8zStsLAl780ltoYoo1WvKUVGBQK+1ifr7g==", "license": "MIT" }, - "node_modules/image-size": { - "version": "0.7.5", - "resolved": "https://registry.npmjs.org/image-size/-/image-size-0.7.5.tgz", - "integrity": "sha512-Hiyv+mXHfFEP7LzUL/llg9RwFxxY+o9N3JVLIeG5E7iFIFAalxvRU9UZthBdYDEVnzHMgjnKJPPpay5BWf1g9g==", - "dev": true, - "license": "MIT", - "optional": true, - "bin": { - "image-size": "bin/image-size.js" - }, - "engines": { - "node": ">=6.9.0" - } - }, "node_modules/image-type": { "version": "5.2.0", "resolved": "https://registry.npmjs.org/image-type/-/image-type-5.2.0.tgz", @@ -13762,17 +12835,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/imul": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/imul/-/imul-1.0.1.tgz", - "integrity": "sha512-WFAgfwPLAjU66EKt6vRdTlKj4nAgIDQzh29JonLa4Bqtl6D8JrIMvWjCnx7xEjVNmP3U0fM5o8ZObk7d0f62bA==", - "dev": true, - "license": "MIT", - "optional": true, - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/imurmurhash": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", @@ -13786,7 +12848,6 @@ "version": "4.0.0", "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", - "dev": true, "license": "MIT", "engines": { "node": ">=8" @@ -14006,29 +13067,6 @@ "dev": true, "license": "MIT" }, - "node_modules/is-my-ip-valid": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-my-ip-valid/-/is-my-ip-valid-1.0.1.tgz", - "integrity": "sha512-jxc8cBcOWbNK2i2aTkCZP6i7wkHF1bqKFrwEHuN5Jtg5BSaZHUZQ/JTOJwoV41YvHnOaRyWWh72T/KvfNz9DJg==", - "dev": true, - "license": "MIT", - "optional": true - }, - "node_modules/is-my-json-valid": { - "version": "2.20.6", - "resolved": "https://registry.npmjs.org/is-my-json-valid/-/is-my-json-valid-2.20.6.tgz", - "integrity": "sha512-1JQwulVNjx8UqkPE/bqDaxtH4PXCe/2VRh/y3p99heOV87HG4Id5/VfDswd+YiAfHcRTfDlWgISycnHuhZq1aw==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "generate-function": "^2.0.0", - "generate-object-property": "^1.1.0", - "is-my-ip-valid": "^1.0.0", - "jsonpointer": "^5.0.0", - "xtend": "^4.0.0" - } - }, "node_modules/is-number": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", @@ -14063,14 +13101,6 @@ "integrity": "sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==", "license": "MIT" }, - "node_modules/is-property": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-property/-/is-property-1.0.2.tgz", - "integrity": "sha512-Ks/IoX00TtClbGQr4TWXemAnktAQvYB7HzcCxDGqEZU6oCmb2INHuOoKxbtR+HFkmYWBKv/dOZtGRiAjDhj92g==", - "dev": true, - "license": "MIT", - "optional": true - }, "node_modules/is-stream": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", @@ -14342,6 +13372,34 @@ "jiti": "bin/jiti.js" } }, + "node_modules/joi": { + "version": "17.13.3", + "resolved": "https://registry.npmjs.org/joi/-/joi-17.13.3.tgz", + "integrity": "sha512-otDA4ldcIx+ZXsKHWmp0YizCweVRZG96J10b0FevjfuncLO1oX59THoAmHkNubYJ+9gWsYsp5k8v4ib6oDv1fA==", + "license": "BSD-3-Clause", + "dependencies": { + "@hapi/hoek": "^9.3.0", + "@hapi/topo": "^5.1.0", + "@sideway/address": "^4.1.5", + "@sideway/formula": "^3.0.1", + "@sideway/pinpoint": "^2.0.0" + } + }, + "node_modules/jose": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/jose/-/jose-2.0.7.tgz", + "integrity": "sha512-5hFWIigKqC+e/lRyQhfnirrAqUdIPMB7SJRqflJaO29dW7q5DFvH1XCSTmv6PQ6pb++0k6MJlLRoS0Wv4s38Wg==", + "license": "MIT", + "dependencies": { + "@panva/asn1.js": "^1.0.0" + }, + "engines": { + "node": ">=10.13.0 < 13 || >=13.7.0" + }, + "funding": { + "url": "https://github.com/sponsors/panva" + } + }, "node_modules/jotai": { "version": "2.11.0", "resolved": "https://registry.npmjs.org/jotai/-/jotai-2.11.0.tgz", @@ -14585,17 +13643,6 @@ "graceful-fs": "^4.1.6" } }, - "node_modules/jsonpointer": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/jsonpointer/-/jsonpointer-5.0.1.tgz", - "integrity": "sha512-p/nXbhSEcu3pZRdkW1OfJhpsVtW1gd4Wa1fnQc9YLiTfAjn0312eMKimbdIQzuZl9aa9xUGaRlP9T/CJE/ditQ==", - "dev": true, - "license": "MIT", - "optional": true, - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/jsplumb": { "version": "2.15.6", "resolved": "https://registry.npmjs.org/jsplumb/-/jsplumb-2.15.6.tgz", @@ -15093,21 +14140,6 @@ "dev": true, "license": "MIT" }, - "node_modules/macos-alias": { - "version": "0.2.12", - "resolved": "https://registry.npmjs.org/macos-alias/-/macos-alias-0.2.12.tgz", - "integrity": "sha512-yiLHa7cfJcGRFq4FrR4tMlpNHb4Vy4mWnpajlSSIFM5k4Lv8/7BbbDLzCAVogWNl0LlLhizRp1drXv0hK9h0Yw==", - "dev": true, - "hasInstallScript": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "dependencies": { - "nan": "^2.4.0" - } - }, "node_modules/magic-string": { "version": "0.30.17", "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.17.tgz", @@ -15146,6 +14178,12 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "license": "ISC" + }, "node_modules/make-fetch-happen": { "version": "10.2.1", "resolved": "https://registry.npmjs.org/make-fetch-happen/-/make-fetch-happen-10.2.1.tgz", @@ -16237,27 +15275,6 @@ "object-assign": "^4.1.1" } }, - "node_modules/murmur-32": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/murmur-32/-/murmur-32-0.2.0.tgz", - "integrity": "sha512-ZkcWZudylwF+ir3Ld1n7gL6bI2mQAzXvSobPwVtu8aYi2sbXeipeSkdcanRLzIofLcM5F53lGaKm2dk7orBi7Q==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "encode-utf8": "^1.0.3", - "fmix": "^0.1.0", - "imul": "^1.0.0" - } - }, - "node_modules/nan": { - "version": "2.22.0", - "resolved": "https://registry.npmjs.org/nan/-/nan-2.22.0.tgz", - "integrity": "sha512-nbajikzWTMwsW+eSsNm3QwlOs7het9gGJU5dDZzRTQGk03vyBOauxgI4VakDzE0PtsGTmXPsXTbbjVhRwR5mpw==", - "dev": true, - "license": "MIT", - "optional": true - }, "node_modules/nanoid": { "version": "3.3.8", "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.8.tgz", @@ -16631,6 +15648,15 @@ "node": ">=0.10.0" } }, + "node_modules/object-hash": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-2.2.0.tgz", + "integrity": "sha512-gScRMn0bS5fH+IuwyIFgnh9zBdo4DV+6GhygmWM9HyNJSgS0hScp1f5vjtm7oIIOiT9trXrShAkLFSc2IqKNgw==", + "license": "MIT", + "engines": { + "node": ">= 6" + } + }, "node_modules/object-inspect": { "version": "1.13.3", "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.3.tgz", @@ -16653,6 +15679,15 @@ "node": ">= 0.4" } }, + "node_modules/oidc-token-hash": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/oidc-token-hash/-/oidc-token-hash-5.1.0.tgz", + "integrity": "sha512-y0W+X7Ppo7oZX6eovsRkuzcSM40Bicg2JEJkDJ4irIt1wsYAP5MLSNv+QAogO8xivMffw/9OvV3um1pxXgt1uA==", + "license": "MIT", + "engines": { + "node": "^10.13.0 || >=12.0.0" + } + }, "node_modules/omggif": { "version": "1.0.10", "resolved": "https://registry.npmjs.org/omggif/-/omggif-1.0.10.tgz", @@ -16720,6 +15755,39 @@ "license": "MIT", "peer": true }, + "node_modules/openid-client": { + "version": "4.9.1", + "resolved": "https://registry.npmjs.org/openid-client/-/openid-client-4.9.1.tgz", + "integrity": "sha512-DYUF07AHjI3QDKqKbn2F7RqozT4hyi4JvmpodLrq0HHoNP7t/AjeG/uqiBK1/N2PZSAQEThVjDLHSmJN4iqu/w==", + "license": "MIT", + "dependencies": { + "aggregate-error": "^3.1.0", + "got": "^11.8.0", + "jose": "^2.0.5", + "lru-cache": "^6.0.0", + "make-error": "^1.3.6", + "object-hash": "^2.0.1", + "oidc-token-hash": "^5.0.1" + }, + "engines": { + "node": "^10.19.0 || >=12.0.0 < 13 || >=13.7.0 < 14 || >= 14.2.0" + }, + "funding": { + "url": "https://github.com/sponsors/panva" + } + }, + "node_modules/openid-client/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "license": "ISC", + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/optionator": { "version": "0.9.4", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", @@ -17008,24 +16076,6 @@ "node": ">=4.0" } }, - "node_modules/parse-color": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/parse-color/-/parse-color-1.0.0.tgz", - "integrity": "sha512-fuDHYgFHJGbpGMgw9skY/bj3HL/Jrn4l/5rSspy00DoT4RyLnDcRvPxdZ+r6OFwIsgAuhDh4I09tAId4mI12bw==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "color-convert": "~0.5.0" - } - }, - "node_modules/parse-color/node_modules/color-convert": { - "version": "0.5.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-0.5.3.tgz", - "integrity": "sha512-RwBeO/B/vZR3dfKL1ye/vx8MHZ40ugzpyfeVG5GsiuGnrlMWe2o8wxBbLCpw9CsxV+wHuzYlCiWnybrIA0ling==", - "dev": true, - "optional": true - }, "node_modules/parse-json": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", @@ -17953,18 +17003,6 @@ "node": ">= 0.8" } }, - "node_modules/random-path": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/random-path/-/random-path-0.1.2.tgz", - "integrity": "sha512-4jY0yoEaQ5v9StCl5kZbNIQlg1QheIDBrdkDn53EynpPb9FgO6//p3X/tgMnrC45XN6QZCzU1Xz/+pSSsJBpRw==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "base32-encode": "^0.1.0 || ^1.0.0", - "murmur-32": "^0.1.0 || ^0.2.0" - } - }, "node_modules/randombytes": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", @@ -18500,17 +17538,6 @@ "integrity": "sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==", "license": "MIT" }, - "node_modules/repeat-string": { - "version": "1.6.1", - "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", - "integrity": "sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==", - "dev": true, - "license": "MIT", - "optional": true, - "engines": { - "node": ">=0.10" - } - }, "node_modules/require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", @@ -19208,15 +18235,6 @@ "node": ">=4" } }, - "node_modules/send/node_modules/toidentifier": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", - "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", - "license": "MIT", - "engines": { - "node": ">=0.6" - } - }, "node_modules/serialize-error": { "version": "7.0.1", "resolved": "https://registry.npmjs.org/serialize-error/-/serialize-error-7.0.1.tgz", @@ -19772,17 +18790,6 @@ "dev": true, "license": "MIT" }, - "node_modules/stream-buffers": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/stream-buffers/-/stream-buffers-2.2.0.tgz", - "integrity": "sha512-uyQK/mx5QjHun80FLJTfaWE7JtwfRMKBLkMne6udYOmvH0CawotVa7TfgYHzAnpphn4+TweIx1QKMnRIbipmUg==", - "dev": true, - "license": "Unlicense", - "optional": true, - "engines": { - "node": ">= 0.10.0" - } - }, "node_modules/stream-throttle": { "version": "0.1.3", "resolved": "https://registry.npmjs.org/stream-throttle/-/stream-throttle-0.1.3.tgz", @@ -20544,6 +19551,12 @@ "tslib": "^2" } }, + "node_modules/time2fa": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/time2fa/-/time2fa-1.4.2.tgz", + "integrity": "sha512-badZQkQpCi8eZWN02HTjoBBg+leBmwiLWFQtweklEhY8+JEGXSgd2Xy6nGBtPi+7HigSczclYTljAEJA4Z9D4g==", + "license": "MIT" + }, "node_modules/tiny-each-async": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/tiny-each-async/-/tiny-each-async-2.0.3.tgz", @@ -20696,28 +19709,6 @@ "tmp": "^0.2.0" } }, - "node_modules/tn1150": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/tn1150/-/tn1150-0.1.0.tgz", - "integrity": "sha512-DbplOfQFkqG5IHcDyyrs/lkvSr3mPUVsFf/RbDppOshs22yTPnSJWEe6FkYd1txAwU/zcnR905ar2fi4kwF29w==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "unorm": "^1.4.1" - }, - "engines": { - "node": ">=0.12" - } - }, - "node_modules/to-data-view": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/to-data-view/-/to-data-view-1.1.0.tgz", - "integrity": "sha512-1eAdufMg6mwgmlojAx3QeMnzB/BTVp7Tbndi3U7ftcT2zCZadjxkkmLmd97zmaxWi+sgGcgWrokmpEoy0Dn0vQ==", - "dev": true, - "license": "MIT", - "optional": true - }, "node_modules/to-regex-range": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", @@ -20731,6 +19722,15 @@ "node": ">=8.0" } }, + "node_modules/toidentifier": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", + "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", + "license": "MIT", + "engines": { + "node": ">=0.6" + } + }, "node_modules/token-types": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/token-types/-/token-types-5.0.1.tgz", @@ -20832,9 +19832,9 @@ } }, "node_modules/ts-api-utils": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.0.1.tgz", - "integrity": "sha512-dnlgjFSVetynI8nzgJ+qF62efpglpWRk8isUEWZGWlJYySCTD6aKvbUDu+zbPeDakk3bg5H4XpitHukgfL1m9w==", + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.1.0.tgz", + "integrity": "sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ==", "dev": true, "license": "MIT", "engines": { @@ -21249,17 +20249,6 @@ "node": ">= 10.0.0" } }, - "node_modules/unorm": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/unorm/-/unorm-1.6.0.tgz", - "integrity": "sha512-b2/KCUlYZUeA7JFUuRJZPUtr4gZvBh7tavtv4fvk4+KV9pfGiR6CQAQAWl49ZpR3ts2dk4FYkP7EIgDJoiOLDA==", - "dev": true, - "license": "MIT or GPL-2.0", - "optional": true, - "engines": { - "node": ">= 0.4.0" - } - }, "node_modules/unpipe": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", @@ -21347,6 +20336,12 @@ "punycode": "^2.1.0" } }, + "node_modules/url-join": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/url-join/-/url-join-4.0.1.tgz", + "integrity": "sha512-jk1+QP6ZJqyOiuEI9AEWQfju/nB2Pw466kbA0LEZljHwKeMgd9WrAEgEGxjPDD2+TNbbb37rTyhEfrCXfuKXnA==", + "license": "MIT" + }, "node_modules/use-callback-ref": { "version": "1.3.3", "resolved": "https://registry.npmjs.org/use-callback-ref/-/use-callback-ref-1.3.3.tgz", @@ -22342,7 +21337,6 @@ "version": "4.0.0", "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", - "dev": true, "license": "ISC" }, "node_modules/yaml": { diff --git a/package.json b/package.json index 1f27509cf8..2589dd8eda 100644 --- a/package.json +++ b/package.json @@ -99,6 +99,7 @@ "escape-html": "1.0.3", "eslint-linter-browserify": "9.23.0", "express": "4.21.2", + "express-openid-connect": "^2.17.1", "express-rate-limit": "7.5.0", "express-session": "1.18.1", "force-graph": "1.49.5", @@ -143,6 +144,7 @@ "strip-bom": "5.0.0", "striptags": "3.2.0", "swagger-ui-express": "5.0.1", + "time2fa": "^1.3.0", "tmp": "0.2.3", "turndown": "7.2.0", "unescape": "1.0.1", diff --git a/src/app.ts b/src/app.ts index 549c02b71e..326c15efdd 100644 --- a/src/app.ts +++ b/src/app.ts @@ -14,6 +14,8 @@ import custom from "./routes/custom.js"; import error_handlers from "./routes/error_handlers.js"; import { startScheduledCleanup } from "./services/erase.js"; import sql_init from "./services/sql_init.js"; +import { auth } from "express-openid-connect"; +import openID from "./services/open_id.js"; import { t } from "i18next"; await import("./services/handlers.js"); @@ -59,6 +61,9 @@ app.use(`/icon.png`, express.static(path.join(scriptDir, "public/icon.png"))); app.use(sessionParser); app.use(favicon(`${scriptDir}/../images/app-icons/icon.ico`)); +if (openID.isOpenIDEnabled()) + app.use(auth(openID.generateOAuthConfig())); + await assets.register(app); routes.register(app); custom.register(app); diff --git a/src/errors/open_id_error.ts b/src/errors/open_id_error.ts new file mode 100644 index 0000000000..0206a17f34 --- /dev/null +++ b/src/errors/open_id_error.ts @@ -0,0 +1,9 @@ +class OpenIdError { + message: string; + + constructor(message: string) { + this.message = message; + } +} + +export default OpenIdError; \ No newline at end of file diff --git a/src/express.d.ts b/src/express.d.ts index eb523dff2a..846c5b9b21 100644 --- a/src/express.d.ts +++ b/src/express.d.ts @@ -4,6 +4,10 @@ export declare module "express-serve-static-core" { interface Request { session: Session & { loggedIn: boolean; + lastAuthState: { + totpEnabled: boolean; + ssoEnabled: boolean; + }; }; headers: { "x-local-date"?: string; diff --git a/src/public/app/widgets/type_widgets/content/backend_log.ts b/src/public/app/widgets/type_widgets/content/backend_log.ts index 03699d8734..902b729aec 100644 --- a/src/public/app/widgets/type_widgets/content/backend_log.ts +++ b/src/public/app/widgets/type_widgets/content/backend_log.ts @@ -36,7 +36,7 @@ export default class BackendLogWidget extends AbstractCodeTypeWidget { await this.load(); } - getExtraOpts() { + getExtraOpts(): Partial { return { lineWrapping: false, readOnly: true diff --git a/src/public/app/widgets/type_widgets/content_widget.ts b/src/public/app/widgets/type_widgets/content_widget.ts index 5fbb49a7a7..c7a7f70865 100644 --- a/src/public/app/widgets/type_widgets/content_widget.ts +++ b/src/public/app/widgets/type_widgets/content_widget.ts @@ -32,6 +32,7 @@ import DatabaseAnonymizationOptions from "./options/advanced/database_anonymizat import BackendLogWidget from "./content/backend_log.js"; import AttachmentErasureTimeoutOptions from "./options/other/attachment_erasure_timeout.js"; import RibbonOptions from "./options/appearance/ribbon.js"; +import MultiFactorAuthenticationOptions from './options/multi_factor_authentication.js'; import LocalizationOptions from "./options/i18n/i18n.js"; import CodeBlockOptions from "./options/appearance/code_block.js"; import EditorOptions from "./options/text_notes/editor.js"; @@ -94,6 +95,7 @@ const CONTENT_WIDGETS: Record = { PasswordOptions, ProtectedSessionTimeoutOptions ], + _optionsMFA: [MultiFactorAuthenticationOptions], _optionsEtapi: [ EtapiOptions ], diff --git a/src/public/app/widgets/type_widgets/options/multi_factor_authentication.ts b/src/public/app/widgets/type_widgets/options/multi_factor_authentication.ts new file mode 100644 index 0000000000..c369e676fb --- /dev/null +++ b/src/public/app/widgets/type_widgets/options/multi_factor_authentication.ts @@ -0,0 +1,342 @@ +import server from "../../../services/server.js"; +import toastService from "../../../services/toast.js"; +import OptionsWidget from "./options_widget.js"; +import type { OptionMap } from "../../../../../services/options_interface.js"; +import { t } from "../../../services/i18n.js"; +import utils from "../../../services/utils.js"; +import dialogService from "../../../services/dialog.js"; + +const TPL = ` +
+

${t("multi_factor_authentication.title")}

+

${t("multi_factor_authentication.description")}

+ +
+ +
+ +
+ + +
+`; + +const TPL_ELECTRON = ` +
+

${t("multi_factor_authentication.title")}

+

${t("multi_factor_authentication.electron_disabled")}

+
+`; + +interface OAuthStatus { + enabled: boolean; + name?: string; + email?: string; + missingVars?: string[]; +} + +interface TOTPStatus { + set: boolean; +} + +interface RecoveryKeysResponse { + success: boolean; + recoveryCodes?: string[]; + keysExist?: boolean; + usedRecoveryCodes?: string[]; +} + +export default class MultiFactorAuthenticationOptions extends OptionsWidget { + private $mfaEnabledCheckbox!: JQuery; + private $mfaOptions!: JQuery; + private $mfaMethodRadios!: JQuery; + private $totpOptions!: JQuery; + private $noTotpSecretWarning!: JQuery; + private $generateTotpButton!: JQuery; + private $generateRecoveryCodeButton!: JQuery; + private $recoveryKeys: JQuery[] = []; + private $oauthOptions!: JQuery; + private $UserAccountName!: JQuery; + private $UserAccountEmail!: JQuery; + private $oauthWarning!: JQuery; + private $missingVars!: JQuery; + + doRender() { + const template = utils.isElectron() ? TPL_ELECTRON : TPL; + this.$widget = $(template); + + if (!utils.isElectron()) { + this.$mfaEnabledCheckbox = this.$widget.find(".mfa-enabled-checkbox"); + this.$mfaOptions = this.$widget.find(".mfa-options"); + this.$mfaMethodRadios = this.$widget.find(".mfa-method-radio"); + this.$totpOptions = this.$widget.find(".totp-options"); + this.$noTotpSecretWarning = this.$widget.find(".no-totp-secret"); + this.$generateTotpButton = this.$widget.find(".generate-totp"); + this.$generateRecoveryCodeButton = this.$widget.find(".generate-recovery-code"); + + this.$oauthOptions = this.$widget.find(".oauth-options"); + this.$UserAccountName = this.$widget.find(".user-account-name"); + this.$UserAccountEmail = this.$widget.find(".user-account-email"); + this.$oauthWarning = this.$widget.find(".oauth-warning"); + this.$missingVars = this.$widget.find(".missing-vars"); + + this.$recoveryKeys = []; + for (let i = 0; i < 8; i++) { + this.$recoveryKeys.push(this.$widget.find(".key_" + i)); + } + + this.$generateRecoveryCodeButton.on("click", async () => { + await this.setRecoveryKeys(); + }); + + this.$generateTotpButton.on("click", async () => { + await this.generateKey(); + }); + + this.displayRecoveryKeys(); + + this.$mfaEnabledCheckbox.on("change", () => { + const isChecked = this.$mfaEnabledCheckbox.is(":checked"); + this.$mfaOptions.toggle(isChecked); + if (!isChecked) { + this.$totpOptions.hide(); + this.$oauthOptions.hide(); + } else { + this.$mfaMethodRadios.filter('[value="totp"]').prop("checked", true); + this.$totpOptions.show(); + this.$oauthOptions.hide(); + } + this.updateCheckboxOption("mfaEnabled", this.$mfaEnabledCheckbox); + }); + + this.$mfaMethodRadios.on("change", () => { + const selectedMethod = this.$mfaMethodRadios.filter(":checked").val(); + this.$totpOptions.toggle(selectedMethod === "totp"); + this.$oauthOptions.toggle(selectedMethod === "oauth"); + this.updateOption("mfaMethod", selectedMethod); + }); + } + } + + async setRecoveryKeys() { + const result = await server.get("totp_recovery/generate"); + if (!result.success) { + toastService.showError(t("multi_factor_authentication.recovery_keys_error")); + return; + } + if (result.recoveryCodes) { + this.keyFiller(result.recoveryCodes); + await server.post("totp_recovery/set", { + recoveryCodes: result.recoveryCodes, + }); + } + } + + async displayRecoveryKeys() { + const result = await server.get("totp_recovery/enabled"); + if (!result.success) { + this.fillKeys(t("multi_factor_authentication.recovery_keys_error")); + return; + } + + if (!result.keysExist) { + this.fillKeys(t("multi_factor_authentication.recovery_keys_no_key_set")); + this.$generateRecoveryCodeButton.text(t("multi_factor_authentication.recovery_keys_generate")); + return; + } + + const usedResult = await server.get("totp_recovery/used"); + + if (usedResult.usedRecoveryCodes) { + this.keyFiller(usedResult.usedRecoveryCodes); + this.$generateRecoveryCodeButton.text(t("multi_factor_authentication.recovery_keys_regenerate")); + } else { + this.fillKeys(t("multi_factor_authentication.recovery_keys_no_key_set")); + } + } + + private keyFiller(values: string[]) { + this.fillKeys(""); + + values.forEach((key, index) => { + if (typeof key === 'string') { + const date = new Date(key.replace(/\//g, '-')); + if (isNaN(date.getTime())) { + this.$recoveryKeys[index].text(key); + } else { + this.$recoveryKeys[index].text(t("multi_factor_authentication.recovery_keys_used", { date: key.replace(/\//g, '-') })); + } + } else { + this.$recoveryKeys[index].text(t("multi_factor_authentication.recovery_keys_unused", { index: key })); + } + }); + } + + private fillKeys(message: string) { + for (let i = 0; i < 8; i++) { + this.$recoveryKeys[i].text(message); + } + } + + async generateKey() { + const totpStatus = await server.get("totp/status"); + + if (totpStatus.set) { + const confirmed = await dialogService.confirm(t("multi_factor_authentication.totp_secret_regenerate_confirm")); + + if (!confirmed) { + return; + } + } + + const result = await server.get<{ success: boolean; message: string }>("totp/generate"); + + if (result.success) { + await dialogService.prompt({ + title: t("multi_factor_authentication.totp_secret_generated"), + message: t("multi_factor_authentication.totp_secret_warning"), + defaultValue: result.message, + shown: ({ $answer }) => { + if ($answer) { + $answer.prop('readonly', true); + } + } + }); + + this.$generateTotpButton.text(t("multi_factor_authentication.totp_secret_regenerate")); + + await this.setRecoveryKeys(); + } else { + toastService.showError(result.message); + } + } + + optionsLoaded(options: OptionMap) { + if (!utils.isElectron()) { + this.$mfaEnabledCheckbox.prop("checked", options.mfaEnabled === "true"); + + this.$mfaOptions.toggle(options.mfaEnabled === "true"); + if (options.mfaEnabled === "true") { + const savedMethod = options.mfaMethod || "totp"; + this.$mfaMethodRadios.filter(`[value="${savedMethod}"]`).prop("checked", true); + this.$totpOptions.toggle(savedMethod === "totp"); + this.$oauthOptions.toggle(savedMethod === "oauth"); + } else { + this.$totpOptions.hide(); + this.$oauthOptions.hide(); + } + + server.get("oauth/status").then((result) => { + if (result.enabled) { + if (result.name) this.$UserAccountName.text(result.name); + if (result.email) this.$UserAccountEmail.text(result.email); + this.$oauthWarning.hide(); + this.$missingVars.hide(); + } else { + this.$UserAccountName.text(t("multi_factor_authentication.oauth_user_not_logged_in")); + this.$UserAccountEmail.text(t("multi_factor_authentication.oauth_user_not_logged_in")); + this.$oauthWarning.show(); + if (result.missingVars && result.missingVars.length > 0) { + this.$missingVars.show(); + const missingVarsList = result.missingVars.map(v => `"${v}"`); + this.$missingVars.html(t("multi_factor_authentication.oauth_missing_vars", { variables: missingVarsList.join(", ") })); + } + } + }); + + server.get("totp/status").then((result) => { + if (result.set) { + this.$generateTotpButton.text(t("multi_factor_authentication.totp_secret_regenerate")); + this.$noTotpSecretWarning.hide(); + } else { + this.$noTotpSecretWarning.show(); + } + }); + } + } +} diff --git a/src/public/app/widgets/type_widgets/read_only_code.ts b/src/public/app/widgets/type_widgets/read_only_code.ts index e4a4bc447c..cf2aaddbf4 100644 --- a/src/public/app/widgets/type_widgets/read_only_code.ts +++ b/src/public/app/widgets/type_widgets/read_only_code.ts @@ -43,7 +43,7 @@ export default class ReadOnlyCodeTypeWidget extends AbstractCodeTypeWidget { this.show(); } - getExtraOpts() { + getExtraOpts(): Partial { return { readOnly: true }; @@ -100,7 +100,7 @@ export default class ReadOnlyCodeTypeWidget extends AbstractCodeTypeWidget { return ret; }); - for (i = pre.length; i--; ) { + for (i = pre.length; i--;) { html = html.replace("<--TEMPPRE" + i + "/-->", pre[i].tag.replace("
", "
\n").replace("
", pre[i].indent + "
")); } diff --git a/src/public/stylesheets/theme-next/pages.css b/src/public/stylesheets/theme-next/pages.css index e5778d4cea..7cc1ee77bd 100644 --- a/src/public/stylesheets/theme-next/pages.css +++ b/src/public/stylesheets/theme-next/pages.css @@ -32,6 +32,31 @@ color: var(--dropdown-item-icon-destructive-color) !important; } +.google-login-btn { + display: flex; + align-items: center; + justify-content: center; + background-color: #fff; + color: #757575; + border: 1px solid #ddd; + border-radius: 4px; + padding: 10px 20px; + font-size: 14px; + font-weight: 500; + cursor: pointer; + margin-bottom: 20px; + text-decoration: none; + transition: background-color 0.3s; +} +.google-login-btn:hover { + background-color: #f5f5f5; +} +.google-login-btn img { + margin-right: 10px; + width: 18px; + height: 18px; +} + /* * SEARCH PAGE */ diff --git a/src/public/translations/cn/translation.json b/src/public/translations/cn/translation.json index 26c4982c68..5ac95e14fb 100644 --- a/src/public/translations/cn/translation.json +++ b/src/public/translations/cn/translation.json @@ -1299,6 +1299,39 @@ "password_mismatch": "新密码不一致。", "password_changed_success": "密码已更改。按 OK 后 Trilium 将重载。" }, + "multi_factor_authentication": { + "title": "多因素认证(MFA)", + "description": "多因素认证(MFA)为您的账户添加了额外的安全层。除了输入密码登录外,MFA还要求您提供一个或多个额外的验证信息来验证您的身份。这样,即使有人获得了您的密码,没有第二个验证信息他们也无法访问您的账户。这就像给您的门添加了一把额外的锁,让他人更难闯入。

请按照以下说明启用 MFA。如果您配置不正确,登录将仅使用密码。", + "mfa_enabled": "启用多因素认证", + "mfa_method": "MFA 方法", + "electron_disabled": "当前桌面版本不支持多因素认证。", + "totp_title": "基于时间的一次性密码(TOTP)", + "totp_description": "TOTP(基于时间的一次性密码)是一种安全功能,它会生成一个每30秒变化的唯一临时代码。您需要使用这个代码和您的密码一起登录账户,这使得他人更难访问您的账户。", + "totp_secret_title": "生成 TOTP 密钥", + "totp_secret_generate": "生成 TOTP 密钥", + "totp_secret_regenerate": "重新生成 TOTP 密钥", + "no_totp_secret_warning": "要启用 TOTP,您需要先生成一个 TOTP 密钥。", + "totp_secret_description_warning": "生成新的 TOTP 密钥后,您需要使用新的 TOTP 密钥重新登录。", + "totp_secret_generated": "TOTP 密钥已生成", + "totp_secret_warning": "请将生成的密钥保存在安全的地方。它将不会再次显示。", + "totp_secret_regenerate_confirm": "您确定要重新生成 TOTP 密钥吗?这将使之前的 TOTP 密钥失效,并使所有现有的恢复代码失效。请将生成的密钥保存在安全的地方。它将不会再次显示。", + "recovery_keys_title": "单点登录恢复密钥", + "recovery_keys_description": "单点登录恢复密钥用于在您无法访问您的认证器代码时登录。离开页面后,恢复密钥将不会再次显示。请将它们保存在安全的地方。", + "recovery_keys_description_warning": "离开页面后,恢复密钥将不会再次显示。请将它们保存在安全的地方。
一旦恢复密钥被使用,它将无法再次使用。", + "recovery_keys_error": "生成恢复代码时出错", + "recovery_keys_no_key_set": "未设置恢复代码", + "recovery_keys_generate": "生成恢复代码", + "recovery_keys_regenerate": "重新生成恢复代码", + "recovery_keys_used": "已使用: {{date}}", + "recovery_keys_unused": "恢复代码 {{index}} 未使用", + "oauth_title": "OAuth/OpenID 认证", + "oauth_description": "OpenID 是一种标准化方式,允许您使用其他服务(如 Google)的账户登录网站,以验证您的身份。请参阅这些 指南 通过 Google 设置 OpenID 服务。", + "oauth_description_warning": "要启用 OAuth/OpenID,您需要设置 config.ini 文件中的 OAuth/OpenID 基础 URL、客户端 ID 和客户端密钥,并重新启动应用程序。如果要从环境变量设置,请设置 TRILIUM_OAUTH_BASE_URL、TRILIUM_OAUTH_CLIENT_ID 和 TRILIUM_OAUTH_CLIENT_SECRET 环境变量。", + "oauth_missing_vars": "缺少以下设置项: {{missingVars}}", + "oauth_user_account": "用户账号:", + "oauth_user_email": "用户邮箱:", + "oauth_user_not_logged_in": "未登录!" + }, "shortcuts": { "keyboard_shortcuts": "快捷键", "multiple_shortcuts": "同一操作的多个快捷键可以用逗号分隔。", diff --git a/src/public/translations/en/translation.json b/src/public/translations/en/translation.json index 998d9f2dc6..13bb74dfd8 100644 --- a/src/public/translations/en/translation.json +++ b/src/public/translations/en/translation.json @@ -1310,6 +1310,39 @@ "password_mismatch": "New passwords are not the same.", "password_changed_success": "Password has been changed. Trilium will be reloaded after you press OK." }, + "multi_factor_authentication": { + "title": "Multi-Factor Authentication", + "description": "Multi-Factor Authentication (MFA) adds an extra layer of security to your account. Instead of just entering a password to log in, MFA requires you to provide one or more additional pieces of evidence to verify your identity. This way, even if someone gets hold of your password, they still can't access your account without the second piece of information. It's like adding an extra lock to your door, making it much harder for anyone else to break in.

Please follow the instructions below to enable MFA. If you don't config correctly, login will fall back to password only.", + "mfa_enabled": "Enable Multi-Factor Authentication", + "mfa_method": "MFA Method", + "electron_disabled": "Multi-Factor Authentication is not supported in the desktop build currently.", + "totp_title": "Time-based One-Time Password (TOTP)", + "totp_description": "TOTP (Time-Based One-Time Password) is a security feature that generates a unique, temporary code which changes every 30 seconds. You use this code, along with your password to log into your account, making it much harder for anyone else to access it.", + "totp_secret_title": "Generate TOTP Secret", + "totp_secret_generate": "Generate TOTP Secret", + "totp_secret_regenerate": "Regenerate TOTP Secret", + "no_totp_secret_warning": "To enable TOTP, you need to generate a TOTP secret first.", + "totp_secret_description_warning": "After generating a new TOTP secret, you will be required to login again with the new TOTP secret.", + "totp_secret_generated": "TOTP Secret Generated", + "totp_secret_warning": "Please save the generated secret in a secure location. It will not be shown again.", + "totp_secret_regenerate_confirm": "Are you sure you want to regenerate the TOTP secret? This will invalidate previous TOTP secret and all existing recovery codes.", + "recovery_keys_title": "Single Sign-on Recovery Keys", + "recovery_keys_description": "Single sign-on recovery keys are used to login in the even you cannot access your Authenticator codes.", + "recovery_keys_description_warning": "Recovery keys won't be shown again after leaving the page, keep them somewhere safe and secure.
After a recovery key is used it cannot be used again.", + "recovery_keys_error": "Error generating recovery codes", + "recovery_keys_no_key_set": "No recovery codes set", + "recovery_keys_generate": "Generate Recovery Codes", + "recovery_keys_regenerate": "Regenerate Recovery Codes", + "recovery_keys_used": "Used: {{date}}", + "recovery_keys_unused": "Recovery code {{index}} is unused", + "oauth_title": "OAuth/OpenID", + "oauth_description": "OpenID is a standardized way to let you log into websites using an account from another service, like Google, to verify your identity. Follow these instructions to setup an OpenID service through Google.", + "oauth_description_warning": "To enable OAuth/OpenID, you need to set the OAuth/OpenID base URL, client ID and client secret in the config.ini file and restart the application. If you want to set from environment variables, please set TRILIUM_OAUTH_BASE_URL, TRILIUM_OAUTH_CLIENT_ID and TRILIUM_OAUTH_CLIENT_SECRET.", + "oauth_missing_vars": "Missing settings: {{variables}}", + "oauth_user_account": "User Account: ", + "oauth_user_email": "User Email: ", + "oauth_user_not_logged_in": "Not logged in!" + }, "shortcuts": { "keyboard_shortcuts": "Keyboard Shortcuts", "multiple_shortcuts": "Multiple shortcuts for the same action can be separated by comma.", @@ -1433,7 +1466,8 @@ "widget": "Widget", "confirm-change": "It is not recommended to change note type when note content is not empty. Do you want to continue anyway?", "geo-map": "Geo Map", - "beta-feature": "Beta" + "beta-feature": "Beta", + "task-list": "Task List" }, "protect_note": { "toggle-on": "Protect the note", diff --git a/src/routes/api/app_info.ts b/src/routes/api/app_info.ts index fb2f84aec8..ece825a30b 100644 --- a/src/routes/api/app_info.ts +++ b/src/routes/api/app_info.ts @@ -1,5 +1,3 @@ -"use strict"; - import appInfo from "../../services/app_info.js"; /** diff --git a/src/routes/api/options.ts b/src/routes/api/options.ts index 1716d51268..140e2448e4 100644 --- a/src/routes/api/options.ts +++ b/src/routes/api/options.ts @@ -80,7 +80,9 @@ const ALLOWED_OPTIONS = new Set([ "allowedHtmlTags", "redirectBareDomain", "showLoginInShareTheme", - "splitEditorOrientation" + "splitEditorOrientation", + "mfaEnabled", + "mfaMethod" ]); function getOptions() { diff --git a/src/routes/api/recovery_codes.ts b/src/routes/api/recovery_codes.ts new file mode 100644 index 0000000000..a8487eab32 --- /dev/null +++ b/src/routes/api/recovery_codes.ts @@ -0,0 +1,65 @@ +import recovery_codes from '../../services/encryption/recovery_codes.js'; +import type { Request } from 'express'; +import { randomBytes } from 'crypto'; + +function setRecoveryCodes(req: Request) { + const success = recovery_codes.setRecoveryCodes(req.body.recoveryCodes.join(',')); + return { success: success, message: 'Recovery codes set!' }; +} + +function veryifyRecoveryCode(req: Request) { + const success = recovery_codes.verifyRecoveryCode(req.body.recovery_code_guess); + + return { success: success }; +} + +function checkForRecoveryKeys() { + return { + success: true, keysExist: recovery_codes.isRecoveryCodeSet() + }; +} + +function generateRecoveryCodes() { + const recoveryKeys = [ + randomBytes(16).toString('base64'), + randomBytes(16).toString('base64'), + randomBytes(16).toString('base64'), + randomBytes(16).toString('base64'), + randomBytes(16).toString('base64'), + randomBytes(16).toString('base64'), + randomBytes(16).toString('base64'), + randomBytes(16).toString('base64') + ]; + + recovery_codes.setRecoveryCodes(recoveryKeys.join(',')); + + return { success: true, recoveryCodes: recoveryKeys }; +} + +function getUsedRecoveryCodes() { + if (!recovery_codes.isRecoveryCodeSet()) { + return [] + } + + const dateRegex = RegExp(/^\d{4}\/\d{2}\/\d{2}T\d{2}:\d{2}:\d{2}.\d{3}Z$/gm); + const recoveryCodes = recovery_codes.getRecoveryCodes(); + const usedStatus: string[] = []; + + recoveryCodes.forEach((recoveryKey: string) => { + if (dateRegex.test(recoveryKey)) usedStatus.push(recoveryKey); + else usedStatus.push(recoveryCodes.indexOf(recoveryKey)); + }); + + return { + success: true, + usedRecoveryCodes: usedStatus + }; +} + +export default { + setRecoveryCodes, + generateRecoveryCodes, + veryifyRecoveryCode, + checkForRecoveryKeys, + getUsedRecoveryCodes +}; \ No newline at end of file diff --git a/src/routes/api/totp.ts b/src/routes/api/totp.ts new file mode 100644 index 0000000000..ece982960d --- /dev/null +++ b/src/routes/api/totp.ts @@ -0,0 +1,19 @@ +import totpService from '../../services/totp.js'; + +function generateTOTPSecret() { + return totpService.createSecret(); +} + +function getTOTPStatus() { + return { success: true, message: totpService.isTotpEnabled(), set: totpService.checkForTotpSecret() }; +} + +function getSecret() { + return totpService.getTotpSecret(); +} + +export default { + generateSecret: generateTOTPSecret, + getTOTPStatus, + getSecret +}; \ No newline at end of file diff --git a/src/routes/login.ts b/src/routes/login.ts index a739faa2e7..7b0ace3aa9 100644 --- a/src/routes/login.ts +++ b/src/routes/login.ts @@ -1,5 +1,3 @@ -"use strict"; - import utils from "../services/utils.js"; import optionService from "../services/options.js"; import myScryptService from "../services/encryption/my_scrypt.js"; @@ -8,13 +6,20 @@ import passwordService from "../services/encryption/password.js"; import assetPath from "../services/asset_path.js"; import appPath from "../services/app_path.js"; import ValidationError from "../errors/validation_error.js"; -import type { Request, Response } from "express"; +import type { Request, Response } from 'express'; +import totp from '../services/totp.js'; +import recoveryCodeService from '../services/encryption/recovery_codes.js'; +import openID from '../services/open_id.js'; +import openIDEncryption from '../services/encryption/open_id_encryption.js'; function loginPage(req: Request, res: Response) { - res.render("login", { - failedAuth: false, - assetPath, - appPath + res.render('login', { + wrongPassword: false, + wrongTotp: false, + totpEnabled: totp.isTotpEnabled(), + ssoEnabled: openID.isOpenIDEnabled(), + assetPath: assetPath, + appPath: appPath, }); } @@ -58,43 +63,95 @@ function setPassword(req: Request, res: Response) { } function login(req: Request, res: Response) { - const { password, rememberMe } = req.body; + if (openID.isOpenIDEnabled()) { + res.oidc.login({ + returnTo: '/', + authorizationParams: { + prompt: 'consent', + access_type: 'offline' + } + }); + return; + } - if (!verifyPassword(password)) { - // note that logged IP address is usually meaningless since the traffic should come from a reverse proxy - log.info(`WARNING: Wrong password from ${req.ip}, rejecting.`); + const submittedPassword = req.body.password; + const submittedTotpToken = req.body.totpToken; - return res.status(401).render("login", { - failedAuth: true, - assetPath, - appPath - }); + if (!verifyPassword(submittedPassword)) { + sendLoginError(req, res, 'password'); + return; } + if (totp.isTotpEnabled()) { + if (!verifyTOTP(submittedTotpToken)) { + sendLoginError(req, res, 'totp'); + return; + } + } + + const rememberMe = req.body.rememberMe; + req.session.regenerate(() => { - if (!rememberMe) { + if (rememberMe) { + req.session.cookie.maxAge = 21 * 24 * 3600000; // 3 weeks + } else { // unset default maxAge set by sessionParser // Cookie becomes non-persistent and expires after current browser session (e.g. when browser is closed) req.session.cookie.maxAge = undefined; } - req.session.loggedIn = true; + req.session.lastAuthState = { + totpEnabled: totp.isTotpEnabled(), + ssoEnabled: openID.isOpenIDEnabled() + }; - res.redirect("."); + req.session.loggedIn = true; + res.redirect('.'); }); } -function verifyPassword(guessedPassword: string) { +function verifyTOTP(submittedTotpToken: string) { + if (totp.validateTOTP(submittedTotpToken)) return true; + + const recoveryCodeValidates = recoveryCodeService.verifyRecoveryCode(submittedTotpToken); + + return recoveryCodeValidates; +} + +function verifyPassword(submittedPassword: string) { const hashed_password = utils.fromBase64(optionService.getOption("passwordVerificationHash")); - const guess_hashed = myScryptService.getVerificationHash(guessedPassword); + const guess_hashed = myScryptService.getVerificationHash(submittedPassword); return guess_hashed.equals(hashed_password); } +function sendLoginError(req: Request, res: Response, errorType: 'password' | 'totp' = 'password') { + // note that logged IP address is usually meaningless since the traffic should come from a reverse proxy + if (totp.isTotpEnabled()) { + log.info(`WARNING: Wrong ${errorType} from ${req.ip}, rejecting.`); + } else { + log.info(`WARNING: Wrong password from ${req.ip}, rejecting.`); + } + + res.render('login', { + wrongPassword: errorType === 'password', + wrongTotp: errorType === 'totp', + totpEnabled: totp.isTotpEnabled(), + ssoEnabled: openID.isOpenIDEnabled(), + assetPath: assetPath, + appPath: appPath, + }); +} + function logout(req: Request, res: Response) { req.session.regenerate(() => { req.session.loggedIn = false; + + if (openID.isOpenIDEnabled() && openIDEncryption.isSubjectIdentifierSaved()) { + res.oidc.logout({ returnTo: '/' }); + } else res.redirect('login'); + res.sendStatus(200); }); } diff --git a/src/routes/routes.ts b/src/routes/routes.ts index de2055d997..abae1acaa6 100644 --- a/src/routes/routes.ts +++ b/src/routes/routes.ts @@ -6,6 +6,9 @@ import log from "../services/log.js"; import express from "express"; const router = express.Router(); import auth from "../services/auth.js"; +import openID from '../services/open_id.js'; +import totp from './api/totp.js'; +import recoveryCodes from './api/recovery_codes.js'; import cls from "../services/cls.js"; import sql from "../services/sql.js"; import entityChangesService from "../services/entity_changes.js"; @@ -70,9 +73,9 @@ import etapiNoteRoutes from "../etapi/notes.js"; import etapiSpecialNoteRoutes from "../etapi/special_notes.js"; import etapiSpecRoute from "../etapi/spec.js"; import etapiBackupRoute from "../etapi/backup.js"; - import apiDocsRoute from "./api_docs.js"; + const MAX_ALLOWED_FILE_SIZE_MB = 250; const GET = "get", PST = "post", @@ -114,8 +117,22 @@ function register(app: express.Application) { route(PST, "/set-password", [auth.checkAppInitialized, auth.checkPasswordNotSet], loginRoute.setPassword); route(GET, "/setup", [], setupRoute.setupPage); - apiRoute(GET, "/api/tree", treeApiRoute.getTree); - apiRoute(PST, "/api/tree/load", treeApiRoute.load); + + apiRoute(GET, '/api/totp/generate', totp.generateSecret); + apiRoute(GET, '/api/totp/status', totp.getTOTPStatus); + apiRoute(GET, '/api/totp/get', totp.getSecret); + + apiRoute(GET, '/api/oauth/status', openID.getOAuthStatus); + apiRoute(GET, '/api/oauth/validate', openID.isTokenValid); + + apiRoute(PST, '/api/totp_recovery/set', recoveryCodes.setRecoveryCodes); + apiRoute(PST, '/api/totp_recovery/verify', recoveryCodes.veryifyRecoveryCode); + apiRoute(GET, '/api/totp_recovery/generate', recoveryCodes.generateRecoveryCodes); + apiRoute(GET, '/api/totp_recovery/enabled', recoveryCodes.checkForRecoveryKeys); + apiRoute(GET, '/api/totp_recovery/used', recoveryCodes.getUsedRecoveryCodes); + + apiRoute(GET, '/api/tree', treeApiRoute.getTree); + apiRoute(PST, '/api/tree/load', treeApiRoute.load); apiRoute(GET, "/api/notes/:noteId", notesApiRoute.getNote); apiRoute(GET, "/api/notes/:noteId/blob", notesApiRoute.getNoteBlob); @@ -492,7 +509,7 @@ function handleException(e: unknown | Error, method: HttpMethod, path: string, r log.error(`${method} ${path} threw exception: '${errMessage}', stack: ${errStack}`); - const resStatusCode = (e instanceof ValidationError || e instanceof NotFoundError) ? e.statusCode : 500; + const resStatusCode = (e instanceof ValidationError || e instanceof NotFoundError) ? e.statusCode : 500; res.status(resStatusCode).json({ message: errMessage diff --git a/src/routes/session_parser.ts b/src/routes/session_parser.ts index 89df0e037e..cc69cc6a22 100644 --- a/src/routes/session_parser.ts +++ b/src/routes/session_parser.ts @@ -3,6 +3,7 @@ import sessionFileStore from "session-file-store"; import sessionSecret from "../services/session_secret.js"; import dataDir from "../services/data_dir.js"; import config from "../services/config.js"; + const FileStore = sessionFileStore(session); const sessionParser = session({ diff --git a/src/services/app_info.ts b/src/services/app_info.ts index 72b1f0fcad..bd1436912b 100644 --- a/src/services/app_info.ts +++ b/src/services/app_info.ts @@ -1,11 +1,9 @@ -"use strict"; - import path from "path"; import build from "./build.js"; import packageJson from "../../package.json" with { type: "json" }; import dataDir from "./data_dir.js"; -const APP_DB_VERSION = 228; +const APP_DB_VERSION = 229; const SYNC_VERSION = 34; const CLIPPER_PROTOCOL_VERSION = "1.0"; diff --git a/src/services/auth.ts b/src/services/auth.ts index 03f40e6e7b..69bffa73be 100644 --- a/src/services/auth.ts +++ b/src/services/auth.ts @@ -1,5 +1,3 @@ -"use strict"; - import etapiTokenService from "./etapi_tokens.js"; import log from "./log.js"; import sqlInit from "./sql_init.js"; @@ -7,6 +5,8 @@ import { isElectron } from "./utils.js"; import passwordEncryptionService from "./encryption/password_encryption.js"; import config from "./config.js"; import passwordService from "./encryption/password.js"; +import totp from "./totp.js"; +import openID from "./open_id.js"; import options from "./options.js"; import attributes from "./attributes.js"; import type { NextFunction, Request, Response } from "express"; @@ -15,8 +15,30 @@ const noAuthentication = config.General && config.General.noAuthentication === t function checkAuth(req: Request, res: Response, next: NextFunction) { if (!sqlInit.isDbInitialized()) { - res.redirect("setup"); - } else if (!req.session.loggedIn && !isElectron && !noAuthentication) { + res.redirect('setup'); + } + + const currentTotpStatus = totp.isTotpEnabled(); + const currentSsoStatus = openID.isOpenIDEnabled(); + const lastAuthState = req.session.lastAuthState || { totpEnabled: false, ssoEnabled: false }; + + if (isElectron) { + next(); + return; + } else if (currentTotpStatus !== lastAuthState.totpEnabled || currentSsoStatus !== lastAuthState.ssoEnabled) { + req.session.destroy((err) => { + if (err) console.error('Error destroying session:', err); + res.redirect('/login'); + }); + return; + } else if (currentSsoStatus) { + if (req.oidc?.isAuthenticated() && req.session.loggedIn) { + next(); + return; + } + res.redirect('/login'); + return; + } else if (!req.session.loggedIn && !noAuthentication) { const redirectToShare = options.getOptionBool("redirectBareDomain"); if (redirectToShare) { // Check if any note has the #shareRoot label diff --git a/src/services/config.ts b/src/services/config.ts index f61eb19f5f..99704e0cf7 100644 --- a/src/services/config.ts +++ b/src/services/config.ts @@ -1,5 +1,3 @@ -"use strict"; - import ini from "ini"; import fs from "fs"; import dataDir from "./data_dir.js"; @@ -41,6 +39,11 @@ export interface TriliumConfig { syncServerTimeout: string; syncProxy: string; }; + MultiFactorAuthentication: { + oauthBaseUrl: string; + oauthClientId: string; + oauthClientSecret: string; + }; } //prettier-ignore @@ -50,13 +53,13 @@ const config: TriliumConfig = { instanceName: process.env.TRILIUM_GENERAL_INSTANCENAME || iniConfig.General.instanceName || "", - noAuthentication: + noAuthentication: envToBoolean(process.env.TRILIUM_GENERAL_NOAUTHENTICATION) || iniConfig.General.noAuthentication || false, - noBackup: + noBackup: envToBoolean(process.env.TRILIUM_GENERAL_NOBACKUP) || iniConfig.General.noBackup || false, - noDesktopIcon: + noDesktopIcon: envToBoolean(process.env.TRILIUM_GENERAL_NODESKTOPICON) || iniConfig.General.noDesktopIcon || false }, @@ -67,14 +70,14 @@ const config: TriliumConfig = { port: process.env.TRILIUM_NETWORK_PORT || iniConfig.Network.port || "3000", - https: + https: envToBoolean(process.env.TRILIUM_NETWORK_HTTPS) || iniConfig.Network.https || false, - certPath: + certPath: process.env.TRILIUM_NETWORK_CERTPATH || iniConfig.Network.certPath || "", - keyPath: - process.env.TRILIUM_NETWORK_KEYPATH || iniConfig.Network.keyPath || "", + keyPath: + process.env.TRILIUM_NETWORK_KEYPATH || iniConfig.Network.keyPath || "", trustedReverseProxy: process.env.TRILIUM_NETWORK_TRUSTEDREVERSEPROXY || iniConfig.Network.trustedReverseProxy || false @@ -98,8 +101,18 @@ const config: TriliumConfig = { syncProxy: // additionally checking in iniConfig for inconsistently named syncProxy for backwards compatibility process.env.TRILIUM_SYNC_SERVER_PROXY || iniConfig?.Sync?.syncProxy || iniConfig?.Sync?.syncServerProxy || "" - } + }, + MultiFactorAuthentication: { + oauthBaseUrl: + process.env.TRILIUM_OAUTH_BASE_URL || iniConfig?.MultiFactorAuthentication?.oauthBaseUrl || "", + + oauthClientId: + process.env.TRILIUM_OAUTH_CLIENT_ID || iniConfig?.MultiFactorAuthentication?.oauthClientId || "", + + oauthClientSecret: + process.env.TRILIUM_OAUTH_CLIENT_SECRET || iniConfig?.MultiFactorAuthentication?.oauthClientSecret || "" + } }; export default config; diff --git a/src/services/encryption/data_encryption.ts b/src/services/encryption/data_encryption.ts index 624a845ede..250db687b2 100644 --- a/src/services/encryption/data_encryption.ts +++ b/src/services/encryption/data_encryption.ts @@ -1,5 +1,3 @@ -"use strict"; - import crypto from "crypto"; import log from "../log.js"; diff --git a/src/services/encryption/my_scrypt.ts b/src/services/encryption/my_scrypt.ts index ec8adc18cd..d1bd9a5361 100644 --- a/src/services/encryption/my_scrypt.ts +++ b/src/services/encryption/my_scrypt.ts @@ -1,7 +1,6 @@ -"use strict"; - import optionService from "../options.js"; import crypto from "crypto"; +import sql from "../sql.js"; function getVerificationHash(password: crypto.BinaryLike) { const salt = optionService.getOption("passwordVerificationSalt"); @@ -21,7 +20,45 @@ function getScryptHash(password: crypto.BinaryLike, salt: crypto.BinaryLike) { return hashed; } +function getSubjectIdentifierVerificationHash( + guessedUserId: string | crypto.BinaryLike, + salt?: string +) { + if (salt != null) return getScryptHash(guessedUserId, salt); + + const savedSalt = sql.getValue("SELECT salt FROM user_data;"); + if (!savedSalt) { + console.error("User salt undefined!"); + return undefined; + } + return getScryptHash(guessedUserId, savedSalt.toString()); +} + +function getSubjectIdentifierDerivedKey( + subjectIdentifer: crypto.BinaryLike, + givenSalt?: string +) { + if (givenSalt !== undefined) { + return getScryptHash(subjectIdentifer, givenSalt.toString()); + } + + const salt = sql.getValue("SELECT salt FROM user_data;"); + if (!salt) return undefined; + + return getScryptHash(subjectIdentifer, salt.toString()); +} + +function createSubjectIdentifierDerivedKey( + subjectIdentifer: string | crypto.BinaryLike, + salt: string | crypto.BinaryLike +) { + return getScryptHash(subjectIdentifer, salt); +} + export default { getVerificationHash, - getPasswordDerivedKey + getPasswordDerivedKey, + getSubjectIdentifierVerificationHash, + getSubjectIdentifierDerivedKey, + createSubjectIdentifierDerivedKey }; diff --git a/src/services/encryption/open_id_encryption.ts b/src/services/encryption/open_id_encryption.ts new file mode 100644 index 0000000000..5dad9c06ba --- /dev/null +++ b/src/services/encryption/open_id_encryption.ts @@ -0,0 +1,145 @@ +import myScryptService from "./my_scrypt.js"; +import utils from "../utils.js"; +import dataEncryptionService from "./data_encryption.js"; +import sql from "../sql.js"; +import sqlInit from "../sql_init.js"; +import OpenIdError from "../../errors/open_id_error.js"; + +function saveUser(subjectIdentifier: string, name: string, email: string) { + if (isUserSaved()) return false; + + const verificationSalt = utils.randomSecureToken(32); + const derivedKeySalt = utils.randomSecureToken(32); + + const verificationHash = myScryptService.getSubjectIdentifierVerificationHash( + subjectIdentifier, + verificationSalt + ); + if (!verificationHash) { + throw new OpenIdError("Verification hash undefined!") + } + + const userIDEncryptedDataKey = setDataKey( + subjectIdentifier, + utils.randomSecureToken(16), + verificationSalt + ); + + if (!userIDEncryptedDataKey) { + console.error("UserID encrypted data key null"); + return undefined; + } + + const data = { + tmpID: 0, + userIDVerificationHash: utils.toBase64(verificationHash), + salt: verificationSalt, + derivedKey: derivedKeySalt, + userIDEncryptedDataKey: userIDEncryptedDataKey, + isSetup: "true", + username: name, + email: email + }; + + sql.upsert("user_data", "tmpID", data); + return true; +} + +function isSubjectIdentifierSaved() { + const value = sql.getValue("SELECT userIDEncryptedDataKey FROM user_data;"); + if (value === undefined || value === null || value === "") return false; + return true; +} + +function isUserSaved() { + const isSaved = sql.getValue("SELECT isSetup FROM user_data;"); + return isSaved === "true" ? true : false; +} + +function verifyOpenIDSubjectIdentifier(subjectIdentifier: string) { + if (!sqlInit.isDbInitialized()) { + throw new OpenIdError("Database not initialized!"); + } + + if (isUserSaved()) { + return false; + } + + const salt = sql.getValue("SELECT salt FROM user_data;"); + if (salt == undefined) { + console.log("Salt undefined"); + return undefined; + } + + const givenHash = myScryptService + .getSubjectIdentifierVerificationHash(subjectIdentifier) + ?.toString("base64"); + if (givenHash === undefined) { + console.log("Sub id hash undefined!"); + return undefined; + } + + const savedHash = sql.getValue( + "SELECT userIDVerificationHash FROM user_data" + ); + if (savedHash === undefined) { + console.log("verification hash undefined"); + return undefined; + } + + console.log("Matches: " + givenHash === savedHash); + return givenHash === savedHash; +} + +function setDataKey( + subjectIdentifier: string, + plainTextDataKey: string | Buffer, + salt: string +) { + const subjectIdentifierDerivedKey = + myScryptService.getSubjectIdentifierDerivedKey(subjectIdentifier, salt); + + if (subjectIdentifierDerivedKey === undefined) { + console.error("SOMETHING WENT WRONG SAVING USER ID DERIVED KEY"); + return undefined; + } + const newEncryptedDataKey = dataEncryptionService.encrypt( + subjectIdentifierDerivedKey, + plainTextDataKey + ); + + return newEncryptedDataKey; +} + +function getDataKey(subjectIdentifier: string) { + const subjectIdentifierDerivedKey = + myScryptService.getSubjectIdentifierDerivedKey(subjectIdentifier); + + const encryptedDataKey = sql.getValue( + "SELECT userIDEncryptedDataKey FROM user_data" + ); + + if (!encryptedDataKey) { + console.error("Encrypted data key empty!"); + return undefined; + } + + if (!subjectIdentifierDerivedKey) { + console.error("SOMETHING WENT WRONG SAVING USER ID DERIVED KEY"); + return undefined; + } + const decryptedDataKey = dataEncryptionService.decrypt( + subjectIdentifierDerivedKey, + encryptedDataKey.toString() + ); + + return decryptedDataKey; +} + +export default { + verifyOpenIDSubjectIdentifier, + getDataKey, + setDataKey, + saveUser, + isSubjectIdentifierSaved, +}; diff --git a/src/services/encryption/password.ts b/src/services/encryption/password.ts index bbfe9577b0..62046f47ac 100644 --- a/src/services/encryption/password.ts +++ b/src/services/encryption/password.ts @@ -1,5 +1,3 @@ -"use strict"; - import sql from "../sql.js"; import optionService from "../options.js"; import myScryptService from "./my_scrypt.js"; diff --git a/src/services/encryption/recovery_codes.ts b/src/services/encryption/recovery_codes.ts new file mode 100644 index 0000000000..72c0012a83 --- /dev/null +++ b/src/services/encryption/recovery_codes.ts @@ -0,0 +1,73 @@ +import sql from '../sql.js'; +import optionService from '../options.js'; +import crypto from 'crypto'; + +function isRecoveryCodeSet() { + return optionService.getOptionBool('encryptedRecoveryCodes'); +} + +function setRecoveryCodes(recoveryCodes: string) { + const iv = crypto.randomBytes(16); + const securityKey = crypto.randomBytes(32); + const cipher = crypto.createCipheriv('aes-256-cbc', securityKey, iv); + let encryptedRecoveryCodes = cipher.update(recoveryCodes, 'utf-8', 'hex'); + + sql.transactional(() => { + optionService.setOption('recoveryCodeInitialVector', iv.toString('hex')); + optionService.setOption('recoveryCodeSecurityKey', securityKey.toString('hex')); + optionService.setOption('recoveryCodesEncrypted', encryptedRecoveryCodes + cipher.final('hex')); + optionService.setOption('encryptedRecoveryCodes', 'true'); + return true; + }); + return false; +} + +function getRecoveryCodes() { + if (!isRecoveryCodeSet()) { + return [] + } + + return sql.transactional(() => { + const iv = Buffer.from(optionService.getOption('recoveryCodeInitialVector'), 'hex'); + const securityKey = Buffer.from(optionService.getOption('recoveryCodeSecurityKey'), 'hex'); + const encryptedRecoveryCodes = optionService.getOption('recoveryCodesEncrypted'); + + const decipher = crypto.createDecipheriv('aes-256-cbc', securityKey, iv); + const decryptedData = decipher.update(encryptedRecoveryCodes, 'hex', 'utf-8'); + + const decryptedString = decryptedData + decipher.final('utf-8'); + return decryptedString.split(','); + }); +} + +function removeRecoveryCode(usedCode: string) { + const oldCodes: string[] = getRecoveryCodes(); + const today = new Date(); + oldCodes[oldCodes.indexOf(usedCode)] = today.toJSON().replace(/-/g, '/'); + setRecoveryCodes(oldCodes.toString()); +} + +function verifyRecoveryCode(recoveryCodeGuess: string) { + const recoveryCodeRegex = RegExp(/^.{22}==$/gm); + if (!recoveryCodeRegex.test(recoveryCodeGuess)) { + return false; + } + + const recoveryCodes = getRecoveryCodes(); + var loginSuccess = false; + recoveryCodes.forEach((recoveryCode: string) => { + if (recoveryCodeGuess === recoveryCode) { + removeRecoveryCode(recoveryCode); + loginSuccess = true; + return; + } + }); + return loginSuccess; +} + +export default { + setRecoveryCodes, + getRecoveryCodes, + verifyRecoveryCode, + isRecoveryCodeSet +}; \ No newline at end of file diff --git a/src/services/encryption/totp_encryption.ts b/src/services/encryption/totp_encryption.ts new file mode 100644 index 0000000000..bf079cc9d2 --- /dev/null +++ b/src/services/encryption/totp_encryption.ts @@ -0,0 +1,83 @@ +import optionService from "../options.js"; +import myScryptService from "./my_scrypt.js"; +import { randomSecureToken, toBase64 } from "../utils.js"; +import dataEncryptionService from "./data_encryption.js"; +import type { OptionNames } from "../options_interface.js"; + +const TOTP_OPTIONS: Record = { + SALT: "totpEncryptionSalt", + ENCRYPTED_SECRET: "totpEncryptedSecret", + VERIFICATION_HASH: "totpVerificationHash" +}; + +function verifyTotpSecret(secret: string): boolean { + const givenSecretHash = toBase64(myScryptService.getVerificationHash(secret)); + const dbSecretHash = optionService.getOptionOrNull(TOTP_OPTIONS.VERIFICATION_HASH); + + if (!dbSecretHash) { + return false; + } + + return givenSecretHash === dbSecretHash; +} + +function setTotpSecret(secret: string) { + if (!secret) { + throw new Error("TOTP secret cannot be empty"); + } + + const encryptionSalt = randomSecureToken(32); + optionService.setOption(TOTP_OPTIONS.SALT, encryptionSalt); + + const verificationHash = toBase64(myScryptService.getVerificationHash(secret)); + optionService.setOption(TOTP_OPTIONS.VERIFICATION_HASH, verificationHash); + + const encryptedSecret = dataEncryptionService.encrypt( + Buffer.from(encryptionSalt), + secret + ); + optionService.setOption(TOTP_OPTIONS.ENCRYPTED_SECRET, encryptedSecret); +} + +function getTotpSecret(): string | null { + const encryptionSalt = optionService.getOptionOrNull(TOTP_OPTIONS.SALT); + const encryptedSecret = optionService.getOptionOrNull(TOTP_OPTIONS.ENCRYPTED_SECRET); + + if (!encryptionSalt || !encryptedSecret) { + return null; + } + + try { + const decryptedSecret = dataEncryptionService.decrypt( + Buffer.from(encryptionSalt), + encryptedSecret + ); + + if (!decryptedSecret) { + return null; + } + + return decryptedSecret.toString(); + } catch (e) { + console.error("Failed to decrypt TOTP secret:", e); + return null; + } +} + +function resetTotpSecret() { + optionService.setOption(TOTP_OPTIONS.SALT, ""); + optionService.setOption(TOTP_OPTIONS.ENCRYPTED_SECRET, ""); + optionService.setOption(TOTP_OPTIONS.VERIFICATION_HASH, ""); +} + +function isTotpSecretSet(): boolean { + return !!optionService.getOptionOrNull(TOTP_OPTIONS.VERIFICATION_HASH); +} + +export default { + verifyTotpSecret, + setTotpSecret, + getTotpSecret, + resetTotpSecret, + isTotpSecretSet +}; diff --git a/src/services/hidden_subtree.ts b/src/services/hidden_subtree.ts index 79c80f29e0..369b23aeae 100644 --- a/src/services/hidden_subtree.ts +++ b/src/services/hidden_subtree.ts @@ -271,6 +271,7 @@ function buildHiddenSubtreeDefinition(helpSubtree: HiddenSubtreeItem[]): HiddenS { id: "_optionsImages", title: t("hidden-subtree.images-title"), type: "contentWidget", icon: "bx-image" }, { id: "_optionsSpellcheck", title: t("hidden-subtree.spellcheck-title"), type: "contentWidget", icon: "bx-check-double" }, { id: "_optionsPassword", title: t("hidden-subtree.password-title"), type: "contentWidget", icon: "bx-lock" }, + { id: '_optionsMFA', title: t('hidden-subtree.multi-factor-authentication-title'), type: 'contentWidget', icon: 'bx-lock ' }, { id: "_optionsEtapi", title: t("hidden-subtree.etapi-title"), type: "contentWidget", icon: "bx-extension" }, { id: "_optionsBackup", title: t("hidden-subtree.backup-title"), type: "contentWidget", icon: "bx-data" }, { id: "_optionsSync", title: t("hidden-subtree.sync-title"), type: "contentWidget", icon: "bx-wifi" }, diff --git a/src/services/open_id.ts b/src/services/open_id.ts new file mode 100644 index 0000000000..e45ed65997 --- /dev/null +++ b/src/services/open_id.ts @@ -0,0 +1,154 @@ +import type { NextFunction, Request, Response } from "express"; +import openIDEncryption from "./encryption/open_id_encryption.js"; +import sqlInit from "./sql_init.js"; +import options from "./options.js"; +import type { Session } from "express-openid-connect"; +import sql from "./sql.js"; +import config from "./config.js"; + + +function checkOpenIDConfig() { + let missingVars: string[] = [] + if (config.MultiFactorAuthentication.oauthBaseUrl === "") { + missingVars.push("oauthBaseUrl"); + } + if (config.MultiFactorAuthentication.oauthClientId === "") { + missingVars.push("oauthClientId"); + } + if (config.MultiFactorAuthentication.oauthClientSecret === "") { + missingVars.push("oauthClientSecret"); + } + return missingVars; +} + +function isOpenIDEnabled() { + return !(checkOpenIDConfig().length > 0) && options.getOptionOrNull('mfaMethod') === 'oauth'; +} + +function isUserSaved() { + const data = sql.getValue("SELECT isSetup FROM user_data;"); + return data === "true" ? true : false; +} + +function getUsername() { + const username = sql.getValue("SELECT username FROM user_data;"); + return username; +} + +function getUserEmail() { + const email = sql.getValue("SELECT email FROM user_data;"); + return email; +} + +function clearSavedUser() { + sql.execute("DELETE FROM user_data"); + options.setOption("userSubjectIdentifierSaved", false); + return { + success: true, + message: "Account data removed." + }; +} + +function getOAuthStatus() { + return { + success: true, + name: getUsername(), + email: getUserEmail(), + enabled: isOpenIDEnabled(), + missingVars: checkOpenIDConfig() + }; +} + +function isTokenValid(req: Request, res: Response, next: NextFunction) { + const userStatus = openIDEncryption.isSubjectIdentifierSaved(); + + if (req.oidc !== undefined) { + const result = req.oidc + .fetchUserInfo() + .then((result) => { + return { + success: true, + message: "Token is valid", + user: userStatus, + }; + }) + .catch((result) => { + return { + success: false, + message: "Token is not valid", + user: userStatus, + }; + }); + return result; + } else { + return { + success: false, + message: "Token not set up", + user: userStatus, + }; + } +} + +function generateOAuthConfig() { + const authRoutes = { + callback: "/callback", + login: "/authenticate", + postLogoutRedirect: "/login", + logout: "/logout", + }; + + const logoutParams = { + }; + + const authConfig = { + authRequired: false, + auth0Logout: false, + baseURL: config.MultiFactorAuthentication.oauthBaseUrl, + clientID: config.MultiFactorAuthentication.oauthClientId, + issuerBaseURL: "https://accounts.google.com", + secret: config.MultiFactorAuthentication.oauthClientSecret, + clientSecret: config.MultiFactorAuthentication.oauthClientSecret, + authorizationParams: { + response_type: "code", + scope: "openid profile email", + access_type: "offline", + prompt: "consent", + state: "random_state_" + Math.random().toString(36).substring(2) + }, + routes: authRoutes, + idpLogout: true, + logoutParams: logoutParams, + afterCallback: async (req: Request, res: Response, session: Session) => { + if (!sqlInit.isDbInitialized()) return session; + + if (!req.oidc.user) { + console.log("user invalid!"); + return session; + } + + openIDEncryption.saveUser( + req.oidc.user.sub.toString(), + req.oidc.user.name.toString(), + req.oidc.user.email.toString() + ); + + req.session.loggedIn = true; + req.session.lastAuthState = { + totpEnabled: false, + ssoEnabled: true + }; + + return session; + }, + }; + return authConfig; +} + +export default { + generateOAuthConfig, + getOAuthStatus, + isOpenIDEnabled, + clearSavedUser, + isTokenValid, + isUserSaved, +}; diff --git a/src/services/options_init.ts b/src/services/options_init.ts index 491edc65f9..14967a6d67 100644 --- a/src/services/options_init.ts +++ b/src/services/options_init.ts @@ -131,6 +131,10 @@ const defaultOptions: DefaultOption[] = [ { name: "customSearchEngineUrl", value: "https://duckduckgo.com/?q={keyword}", isSynced: true }, { name: "promotedAttributesOpenInRibbon", value: "true", isSynced: true }, { name: "editedNotesOpenInRibbon", value: "true", isSynced: true }, + { name: "mfaEnabled", value: "false", isSynced: false }, + { name: "mfaMethod", value: "totp", isSynced: false }, + { name: "encryptedRecoveryCodes", value: "false", isSynced: false }, + { name: "userSubjectIdentifierSaved", value: "false", isSynced: false }, // Appearance { name: "splitEditorOrientation", value: "horizontal", isSynced: true }, diff --git a/src/services/options_interface.ts b/src/services/options_interface.ts index 26c4f1b40f..cdd9184fb6 100644 --- a/src/services/options_interface.ts +++ b/src/services/options_interface.ts @@ -48,6 +48,18 @@ export interface OptionDefinitions extends KeyboardShortcutsOptions