diff --git a/.env b/.env
index 3e76ccab6..123e4bd74 100644
--- a/.env
+++ b/.env
@@ -11,11 +11,13 @@ APP_NAME=Acme Co.
APP_HOSTNAME=localhost
APP_ORIGIN=http://localhost:5173
API_ORIGIN=https://api-mcfytwakla-uc.a.run.app
+APP_STORAGE_BUCKET=example.com
# Google Cloud
# https://console.cloud.google.com/
GOOGLE_CLOUD_PROJECT=kriasoft
GOOGLE_CLOUD_REGION=us-central1
+GOOGLE_CLOUD_DATABASE="(default)"
GOOGLE_CLOUD_CREDENTIALS={"type":"service_account","project_id":"example","private_key_id":"xxx","private_key":"-----BEGIN PRIVATE KEY-----\nxxxxx\n-----END PRIVATE KEY-----\n","client_email":"application@exmaple.iam.gserviceaccount.com","client_id":"xxxxx","auth_uri":"https://accounts.google.com/o/oauth2/auth","token_uri":"https://oauth2.googleapis.com/token","auth_provider_x509_cert_url":"https://www.googleapis.com/oauth2/v1/certs","client_x509_cert_url":"https://www.googleapis.com/robot/v1/metadata/x509/application%40example.iam.gserviceaccount.com"}
# Firebase
@@ -24,6 +26,11 @@ FIREBASE_APP_ID=1:736557952746:web:b5ee23841e24c0b883b193
FIREBASE_API_KEY=AIzaSyAZDmdeRWvlYgZpwm6LBCkYJM6ySIMF2Hw
FIREBASE_AUTH_DOMAIN=kriasoft.web.app
+# OpenAI
+# https://platform.openai.com/
+OPENAI_ORGANIZATION=xxxxx
+OPENAI_API_KEY=xxxxx
+
# Cloudflare
# https://dash.cloudflare.com/
# https://developers.cloudflare.com/api/tokens/create
diff --git a/.vscode/settings.json b/.vscode/settings.json
index 5162a0e85..d2ac34e86 100644
--- a/.vscode/settings.json
+++ b/.vscode/settings.json
@@ -61,6 +61,7 @@
"firestore",
"globby",
"hono",
+ "identitytoolkit",
"jamstack",
"kriasoft",
"localforage",
@@ -68,16 +69,20 @@
"nodenext",
"notistack",
"oidc",
+ "openai",
"pathinfo",
+ "pino",
"pnpify",
"reactstarter",
"refetch",
"refetchable",
+ "relyingparty",
"sendgrid",
"signup",
"sourcemap",
"spdx",
"swapi",
+ "trpc",
"tslib",
"typechecking",
"vite",
diff --git a/README.md b/README.md
index 0e53aa73a..ae0224eb3 100644
--- a/README.md
+++ b/README.md
@@ -31,9 +31,11 @@ Be sure to join our [Discord channel](https://discord.com/invite/2nKEnKq) for as
`├──`[`.github`](.github) — GitHub configuration including CI/CD workflows
`├──`[`.vscode`](.vscode) — VSCode settings including code snippets, recommended extensions etc.
`├──`[`app`](./app) — Web application front-end built with [React](https://react.dev/) and [Joy UI](https://mui.com/joy-ui/getting-started/)
+`├──`[`db`](./db) — Firestore database schema, seed data, and admin tools
`├──`[`edge`](./edge) — Cloudflare Workers (CDN) edge endpoint
`├──`[`env`](./env) — Application settings, API keys, etc.
`├──`[`scripts`](./scripts) — Automation scripts such as `yarn deploy`
+`├──`[`server`](./server) — Node.js application server built with [tRPC](https://trpc.io/)
`├──`[`tsconfig.base.json`](./tsconfig.base.json) — The common/shared TypeScript configuration
`└──`[`tsconfig.json`](./tsconfig.json) — The root TypeScript configuration
diff --git a/app/package.json b/app/package.json
index 524f4bcb9..50fde8331 100644
--- a/app/package.json
+++ b/app/package.json
@@ -36,7 +36,7 @@
"devDependencies": {
"@babel/core": "^7.23.7",
"@emotion/babel-plugin": "^11.11.0",
- "@types/node": "^20.10.6",
+ "@types/node": "^20.10.7",
"@types/react": "^18.2.47",
"@types/react-dom": "^18.2.18",
"@vitejs/plugin-react": "^4.2.1",
diff --git a/db/README.md b/db/README.md
new file mode 100644
index 000000000..12d758756
--- /dev/null
+++ b/db/README.md
@@ -0,0 +1,20 @@
+# Firestore Database
+
+Database schema, security rules, indexes, and seed data for the [Firestore](https://cloud.google.com/firestore) database.
+
+## Directory Structure
+
+- [`/models`](./models/) — Database schema definitions using [Zod](https://zod.dev/).
+- [`/seeds`](./seeds/) — Sample / reference data for the database.
+- [`/scripts`](./scripts/) — Scripts for managing the database.
+- [`/firestore.indexes.json`](./firestore.indexes.json) — Firestore indexes.
+- [`/firestore.rules`](./firestore.rules) — Firestore security rules.
+
+## Scripts
+
+- `yarn workspace db seed` - Seed the database with data from [`/seeds`](./seeds/).
+
+## References
+
+- https://zod.dev/
+- https://cloud.google.com/firestore
diff --git a/db/firestore.indexes.json b/db/firestore.indexes.json
new file mode 100644
index 000000000..19272370d
--- /dev/null
+++ b/db/firestore.indexes.json
@@ -0,0 +1,13 @@
+{
+ "indexes": [
+ {
+ "collectionGroup": "workspace",
+ "queryScope": "COLLECTION",
+ "fields": [
+ { "fieldPath": "ownerId", "order": "ASCENDING" },
+ { "fieldPath": "archived", "order": "DESCENDING" }
+ ]
+ }
+ ],
+ "fieldOverrides": []
+}
diff --git a/db/firestore.rules b/db/firestore.rules
new file mode 100644
index 000000000..23b616177
--- /dev/null
+++ b/db/firestore.rules
@@ -0,0 +1,19 @@
+// Firestore security rules.
+// https://cloud.google.com/firestore/docs/security/get-started
+
+rules_version = '2';
+
+service cloud.firestore {
+ match /databases/{database}/documents {
+ match /workspace/{id} {
+ allow read: if request.auth != null && (
+ resource.data.ownerId = request.auth.uid ||
+ request.auth.token.admin == true
+ );
+ }
+
+ match /{document=**} {
+ allow read, write: if false;
+ }
+ }
+}
diff --git a/db/index.ts b/db/index.ts
new file mode 100644
index 000000000..a4c50fa4b
--- /dev/null
+++ b/db/index.ts
@@ -0,0 +1,6 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+export * from "./models";
+export { testUsers } from "./seeds/01-users";
+export { testWorkspaces } from "./seeds/02-workspaces";
diff --git a/db/models/index.ts b/db/models/index.ts
new file mode 100644
index 000000000..0690c018f
--- /dev/null
+++ b/db/models/index.ts
@@ -0,0 +1,4 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+export * from "./workspace";
diff --git a/db/models/workspace.ts b/db/models/workspace.ts
new file mode 100644
index 000000000..7794839ed
--- /dev/null
+++ b/db/models/workspace.ts
@@ -0,0 +1,16 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { Timestamp } from "@google-cloud/firestore";
+import { z } from "zod";
+
+export const WorkspaceSchema = z.object({
+ name: z.string().max(100),
+ ownerId: z.string().max(50),
+ created: z.instanceof(Timestamp),
+ updated: z.instanceof(Timestamp),
+ archived: z.instanceof(Timestamp).nullable(),
+});
+
+export type Workspace = z.output;
+export type WorkspaceInput = z.input;
diff --git a/db/package.json b/db/package.json
new file mode 100644
index 000000000..c8c41afa7
--- /dev/null
+++ b/db/package.json
@@ -0,0 +1,30 @@
+{
+ "name": "db",
+ "version": "0.0.0",
+ "private": true,
+ "type": "module",
+ "exports": {
+ ".": {
+ "default": "./index.ts"
+ },
+ "./package.json": "./package.json"
+ },
+ "scripts": {
+ "seed": "vite-node ./scripts/seed.ts",
+ "test": "vitest"
+ },
+ "dependencies": {
+ "@google-cloud/firestore": "^7.1.0",
+ "@googleapis/identitytoolkit": "^8.0.0",
+ "zod": "^3.22.4"
+ },
+ "devDependencies": {
+ "@types/node": "^20.10.7",
+ "dotenv": "^16.3.1",
+ "ora": "^8.0.1",
+ "typescript": "~5.3.3",
+ "vite": "~5.0.11",
+ "vite-node": "~1.1.3",
+ "vitest": "~1.1.3"
+ }
+}
diff --git a/db/scripts/seed.ts b/db/scripts/seed.ts
new file mode 100644
index 000000000..5fa0eafda
--- /dev/null
+++ b/db/scripts/seed.ts
@@ -0,0 +1,43 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { Firestore } from "@google-cloud/firestore";
+import { configDotenv } from "dotenv";
+import { relative, resolve } from "node:path";
+import { oraPromise } from "ora";
+
+const rootDir = resolve(__dirname, "../..");
+
+// Load environment variables from .env files.
+configDotenv({ path: resolve(rootDir, ".env.local") });
+configDotenv({ path: resolve(rootDir, ".env") });
+
+let db: Firestore | null = null;
+
+// Seed the database with test / sample data.
+try {
+ db = new Firestore({
+ projectId: process.env.GOOGLE_CLOUD_PROJECT,
+ databaseId: process.env.GOOGLE_CLOUD_DATABASE,
+ });
+
+ // Import all seed modules from the `/seeds` folder.
+ const files = import.meta.glob("../seeds/*.ts");
+
+ // Sequentially seed the database with data from each module.
+ for (const [path, load] of Object.entries(files)) {
+ const message = `Seeding ${relative("../seeds", path)}`;
+ const action = (async () => {
+ const { seed } = await load();
+ await seed(db);
+ })();
+
+ await oraPromise(action, message);
+ }
+} finally {
+ await db?.terminate();
+}
+
+type SeedModule = {
+ seed: (db: Firestore) => Promise;
+};
diff --git a/db/seeds/01-users.ts b/db/seeds/01-users.ts
new file mode 100644
index 000000000..09f23fb58
--- /dev/null
+++ b/db/seeds/01-users.ts
@@ -0,0 +1,87 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import {
+ AuthPlus,
+ identitytoolkit,
+ identitytoolkit_v3,
+} from "@googleapis/identitytoolkit";
+
+/**
+ * Test user accounts generated by https://randomuser.me/.
+ */
+export const testUsers: identitytoolkit_v3.Schema$UserInfo[] = [
+ {
+ localId: "test-erika",
+ screenName: "erika",
+ email: "erika.pearson@example.com",
+ emailVerified: true,
+ phoneNumber: "+14788078434",
+ displayName: "Erika Pearson",
+ photoUrl: "https://randomuser.me/api/portraits/women/29.jpg",
+ rawPassword: "paloma",
+ createdAt: new Date("2024-01-01T12:00:00Z").getTime().toString(),
+ lastLoginAt: new Date("2024-01-01T12:00:00Z").getTime().toString(),
+ },
+ {
+ localId: "test-ryan",
+ screenName: "ryan",
+ email: "ryan.hunt@example.com",
+ emailVerified: true,
+ phoneNumber: "+16814758216",
+ displayName: "Ryan Hunt",
+ photoUrl: "https://randomuser.me/api/portraits/men/20.jpg",
+ rawPassword: "baggins",
+ createdAt: new Date("2024-01-02T12:00:00Z").getTime().toString(),
+ lastLoginAt: new Date("2024-01-02T12:00:00Z").getTime().toString(),
+ },
+ {
+ localId: "test-marian",
+ screenName: "marian",
+ email: "marian.stone@example.com",
+ emailVerified: true,
+ phoneNumber: "+19243007975",
+ displayName: "Marian Stone",
+ photoUrl: "https://randomuser.me/api/portraits/women/2.jpg",
+ rawPassword: "winter1",
+ createdAt: new Date("2024-01-03T12:00:00Z").getTime().toString(),
+ lastLoginAt: new Date("2024-01-03T12:00:00Z").getTime().toString(),
+ },
+ {
+ localId: "test-kurt",
+ screenName: "kurt",
+ email: "kurt.howward@example.com",
+ emailVerified: true,
+ phoneNumber: "+19243007975",
+ displayName: "Kurt Howard",
+ photoUrl: "https://randomuser.me/api/portraits/men/23.jpg",
+ rawPassword: "mayday",
+ createdAt: new Date("2024-01-04T12:00:00Z").getTime().toString(),
+ lastLoginAt: new Date("2024-01-04T12:00:00Z").getTime().toString(),
+ },
+ {
+ localId: "test-dan",
+ screenName: "dan",
+ email: "dan.day@example.com",
+ emailVerified: true,
+ phoneNumber: "+12046748092",
+ displayName: "Dan Day",
+ photoUrl: "https://randomuser.me/api/portraits/men/65.jpg",
+ rawPassword: "teresa",
+ createdAt: new Date("2024-01-05T12:00:00Z").getTime().toString(),
+ lastLoginAt: new Date("2024-01-05T12:00:00Z").getTime().toString(),
+ customAttributes: JSON.stringify({ admin: true }),
+ },
+];
+
+/**
+ * Seeds the Google Identity Platform (Firebase Auth) with test user accounts.
+ *
+ * @see https://randomuser.me/
+ * @see https://cloud.google.com/identity-platform
+ */
+export async function seed() {
+ const auth = new AuthPlus();
+ const { relyingparty } = identitytoolkit({ version: "v3", auth });
+ await relyingparty.uploadAccount({ requestBody: { users: testUsers } });
+}
diff --git a/db/seeds/02-workspaces.ts b/db/seeds/02-workspaces.ts
new file mode 100644
index 000000000..1455ec2d7
--- /dev/null
+++ b/db/seeds/02-workspaces.ts
@@ -0,0 +1,63 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { Firestore, Timestamp } from "@google-cloud/firestore";
+import { WorkspaceInput } from "../models";
+import { testUsers as users } from "./01-users";
+
+/**
+ * Test workspaces.
+ */
+export const testWorkspaces: (WorkspaceInput & { id: string })[] = [
+ {
+ id: "DwYchGFGpk",
+ ownerId: users[0].localId!,
+ name: "Personal workspace",
+ created: Timestamp.fromDate(new Date(+users[0].createdAt!)),
+ updated: Timestamp.fromDate(new Date(+users[0].createdAt!)),
+ archived: null,
+ },
+ {
+ id: "YfYKTcO9q9",
+ ownerId: users[1].localId!,
+ name: "Personal workspace",
+ created: Timestamp.fromDate(new Date(+users[1].createdAt!)),
+ updated: Timestamp.fromDate(new Date(+users[1].createdAt!)),
+ archived: null,
+ },
+ {
+ id: "c2OsmUvFMY",
+ ownerId: users[2].localId!,
+ name: "Personal workspace",
+ created: Timestamp.fromDate(new Date(+users[2].createdAt!)),
+ updated: Timestamp.fromDate(new Date(+users[2].createdAt!)),
+ archived: null,
+ },
+ {
+ id: "uTqcGw4qn7",
+ ownerId: users[3].localId!,
+ name: "Personal workspace",
+ created: Timestamp.fromDate(new Date(+users[3].createdAt!)),
+ updated: Timestamp.fromDate(new Date(+users[3].createdAt!)),
+ archived: null,
+ },
+ {
+ id: "vBHHgg5ydn",
+ ownerId: users[4].localId!,
+ name: "Personal workspace",
+ created: Timestamp.fromDate(new Date(+users[4].createdAt!)),
+ updated: Timestamp.fromDate(new Date(+users[4].createdAt!)),
+ archived: null,
+ },
+];
+
+export async function seed(db: Firestore) {
+ const batch = db.batch();
+
+ for (const { id, ...workspace } of testWorkspaces) {
+ const ref = db.doc(`workspace/${id}`);
+ batch.set(ref, workspace, { merge: true });
+ }
+
+ await batch.commit();
+}
diff --git a/db/tsconfig.json b/db/tsconfig.json
new file mode 100644
index 000000000..22d7921eb
--- /dev/null
+++ b/db/tsconfig.json
@@ -0,0 +1,14 @@
+{
+ "extends": "../tsconfig.base.json",
+ "compilerOptions": {
+ "composite": true,
+ "emitDeclarationOnly": true,
+ "module": "ESNext",
+ "moduleResolution": "Bundler",
+ "noEmit": false,
+ "outDir": "../.cache/ts-db",
+ "types": ["node", "vite/client"]
+ },
+ "include": ["**/*.ts", "**/*.json"],
+ "exclude": ["**/dist/**/*", "**/node_modules/**/*"]
+}
diff --git a/edge/package.json b/edge/package.json
index 2bb7a65cf..5af88ab69 100644
--- a/edge/package.json
+++ b/edge/package.json
@@ -26,7 +26,7 @@
},
"devDependencies": {
"@cloudflare/workers-types": "^4.20231218.0",
- "@types/node": "^20.10.6",
+ "@types/node": "^20.10.7",
"happy-dom": "^12.10.3",
"toml": "^3.0.0",
"typescript": "~5.3.3",
diff --git a/package.json b/package.json
index c9191ffb9..70e0fb324 100644
--- a/package.json
+++ b/package.json
@@ -1,16 +1,18 @@
{
"name": "root",
"version": "0.0.0",
- "private": true,
"packageManager": "yarn@4.0.2",
+ "private": true,
"type": "module",
"workspaces": [
"app",
+ "db",
"edge",
- "scripts"
+ "scripts",
+ "server"
],
"scripts": {
- "postinstall": "husky install && node ./scripts/postinstall.js",
+ "postinstall": "husky install && node ./scripts/post-install.js",
"start": "yarn workspace app start",
"lint": "eslint --cache --report-unused-disable-directives .",
"test": "vitest",
@@ -21,8 +23,8 @@
"@emotion/babel-plugin": "^11.11.0",
"@emotion/eslint-plugin": "^11.11.0",
"@types/eslint": "^8.56.1",
- "@typescript-eslint/eslint-plugin": "^6.17.0",
- "@typescript-eslint/parser": "^6.17.0",
+ "@typescript-eslint/eslint-plugin": "^6.18.0",
+ "@typescript-eslint/parser": "^6.18.0",
"eslint": "^8.56.0",
"eslint-config-prettier": "^9.1.0",
"eslint-import-resolver-typescript": "^3.6.1",
diff --git a/scripts/bundle-yarn.js b/scripts/bundle-yarn.js
new file mode 100644
index 000000000..81f85e741
--- /dev/null
+++ b/scripts/bundle-yarn.js
@@ -0,0 +1,33 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { execa } from "execa";
+import { copyFile, readFile, writeFile } from "node:fs/promises";
+import { resolve } from "node:path";
+import { fileURLToPath } from "node:url";
+
+const rootDir = resolve(fileURLToPath(import.meta.url), "../..");
+const rootPkg = JSON.parse(await readFile(`${rootDir}/package.json`, "utf-8"));
+const pkg = JSON.parse(await readFile("./package.json", "utf-8"));
+
+pkg.packageManager = rootPkg.packageManager;
+delete pkg.scripts;
+delete pkg.devDependencies;
+delete pkg.dependencies.db;
+
+// Create ./dist/package.json
+await writeFile("./dist/package.json", JSON.stringify(pkg, null, 2), "utf-8");
+
+// Create ./dist/yarn.lock
+await copyFile(`${rootDir}/yarn.lock`, "./dist/yarn.lock");
+
+// Install production dependencies
+await execa("yarn", ["install", "--mode=update-lockfile"], {
+ env: {
+ ...process.env,
+ NODE_OPTIONS: undefined,
+ YARN_ENABLE_IMMUTABLE_INSTALLS: "false",
+ },
+ cwd: "./dist",
+ stdio: "inherit",
+});
diff --git a/scripts/package.json b/scripts/package.json
index a479911a0..95f50a766 100644
--- a/scripts/package.json
+++ b/scripts/package.json
@@ -7,7 +7,7 @@
"dotenv": "^16.3.1",
"execa": "^8.0.1",
"get-port": "^7.0.0",
- "got": "^14.0.0",
+ "got": "^13.0.0",
"graphql": "^16.8.1",
"lodash-es": "^4.17.21",
"miniflare": "^3.20231218.1",
diff --git a/scripts/post-install.js b/scripts/post-install.js
new file mode 100644
index 000000000..447844f92
--- /dev/null
+++ b/scripts/post-install.js
@@ -0,0 +1,31 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { execa } from "execa";
+import fs from "node:fs";
+import { EOL } from "node:os";
+
+// Create Git-ignored files for environment variable overrides
+if (!fs.existsSync("./.env.local")) {
+ await fs.writeFile(
+ "./.env.local",
+ [
+ `# Overrides for the ``.env`` file in the root folder.`,
+ "#",
+ "# CLOUDFLARE_API_TOKEN=xxxxx",
+ "# GOOGLE_CLOUD_CREDENTIALS=xxxxx",
+ "# SENDGRID_API_KEY=xxxxx",
+ "#",
+ "",
+ "API_URL=http://localhost:8080",
+ "",
+ ].join(EOL),
+ "utf-8",
+ );
+}
+
+try {
+ await execa("yarn", ["tsc", "--build"], { stdin: "inherit" });
+} catch (err) {
+ console.error(err);
+}
diff --git a/scripts/postinstall.js b/scripts/postinstall.js
deleted file mode 100644
index 51cd15cb4..000000000
--- a/scripts/postinstall.js
+++ /dev/null
@@ -1,39 +0,0 @@
-/* SPDX-FileCopyrightText: 2014-present Kriasoft */
-/* SPDX-License-Identifier: MIT */
-
-import { execa } from "execa";
-import fs from "node:fs";
-import { EOL } from "node:os";
-
-const environments = [
- { name: "local", description: "development" },
- { name: "test", description: "staging/QA" },
- { name: "prod", description: "production" },
-];
-
-// Create Git-ignored files for environment variable overrides
-for (const env of environments) {
- const filename = `./env/.${env.name}.override.env`;
-
- if (!fs.existsSync(filename)) {
- await fs.writeFile(
- filename,
- [
- `# Overrides for the "${env.name}" (${env.description}) environment`,
- "#",
- "# CLOUDFLARE_API_TOKEN=xxxxx",
- "# GOOGLE_CLOUD_CREDENTIALS=xxxxx",
- "# SENDGRID_API_KEY=xxxxx",
- "#",
- "",
- ].join(EOL),
- "utf-8",
- );
- }
-}
-
-try {
- await execa("yarn", ["tsc", "--build"], { stdin: "inherit" });
-} catch (err) {
- console.error(err);
-}
diff --git a/server/Dockerfile b/server/Dockerfile
new file mode 100644
index 000000000..66b13e51a
--- /dev/null
+++ b/server/Dockerfile
@@ -0,0 +1,34 @@
+# SPDX-FileCopyrightText: 2014-present Kriasoft
+# SPDX-License-Identifier: MIT
+
+# Docker image for a Cloud Run service.
+#
+# https://cloud.google.com/run/docs/container-contract
+# https://cloud.google.com/run/docs/quickstarts/build-and-deploy/nodejs
+# https://github.com/GoogleCloudPlatform/cloud-run-microservice-template-nodejs/blob/main/Dockerfile
+
+# Use the official lightweight Node.js image.
+# https://hub.docker.com/_/node
+FROM node:20.10.0-slim
+
+# Upgrade OS packages.
+RUN apt-get update && apt-get upgrade -y
+
+# Set environment variables.
+ENV NODE_ENV=production
+
+# Create and change to the app directory.
+WORKDIR /usr/src/app
+
+# Copy application dependency manifests to the container image.
+# Copying this separately prevents re-running npm install on every code change.
+COPY dist/package.json dist/yarn.lock ./
+
+# Install dependencies.
+RUN corepack enable && yarn config set nodeLinker node-modules && yarn install --immutable
+
+# Copy compiled code to the container image.
+COPY ./dist .
+
+# Run the web service on container startup.
+CMD [ "node", "index.js" ]
diff --git a/server/README.md b/server/README.md
new file mode 100644
index 000000000..6fa39a911
--- /dev/null
+++ b/server/README.md
@@ -0,0 +1,9 @@
+# Application Server
+
+Node.js application server for web and mobile clients using [tRPC](https://trpc.io/) with HTTP and WebSocket transports.
+
+## Scripts
+
+- `yarn workspace server start` — Start the server in development mode.
+- `yarn workspace server build` — Build the server for production.
+- `yarn workspace server test` — Run tests.
diff --git a/server/app.test.ts b/server/app.test.ts
new file mode 100644
index 000000000..b8c7854ac
--- /dev/null
+++ b/server/app.test.ts
@@ -0,0 +1,30 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import supertest from "supertest";
+import { describe, expect, it } from "vitest";
+import { app } from "./app";
+
+describe("GET /trpc", () => {
+ it("returns NOT_FOUND response for unknown requests", async () => {
+ const res = await supertest(app).get("/trpc");
+
+ expect({
+ statusCode: res.statusCode,
+ body: res.body,
+ }).toEqual({
+ statusCode: 404,
+ body: {
+ error: {
+ code: -32004,
+ data: {
+ code: "NOT_FOUND",
+ httpStatus: 404,
+ path: "",
+ },
+ message: 'No "query"-procedure on path ""',
+ },
+ },
+ });
+ });
+});
diff --git a/server/app.ts b/server/app.ts
new file mode 100644
index 000000000..ddf6d6f90
--- /dev/null
+++ b/server/app.ts
@@ -0,0 +1,54 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { createExpressMiddleware } from "@trpc/server/adapters/express";
+import express from "express";
+import { createProxyMiddleware } from "http-proxy-middleware";
+import { sessionMiddleware } from "./core/auth";
+import { env } from "./core/env";
+import { loggerMiddleware } from "./core/logging";
+import { createContext } from "./core/trpc";
+import { router } from "./routes/index";
+
+export const app = express();
+
+app.disable("x-powered-by");
+app.set("trust proxy", 1); // Trust first proxy
+
+app.use(loggerMiddleware);
+app.use(sessionMiddleware);
+
+/**
+ * tRPC HTTP handler for Express.js.
+ *
+ * @see https://trpc.io/docs/getting-started
+ * @see https://trpc.io/docs/server/adapters/express
+ */
+app.use("/trpc", createExpressMiddleware({ router, createContext }));
+
+/**
+ * Proxy auth requests to Google Identity Platform.
+ *
+ * @see https://firebase.google.com/docs/auth/web/redirect-best-practices
+ */
+app.use(
+ createProxyMiddleware("/__", {
+ target: `https://${env.GOOGLE_CLOUD_PROJECT}.firebaseapp.com`,
+ changeOrigin: true,
+ logLevel: "warn",
+ }),
+);
+
+/**
+ * Proxy static assets to Google Cloud Storage.
+ */
+app.use(
+ createProxyMiddleware("/", {
+ target: "https://c.storage.googleapis.com",
+ changeOrigin: true,
+ logLevel: "warn",
+ onProxyReq(proxyReq) {
+ proxyReq.setHeader("host", env.APP_STORAGE_BUCKET);
+ },
+ }),
+);
diff --git a/server/core/auth.test.ts b/server/core/auth.test.ts
new file mode 100644
index 000000000..248d4f757
--- /dev/null
+++ b/server/core/auth.test.ts
@@ -0,0 +1,52 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import express from "express";
+import supertest from "supertest";
+import { describe, expect, it } from "vitest";
+import { fetchCertificates, sessionMiddleware } from "./auth";
+import { env } from "./env";
+
+describe("fetchCertificates()", () => {
+ it("should fetch certificates from Google", async () => {
+ const certs = await fetchCertificates();
+ expect(certs).toEqual(expect.any(Object));
+ expect(Object.values(certs)).toEqual(
+ expect.arrayContaining([
+ expect.stringMatching(/^-----BEGIN CERTIFICATE-----/),
+ ]),
+ );
+ });
+});
+
+describe("sessionMiddleware()", () => {
+ it.skip("should verify a valid ID token", async () => {
+ const app = express();
+ app.get("/", sessionMiddleware, (req, res) => {
+ res.type("json");
+ res.send(JSON.stringify(req.token));
+ });
+
+ const idToken = "eyJhbGciOiJSUzI1NiIsImtpZCI6ImQxNjg5NDE1ZWMyM2EzMzdlMmJiYWE1ZTNlNjhiNjZkYzk5MzY4ODQiLCJ0eXAiOiJKV1QifQ.eyJuYW1lIjoiS3JpYXNvZnQiLCJwaWN0dXJlIjoiaHR0cHM6Ly9saDMuZ29vZ2xldXNlcmNvbnRlbnQuY29tL2EvQUNnOG9jS0hBRVhvRXpJMnpoMWNpN3lna0FVZXFiRWppQ1ZxUE96VkZQYnZmUGtXcGc9czk2LWMiLCJpc3MiOiJodHRwczovL3NlY3VyZXRva2VuLmdvb2dsZS5jb20va3JpYXNvZnQiLCJhdWQiOiJrcmlhc29mdCIsImF1dGhfdGltZSI6MTcwNDU0MjYyOCwidXNlcl9pZCI6IkpRdWxWbjhaMUhZelRIR0J5YnhlcndGRUhsNzIiLCJzdWIiOiJKUXVsVm44WjFIWXpUSEdCeWJ4ZXJ3RkVIbDcyIiwiaWF0IjoxNzA0NTQyNjI4LCJleHAiOjE3MDQ1NDYyMjgsImVtYWlsIjoia3JpYXNvZnRAZ21haWwuY29tIiwiZW1haWxfdmVyaWZpZWQiOnRydWUsImZpcmViYXNlIjp7ImlkZW50aXRpZXMiOnsiZ29vZ2xlLmNvbSI6WyIxMDI2NTQwODA3NzUxNTYyOTI2ODYiXSwiZW1haWwiOlsia3JpYXNvZnRAZ21haWwuY29tIl19LCJzaWduX2luX3Byb3ZpZGVyIjoiZ29vZ2xlLmNvbSJ9fQ.E1sOoIeLH0269m4K4DXfXOJk97cxc8h3D62u3q9Kqyk0AsmwQfKURRl34IiOtNEzizjesLex6EHSetFr8kS1GSgVrW6yxHowmJZCY8tsgSWfunZ7Vj8l1kG4y0iM7hdFw3t0dhilK8-vDlKpLeRfLVHG8qgt46qI7Rxmdb928llJoa7H6NuuS5heavNJadLfiYItJyUq7i5kjys6-WfndQQcRb7kTt07arHb_1w2jtZnyjZE_S3ErZcIgwnE9M_gqXZ4y1MucpGPR2_nHzicRBBYOMwZDVG7Y0tI9IWRaTyTF3Psd7XKisE6GorZ_X1cDwkaT5ffoXZ1tkBOjeMjfw"; // prettier-ignore
+ const res = await supertest(app).get("/").auth(idToken, { type: "bearer" });
+
+ expect({ status: res.status, body: res.body }).toEqual({
+ status: 200,
+ body: expect.objectContaining({
+ name: expect.any(String),
+ picture: expect.any(String),
+ iss: `https://securetoken.google.com/${env.GOOGLE_CLOUD_PROJECT}`,
+ aud: env.GOOGLE_CLOUD_PROJECT,
+ auth_time: expect.any(Number),
+ sub: expect.any(String),
+ iat: expect.any(Number),
+ exp: expect.any(Number),
+ email: expect.any(String),
+ email_verified: true,
+ firebase: expect.objectContaining({
+ sign_in_provider: "google.com",
+ }),
+ }),
+ });
+ });
+});
diff --git a/server/core/auth.ts b/server/core/auth.ts
new file mode 100644
index 000000000..9affeaea2
--- /dev/null
+++ b/server/core/auth.ts
@@ -0,0 +1,188 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { NextFunction, Request, Response } from "express";
+import {
+ Certificates,
+ GoogleAuth,
+ IdTokenClient,
+ TokenPayload,
+} from "google-auth-library";
+import { got } from "got";
+import { env } from "./env";
+
+export const auth = new GoogleAuth({
+ scopes: ["https://www.googleapis.com/auth/cloud-platform"],
+});
+
+const certificatesURL = "https://www.googleapis.com/service_accounts/v1/metadata/x509/securetoken@system.gserviceaccount.com"; // prettier-ignore
+const certificatesCache = new Map();
+
+/**
+ * Fetches the latest Google Cloud Identity Platform certificates.
+ */
+export function fetchCertificates(options?: { signal: AbortSignal }) {
+ return got.get(certificatesURL, {
+ cache: certificatesCache,
+ resolveBodyOnly: true,
+ responseType: "json",
+ signal: options?.signal,
+ });
+}
+
+// Refresh certificates every 6 hours.
+const cleanup = (() => {
+ const ac = new AbortController();
+ const int = setInterval(() => fetchCertificates(), 2.16e7);
+ fetchCertificates({ signal: ac.signal });
+ return () => {
+ clearInterval(int);
+ ac.abort();
+ };
+})();
+
+process.on("SIGTERM", cleanup);
+process.on("SIGINT", cleanup);
+
+const idTokenClients = new Map();
+
+/**
+ * Express middleware that verifies that the request has a valid Firebase ID
+ * token attached, and adds the decoded token to `req.token`.
+ */
+export async function sessionMiddleware(
+ req: Request,
+ res: Response,
+ next: NextFunction,
+) {
+ try {
+ req.token = null;
+ const idToken = req.headers.authorization?.replace(/^Bearer /i, "");
+
+ if (idToken) {
+ const certificatesPromise = fetchCertificates();
+ const audience = env.GOOGLE_CLOUD_PROJECT;
+ let idTokenClient = idTokenClients.get(audience);
+
+ if (!idTokenClient) {
+ idTokenClient = await auth.getIdTokenClient(audience);
+ idTokenClients.set(audience, idTokenClient);
+ }
+
+ const ticket = await idTokenClient.verifySignedJwtWithCertsAsync(
+ idToken,
+ await certificatesPromise,
+ audience,
+ [`https://securetoken.google.com/${env.GOOGLE_CLOUD_PROJECT}`],
+ );
+
+ const token = ticket.getPayload();
+
+ if (token) {
+ if ("user_id" in token) delete token.user_id;
+ Object.assign(token, { uid: token.sub });
+ req.token = token as DecodedIdToken;
+ }
+ }
+
+ next();
+ } catch (err) {
+ req.log?.warn(err);
+ next();
+ }
+}
+
+// #region Types
+
+/**
+ * Interface representing a decoded Firebase ID token, returned from the
+ * {@link verifyIdToken} method.
+ *
+ * Firebase ID tokens are OpenID Connect spec-compliant JSON Web Tokens (JWTs).
+ * See the
+ * [ID Token section of the OpenID Connect spec](http://openid.net/specs/openid-connect-core-1_0.html#IDToken)
+ * for more information about the specific properties below.
+ */
+export interface DecodedIdToken extends TokenPayload {
+ /**
+ * Time, in seconds since the Unix epoch, when the end-user authentication
+ * occurred.
+ *
+ * This value is not set when this particular ID token was created, but when the
+ * user initially logged in to this session. In a single session, the Firebase
+ * SDKs will refresh a user's ID tokens every hour. Each ID token will have a
+ * different [`iat`](#iat) value, but the same `auth_time` value.
+ */
+ auth_time: number;
+
+ /**
+ * Information about the sign in event, including which sign in provider was
+ * used and provider-specific identity details.
+ *
+ * This data is provided by the Firebase Authentication service and is a
+ * reserved claim in the ID token.
+ */
+ firebase: {
+ /**
+ * Provider-specific identity details corresponding
+ * to the provider used to sign in the user.
+ */
+ identities: {
+ /* eslint-disable-next-line @typescript-eslint/no-explicit-any */
+ [key: string]: any;
+ };
+
+ /**
+ * The ID of the provider used to sign in the user.
+ * One of `"anonymous"`, `"password"`, `"facebook.com"`, `"github.com"`,
+ * `"google.com"`, `"twitter.com"`, `"apple.com"`, `"microsoft.com"`,
+ * `"yahoo.com"`, `"phone"`, `"playgames.google.com"`, `"gc.apple.com"`,
+ * or `"custom"`.
+ *
+ * Additional Identity Platform provider IDs include `"linkedin.com"`,
+ * OIDC and SAML identity providers prefixed with `"saml."` and `"oidc."`
+ * respectively.
+ */
+ sign_in_provider: string;
+
+ /**
+ * The type identifier or `factorId` of the second factor, provided the
+ * ID token was obtained from a multi-factor authenticated user.
+ * For phone, this is `"phone"`.
+ */
+ sign_in_second_factor?: string;
+
+ /**
+ * The `uid` of the second factor used to sign in, provided the
+ * ID token was obtained from a multi-factor authenticated user.
+ */
+ second_factor_identifier?: string;
+
+ /**
+ * The ID of the tenant the user belongs to, if available.
+ */
+ tenant?: string;
+ /* eslint-disable-next-line @typescript-eslint/no-explicit-any */
+ [key: string]: any;
+ };
+
+ /**
+ * The phone number of the user to whom the ID token belongs, if available.
+ */
+ phone_number?: string;
+
+ /**
+ * The `uid` corresponding to the user who the ID token belonged to.
+ *
+ * This value is not actually in the JWT token claims itself. It is added as a
+ * convenience, and is set as the value of the [`sub`](#sub) property.
+ */
+ uid: string;
+
+ /**
+ * Indicates whether or not the user is an admin.
+ */
+ admin?: boolean;
+}
+
+// #endregion
diff --git a/server/core/env.ts b/server/core/env.ts
new file mode 100644
index 000000000..6feccc153
--- /dev/null
+++ b/server/core/env.ts
@@ -0,0 +1,21 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { cleanEnv, str } from "envalid";
+
+/**
+ * Environment variables that has been validated and sanitized.
+ *
+ * @see https://github.com/ilyakaznacheev/cleanenv#readme
+ */
+export const env = cleanEnv(process.env, {
+ VERSION: str({ default: "latest" }),
+
+ APP_STORAGE_BUCKET: str(),
+
+ GOOGLE_CLOUD_PROJECT: str(),
+ GOOGLE_CLOUD_DATABASE: str(),
+
+ OPENAI_ORGANIZATION: str(),
+ OPENAI_API_KEY: str(),
+});
diff --git a/server/core/firestore.ts b/server/core/firestore.ts
new file mode 100644
index 000000000..5c1965721
--- /dev/null
+++ b/server/core/firestore.ts
@@ -0,0 +1,18 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { Firestore } from "@google-cloud/firestore";
+import { env } from "./env";
+
+let db: Firestore | undefined;
+
+export function getFirestore() {
+ if (!db) {
+ db = new Firestore({
+ projectId: env.GOOGLE_CLOUD_PROJECT,
+ databaseId: env.GOOGLE_CLOUD_DATABASE,
+ });
+ }
+
+ return db;
+}
diff --git a/server/core/logging.ts b/server/core/logging.ts
new file mode 100644
index 000000000..b8ca332db
--- /dev/null
+++ b/server/core/logging.ts
@@ -0,0 +1,60 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { Request, Response } from "express";
+import { pino } from "pino";
+import { pinoHttp } from "pino-http";
+import { env } from "./env";
+
+/**
+ * Low overhead Node.js logger.
+ *
+ * @see https://github.com/pinojs/pino
+ */
+export const logger = pino({
+ // Custom formatter to set the "severity" property in the JSON payload
+ // to the log level to be automatically parsed.
+ // https://cloud.google.com/run/docs/logging#special-fields
+ formatters: {
+ level(label) {
+ return { severity: label };
+ },
+ },
+ transport: {
+ // Enable pretty printing in development.
+ // https://github.com/pinojs/pino-pretty#readme
+ target: env.isProduction ? "pino/file" : "pino-pretty",
+ options: {
+ ...(!env.isProduction && { colorize: true }),
+ ignore: env.isProduction
+ ? "pid,hostname"
+ : "pid,hostname,req.headers,req.remoteAddress,req.remotePort,res.headers",
+ },
+ },
+});
+
+/**
+ * Creates a request-based logger with trace ID field for logging correlation.
+ *
+ * @see https://cloud.google.com/run/docs/logging#correlate-logs
+ */
+export const loggerMiddleware = pinoHttp({
+ logger,
+ customProps(req) {
+ const traceHeader = req.header("X-Cloud-Trace-Context");
+
+ let trace;
+
+ if (traceHeader) {
+ const [traceId] = traceHeader.split("/");
+ trace = `projects/${env.GOOGLE_CLOUD_PROJECT}/traces/${traceId}`;
+ }
+
+ return {
+ "logging.googleapis.com/trace": trace,
+ };
+ },
+ redact: {
+ paths: ["req.headers.authorization", "req.headers.cookie"],
+ },
+});
diff --git a/server/core/openai.ts b/server/core/openai.ts
new file mode 100644
index 000000000..c3fe75560
--- /dev/null
+++ b/server/core/openai.ts
@@ -0,0 +1,15 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { OpenAI } from "openai";
+import { env } from "./env";
+
+/**
+ * OpenAI API client.
+ *
+ * @see https://github.com/openai/openai-node#readme
+ */
+export const openai = new OpenAI({
+ organization: env.OPENAI_ORGANIZATION,
+ apiKey: env.OPENAI_API_KEY,
+});
diff --git a/server/core/trpc.ts b/server/core/trpc.ts
new file mode 100644
index 000000000..31c67c604
--- /dev/null
+++ b/server/core/trpc.ts
@@ -0,0 +1,102 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { Firestore } from "@google-cloud/firestore";
+import { TRPCError, initTRPC } from "@trpc/server";
+import { CreateExpressContextOptions } from "@trpc/server/adapters/express";
+import { Logger } from "pino";
+import { SetNonNullable } from "type-fest";
+import { ZodError } from "zod";
+import { DecodedIdToken } from "./auth";
+import { env } from "./env";
+import { getFirestore } from "./firestore";
+import { logger } from "./logging";
+
+/**
+ * tRPC instance.
+ *
+ * @see https://trpc.io/docs/quickstart
+ */
+export const t = initTRPC.context().create({
+ isDev: env.isDev,
+
+ // https://trpc.io/docs/server/error-formatting
+ errorFormatter(opts) {
+ const { shape, error } = opts;
+ return {
+ ...shape,
+ data: {
+ ...shape.data,
+ ...(error.code === "BAD_REQUEST" &&
+ error.cause instanceof ZodError && {
+ zodError: error.cause.flatten(),
+ }),
+ },
+ };
+ },
+});
+
+/**
+ * Creates a tRPC context for an incoming HTTP request.
+ */
+export async function createContext(
+ ctx: CreateExpressContextOptions,
+): Promise {
+ return new HttpContext(getFirestore(), ctx.req.log, ctx.req.token);
+}
+
+/**
+ * Creates a tRPC context for an incoming WebSocket request.
+ */
+export async function createWsContext(): Promise {
+ return new WsContext(getFirestore(), logger);
+}
+
+class HttpContext {
+ constructor(
+ readonly db: Firestore,
+ readonly log: Logger,
+ readonly token: DecodedIdToken | null,
+ ) {}
+}
+
+class WsContext {
+ constructor(
+ readonly db: Firestore,
+ readonly log: Logger,
+ ) {}
+
+ get token(): DecodedIdToken | null {
+ throw new Error("ID token is not available in WebSocket context.");
+ }
+}
+
+/**
+ * Ensures that the user is authenticated.
+ */
+export const authorize = t.middleware((opts) => {
+ if (!opts.ctx.token) {
+ throw new TRPCError({ code: "UNAUTHORIZED" });
+ }
+
+ return opts.next({
+ ...opts,
+ ctx: opts.ctx as SetNonNullable,
+ });
+});
+
+/**
+ * Ensures that the user is an admin.
+ */
+export const authorizeAdmin = authorize.unstable_pipe((opts) => {
+ if (!opts.ctx.token.admin) {
+ throw new TRPCError({ code: "FORBIDDEN" });
+ }
+
+ return opts.next(opts);
+});
+
+/**
+ * tRPC context.
+ */
+export type Context = HttpContext;
diff --git a/server/core/utils.ts b/server/core/utils.ts
new file mode 100644
index 000000000..ce7480e6b
--- /dev/null
+++ b/server/core/utils.ts
@@ -0,0 +1,13 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { customAlphabet, urlAlphabet } from "nanoid";
+
+const shortIdAlphabet = urlAlphabet.replace(/[-_]/g, "");
+
+/**
+ * Generates secure unique ID using URL friendly characters.
+ *
+ * @see https://github.com/ai/nanoid#readme
+ */
+export const newId = customAlphabet(shortIdAlphabet, 10);
diff --git a/server/global.d.ts b/server/global.d.ts
new file mode 100644
index 000000000..4d3c00163
--- /dev/null
+++ b/server/global.d.ts
@@ -0,0 +1,13 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import "express";
+import { DecodedIdToken } from "./core/auth";
+
+declare global {
+ namespace Express {
+ interface Request {
+ token: DecodedIdToken | null;
+ }
+ }
+}
diff --git a/server/index.ts b/server/index.ts
new file mode 100644
index 000000000..6000ae8f9
--- /dev/null
+++ b/server/index.ts
@@ -0,0 +1,65 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { applyWSSHandler } from "@trpc/server/adapters/ws";
+import { WebSocketServer } from "ws";
+import { app } from "./app";
+import { logger } from "./core/logging";
+import { createWsContext as createContext } from "./core/trpc";
+import { router } from "./routes";
+
+// Detect if running in Google Cloud environment
+const isCloudRun = !!process.env.K_SERVICE;
+
+/**
+ * Starts the HTTP and WebSocket servers.
+ */
+export function listen(port: number) {
+ const server = app.listen(port, () => {
+ logger.info(`API listening on ${port}`);
+ });
+
+ const wss = new WebSocketServer({ server, path: "/trpc" });
+ const handler = applyWSSHandler({ wss, router, createContext });
+
+ wss.on("connection", (ws) => {
+ logger.info({ clients: wss.clients.size }, "wss:connection");
+ ws.once("close", () => {
+ logger.info({ clients: wss.clients.size }, "wss:close");
+ });
+ });
+
+ return function dispose(cb?: () => void) {
+ handler.broadcastReconnectNotification();
+ wss.close((err) => {
+ if (err) logger.error(err);
+ if (isCloudRun) logger.info("WebSocket server closed");
+ server.close((err) => {
+ if (err) logger.error(err);
+ if (isCloudRun) logger.info("HTTP server closed");
+ logger.flush((err) => {
+ if (err) console.error(err);
+ if (isCloudRun) {
+ process.exit(0);
+ } else {
+ cb?.();
+ }
+ });
+ });
+ });
+ };
+}
+
+if (process.env.PORT && process.env.K_SERVICE?.startsWith("server")) {
+ const port = parseInt(process.env.PORT);
+ const dispose = listen(port);
+
+ /* eslint-disable-next-line no-inner-declarations */
+ function handleClose(code: NodeJS.Signals) {
+ logger.info(`${code} signal received`);
+ dispose();
+ }
+
+ process.on("SIGINT", handleClose);
+ process.on("SIGTERM", handleClose);
+}
diff --git a/server/package.json b/server/package.json
new file mode 100644
index 000000000..92bdcaa12
--- /dev/null
+++ b/server/package.json
@@ -0,0 +1,52 @@
+{
+ "name": "server",
+ "version": "0.0.0",
+ "private": true,
+ "type": "module",
+ "exports": {
+ ".": {
+ "types": "./types.ts"
+ },
+ "./package.json": "./package.json"
+ },
+ "scripts": {
+ "start": "vite-node --watch ./start.ts",
+ "build": "vite build && yarn node ../scripts/bundle-yarn.js",
+ "test": "vitest"
+ },
+ "dependencies": {
+ "@google-cloud/firestore": "^7.1.0",
+ "@googleapis/identitytoolkit": "^8.0.0",
+ "@trpc/server": "^10.45.0",
+ "db": "workspace:*",
+ "envalid": "^8.0.0",
+ "express": "^4.18.2",
+ "google-auth-library": "^9.4.1",
+ "got": "^13.0.0",
+ "http-errors": "^2.0.0",
+ "http-proxy-middleware": "^2.0.6",
+ "nanoid": "^5.0.4",
+ "openai": "^4.24.1",
+ "pino": "^8.17.2",
+ "pino-http": "^9.0.0",
+ "pino-pretty": "^10.3.1",
+ "type-fest": "^4.9.0",
+ "ws": "^8.16.0",
+ "zod": "^3.22.4"
+ },
+ "devDependencies": {
+ "@types/express": "^4.17.21",
+ "@types/http-errors": "^2.0.4",
+ "@types/node": "^20.10.7",
+ "@types/supertest": "^6.0.2",
+ "@types/ws": "^8.5.10",
+ "envars": "^1.0.2",
+ "get-port": "^7.0.0",
+ "supertest": "^6.3.3",
+ "type-fest": "^4.9.0",
+ "typescript": "~5.3.3",
+ "vite": "~5.0.11",
+ "vite-node": "~1.1.3",
+ "vitest": "~1.1.3"
+ }
+}
diff --git a/server/routes/index.ts b/server/routes/index.ts
new file mode 100644
index 000000000..93a7b37e6
--- /dev/null
+++ b/server/routes/index.ts
@@ -0,0 +1,15 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { t } from "../core/trpc";
+import { workspace } from "./workspace";
+
+/**
+ * The root tRPC router.
+ * @see https://trpc.io/docs/quickstart
+ */
+export const router = t.router({
+ workspace,
+});
+
+export type AppRouter = typeof router;
diff --git a/server/routes/workspace.test.ts b/server/routes/workspace.test.ts
new file mode 100644
index 000000000..6eea0ea0e
--- /dev/null
+++ b/server/routes/workspace.test.ts
@@ -0,0 +1,63 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { TRPCError } from "@trpc/server";
+import { testWorkspaces } from "db";
+import { describe, expect, it } from "vitest";
+import { createClient } from "../test/context";
+import * as router from "./workspace";
+
+describe("workspace.update()", () => {
+ it("requires authentication", async () => {
+ const [client] = createClient(router.workspace);
+ const action = client.update({
+ id: "xxxxxxxxxx",
+ name: "My Workspace",
+ });
+
+ await expect(action).rejects.toThrowError(
+ new TRPCError({ code: "UNAUTHORIZED" }),
+ );
+ });
+
+ it("workspace must exist", async () => {
+ const [client] = createClient(router.workspace, { user: "erika" });
+ const action = client.update({
+ id: "xxxxxxxxxx", // Non-existent workspace ID
+ name: "My Workspace",
+ });
+
+ await expect(action).rejects.toThrowError(
+ new TRPCError({ code: "NOT_FOUND" }),
+ );
+ });
+
+ it("workspace must belong to the user", async () => {
+ const [client, ctx] = createClient(router.workspace, { user: "erika" });
+ const id = testWorkspaces.find((x) => x.ownerId !== ctx.token?.uid)?.id;
+ expect(id).toEqual(expect.any(String));
+
+ const action = client.update({
+ id: id!, // Workspace ID that belongs to another user
+ name: "My Workspace",
+ });
+
+ await expect(action).rejects.toThrowError(
+ new TRPCError({ code: "NOT_FOUND" }),
+ );
+ });
+
+ it("updates the workspace", async () => {
+ const [client, ctx] = createClient(router.workspace, { user: "erika" });
+ const id = testWorkspaces.find((x) => x.ownerId === ctx.token?.uid)?.id;
+ expect(id).toEqual(expect.any(String));
+
+ await client.update({ id: id!, name: "My Workspace" });
+ const doc1 = await ctx.db.doc(`workspace/${id}`).get();
+ expect(doc1.data()?.name).toBe("My Workspace");
+
+ await client.update({ id: id!, name: "Personal workspace" });
+ const doc2 = await ctx.db.doc(`workspace/${id}`).get();
+ expect(doc2.data()?.name).toBe("Personal workspace");
+ });
+});
diff --git a/server/routes/workspace.ts b/server/routes/workspace.ts
new file mode 100644
index 000000000..c98eb4835
--- /dev/null
+++ b/server/routes/workspace.ts
@@ -0,0 +1,39 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { Timestamp } from "@google-cloud/firestore";
+import { TRPCError } from "@trpc/server";
+import { z } from "zod";
+import { authorize, t } from "../core/trpc";
+
+/**
+ * Workspace API.
+ */
+export const workspace = t.router({
+ /**
+ * Updates the workspace.
+ */
+ update: t.procedure
+ .use(authorize)
+ .input(
+ z.object({
+ id: z.string(),
+ name: z.string().max(100),
+ }),
+ )
+ .query(async ({ input, ctx }) => {
+ const { db } = ctx;
+ const doc = await db.doc(`workspace/${input.id}`).get();
+
+ if (!doc.exists || doc.data()?.ownerId !== ctx.token.uid) {
+ throw new TRPCError({ code: "NOT_FOUND" });
+ }
+
+ await doc.ref.update({
+ name: input.name,
+ updated: Timestamp.now(),
+ });
+ }),
+});
+
+export type WorkspaceRouter = typeof workspace;
diff --git a/server/start.ts b/server/start.ts
new file mode 100644
index 000000000..6d73dbb02
--- /dev/null
+++ b/server/start.ts
@@ -0,0 +1,26 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import getPort, { portNumbers } from "get-port";
+import { listen } from "./index";
+
+const port = await getPort({ port: portNumbers(8080, 9000) });
+
+let dispose = listen(port);
+
+if (import.meta.hot) {
+ import.meta.hot.accept("/index.ts", () => {
+ dispose(() => {
+ import("./index").then(({ listen }) => {
+ dispose = listen(port);
+ });
+ });
+ });
+}
+
+function cleanUp() {
+ dispose(() => process.exit());
+}
+
+process.on("SIGINT", cleanUp);
+process.on("SIGTERM", cleanUp);
diff --git a/server/test/context.ts b/server/test/context.ts
new file mode 100644
index 000000000..c74b7eeac
--- /dev/null
+++ b/server/test/context.ts
@@ -0,0 +1,80 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { Firestore } from "@google-cloud/firestore";
+import {
+ AnyRootConfig,
+ AnyRouterDef,
+ Router,
+ createCallerFactory,
+} from "@trpc/server";
+import { testUsers } from "db";
+import { IdTokenClient } from "google-auth-library";
+import { auth } from "../core/auth";
+import { env } from "../core/env";
+import { getFirestore } from "../core/firestore";
+import { logger } from "../core/logging";
+import { Context } from "../core/trpc";
+
+let idTokenClient: IdTokenClient | undefined;
+
+export async function getIdToken(audience: string) {
+ idTokenClient = idTokenClient ?? (await auth.getIdTokenClient(audience));
+ return await idTokenClient.idTokenProvider.fetchIdToken(audience);
+}
+
+/**
+ * Create a tRPC context for unit testing.
+ */
+export function createContext(options?: ContextOptions): Context {
+ const user = testUsers.find((u) => u.screenName === options?.user);
+ const now = Math.floor(Date.now() / 1000);
+
+ if (options?.user && !user) {
+ throw new Error(`User not found: ${options.user}`);
+ }
+
+ return {
+ db: options?.db ?? getFirestore(),
+ log: logger,
+ token: user
+ ? {
+ uid: user.localId,
+ sub: user.localId,
+ email: user.email,
+ email_verified: user.emailVerified,
+ aud: env.GOOGLE_CLOUD_PROJECT,
+ iss: `https://securetoken.google.com/${env.GOOGLE_CLOUD_PROJECT}`,
+ iat: now,
+ exp: now + 3600,
+ auth_time: parseInt(user.lastLoginAt!, 10),
+ firebase: {
+ identities: {},
+ sign_in_provider: "google.com",
+ },
+ ...(user.customAttributes && JSON.parse(user.customAttributes)),
+ }
+ : null,
+ };
+}
+
+export function createClient<
+ TRouter extends Router>,
+>(router: TRouter, options?: ContextOptions) {
+ const createCaller = createCallerFactory();
+ const ctx = createContext(options);
+ const client = createCaller(router)(ctx);
+ return [client, ctx] as const;
+}
+
+// #region Types
+
+type ContextOptions = {
+ /**
+ * The user to impersonate. Use `dan` if you need admin permissions.
+ */
+ user?: "erika" | "ryan" | "marian" | "kurt" | "dan";
+ db?: Firestore;
+};
+
+// #endregion Types
diff --git a/server/tsconfig.json b/server/tsconfig.json
new file mode 100644
index 000000000..74dd02c1a
--- /dev/null
+++ b/server/tsconfig.json
@@ -0,0 +1,15 @@
+{
+ "extends": "../tsconfig.base.json",
+ "compilerOptions": {
+ "composite": true,
+ "emitDeclarationOnly": true,
+ "module": "ESNext",
+ "moduleResolution": "Bundler",
+ "noEmit": false,
+ "outDir": "../.cache/ts-server",
+ "types": ["node", "vite/client"]
+ },
+ "include": ["**/*.ts", "**/*.json"],
+ "exclude": ["**/dist/**/*", "**/node_modules/**/*"],
+ "references": [{ "path": "../db" }]
+}
diff --git a/server/types.ts b/server/types.ts
new file mode 100644
index 000000000..97af82379
--- /dev/null
+++ b/server/types.ts
@@ -0,0 +1,4 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+export type { AppRouter } from "./routes/index";
diff --git a/server/vite.config.ts b/server/vite.config.ts
new file mode 100644
index 000000000..34c038937
--- /dev/null
+++ b/server/vite.config.ts
@@ -0,0 +1,38 @@
+/* SPDX-FileCopyrightText: 2014-present Kriasoft */
+/* SPDX-License-Identifier: MIT */
+
+import { loadEnv } from "envars";
+import { defineConfig } from "vitest/config";
+
+/**
+ * Vite configuration for the server-side bundle.
+ *
+ * @see https://vitejs.dev/config/
+ */
+export default defineConfig(async ({ mode }) => {
+ await loadEnv(mode, {
+ root: "..",
+ schema: "./core/env.ts",
+ mergeTo: process.env,
+ });
+
+ return {
+ cacheDir: "../.cache/vite-server",
+
+ build: {
+ ssr: "./index.ts",
+ sourcemap: true,
+ },
+
+ ssr: {
+ ...(mode === "production" && {
+ noExternal: ["http-errors"],
+ }),
+ },
+
+ test: {
+ environment: "node",
+ testTimeout: 20000,
+ },
+ };
+});
diff --git a/tsconfig.json b/tsconfig.json
index 8c8c9f712..7e02dcc24 100644
--- a/tsconfig.json
+++ b/tsconfig.json
@@ -2,7 +2,9 @@
"files": [],
"references": [
{ "path": "./app" },
+ { "path": "./db" },
{ "path": "./edge" },
- { "path": "./scripts" }
+ { "path": "./scripts" },
+ { "path": "./server" }
]
}
diff --git a/yarn.lock b/yarn.lock
index cfcd7bde8..3322766c8 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -1443,11 +1443,21 @@ __metadata:
linkType: hard
"@floating-ui/utils@npm:^0.2.0":
- version: 0.2.0
- resolution: "@floating-ui/utils@npm:0.2.0"
- peerDependencies:
- react: ">=16.8.0"
- checksum: 892a7a272a350f32ceab68bf0be7773378bd0d7ccf1c27385f935f18dc38327fc44633e8286191f9a836ae063684386d48e43ee6026f17be149bf02ca20433d7
+ version: 0.2.1
+ resolution: "@floating-ui/utils@npm:0.2.1"
+ checksum: 33c9ab346e7b05c5a1e6a95bc902aafcfc2c9d513a147e2491468843bd5607531b06d0b9aa56aa491cbf22a6c2495c18ccfc4c0344baec54a689a7bb8e4898d6
+ languageName: node
+ linkType: hard
+
+"@google-cloud/firestore@npm:^7.1.0":
+ version: 7.1.0
+ resolution: "@google-cloud/firestore@npm:7.1.0"
+ dependencies:
+ fast-deep-equal: "npm:^3.1.1"
+ functional-red-black-tree: "npm:^1.0.1"
+ google-gax: "npm:^4.0.4"
+ protobufjs: "npm:^7.2.5"
+ checksum: 141c03bec6ba55c80308ea105ca1319d29a7fbea1ecb95a5c2f58aa454d7f1d44112b67440a960639344b6562f0c89ffbda206fd74d875023e7addc4cdfa876f
languageName: node
linkType: hard
@@ -1460,6 +1470,15 @@ __metadata:
languageName: node
linkType: hard
+"@googleapis/identitytoolkit@npm:^8.0.0":
+ version: 8.0.0
+ resolution: "@googleapis/identitytoolkit@npm:8.0.0"
+ dependencies:
+ googleapis-common: "npm:^7.0.0"
+ checksum: 59b20c6c130d00cbeeaaca0bdf2efc0e8a06ebb20725c96b5a63efbee24b50d4d7ced9f032c691fce1acb13cc906eaf698b6e21b03d63a17fb98287a9e42a135
+ languageName: node
+ linkType: hard
+
"@grpc/grpc-js@npm:~1.9.0, @grpc/grpc-js@npm:~1.9.6":
version: 1.9.13
resolution: "@grpc/grpc-js@npm:1.9.13"
@@ -2260,10 +2279,10 @@ __metadata:
languageName: node
linkType: hard
-"@sindresorhus/is@npm:^6.1.0":
- version: 6.1.0
- resolution: "@sindresorhus/is@npm:6.1.0"
- checksum: e15c4abdca53a938e177d70c4e0562234a7ae32fa7c09ca318919d6ad28aa4ba68f58bd5339c3c38f2e912d40a201df169b9ed84579f2ef6f25d9da2d7b05334
+"@sindresorhus/is@npm:^5.2.0":
+ version: 5.6.0
+ resolution: "@sindresorhus/is@npm:5.6.0"
+ checksum: b077c325acec98e30f7d86df158aaba2e7af2acb9bb6a00fda4b91578539fbff4ecebe9b934e24fec0e6950de3089d89d79ec02d9062476b20ce185be0e01bd6
languageName: node
linkType: hard
@@ -2283,6 +2302,13 @@ __metadata:
languageName: node
linkType: hard
+"@trpc/server@npm:^10.45.0":
+ version: 10.45.0
+ resolution: "@trpc/server@npm:10.45.0"
+ checksum: 87f3c0f0169d73e7a4ce4df0cbbe5a7dd18a67a29e8a932db135bfa72301adbc656558e9bc8478317ec4007b32d6988f456aefd37a66624041714333d64d81e0
+ languageName: node
+ linkType: hard
+
"@types/babel__core@npm:^7.20.5":
version: 7.20.5
resolution: "@types/babel__core@npm:7.20.5"
@@ -2333,6 +2359,16 @@ __metadata:
languageName: node
linkType: hard
+"@types/body-parser@npm:*":
+ version: 1.19.5
+ resolution: "@types/body-parser@npm:1.19.5"
+ dependencies:
+ "@types/connect": "npm:*"
+ "@types/node": "npm:*"
+ checksum: 1e251118c4b2f61029cc43b0dc028495f2d1957fe8ee49a707fb940f86a9bd2f9754230805598278fe99958b49e9b7e66eec8ef6a50ab5c1f6b93e1ba2aaba82
+ languageName: node
+ linkType: hard
+
"@types/caseless@npm:*":
version: 0.12.5
resolution: "@types/caseless@npm:0.12.5"
@@ -2340,6 +2376,22 @@ __metadata:
languageName: node
linkType: hard
+"@types/connect@npm:*":
+ version: 3.4.38
+ resolution: "@types/connect@npm:3.4.38"
+ dependencies:
+ "@types/node": "npm:*"
+ checksum: 7eb1bc5342a9604facd57598a6c62621e244822442976c443efb84ff745246b10d06e8b309b6e80130026a396f19bf6793b7cecd7380169f369dac3bfc46fb99
+ languageName: node
+ linkType: hard
+
+"@types/cookiejar@npm:^2.1.5":
+ version: 2.1.5
+ resolution: "@types/cookiejar@npm:2.1.5"
+ checksum: 04d5990e87b6387532d15a87d9ec9b2eb783039291193863751dcfd7fc723a3b3aa30ce4c06b03975cba58632e933772f1ff031af23eaa3ac7f94e71afa6e073
+ languageName: node
+ linkType: hard
+
"@types/eslint@npm:^8.56.1":
version: 8.56.1
resolution: "@types/eslint@npm:8.56.1"
@@ -2357,6 +2409,30 @@ __metadata:
languageName: node
linkType: hard
+"@types/express-serve-static-core@npm:^4.17.33":
+ version: 4.17.41
+ resolution: "@types/express-serve-static-core@npm:4.17.41"
+ dependencies:
+ "@types/node": "npm:*"
+ "@types/qs": "npm:*"
+ "@types/range-parser": "npm:*"
+ "@types/send": "npm:*"
+ checksum: 7647e19d9c3d57ddd18947d2b161b90ef0aedd15875140e5b824209be41c1084ae942d4fb43cd5f2051a6a5f8c044519ef6c9ac1b2ad86b9aa546b4f1f023303
+ languageName: node
+ linkType: hard
+
+"@types/express@npm:^4.17.21":
+ version: 4.17.21
+ resolution: "@types/express@npm:4.17.21"
+ dependencies:
+ "@types/body-parser": "npm:*"
+ "@types/express-serve-static-core": "npm:^4.17.33"
+ "@types/qs": "npm:*"
+ "@types/serve-static": "npm:*"
+ checksum: 7a6d26cf6f43d3151caf4fec66ea11c9d23166e4f3102edfe45a94170654a54ea08cf3103d26b3928d7ebcc24162c90488e33986b7e3a5f8941225edd5eb18c7
+ languageName: node
+ linkType: hard
+
"@types/fs-extra@npm:^11.0.1":
version: 11.0.4
resolution: "@types/fs-extra@npm:11.0.4"
@@ -2374,6 +2450,22 @@ __metadata:
languageName: node
linkType: hard
+"@types/http-errors@npm:*, @types/http-errors@npm:^2.0.4":
+ version: 2.0.4
+ resolution: "@types/http-errors@npm:2.0.4"
+ checksum: 1f3d7c3b32c7524811a45690881736b3ef741bf9849ae03d32ad1ab7062608454b150a4e7f1351f83d26a418b2d65af9bdc06198f1c079d75578282884c4e8e3
+ languageName: node
+ linkType: hard
+
+"@types/http-proxy@npm:^1.17.8":
+ version: 1.17.14
+ resolution: "@types/http-proxy@npm:1.17.14"
+ dependencies:
+ "@types/node": "npm:*"
+ checksum: aa1a3e66cd43cbf06ea5901bf761d2031200a0ab42ba7e462a15c752e70f8669f21fb3be7c2f18fefcb83b95132dfa15740282e7421b856745598fbaea8e3a42
+ languageName: node
+ linkType: hard
+
"@types/json-schema@npm:*, @types/json-schema@npm:^7.0.12":
version: 7.0.15
resolution: "@types/json-schema@npm:7.0.15"
@@ -2404,6 +2496,27 @@ __metadata:
languageName: node
linkType: hard
+"@types/methods@npm:^1.1.4":
+ version: 1.1.4
+ resolution: "@types/methods@npm:1.1.4"
+ checksum: ad2a7178486f2fd167750f3eb920ab032a947ff2e26f55c86670a6038632d790b46f52e5b6ead5823f1e53fc68028f1e9ddd15cfead7903e04517c88debd72b1
+ languageName: node
+ linkType: hard
+
+"@types/mime@npm:*":
+ version: 3.0.4
+ resolution: "@types/mime@npm:3.0.4"
+ checksum: a6139c8e1f705ef2b064d072f6edc01f3c099023ad7c4fce2afc6c2bf0231888202adadbdb48643e8e20da0ce409481a49922e737eca52871b3dc08017455843
+ languageName: node
+ linkType: hard
+
+"@types/mime@npm:^1":
+ version: 1.3.5
+ resolution: "@types/mime@npm:1.3.5"
+ checksum: e29a5f9c4776f5229d84e525b7cd7dd960b51c30a0fb9a028c0821790b82fca9f672dab56561e2acd9e8eed51d431bde52eafdfef30f643586c4162f1aecfc78
+ languageName: node
+ linkType: hard
+
"@types/minimist@npm:^1.2.2":
version: 1.2.5
resolution: "@types/minimist@npm:1.2.5"
@@ -2411,6 +2524,16 @@ __metadata:
languageName: node
linkType: hard
+"@types/node-fetch@npm:^2.6.4":
+ version: 2.6.10
+ resolution: "@types/node-fetch@npm:2.6.10"
+ dependencies:
+ "@types/node": "npm:*"
+ form-data: "npm:^4.0.0"
+ checksum: 121c0c2d4d28e784006cbff26383b31538aa4f113475dcf05a2744d18b4f44d6b30184fb2460a6ef8ec66b289abc6d9b80bf54518f8a5803fec614514b022c6b
+ languageName: node
+ linkType: hard
+
"@types/node-forge@npm:^1.3.0":
version: 1.3.11
resolution: "@types/node-forge@npm:1.3.11"
@@ -2420,21 +2543,21 @@ __metadata:
languageName: node
linkType: hard
-"@types/node@npm:*, @types/node@npm:>=12.12.47, @types/node@npm:>=13.7.0, @types/node@npm:^20.10.6":
- version: 20.10.6
- resolution: "@types/node@npm:20.10.6"
+"@types/node@npm:*, @types/node@npm:>=12.12.47, @types/node@npm:>=13.7.0, @types/node@npm:^20.10.7":
+ version: 20.10.7
+ resolution: "@types/node@npm:20.10.7"
dependencies:
undici-types: "npm:~5.26.4"
- checksum: 08471220d3cbbb6669835c4b78541edf5eface8f2c2e36c550cfa4ff73da73071c90e200a06359fac25d6564127597c23e178128058fb676824ec23d5178a017
+ checksum: 4a1ba3fb7fc0a262c6b2d5ec763cec7abf5b5d6b2d277798ddbf05227f5778f9a806987821fd4d3eacd6f37c946ac61a9c64fb79ae4b84daaec12354158835a3
languageName: node
linkType: hard
-"@types/node@npm:^18.16.3":
- version: 18.19.4
- resolution: "@types/node@npm:18.19.4"
+"@types/node@npm:^18.11.18, @types/node@npm:^18.16.3":
+ version: 18.19.5
+ resolution: "@types/node@npm:18.19.5"
dependencies:
undici-types: "npm:~5.26.4"
- checksum: 4eaf6e1bb588da960c7e5974e8a0112633138b5468f0e48fc959d5585863baca549bc8c5f5f7b751c61dd282b8c2879842220f6eeed33aa54c255e876ec16536
+ checksum: 215bc65e464e5c9f042dc2e475298851dc4ba9cb68a3648ad816075b515fd1051bc19dcfab49f661938a64f40d62cc462e9c469c7aff971d1cda7095b52be9ec
languageName: node
linkType: hard
@@ -2466,6 +2589,20 @@ __metadata:
languageName: node
linkType: hard
+"@types/qs@npm:*":
+ version: 6.9.11
+ resolution: "@types/qs@npm:6.9.11"
+ checksum: 620ca1628bf3da65662c54ed6ebb120b18a3da477d0bfcc872b696685a9bb1893c3c92b53a1190a8f54d52eaddb6af8b2157755699ac83164604329935e8a7f2
+ languageName: node
+ linkType: hard
+
+"@types/range-parser@npm:*":
+ version: 1.2.7
+ resolution: "@types/range-parser@npm:1.2.7"
+ checksum: 95640233b689dfbd85b8c6ee268812a732cf36d5affead89e806fe30da9a430767af8ef2cd661024fd97e19d61f3dec75af2df5e80ec3bea000019ab7028629a
+ languageName: node
+ linkType: hard
+
"@types/react-dom@npm:^18.2.18":
version: 18.2.18
resolution: "@types/react-dom@npm:18.2.18"
@@ -2521,6 +2658,48 @@ __metadata:
languageName: node
linkType: hard
+"@types/send@npm:*":
+ version: 0.17.4
+ resolution: "@types/send@npm:0.17.4"
+ dependencies:
+ "@types/mime": "npm:^1"
+ "@types/node": "npm:*"
+ checksum: 28320a2aa1eb704f7d96a65272a07c0bf3ae7ed5509c2c96ea5e33238980f71deeed51d3631927a77d5250e4091b3e66bce53b42d770873282c6a20bb8b0280d
+ languageName: node
+ linkType: hard
+
+"@types/serve-static@npm:*":
+ version: 1.15.5
+ resolution: "@types/serve-static@npm:1.15.5"
+ dependencies:
+ "@types/http-errors": "npm:*"
+ "@types/mime": "npm:*"
+ "@types/node": "npm:*"
+ checksum: 49aa21c367fffe4588fc8c57ea48af0ea7cbadde7418bc53cde85d8bd57fd2a09a293970d9ea86e79f17a87f8adeb3e20da76aab38e1c4d1567931fa15c8af38
+ languageName: node
+ linkType: hard
+
+"@types/superagent@npm:^8.1.0":
+ version: 8.1.1
+ resolution: "@types/superagent@npm:8.1.1"
+ dependencies:
+ "@types/cookiejar": "npm:^2.1.5"
+ "@types/methods": "npm:^1.1.4"
+ "@types/node": "npm:*"
+ checksum: 02b987833cf0d85da9b137fd296fe8ad25a470d60f7e9d81a6ed3f8f8a5d6bace8780816bd35885e2928f467e819a4aa509879a7da0f28018ab1453845eb91e2
+ languageName: node
+ linkType: hard
+
+"@types/supertest@npm:^6.0.2":
+ version: 6.0.2
+ resolution: "@types/supertest@npm:6.0.2"
+ dependencies:
+ "@types/methods": "npm:^1.1.4"
+ "@types/superagent": "npm:^8.1.0"
+ checksum: 4b67fb2d1bfbb7ff0a7dfaaf190cdf2e0014522615fb2dc53c214bdac95b4ee42696dd1df13332c90a7765cc52934c9cc0c428bf0f9e8189167aef01042e7448
+ languageName: node
+ linkType: hard
+
"@types/tough-cookie@npm:*":
version: 4.0.5
resolution: "@types/tough-cookie@npm:4.0.5"
@@ -2535,15 +2714,24 @@ __metadata:
languageName: node
linkType: hard
-"@typescript-eslint/eslint-plugin@npm:^6.17.0":
- version: 6.17.0
- resolution: "@typescript-eslint/eslint-plugin@npm:6.17.0"
+"@types/ws@npm:^8.5.10":
+ version: 8.5.10
+ resolution: "@types/ws@npm:8.5.10"
+ dependencies:
+ "@types/node": "npm:*"
+ checksum: 9b414dc5e0b6c6f1ea4b1635b3568c58707357f68076df9e7cd33194747b7d1716d5189c0dbdd68c8d2521b148e88184cf881bac7429eb0e5c989b001539ed31
+ languageName: node
+ linkType: hard
+
+"@typescript-eslint/eslint-plugin@npm:^6.18.0":
+ version: 6.18.0
+ resolution: "@typescript-eslint/eslint-plugin@npm:6.18.0"
dependencies:
"@eslint-community/regexpp": "npm:^4.5.1"
- "@typescript-eslint/scope-manager": "npm:6.17.0"
- "@typescript-eslint/type-utils": "npm:6.17.0"
- "@typescript-eslint/utils": "npm:6.17.0"
- "@typescript-eslint/visitor-keys": "npm:6.17.0"
+ "@typescript-eslint/scope-manager": "npm:6.18.0"
+ "@typescript-eslint/type-utils": "npm:6.18.0"
+ "@typescript-eslint/utils": "npm:6.18.0"
+ "@typescript-eslint/visitor-keys": "npm:6.18.0"
debug: "npm:^4.3.4"
graphemer: "npm:^1.4.0"
ignore: "npm:^5.2.4"
@@ -2556,44 +2744,44 @@ __metadata:
peerDependenciesMeta:
typescript:
optional: true
- checksum: f2a5774e9cc03e491a5a488501e5622c7eebd766f5a4fc2c30642864a3b89b0807946bde33a678f326ba7032f3f6a51aa0bf9c2d10adc823804fc9fb47db55a6
+ checksum: 7810a84f6d9cb735f6848aa4a7ef64c77740812afc0248fda63ec182910a1d045cd9a32d8e49b0e5323266db6da12a2fe50171147927b3186815a56f12c16ee7
languageName: node
linkType: hard
-"@typescript-eslint/parser@npm:^6.17.0":
- version: 6.17.0
- resolution: "@typescript-eslint/parser@npm:6.17.0"
+"@typescript-eslint/parser@npm:^6.18.0":
+ version: 6.18.0
+ resolution: "@typescript-eslint/parser@npm:6.18.0"
dependencies:
- "@typescript-eslint/scope-manager": "npm:6.17.0"
- "@typescript-eslint/types": "npm:6.17.0"
- "@typescript-eslint/typescript-estree": "npm:6.17.0"
- "@typescript-eslint/visitor-keys": "npm:6.17.0"
+ "@typescript-eslint/scope-manager": "npm:6.18.0"
+ "@typescript-eslint/types": "npm:6.18.0"
+ "@typescript-eslint/typescript-estree": "npm:6.18.0"
+ "@typescript-eslint/visitor-keys": "npm:6.18.0"
debug: "npm:^4.3.4"
peerDependencies:
eslint: ^7.0.0 || ^8.0.0
peerDependenciesMeta:
typescript:
optional: true
- checksum: 2ed0ed4a5b30e953430ce3279df3655af09fa1caed2abf11804d239717daefc32a22864f6620ef57bb9c684c74a99a13241384fea5096e961385e3678fc2e920
+ checksum: 6798332819f839454a8405e31cfaa0fe908d5966be929bef55e78ac51a0ff3868feb42b38e7772cedf88277f6b2841b3d91f6c573eacb945e2741ecae94047c7
languageName: node
linkType: hard
-"@typescript-eslint/scope-manager@npm:6.17.0":
- version: 6.17.0
- resolution: "@typescript-eslint/scope-manager@npm:6.17.0"
+"@typescript-eslint/scope-manager@npm:6.18.0":
+ version: 6.18.0
+ resolution: "@typescript-eslint/scope-manager@npm:6.18.0"
dependencies:
- "@typescript-eslint/types": "npm:6.17.0"
- "@typescript-eslint/visitor-keys": "npm:6.17.0"
- checksum: fe09c628553c9336e6a36d32c1d34e78ebd20aa02130a6bf535329621ba5a98aaac171f607bc6e4d17b3478c42e7de6476376636897ce3f227c754eb99acd07e
+ "@typescript-eslint/types": "npm:6.18.0"
+ "@typescript-eslint/visitor-keys": "npm:6.18.0"
+ checksum: c2c465e6803f78d3300142167a8a79dd2613c64cf464a40a9cf6b13a2c10c3d82ca30bb9c2d26aba7f054b8740b38e1d25f377fcdf917aba489d5b5ea2550858
languageName: node
linkType: hard
-"@typescript-eslint/type-utils@npm:6.17.0":
- version: 6.17.0
- resolution: "@typescript-eslint/type-utils@npm:6.17.0"
+"@typescript-eslint/type-utils@npm:6.18.0":
+ version: 6.18.0
+ resolution: "@typescript-eslint/type-utils@npm:6.18.0"
dependencies:
- "@typescript-eslint/typescript-estree": "npm:6.17.0"
- "@typescript-eslint/utils": "npm:6.17.0"
+ "@typescript-eslint/typescript-estree": "npm:6.18.0"
+ "@typescript-eslint/utils": "npm:6.18.0"
debug: "npm:^4.3.4"
ts-api-utils: "npm:^1.0.1"
peerDependencies:
@@ -2601,23 +2789,23 @@ __metadata:
peerDependenciesMeta:
typescript:
optional: true
- checksum: dc7938429193acfda61b7282197ec046039e2c4da41cdcddf4daaf300d10229e4e23bb0fcf0503b19c0b99a874849c8a9f5bb35ce106260f56a14106d2b41d8c
+ checksum: 9f4a392fe7b7f6b1fb02acbdaa331e764775f6404c29ca66774419e39552523d878227020526ca9c438d997555a99600f8d711496e9a435fb14a779e25ed094e
languageName: node
linkType: hard
-"@typescript-eslint/types@npm:6.17.0":
- version: 6.17.0
- resolution: "@typescript-eslint/types@npm:6.17.0"
- checksum: 87ab1b5a3270ab34b917c22a2fb90a9ad7d9f3b19d73a337bc9efbe65f924da13482c97e8ccbe3bd3d081aa96039eeff50de41c1da2a2128066429b931cdb21d
+"@typescript-eslint/types@npm:6.18.0":
+ version: 6.18.0
+ resolution: "@typescript-eslint/types@npm:6.18.0"
+ checksum: fc507ca7a1bfec04467087165ff6722f7b9aa9a089ecf0c17656824a951b92ca014766e480122de850057c63a3066628985eb0681c5bbb80ab41d94e7bb52288
languageName: node
linkType: hard
-"@typescript-eslint/typescript-estree@npm:6.17.0":
- version: 6.17.0
- resolution: "@typescript-eslint/typescript-estree@npm:6.17.0"
+"@typescript-eslint/typescript-estree@npm:6.18.0":
+ version: 6.18.0
+ resolution: "@typescript-eslint/typescript-estree@npm:6.18.0"
dependencies:
- "@typescript-eslint/types": "npm:6.17.0"
- "@typescript-eslint/visitor-keys": "npm:6.17.0"
+ "@typescript-eslint/types": "npm:6.18.0"
+ "@typescript-eslint/visitor-keys": "npm:6.18.0"
debug: "npm:^4.3.4"
globby: "npm:^11.1.0"
is-glob: "npm:^4.0.3"
@@ -2627,34 +2815,34 @@ __metadata:
peerDependenciesMeta:
typescript:
optional: true
- checksum: 1671b0d2f2fdf07074fb1e2524d61935cec173bd8db6e482cc5b2dcc77aed3ffa831396736ffa0ee2fdbddd8585ae9ca8d6c97bcaea1385b23907a1ec0508f83
+ checksum: b65392e944baba97ed98e99a1e7122b7b05fa0d9a082b48d0190b377ae9e2ae4ed72d505a2f0f05f2ca78908f0e4b0f1acd44d345c7f4f4415fcec6bb2c57791
languageName: node
linkType: hard
-"@typescript-eslint/utils@npm:6.17.0":
- version: 6.17.0
- resolution: "@typescript-eslint/utils@npm:6.17.0"
+"@typescript-eslint/utils@npm:6.18.0":
+ version: 6.18.0
+ resolution: "@typescript-eslint/utils@npm:6.18.0"
dependencies:
"@eslint-community/eslint-utils": "npm:^4.4.0"
"@types/json-schema": "npm:^7.0.12"
"@types/semver": "npm:^7.5.0"
- "@typescript-eslint/scope-manager": "npm:6.17.0"
- "@typescript-eslint/types": "npm:6.17.0"
- "@typescript-eslint/typescript-estree": "npm:6.17.0"
+ "@typescript-eslint/scope-manager": "npm:6.18.0"
+ "@typescript-eslint/types": "npm:6.18.0"
+ "@typescript-eslint/typescript-estree": "npm:6.18.0"
semver: "npm:^7.5.4"
peerDependencies:
eslint: ^7.0.0 || ^8.0.0
- checksum: 37c63afcf66124bf84808699997953b8c84a378aa2c490a771b611d82cdac8499c58fac8eeb8378528e97660b59563d99297bfec4b982cd68760b0ffe54aa714
+ checksum: f91798069e337ed42b7e415aabded833d540dab9adf66ae88183003428892584946f20a5496aae2dc65fc0f65b66496baed785a3470ee782b6e2dd25b9200c6c
languageName: node
linkType: hard
-"@typescript-eslint/visitor-keys@npm:6.17.0":
- version: 6.17.0
- resolution: "@typescript-eslint/visitor-keys@npm:6.17.0"
+"@typescript-eslint/visitor-keys@npm:6.18.0":
+ version: 6.18.0
+ resolution: "@typescript-eslint/visitor-keys@npm:6.18.0"
dependencies:
- "@typescript-eslint/types": "npm:6.17.0"
+ "@typescript-eslint/types": "npm:6.18.0"
eslint-visitor-keys: "npm:^3.4.1"
- checksum: a2aed0e1437fdab8858ab9c7c8e355f8b72a5fa4b0adc54f28b8a2bbc29d4bb93214968ee940f83d013d0a4b83d00cd4eeeb05fb4c2c7d0ead324c6793f7d6d4
+ checksum: bf34a357549d515607c761f385b7c7c82aaa07795cada0be2e1e3860c5103fbf731408ac07eaeb0517b51426d77ef9b194dfb94f205c776107a46e0d0027b452
languageName: node
linkType: hard
@@ -2750,6 +2938,16 @@ __metadata:
languageName: node
linkType: hard
+"accepts@npm:~1.3.8":
+ version: 1.3.8
+ resolution: "accepts@npm:1.3.8"
+ dependencies:
+ mime-types: "npm:~2.1.34"
+ negotiator: "npm:0.6.3"
+ checksum: 67eaaa90e2917c58418e7a9b89392002d2b1ccd69bcca4799135d0c632f3b082f23f4ae4ddeedbced5aa59bcc7bdf4699c69ebed4593696c922462b7bc5744d6
+ languageName: node
+ linkType: hard
+
"acorn-jsx@npm:^5.3.2":
version: 5.3.2
resolution: "acorn-jsx@npm:5.3.2"
@@ -2793,6 +2991,15 @@ __metadata:
languageName: node
linkType: hard
+"agentkeepalive@npm:^4.2.1":
+ version: 4.5.0
+ resolution: "agentkeepalive@npm:4.5.0"
+ dependencies:
+ humanize-ms: "npm:^1.2.1"
+ checksum: dd210ba2a2e2482028f027b1156789744aadbfd773a6c9dd8e4e8001930d5af82382abe19a69240307b1d8003222ce6b0542935038313434b900e351914fc15f
+ languageName: node
+ linkType: hard
+
"aggregate-error@npm:^3.0.0":
version: 3.1.0
resolution: "aggregate-error@npm:3.1.0"
@@ -2885,7 +3092,7 @@ __metadata:
"@mui/joy": "npm:^5.0.0-beta.21"
"@mui/lab": "npm:^5.0.0-alpha.159"
"@mui/material": "npm:^5.15.3"
- "@types/node": "npm:^20.10.6"
+ "@types/node": "npm:^20.10.7"
"@types/react": "npm:^18.2.47"
"@types/react-dom": "npm:^18.2.18"
"@vitejs/plugin-react": "npm:^4.2.1"
@@ -2940,6 +3147,13 @@ __metadata:
languageName: node
linkType: hard
+"array-flatten@npm:1.1.1":
+ version: 1.1.1
+ resolution: "array-flatten@npm:1.1.1"
+ checksum: e13c9d247241be82f8b4ec71d035ed7204baa82fae820d4db6948d30d3c4a9f2b3905eb2eec2b937d4aa3565200bd3a1c500480114cff649fa748747d2a50feb
+ languageName: node
+ linkType: hard
+
"array-includes@npm:^3.1.6, array-includes@npm:^3.1.7":
version: 3.1.7
resolution: "array-includes@npm:3.1.7"
@@ -3034,6 +3248,13 @@ __metadata:
languageName: node
linkType: hard
+"asap@npm:^2.0.0":
+ version: 2.0.6
+ resolution: "asap@npm:2.0.6"
+ checksum: b244c0458c571945e4b3be0b14eb001bea5596f9868cc50cc711dc03d58a7e953517d3f0dad81ccde3ff37d1f074701fa76a6f07d41aaa992d7204a37b915dda
+ languageName: node
+ linkType: hard
+
"assertion-error@npm:^1.1.0":
version: 1.1.0
resolution: "assertion-error@npm:1.1.0"
@@ -3064,6 +3285,13 @@ __metadata:
languageName: node
linkType: hard
+"atomic-sleep@npm:^1.0.0":
+ version: 1.0.0
+ resolution: "atomic-sleep@npm:1.0.0"
+ checksum: 3ab6d2cf46b31394b4607e935ec5c1c3c4f60f3e30f0913d35ea74b51b3585e84f590d09e58067f11762eec71c87d25314ce859030983dc0e4397eed21daa12e
+ languageName: node
+ linkType: hard
+
"available-typed-arrays@npm:^1.0.5":
version: 1.0.5
resolution: "available-typed-arrays@npm:1.0.5"
@@ -3105,7 +3333,14 @@ __metadata:
languageName: node
linkType: hard
-"base64-js@npm:^1.3.0":
+"base-64@npm:^0.1.0":
+ version: 0.1.0
+ resolution: "base-64@npm:0.1.0"
+ checksum: 5a42938f82372ab5392cbacc85a5a78115cbbd9dbef9f7540fa47d78763a3a8bd7d598475f0d92341f66285afd377509851a9bb5c67bbecb89686e9255d5b3eb
+ languageName: node
+ linkType: hard
+
+"base64-js@npm:^1.3.0, base64-js@npm:^1.3.1":
version: 1.5.1
resolution: "base64-js@npm:1.5.1"
checksum: 669632eb3745404c2f822a18fc3a0122d2f9a7a13f7fb8b5823ee19d1d2ff9ee5b52c53367176ea4ad093c332fd5ab4bd0ebae5a8e27917a4105a4cfc86b1005
@@ -3133,6 +3368,26 @@ __metadata:
languageName: node
linkType: hard
+"body-parser@npm:1.20.1":
+ version: 1.20.1
+ resolution: "body-parser@npm:1.20.1"
+ dependencies:
+ bytes: "npm:3.1.2"
+ content-type: "npm:~1.0.4"
+ debug: "npm:2.6.9"
+ depd: "npm:2.0.0"
+ destroy: "npm:1.2.0"
+ http-errors: "npm:2.0.0"
+ iconv-lite: "npm:0.4.24"
+ on-finished: "npm:2.4.1"
+ qs: "npm:6.11.0"
+ raw-body: "npm:2.5.1"
+ type-is: "npm:~1.6.18"
+ unpipe: "npm:1.0.0"
+ checksum: 5f8d128022a2fb8b6e7990d30878a0182f300b70e46b3f9d358a9433ad6275f0de46add6d63206da3637c01c3b38b6111a7480f7e7ac2e9f7b989f6133fe5510
+ languageName: node
+ linkType: hard
+
"brace-expansion@npm:^1.1.7":
version: 1.1.11
resolution: "brace-expansion@npm:1.1.11"
@@ -3182,6 +3437,16 @@ __metadata:
languageName: node
linkType: hard
+"buffer@npm:^6.0.3":
+ version: 6.0.3
+ resolution: "buffer@npm:6.0.3"
+ dependencies:
+ base64-js: "npm:^1.3.1"
+ ieee754: "npm:^1.2.1"
+ checksum: b6bc68237ebf29bdacae48ce60e5e28fc53ae886301f2ad9496618efac49427ed79096750033e7eab1897a4f26ae374ace49106a5758f38fb70c78c9fda2c3b1
+ languageName: node
+ linkType: hard
+
"builtins@npm:^5.0.0":
version: 5.0.1
resolution: "builtins@npm:5.0.1"
@@ -3200,6 +3465,13 @@ __metadata:
languageName: node
linkType: hard
+"bytes@npm:3.1.2":
+ version: 3.1.2
+ resolution: "bytes@npm:3.1.2"
+ checksum: a10abf2ba70c784471d6b4f58778c0beeb2b5d405148e66affa91f23a9f13d07603d0a0354667310ae1d6dc141474ffd44e2a074be0f6e2254edb8fc21445388
+ languageName: node
+ linkType: hard
+
"cac@npm:^6.7.14":
version: 6.7.14
resolution: "cac@npm:6.7.14"
@@ -3234,7 +3506,7 @@ __metadata:
languageName: node
linkType: hard
-"cacheable-request@npm:^10.2.14":
+"cacheable-request@npm:^10.2.8":
version: 10.2.14
resolution: "cacheable-request@npm:10.2.14"
dependencies:
@@ -3300,9 +3572,9 @@ __metadata:
linkType: hard
"caniuse-lite@npm:^1.0.30001565":
- version: 1.0.30001574
- resolution: "caniuse-lite@npm:1.0.30001574"
- checksum: 159ebd04d9bbef11bd08499f058f70bf795a55641929be5efadf0f6b17216d4b923506778e59bbb939246834304b753b2e88ff1e2430f6a5aef0a86971f98bd3
+ version: 1.0.30001576
+ resolution: "caniuse-lite@npm:1.0.30001576"
+ checksum: 51632942733593f310e581bd91c9558b8d75fbf67160a39f8036d2976cd7df9183e96d4c9d9e6f18e0205950b940d9c761bcfb7810962d7899f8a1179fde6e3f
languageName: node
linkType: hard
@@ -3352,13 +3624,20 @@ __metadata:
languageName: node
linkType: hard
-"chalk@npm:^5.2.0":
+"chalk@npm:^5.2.0, chalk@npm:^5.3.0":
version: 5.3.0
resolution: "chalk@npm:5.3.0"
checksum: 6373caaab21bd64c405bfc4bd9672b145647fc9482657b5ea1d549b3b2765054e9d3d928870cdf764fb4aad67555f5061538ff247b8310f110c5c888d92397ea
languageName: node
linkType: hard
+"charenc@npm:0.0.2":
+ version: 0.0.2
+ resolution: "charenc@npm:0.0.2"
+ checksum: 81dcadbe57e861d527faf6dd3855dc857395a1c4d6781f4847288ab23cffb7b3ee80d57c15bba7252ffe3e5e8019db767757ee7975663ad2ca0939bb8fcaf2e5
+ languageName: node
+ linkType: hard
+
"check-error@npm:^1.0.3":
version: 1.0.3
resolution: "check-error@npm:1.0.3"
@@ -3401,6 +3680,22 @@ __metadata:
languageName: node
linkType: hard
+"cli-cursor@npm:^4.0.0":
+ version: 4.0.0
+ resolution: "cli-cursor@npm:4.0.0"
+ dependencies:
+ restore-cursor: "npm:^4.0.0"
+ checksum: ab3f3ea2076e2176a1da29f9d64f72ec3efad51c0960898b56c8a17671365c26e67b735920530eaf7328d61f8bd41c27f46b9cf6e4e10fe2fa44b5e8c0e392cc
+ languageName: node
+ linkType: hard
+
+"cli-spinners@npm:^2.9.2":
+ version: 2.9.2
+ resolution: "cli-spinners@npm:2.9.2"
+ checksum: a0a863f442df35ed7294424f5491fa1756bd8d2e4ff0c8736531d886cec0ece4d85e8663b77a5afaf1d296e3cbbebff92e2e99f52bbea89b667cbe789b994794
+ languageName: node
+ linkType: hard
+
"cliui@npm:^8.0.1":
version: 8.0.1
resolution: "cliui@npm:8.0.1"
@@ -3458,7 +3753,14 @@ __metadata:
languageName: node
linkType: hard
-"combined-stream@npm:^1.0.6":
+"colorette@npm:^2.0.7":
+ version: 2.0.20
+ resolution: "colorette@npm:2.0.20"
+ checksum: 0b8de48bfa5d10afc160b8eaa2b9938f34a892530b2f7d7897e0458d9535a066e3998b49da9d21161c78225b272df19ae3a64d6df28b4c9734c0e55bbd02406f
+ languageName: node
+ linkType: hard
+
+"combined-stream@npm:^1.0.6, combined-stream@npm:^1.0.8":
version: 1.0.8
resolution: "combined-stream@npm:1.0.8"
dependencies:
@@ -3467,6 +3769,13 @@ __metadata:
languageName: node
linkType: hard
+"component-emitter@npm:^1.3.0":
+ version: 1.3.1
+ resolution: "component-emitter@npm:1.3.1"
+ checksum: 94550aa462c7bd5a61c1bc480e28554aa306066930152d1b1844a0dd3845d4e5db7e261ddec62ae184913b3e59b55a2ad84093b9d3596a8f17c341514d6c483d
+ languageName: node
+ linkType: hard
+
"concat-map@npm:0.0.1":
version: 0.0.1
resolution: "concat-map@npm:0.0.1"
@@ -3474,6 +3783,22 @@ __metadata:
languageName: node
linkType: hard
+"content-disposition@npm:0.5.4":
+ version: 0.5.4
+ resolution: "content-disposition@npm:0.5.4"
+ dependencies:
+ safe-buffer: "npm:5.2.1"
+ checksum: b7f4ce176e324f19324be69b05bf6f6e411160ac94bc523b782248129eb1ef3be006f6cff431aaea5e337fe5d176ce8830b8c2a1b721626ead8933f0cbe78720
+ languageName: node
+ linkType: hard
+
+"content-type@npm:~1.0.4":
+ version: 1.0.5
+ resolution: "content-type@npm:1.0.5"
+ checksum: 585847d98dc7fb8035c02ae2cb76c7a9bd7b25f84c447e5ed55c45c2175e83617c8813871b4ee22f368126af6b2b167df655829007b21aa10302873ea9c62662
+ languageName: node
+ linkType: hard
+
"convert-source-map@npm:^1.5.0":
version: 1.9.0
resolution: "convert-source-map@npm:1.9.0"
@@ -3488,13 +3813,27 @@ __metadata:
languageName: node
linkType: hard
-"cookie@npm:^0.5.0":
+"cookie-signature@npm:1.0.6":
+ version: 1.0.6
+ resolution: "cookie-signature@npm:1.0.6"
+ checksum: f4e1b0a98a27a0e6e66fd7ea4e4e9d8e038f624058371bf4499cfcd8f3980be9a121486995202ba3fca74fbed93a407d6d54d43a43f96fd28d0bd7a06761591a
+ languageName: node
+ linkType: hard
+
+"cookie@npm:0.5.0, cookie@npm:^0.5.0":
version: 0.5.0
resolution: "cookie@npm:0.5.0"
checksum: aae7911ddc5f444a9025fbd979ad1b5d60191011339bce48e555cb83343d0f98b865ff5c4d71fecdfb8555a5cafdc65632f6fce172f32aaf6936830a883a0380
languageName: node
linkType: hard
+"cookiejar@npm:^2.1.4":
+ version: 2.1.4
+ resolution: "cookiejar@npm:2.1.4"
+ checksum: 4a184f5a0591df8b07d22a43ea5d020eacb4572c383e853a33361a99710437eaa0971716c688684075bbf695b484f5872e9e3f562382e46858716cb7fc8ce3f4
+ languageName: node
+ linkType: hard
+
"cosmiconfig@npm:^5.0.5":
version: 5.2.1
resolution: "cosmiconfig@npm:5.2.1"
@@ -3531,6 +3870,13 @@ __metadata:
languageName: node
linkType: hard
+"crypt@npm:0.0.2":
+ version: 0.0.2
+ resolution: "crypt@npm:0.0.2"
+ checksum: 2c72768de3d28278c7c9ffd81a298b26f87ecdfe94415084f339e6632f089b43fe039f2c93f612bcb5ffe447238373d93b2e8c90894cba6cfb0ac7a74616f8b9
+ languageName: node
+ linkType: hard
+
"css.escape@npm:^1.5.1":
version: 1.5.1
resolution: "css.escape@npm:1.5.1"
@@ -3566,6 +3912,39 @@ __metadata:
languageName: node
linkType: hard
+"dateformat@npm:^4.6.3":
+ version: 4.6.3
+ resolution: "dateformat@npm:4.6.3"
+ checksum: 5c149c91bf9ce2142c89f84eee4c585f0cb1f6faf2536b1af89873f862666a28529d1ccafc44750aa01384da2197c4f76f4e149a3cc0c1cb2c46f5cc45f2bcb5
+ languageName: node
+ linkType: hard
+
+"db@workspace:*, db@workspace:db":
+ version: 0.0.0-use.local
+ resolution: "db@workspace:db"
+ dependencies:
+ "@google-cloud/firestore": "npm:^7.1.0"
+ "@googleapis/identitytoolkit": "npm:^8.0.0"
+ "@types/node": "npm:^20.10.7"
+ dotenv: "npm:^16.3.1"
+ ora: "npm:^8.0.1"
+ typescript: "npm:~5.3.3"
+ vite: "npm:~5.0.11"
+ vite-node: "npm:~1.1.3"
+ vitest: "npm:~1.1.3"
+ zod: "npm:^3.22.4"
+ languageName: unknown
+ linkType: soft
+
+"debug@npm:2.6.9":
+ version: 2.6.9
+ resolution: "debug@npm:2.6.9"
+ dependencies:
+ ms: "npm:2.0.0"
+ checksum: e07005f2b40e04f1bd14a3dd20520e9c4f25f60224cb006ce9d6781732c917964e9ec029fc7f1a151083cd929025ad5133814d4dc624a9aaf020effe4914ed14
+ languageName: node
+ linkType: hard
+
"debug@npm:4, debug@npm:^4.1.0, debug@npm:^4.1.1, debug@npm:^4.3.1, debug@npm:^4.3.2, debug@npm:^4.3.4":
version: 4.3.4
resolution: "debug@npm:4.3.4"
@@ -3648,6 +4027,13 @@ __metadata:
languageName: node
linkType: hard
+"depd@npm:2.0.0":
+ version: 2.0.0
+ resolution: "depd@npm:2.0.0"
+ checksum: c0c8ff36079ce5ada64f46cc9d6fd47ebcf38241105b6e0c98f412e8ad91f084bcf906ff644cc3a4bd876ca27a62accb8b0fff72ea6ed1a414b89d8506f4a5ca
+ languageName: node
+ linkType: hard
+
"dequal@npm:^2.0.3":
version: 2.0.3
resolution: "dequal@npm:2.0.3"
@@ -3655,6 +4041,23 @@ __metadata:
languageName: node
linkType: hard
+"destroy@npm:1.2.0":
+ version: 1.2.0
+ resolution: "destroy@npm:1.2.0"
+ checksum: 0acb300b7478a08b92d810ab229d5afe0d2f4399272045ab22affa0d99dbaf12637659411530a6fcd597a9bdac718fc94373a61a95b4651bbc7b83684a565e38
+ languageName: node
+ linkType: hard
+
+"dezalgo@npm:^1.0.4":
+ version: 1.0.4
+ resolution: "dezalgo@npm:1.0.4"
+ dependencies:
+ asap: "npm:^2.0.0"
+ wrappy: "npm:1"
+ checksum: 895389c6aead740d2ab5da4d3466d20fa30f738010a4d3f4dcccc9fc645ca31c9d10b7e1804ae489b1eb02c7986f9f1f34ba132d409b043082a86d9a4e745624
+ languageName: node
+ linkType: hard
+
"diff-sequences@npm:^29.6.3":
version: 29.6.3
resolution: "diff-sequences@npm:29.6.3"
@@ -3662,6 +4065,16 @@ __metadata:
languageName: node
linkType: hard
+"digest-fetch@npm:^1.3.0":
+ version: 1.3.0
+ resolution: "digest-fetch@npm:1.3.0"
+ dependencies:
+ base-64: "npm:^0.1.0"
+ md5: "npm:^2.3.0"
+ checksum: 5a90f350ed19600aab44753ad19e8b2e5409b0a07a2a6949c9eed589bfc7a735308be020d63c6ddff97ad83d0b5d56405e0577ed4c291baccadec3e61ebca189
+ languageName: node
+ linkType: hard
+
"dir-glob@npm:^3.0.1":
version: 3.0.1
resolution: "dir-glob@npm:3.0.1"
@@ -3761,7 +4174,7 @@ __metadata:
dependencies:
"@cloudflare/workers-types": "npm:^4.20231218.0"
"@hono/zod-validator": "npm:^0.1.11"
- "@types/node": "npm:^20.10.6"
+ "@types/node": "npm:^20.10.7"
happy-dom: "npm:^12.10.3"
hono: "npm:^3.12.0"
jose: "npm:^5.2.0"
@@ -3775,6 +4188,13 @@ __metadata:
languageName: unknown
linkType: soft
+"ee-first@npm:1.1.1":
+ version: 1.1.1
+ resolution: "ee-first@npm:1.1.1"
+ checksum: 1b4cac778d64ce3b582a7e26b218afe07e207a0f9bfe13cc7395a6d307849cfe361e65033c3251e00c27dd060cab43014c2d6b2647676135e18b77d2d05b3f4f
+ languageName: node
+ linkType: hard
+
"electron-to-chromium@npm:^1.4.601":
version: 1.4.623
resolution: "electron-to-chromium@npm:1.4.623"
@@ -3782,6 +4202,13 @@ __metadata:
languageName: node
linkType: hard
+"emoji-regex@npm:^10.3.0":
+ version: 10.3.0
+ resolution: "emoji-regex@npm:10.3.0"
+ checksum: b9b084ebe904f13bb4b66ee4c29fb41a7a4a1165adcc33c1ce8056c0194b882cc91ebdc782f1a779b5d7ea7375c5064643a7734893d7c657b44c5c6b9d7bf1e7
+ languageName: node
+ linkType: hard
+
"emoji-regex@npm:^8.0.0":
version: 8.0.0
resolution: "emoji-regex@npm:8.0.0"
@@ -3796,6 +4223,13 @@ __metadata:
languageName: node
linkType: hard
+"encodeurl@npm:~1.0.2":
+ version: 1.0.2
+ resolution: "encodeurl@npm:1.0.2"
+ checksum: e50e3d508cdd9c4565ba72d2012e65038e5d71bdc9198cb125beb6237b5b1ade6c0d343998da9e170fb2eae52c1bed37d4d6d98a46ea423a0cddbed5ac3f780c
+ languageName: node
+ linkType: hard
+
"encoding@npm:^0.1.13":
version: 0.1.13
resolution: "encoding@npm:0.1.13"
@@ -3805,7 +4239,7 @@ __metadata:
languageName: node
linkType: hard
-"end-of-stream@npm:^1.4.1":
+"end-of-stream@npm:^1.1.0, end-of-stream@npm:^1.4.1":
version: 1.4.4
resolution: "end-of-stream@npm:1.4.4"
dependencies:
@@ -3838,6 +4272,15 @@ __metadata:
languageName: node
linkType: hard
+"envalid@npm:^8.0.0":
+ version: 8.0.0
+ resolution: "envalid@npm:8.0.0"
+ dependencies:
+ tslib: "npm:2.6.2"
+ checksum: cf2159a070c7335ecc9cd01523654fb7c8fdc63b92bef47d18027c406dfcf2b4a72663e70ddd91fa91967fa71c7716fe643dcd4a652f9ab0f93722feddb9e1b2
+ languageName: node
+ linkType: hard
+
"envars@npm:^1.0.2":
version: 1.0.2
resolution: "envars@npm:1.0.2"
@@ -4350,6 +4793,13 @@ __metadata:
languageName: node
linkType: hard
+"escape-html@npm:~1.0.3":
+ version: 1.0.3
+ resolution: "escape-html@npm:1.0.3"
+ checksum: 6213ca9ae00d0ab8bccb6d8d4e0a98e76237b2410302cf7df70aaa6591d509a2a37ce8998008cbecae8fc8ffaadf3fb0229535e6a145f3ce0b211d060decbb24
+ languageName: node
+ linkType: hard
+
"escape-string-regexp@npm:^1.0.5":
version: 1.0.5
resolution: "escape-string-regexp@npm:1.0.5"
@@ -4638,6 +5088,13 @@ __metadata:
languageName: node
linkType: hard
+"etag@npm:~1.8.1":
+ version: 1.8.1
+ resolution: "etag@npm:1.8.1"
+ checksum: 571aeb3dbe0f2bbd4e4fadbdb44f325fc75335cd5f6f6b6a091e6a06a9f25ed5392f0863c5442acb0646787446e816f13cbfc6edce5b07658541dff573cab1ff
+ languageName: node
+ linkType: hard
+
"event-stream@npm:=3.3.4":
version: 3.3.4
resolution: "event-stream@npm:3.3.4"
@@ -4660,6 +5117,20 @@ __metadata:
languageName: node
linkType: hard
+"eventemitter3@npm:^4.0.0":
+ version: 4.0.7
+ resolution: "eventemitter3@npm:4.0.7"
+ checksum: 8030029382404942c01d0037079f1b1bc8fed524b5849c237b80549b01e2fc49709e1d0c557fa65ca4498fc9e24cff1475ef7b855121fcc15f9d61f93e282346
+ languageName: node
+ linkType: hard
+
+"events@npm:^3.3.0":
+ version: 3.3.0
+ resolution: "events@npm:3.3.0"
+ checksum: a3d47e285e28d324d7180f1e493961a2bbb4cad6412090e4dec114f4db1f5b560c7696ee8e758f55e23913ede856e3689cd3aa9ae13c56b5d8314cd3b3ddd1be
+ languageName: node
+ linkType: hard
+
"execa@npm:^6.1.0":
version: 6.1.0
resolution: "execa@npm:6.1.0"
@@ -4708,6 +5179,45 @@ __metadata:
languageName: node
linkType: hard
+"express@npm:^4.18.2":
+ version: 4.18.2
+ resolution: "express@npm:4.18.2"
+ dependencies:
+ accepts: "npm:~1.3.8"
+ array-flatten: "npm:1.1.1"
+ body-parser: "npm:1.20.1"
+ content-disposition: "npm:0.5.4"
+ content-type: "npm:~1.0.4"
+ cookie: "npm:0.5.0"
+ cookie-signature: "npm:1.0.6"
+ debug: "npm:2.6.9"
+ depd: "npm:2.0.0"
+ encodeurl: "npm:~1.0.2"
+ escape-html: "npm:~1.0.3"
+ etag: "npm:~1.8.1"
+ finalhandler: "npm:1.2.0"
+ fresh: "npm:0.5.2"
+ http-errors: "npm:2.0.0"
+ merge-descriptors: "npm:1.0.1"
+ methods: "npm:~1.1.2"
+ on-finished: "npm:2.4.1"
+ parseurl: "npm:~1.3.3"
+ path-to-regexp: "npm:0.1.7"
+ proxy-addr: "npm:~2.0.7"
+ qs: "npm:6.11.0"
+ range-parser: "npm:~1.2.1"
+ safe-buffer: "npm:5.2.1"
+ send: "npm:0.18.0"
+ serve-static: "npm:1.15.0"
+ setprototypeof: "npm:1.2.0"
+ statuses: "npm:2.0.1"
+ type-is: "npm:~1.6.18"
+ utils-merge: "npm:1.0.1"
+ vary: "npm:~1.1.2"
+ checksum: 869ae89ed6ff4bed7b373079dc58e5dddcf2915a2669b36037ff78c99d675ae930e5fe052b35c24f56557d28a023bb1cbe3e2f2fb87eaab96a1cedd7e597809d
+ languageName: node
+ linkType: hard
+
"extend@npm:^3.0.2":
version: 3.0.2
resolution: "extend@npm:3.0.2"
@@ -4715,6 +5225,13 @@ __metadata:
languageName: node
linkType: hard
+"fast-copy@npm:^3.0.0":
+ version: 3.0.1
+ resolution: "fast-copy@npm:3.0.1"
+ checksum: 2f655f1e84440f990cddf7895f0acce38b2eb090a27dc0f97f1654cd6f2e38f67d9603471856c2af13e0bfbdf04c2c0b8d446fee1dd1f6f485992e4cc4693c7a
+ languageName: node
+ linkType: hard
+
"fast-deep-equal@npm:^3.1.1, fast-deep-equal@npm:^3.1.3":
version: 3.1.3
resolution: "fast-deep-equal@npm:3.1.3"
@@ -4749,12 +5266,26 @@ __metadata:
languageName: node
linkType: hard
-"fastq@npm:^1.6.0":
- version: 1.16.0
- resolution: "fastq@npm:1.16.0"
- dependencies:
- reusify: "npm:^1.0.4"
- checksum: de151543aab9d91900ed5da88860c46987ece925c628df586fac664235f25e020ec20729e1c032edb5fd2520fd4aa5b537d69e39b689e65e82112cfbecb4479e
+"fast-redact@npm:^3.1.1":
+ version: 3.3.0
+ resolution: "fast-redact@npm:3.3.0"
+ checksum: a69c5cb52396eafc4f466f46864406cbd4a6ead6782caf74750ce817794829048baaa933ad98543e744dd54ffb4cddff71f3e75e465a86e3d887894e281ec154
+ languageName: node
+ linkType: hard
+
+"fast-safe-stringify@npm:^2.1.1":
+ version: 2.1.1
+ resolution: "fast-safe-stringify@npm:2.1.1"
+ checksum: dc1f063c2c6ac9533aee14d406441f86783a8984b2ca09b19c2fe281f9ff59d315298bc7bc22fd1f83d26fe19ef2f20e2ddb68e96b15040292e555c5ced0c1e4
+ languageName: node
+ linkType: hard
+
+"fastq@npm:^1.6.0":
+ version: 1.16.0
+ resolution: "fastq@npm:1.16.0"
+ dependencies:
+ reusify: "npm:^1.0.4"
+ checksum: de151543aab9d91900ed5da88860c46987ece925c628df586fac664235f25e020ec20729e1c032edb5fd2520fd4aa5b537d69e39b689e65e82112cfbecb4479e
languageName: node
linkType: hard
@@ -4795,6 +5326,21 @@ __metadata:
languageName: node
linkType: hard
+"finalhandler@npm:1.2.0":
+ version: 1.2.0
+ resolution: "finalhandler@npm:1.2.0"
+ dependencies:
+ debug: "npm:2.6.9"
+ encodeurl: "npm:~1.0.2"
+ escape-html: "npm:~1.0.3"
+ on-finished: "npm:2.4.1"
+ parseurl: "npm:~1.3.3"
+ statuses: "npm:2.0.1"
+ unpipe: "npm:~1.0.0"
+ checksum: 635718cb203c6d18e6b48dfbb6c54ccb08ea470e4f474ddcef38c47edcf3227feec316f886dd701235997d8af35240cae49856721ce18f539ad038665ebbf163
+ languageName: node
+ linkType: hard
+
"find-root@npm:^1.1.0":
version: 1.1.0
resolution: "find-root@npm:1.1.0"
@@ -4874,6 +5420,16 @@ __metadata:
languageName: node
linkType: hard
+"follow-redirects@npm:^1.0.0":
+ version: 1.15.4
+ resolution: "follow-redirects@npm:1.15.4"
+ peerDependenciesMeta:
+ debug:
+ optional: true
+ checksum: 2e8f5f259a6b02dfa8dc199e08431848a7c3beed32eb4c19945966164a52c89f07b86c3afcc32ebe4279cf0a960520e45a63013d6350309c5ec90133c5d9351a
+ languageName: node
+ linkType: hard
+
"for-each@npm:^0.3.3":
version: 0.3.3
resolution: "for-each@npm:0.3.3"
@@ -4893,10 +5449,17 @@ __metadata:
languageName: node
linkType: hard
-"form-data-encoder@npm:^4.0.2":
- version: 4.0.2
- resolution: "form-data-encoder@npm:4.0.2"
- checksum: 71a00a1e954267f8b87d4301936dda7177cfe18714a92930c87ebc132acb8b19ed12e9b9f8ad47af52c1cc582fd45c1771f66f850a6d319147731160f6b9b3fa
+"form-data-encoder@npm:1.7.2":
+ version: 1.7.2
+ resolution: "form-data-encoder@npm:1.7.2"
+ checksum: 227bf2cea083284411fd67472ccc22f5cb354ca92c00690e11ff5ed942d993c13ac99dea365046306200f8bd71e1a7858d2d99e236de694b806b1f374a4ee341
+ languageName: node
+ linkType: hard
+
+"form-data-encoder@npm:^2.1.2":
+ version: 2.1.4
+ resolution: "form-data-encoder@npm:2.1.4"
+ checksum: 3778e7db3c21457296e6fdbc4200642a6c01e8be9297256e845ee275f9ddaecb5f49bfb0364690ad216898c114ec59bf85f01ec823a70670b8067273415d62f6
languageName: node
linkType: hard
@@ -4911,6 +5474,27 @@ __metadata:
languageName: node
linkType: hard
+"form-data@npm:^4.0.0":
+ version: 4.0.0
+ resolution: "form-data@npm:4.0.0"
+ dependencies:
+ asynckit: "npm:^0.4.0"
+ combined-stream: "npm:^1.0.8"
+ mime-types: "npm:^2.1.12"
+ checksum: 7264aa760a8cf09482816d8300f1b6e2423de1b02bba612a136857413fdc96d7178298ced106817655facc6b89036c6e12ae31c9eb5bdc16aabf502ae8a5d805
+ languageName: node
+ linkType: hard
+
+"formdata-node@npm:^4.3.2":
+ version: 4.4.1
+ resolution: "formdata-node@npm:4.4.1"
+ dependencies:
+ node-domexception: "npm:1.0.0"
+ web-streams-polyfill: "npm:4.0.0-beta.3"
+ checksum: 29622f75533107c1bbcbe31fda683e6a55859af7f48ec354a9800591ce7947ed84cd3ef2b2fcb812047a884f17a1bac75ce098ffc17e23402cd373e49c1cd335
+ languageName: node
+ linkType: hard
+
"formdata-polyfill@npm:^4.0.10":
version: 4.0.10
resolution: "formdata-polyfill@npm:4.0.10"
@@ -4920,6 +5504,32 @@ __metadata:
languageName: node
linkType: hard
+"formidable@npm:^2.1.2":
+ version: 2.1.2
+ resolution: "formidable@npm:2.1.2"
+ dependencies:
+ dezalgo: "npm:^1.0.4"
+ hexoid: "npm:^1.0.0"
+ once: "npm:^1.4.0"
+ qs: "npm:^6.11.0"
+ checksum: d385180e0461f65e6f7b70452859fe1c32aa97a290c2ca33f00cdc33145ef44fa68bbc9b93af2c3af73ae726e42c3477c6619c49f3c34b49934e9481275b7b4c
+ languageName: node
+ linkType: hard
+
+"forwarded@npm:0.2.0":
+ version: 0.2.0
+ resolution: "forwarded@npm:0.2.0"
+ checksum: 29ba9fd347117144e97cbb8852baae5e8b2acb7d1b591ef85695ed96f5b933b1804a7fac4a15dd09ca7ac7d0cdc104410e8102aae2dd3faa570a797ba07adb81
+ languageName: node
+ linkType: hard
+
+"fresh@npm:0.5.2":
+ version: 0.5.2
+ resolution: "fresh@npm:0.5.2"
+ checksum: 64c88e489b5d08e2f29664eb3c79c705ff9a8eb15d3e597198ef76546d4ade295897a44abb0abd2700e7ef784b2e3cbf1161e4fbf16f59129193fd1030d16da1
+ languageName: node
+ linkType: hard
+
"from@npm:~0":
version: 0.1.7
resolution: "from@npm:0.1.7"
@@ -5001,6 +5611,13 @@ __metadata:
languageName: node
linkType: hard
+"functional-red-black-tree@npm:^1.0.1":
+ version: 1.0.1
+ resolution: "functional-red-black-tree@npm:1.0.1"
+ checksum: debe73e92204341d1fa5f89614e44284d3add26dee660722978d8c50829170f87d1c74768f68c251d215ae461c11db7bac13101c77f4146ff051da75466f7a12
+ languageName: node
+ linkType: hard
+
"functions-have-names@npm:^1.2.3":
version: 1.2.3
resolution: "functions-have-names@npm:1.2.3"
@@ -5017,7 +5634,7 @@ __metadata:
languageName: node
linkType: hard
-"gaxios@npm:^6.0.0, gaxios@npm:^6.1.1":
+"gaxios@npm:^6.0.0, gaxios@npm:^6.0.3, gaxios@npm:^6.1.1":
version: 6.1.1
resolution: "gaxios@npm:6.1.1"
dependencies:
@@ -5053,6 +5670,13 @@ __metadata:
languageName: node
linkType: hard
+"get-east-asian-width@npm:^1.0.0":
+ version: 1.2.0
+ resolution: "get-east-asian-width@npm:1.2.0"
+ checksum: c9b280e7c7c67fb89fa17e867c4a9d1c9f1321aba2a9ee27bff37fb6ca9552bccda328c70a80c1f83a0e39ba1b7e3427e60f47823402d19e7a41b83417ec047a
+ languageName: node
+ linkType: hard
+
"get-func-name@npm:^2.0.1, get-func-name@npm:^2.0.2":
version: 2.0.2
resolution: "get-func-name@npm:2.0.2"
@@ -5237,7 +5861,7 @@ __metadata:
languageName: node
linkType: hard
-"google-auth-library@npm:^9.0.0":
+"google-auth-library@npm:^9.0.0, google-auth-library@npm:^9.4.1":
version: 9.4.1
resolution: "google-auth-library@npm:9.4.1"
dependencies:
@@ -5251,7 +5875,7 @@ __metadata:
languageName: node
linkType: hard
-"google-gax@npm:^4.0.3":
+"google-gax@npm:^4.0.3, google-gax@npm:^4.0.4":
version: 4.0.5
resolution: "google-gax@npm:4.0.5"
dependencies:
@@ -5270,6 +5894,20 @@ __metadata:
languageName: node
linkType: hard
+"googleapis-common@npm:^7.0.0":
+ version: 7.0.1
+ resolution: "googleapis-common@npm:7.0.1"
+ dependencies:
+ extend: "npm:^3.0.2"
+ gaxios: "npm:^6.0.3"
+ google-auth-library: "npm:^9.0.0"
+ qs: "npm:^6.7.0"
+ url-template: "npm:^2.0.8"
+ uuid: "npm:^9.0.0"
+ checksum: ae44d20c81183d72bf3ce85bfca3915b5370310ce5387b7b8c5b3e9348b1eb14c5f2b8e187aa6d088cd7a64153b3f7b797e0e2ee2894068f0c295287edeaafe3
+ languageName: node
+ linkType: hard
+
"gopd@npm:^1.0.1":
version: 1.0.1
resolution: "gopd@npm:1.0.1"
@@ -5279,22 +5917,22 @@ __metadata:
languageName: node
linkType: hard
-"got@npm:^14.0.0":
- version: 14.0.0
- resolution: "got@npm:14.0.0"
+"got@npm:^13.0.0":
+ version: 13.0.0
+ resolution: "got@npm:13.0.0"
dependencies:
- "@sindresorhus/is": "npm:^6.1.0"
+ "@sindresorhus/is": "npm:^5.2.0"
"@szmarczak/http-timer": "npm:^5.0.1"
cacheable-lookup: "npm:^7.0.0"
- cacheable-request: "npm:^10.2.14"
+ cacheable-request: "npm:^10.2.8"
decompress-response: "npm:^6.0.0"
- form-data-encoder: "npm:^4.0.2"
- get-stream: "npm:^8.0.1"
- http2-wrapper: "npm:^2.2.1"
+ form-data-encoder: "npm:^2.1.2"
+ get-stream: "npm:^6.0.1"
+ http2-wrapper: "npm:^2.1.10"
lowercase-keys: "npm:^3.0.0"
- p-cancelable: "npm:^4.0.1"
+ p-cancelable: "npm:^3.0.0"
responselike: "npm:^3.0.0"
- checksum: 3afcf60c8313cacb3bdc1a153b9b48e6c974241dab09ceb9155bfd4f2aad161a32e9b2f541584c78e8ac2e9615b241e453d06cfa2a4518b7e3899b46948ab9f3
+ checksum: 35ac9fe37daca3d0a4f90305d8e64626268ef5a42584f5bcb42eea3cb9bbeb691cf9041d5ea72133a7295d1291684789a3148ff89a95f3d3ce3d0ebb6fb2f680
languageName: node
linkType: hard
@@ -5419,6 +6057,20 @@ __metadata:
languageName: node
linkType: hard
+"help-me@npm:^5.0.0":
+ version: 5.0.0
+ resolution: "help-me@npm:5.0.0"
+ checksum: 5f99bd91dae93d02867175c3856c561d7e3a24f16999b08f5fc79689044b938d7ed58457f4d8c8744c01403e6e0470b7896baa344d112b2355842fd935a75d69
+ languageName: node
+ linkType: hard
+
+"hexoid@npm:^1.0.0":
+ version: 1.0.0
+ resolution: "hexoid@npm:1.0.0"
+ checksum: f2271b8b6b0e13fb5a1eccf740f53ce8bae689c80b9498b854c447f9dc94f75f44e0de064c0e4660ecdbfa8942bb2b69973fdcb080187b45bbb409a3c71f19d4
+ languageName: node
+ linkType: hard
+
"hoist-non-react-statics@npm:^3.3.1":
version: 3.3.2
resolution: "hoist-non-react-statics@npm:3.3.2"
@@ -5458,6 +6110,19 @@ __metadata:
languageName: node
linkType: hard
+"http-errors@npm:2.0.0, http-errors@npm:^2.0.0":
+ version: 2.0.0
+ resolution: "http-errors@npm:2.0.0"
+ dependencies:
+ depd: "npm:2.0.0"
+ inherits: "npm:2.0.4"
+ setprototypeof: "npm:1.2.0"
+ statuses: "npm:2.0.1"
+ toidentifier: "npm:1.0.1"
+ checksum: 0e7f76ee8ff8a33e58a3281a469815b893c41357378f408be8f6d4aa7d1efafb0da064625518e7078381b6a92325949b119dc38fcb30bdbc4e3a35f78c44c439
+ languageName: node
+ linkType: hard
+
"http-parser-js@npm:>=0.5.1":
version: 0.5.8
resolution: "http-parser-js@npm:0.5.8"
@@ -5486,7 +6151,36 @@ __metadata:
languageName: node
linkType: hard
-"http2-wrapper@npm:^2.2.1":
+"http-proxy-middleware@npm:^2.0.6":
+ version: 2.0.6
+ resolution: "http-proxy-middleware@npm:2.0.6"
+ dependencies:
+ "@types/http-proxy": "npm:^1.17.8"
+ http-proxy: "npm:^1.18.1"
+ is-glob: "npm:^4.0.1"
+ is-plain-obj: "npm:^3.0.0"
+ micromatch: "npm:^4.0.2"
+ peerDependencies:
+ "@types/express": ^4.17.13
+ peerDependenciesMeta:
+ "@types/express":
+ optional: true
+ checksum: 768e7ae5a422bbf4b866b64105b4c2d1f468916b7b0e9c96750551c7732383069b411aa7753eb7b34eab113e4f77fb770122cb7fb9c8ec87d138d5ddaafda891
+ languageName: node
+ linkType: hard
+
+"http-proxy@npm:^1.18.1":
+ version: 1.18.1
+ resolution: "http-proxy@npm:1.18.1"
+ dependencies:
+ eventemitter3: "npm:^4.0.0"
+ follow-redirects: "npm:^1.0.0"
+ requires-port: "npm:^1.0.0"
+ checksum: 2489e98aba70adbfd8b9d41ed1ff43528be4598c88616c558b109a09eaffe4bb35e551b6c75ac42ed7d948bb7530a22a2be6ef4f0cecacb5927be139f4274594
+ languageName: node
+ linkType: hard
+
+"http2-wrapper@npm:^2.1.10":
version: 2.2.1
resolution: "http2-wrapper@npm:2.2.1"
dependencies:
@@ -5530,6 +6224,15 @@ __metadata:
languageName: node
linkType: hard
+"humanize-ms@npm:^1.2.1":
+ version: 1.2.1
+ resolution: "humanize-ms@npm:1.2.1"
+ dependencies:
+ ms: "npm:^2.0.0"
+ checksum: 9c7a74a2827f9294c009266c82031030eae811ca87b0da3dceb8d6071b9bde22c9f3daef0469c3c533cc67a97d8a167cd9fc0389350e5f415f61a79b171ded16
+ languageName: node
+ linkType: hard
+
"husky@npm:^8.0.3":
version: 8.0.3
resolution: "husky@npm:8.0.3"
@@ -5539,6 +6242,15 @@ __metadata:
languageName: node
linkType: hard
+"iconv-lite@npm:0.4.24":
+ version: 0.4.24
+ resolution: "iconv-lite@npm:0.4.24"
+ dependencies:
+ safer-buffer: "npm:>= 2.1.2 < 3"
+ checksum: 6d3a2dac6e5d1fb126d25645c25c3a1209f70cceecc68b8ef51ae0da3cdc078c151fade7524a30b12a3094926336831fca09c666ef55b37e2c69638b5d6bd2e3
+ languageName: node
+ linkType: hard
+
"iconv-lite@npm:0.6.3, iconv-lite@npm:^0.6.2, iconv-lite@npm:^0.6.3":
version: 0.6.3
resolution: "iconv-lite@npm:0.6.3"
@@ -5562,6 +6274,13 @@ __metadata:
languageName: node
linkType: hard
+"ieee754@npm:^1.2.1":
+ version: 1.2.1
+ resolution: "ieee754@npm:1.2.1"
+ checksum: d9f2557a59036f16c282aaeb107832dc957a93d73397d89bbad4eb1130560560eb695060145e8e6b3b498b15ab95510226649a0b8f52ae06583575419fe10fc4
+ languageName: node
+ linkType: hard
+
"ignore@npm:^5.2.0, ignore@npm:^5.2.4":
version: 5.3.0
resolution: "ignore@npm:5.3.0"
@@ -5620,7 +6339,7 @@ __metadata:
languageName: node
linkType: hard
-"inherits@npm:2, inherits@npm:^2.0.3":
+"inherits@npm:2, inherits@npm:2.0.4, inherits@npm:^2.0.3":
version: 2.0.4
resolution: "inherits@npm:2.0.4"
checksum: cd45e923bee15186c07fa4c89db0aace24824c482fb887b528304694b2aa6ff8a898da8657046a5dcf3e46cd6db6c61629551f9215f208d7c3f157cf9b290521
@@ -5645,6 +6364,13 @@ __metadata:
languageName: node
linkType: hard
+"ipaddr.js@npm:1.9.1":
+ version: 1.9.1
+ resolution: "ipaddr.js@npm:1.9.1"
+ checksum: 864d0cced0c0832700e9621913a6429ccdc67f37c1bd78fb8c6789fff35c9d167cb329134acad2290497a53336813ab4798d2794fd675d5eb33b5fdf0982b9ca
+ languageName: node
+ linkType: hard
+
"is-array-buffer@npm:^3.0.1, is-array-buffer@npm:^3.0.2":
version: 3.0.2
resolution: "is-array-buffer@npm:3.0.2"
@@ -5700,6 +6426,13 @@ __metadata:
languageName: node
linkType: hard
+"is-buffer@npm:~1.1.6":
+ version: 1.1.6
+ resolution: "is-buffer@npm:1.1.6"
+ checksum: f63da109e74bbe8947036ed529d43e4ae0c5fcd0909921dce4917ad3ea212c6a87c29f525ba1d17c0858c18331cf1046d4fc69ef59ed26896b25c8288a627133
+ languageName: node
+ linkType: hard
+
"is-callable@npm:^1.1.3, is-callable@npm:^1.1.4, is-callable@npm:^1.2.7":
version: 1.2.7
resolution: "is-callable@npm:1.2.7"
@@ -5773,6 +6506,13 @@ __metadata:
languageName: node
linkType: hard
+"is-interactive@npm:^2.0.0":
+ version: 2.0.0
+ resolution: "is-interactive@npm:2.0.0"
+ checksum: e8d52ad490bed7ae665032c7675ec07732bbfe25808b0efbc4d5a76b1a1f01c165f332775c63e25e9a03d319ebb6b24f571a9e902669fc1e40b0a60b5be6e26c
+ languageName: node
+ linkType: hard
+
"is-lambda@npm:^1.0.1":
version: 1.0.1
resolution: "is-lambda@npm:1.0.1"
@@ -5817,6 +6557,13 @@ __metadata:
languageName: node
linkType: hard
+"is-plain-obj@npm:^3.0.0":
+ version: 3.0.0
+ resolution: "is-plain-obj@npm:3.0.0"
+ checksum: a6ebdf8e12ab73f33530641972a72a4b8aed6df04f762070d823808303e4f76d87d5ea5bd76f96a7bbe83d93f04ac7764429c29413bd9049853a69cb630fb21c
+ languageName: node
+ linkType: hard
+
"is-regex@npm:^1.1.4":
version: 1.1.4
resolution: "is-regex@npm:1.1.4"
@@ -5884,6 +6631,20 @@ __metadata:
languageName: node
linkType: hard
+"is-unicode-supported@npm:^1.3.0":
+ version: 1.3.0
+ resolution: "is-unicode-supported@npm:1.3.0"
+ checksum: 20a1fc161afafaf49243551a5ac33b6c4cf0bbcce369fcd8f2951fbdd000c30698ce320de3ee6830497310a8f41880f8066d440aa3eb0a853e2aa4836dd89abc
+ languageName: node
+ linkType: hard
+
+"is-unicode-supported@npm:^2.0.0":
+ version: 2.0.0
+ resolution: "is-unicode-supported@npm:2.0.0"
+ checksum: 000b80639dedaf59a385f1c0a57f97a4d1435e0723716f24cc19ad94253a7a0a9f838bdc9ac49b10a29ac93b01f52ae9b2ed358a8876caf1eb74d73b4ede92b2
+ languageName: node
+ linkType: hard
+
"is-weakmap@npm:^2.0.1":
version: 2.0.1
resolution: "is-weakmap@npm:2.0.1"
@@ -5995,6 +6756,13 @@ __metadata:
languageName: node
linkType: hard
+"joycon@npm:^3.1.1":
+ version: 3.1.1
+ resolution: "joycon@npm:3.1.1"
+ checksum: 4b36e3479144ec196425f46b3618f8a96ce7e1b658f091a309cd4906215f5b7a402d7df331a3e0a09681381a658d0c5f039cb3cf6907e0a1e17ed847f5d37775
+ languageName: node
+ linkType: hard
+
"js-tokens@npm:^3.0.0 || ^4.0.0, js-tokens@npm:^4.0.0":
version: 4.0.0
resolution: "js-tokens@npm:4.0.0"
@@ -6281,6 +7049,16 @@ __metadata:
languageName: node
linkType: hard
+"log-symbols@npm:^6.0.0":
+ version: 6.0.0
+ resolution: "log-symbols@npm:6.0.0"
+ dependencies:
+ chalk: "npm:^5.3.0"
+ is-unicode-supported: "npm:^1.3.0"
+ checksum: 510cdda36700cbcd87a2a691ea08d310a6c6b449084018f7f2ec4f732ca5e51b301ff1327aadd96f53c08318e616276c65f7fe22f2a16704fb0715d788bc3c33
+ languageName: node
+ linkType: hard
+
"long@npm:^5.0.0":
version: 5.2.3
resolution: "long@npm:5.2.3"
@@ -6384,6 +7162,31 @@ __metadata:
languageName: node
linkType: hard
+"md5@npm:^2.3.0":
+ version: 2.3.0
+ resolution: "md5@npm:2.3.0"
+ dependencies:
+ charenc: "npm:0.0.2"
+ crypt: "npm:0.0.2"
+ is-buffer: "npm:~1.1.6"
+ checksum: 88dce9fb8df1a084c2385726dcc18c7f54e0b64c261b5def7cdfe4928c4ee1cd68695c34108b4fab7ecceb05838c938aa411c6143df9fdc0026c4ddb4e4e72fa
+ languageName: node
+ linkType: hard
+
+"media-typer@npm:0.3.0":
+ version: 0.3.0
+ resolution: "media-typer@npm:0.3.0"
+ checksum: 38e0984db39139604756903a01397e29e17dcb04207bb3e081412ce725ab17338ecc47220c1b186b6bbe79a658aad1b0d41142884f5a481f36290cdefbe6aa46
+ languageName: node
+ linkType: hard
+
+"merge-descriptors@npm:1.0.1":
+ version: 1.0.1
+ resolution: "merge-descriptors@npm:1.0.1"
+ checksum: 5abc259d2ae25bb06d19ce2b94a21632583c74e2a9109ee1ba7fd147aa7362b380d971e0251069f8b3eb7d48c21ac839e21fa177b335e82c76ec172e30c31a26
+ languageName: node
+ linkType: hard
+
"merge-stream@npm:^2.0.0":
version: 2.0.0
resolution: "merge-stream@npm:2.0.0"
@@ -6398,7 +7201,14 @@ __metadata:
languageName: node
linkType: hard
-"micromatch@npm:^4.0.4":
+"methods@npm:^1.1.2, methods@npm:~1.1.2":
+ version: 1.1.2
+ resolution: "methods@npm:1.1.2"
+ checksum: a385dd974faa34b5dd021b2bbf78c722881bf6f003bfe6d391d7da3ea1ed625d1ff10ddd13c57531f628b3e785be38d3eed10ad03cebd90b76932413df9a1820
+ languageName: node
+ linkType: hard
+
+"micromatch@npm:^4.0.2, micromatch@npm:^4.0.4":
version: 4.0.5
resolution: "micromatch@npm:4.0.5"
dependencies:
@@ -6415,7 +7225,7 @@ __metadata:
languageName: node
linkType: hard
-"mime-types@npm:^2.1.12":
+"mime-types@npm:^2.1.12, mime-types@npm:~2.1.24, mime-types@npm:~2.1.34":
version: 2.1.35
resolution: "mime-types@npm:2.1.35"
dependencies:
@@ -6424,6 +7234,24 @@ __metadata:
languageName: node
linkType: hard
+"mime@npm:1.6.0":
+ version: 1.6.0
+ resolution: "mime@npm:1.6.0"
+ bin:
+ mime: cli.js
+ checksum: b7d98bb1e006c0e63e2c91b590fe1163b872abf8f7ef224d53dd31499c2197278a6d3d0864c45239b1a93d22feaf6f9477e9fc847eef945838150b8c02d03170
+ languageName: node
+ linkType: hard
+
+"mime@npm:2.6.0":
+ version: 2.6.0
+ resolution: "mime@npm:2.6.0"
+ bin:
+ mime: cli.js
+ checksum: 7da117808b5cd0203bb1b5e33445c330fe213f4d8ee2402a84d62adbde9716ca4fb90dd6d9ab4e77a4128c6c5c24a9c4c9f6a4d720b095b1b342132d02dba58d
+ languageName: node
+ linkType: hard
+
"mime@npm:^3.0.0":
version: 3.0.0
resolution: "mime@npm:3.0.0"
@@ -6433,6 +7261,13 @@ __metadata:
languageName: node
linkType: hard
+"mimic-fn@npm:^2.1.0":
+ version: 2.1.0
+ resolution: "mimic-fn@npm:2.1.0"
+ checksum: d2421a3444848ce7f84bd49115ddacff29c15745db73f54041edc906c14b131a38d05298dae3081667627a59b2eb1ca4b436ff2e1b80f69679522410418b478a
+ languageName: node
+ linkType: hard
+
"mimic-fn@npm:^4.0.0":
version: 4.0.0
resolution: "mimic-fn@npm:4.0.0"
@@ -6616,6 +7451,13 @@ __metadata:
languageName: node
linkType: hard
+"ms@npm:2.0.0":
+ version: 2.0.0
+ resolution: "ms@npm:2.0.0"
+ checksum: 0e6a22b8b746d2e0b65a430519934fefd41b6db0682e3477c10f60c76e947c4c0ad06f63ffdf1d78d335f83edee8c0aa928aa66a36c7cd95b69b26f468d527f4
+ languageName: node
+ linkType: hard
+
"ms@npm:2.1.2":
version: 2.1.2
resolution: "ms@npm:2.1.2"
@@ -6623,7 +7465,7 @@ __metadata:
languageName: node
linkType: hard
-"ms@npm:^2.1.1":
+"ms@npm:2.1.3, ms@npm:^2.0.0, ms@npm:^2.1.1":
version: 2.1.3
resolution: "ms@npm:2.1.3"
checksum: aa92de608021b242401676e35cfa5aa42dd70cbdc082b916da7fb925c542173e36bce97ea3e804923fe92c0ad991434e4a38327e15a1b5b5f945d66df615ae6d
@@ -6648,6 +7490,15 @@ __metadata:
languageName: node
linkType: hard
+"nanoid@npm:^5.0.4":
+ version: 5.0.4
+ resolution: "nanoid@npm:5.0.4"
+ bin:
+ nanoid: bin/nanoid.js
+ checksum: cf09cca3774f3147100948f7478f75f4c9ee97a4af65c328dd9abbd83b12f8bb35cf9f89a21c330f3b759d667a4cd0140ed84aa5fdd522c61e0d341aeaa7fb6f
+ languageName: node
+ linkType: hard
+
"natural-compare@npm:^1.4.0":
version: 1.4.0
resolution: "natural-compare@npm:1.4.0"
@@ -6655,14 +7506,14 @@ __metadata:
languageName: node
linkType: hard
-"negotiator@npm:^0.6.3":
+"negotiator@npm:0.6.3, negotiator@npm:^0.6.3":
version: 0.6.3
resolution: "negotiator@npm:0.6.3"
checksum: 2723fb822a17ad55c93a588a4bc44d53b22855bf4be5499916ca0cab1e7165409d0b288ba2577d7b029f10ce18cf2ed8e703e5af31c984e1e2304277ef979837
languageName: node
linkType: hard
-"node-domexception@npm:^1.0.0":
+"node-domexception@npm:1.0.0, node-domexception@npm:^1.0.0":
version: 1.0.0
resolution: "node-domexception@npm:1.0.0"
checksum: e332522f242348c511640c25a6fc7da4f30e09e580c70c6b13cb0be83c78c3e71c8d4665af2527e869fc96848924a4316ae7ec9014c091e2156f41739d4fa233
@@ -6680,7 +7531,7 @@ __metadata:
languageName: node
linkType: hard
-"node-fetch@npm:^2.6.1, node-fetch@npm:^2.6.9":
+"node-fetch@npm:^2.6.1, node-fetch@npm:^2.6.7, node-fetch@npm:^2.6.9":
version: 2.7.0
resolution: "node-fetch@npm:2.7.0"
dependencies:
@@ -6894,7 +7745,23 @@ __metadata:
languageName: node
linkType: hard
-"once@npm:^1.3.0, once@npm:^1.4.0":
+"on-exit-leak-free@npm:^2.1.0":
+ version: 2.1.2
+ resolution: "on-exit-leak-free@npm:2.1.2"
+ checksum: f7b4b7200026a08f6e4a17ba6d72e6c5cbb41789ed9cf7deaf9d9e322872c7dc5a7898549a894651ee0ee9ae635d34a678115bf8acdfba8ebd2ba2af688b563c
+ languageName: node
+ linkType: hard
+
+"on-finished@npm:2.4.1":
+ version: 2.4.1
+ resolution: "on-finished@npm:2.4.1"
+ dependencies:
+ ee-first: "npm:1.1.1"
+ checksum: 8e81472c5028125c8c39044ac4ab8ba51a7cdc19a9fbd4710f5d524a74c6d8c9ded4dd0eed83f28d3d33ac1d7a6a439ba948ccb765ac6ce87f30450a26bfe2ea
+ languageName: node
+ linkType: hard
+
+"once@npm:^1.3.0, once@npm:^1.3.1, once@npm:^1.4.0":
version: 1.4.0
resolution: "once@npm:1.4.0"
dependencies:
@@ -6903,6 +7770,15 @@ __metadata:
languageName: node
linkType: hard
+"onetime@npm:^5.1.0":
+ version: 5.1.2
+ resolution: "onetime@npm:5.1.2"
+ dependencies:
+ mimic-fn: "npm:^2.1.0"
+ checksum: e9fd0695a01cf226652f0385bf16b7a24153dbbb2039f764c8ba6d2306a8506b0e4ce570de6ad99c7a6eb49520743afdb66edd95ee979c1a342554ed49a9aadd
+ languageName: node
+ linkType: hard
+
"onetime@npm:^6.0.0":
version: 6.0.0
resolution: "onetime@npm:6.0.0"
@@ -6912,6 +7788,25 @@ __metadata:
languageName: node
linkType: hard
+"openai@npm:^4.24.1":
+ version: 4.24.1
+ resolution: "openai@npm:4.24.1"
+ dependencies:
+ "@types/node": "npm:^18.11.18"
+ "@types/node-fetch": "npm:^2.6.4"
+ abort-controller: "npm:^3.0.0"
+ agentkeepalive: "npm:^4.2.1"
+ digest-fetch: "npm:^1.3.0"
+ form-data-encoder: "npm:1.7.2"
+ formdata-node: "npm:^4.3.2"
+ node-fetch: "npm:^2.6.7"
+ web-streams-polyfill: "npm:^3.2.1"
+ bin:
+ openai: bin/cli
+ checksum: 30769c66e62129bfe0e5caedb760d9885019cf69bea9d8bcd370e88b6c32311c81a4d895e84ada96da035b6710dd880d67b747a630369c4e3c2b2a08bc8484e9
+ languageName: node
+ linkType: hard
+
"optionator@npm:^0.9.3":
version: 0.9.3
resolution: "optionator@npm:0.9.3"
@@ -6926,10 +7821,27 @@ __metadata:
languageName: node
linkType: hard
-"p-cancelable@npm:^4.0.1":
- version: 4.0.1
- resolution: "p-cancelable@npm:4.0.1"
- checksum: 64de7b0be4c8bacc006488e0e90aa66fbcceb4da4f6fb84584573145f015f9650fe6ac26470897b3e82a3b528f6c60ea276b84cc315e35c45e9f12dec062a295
+"ora@npm:^8.0.1":
+ version: 8.0.1
+ resolution: "ora@npm:8.0.1"
+ dependencies:
+ chalk: "npm:^5.3.0"
+ cli-cursor: "npm:^4.0.0"
+ cli-spinners: "npm:^2.9.2"
+ is-interactive: "npm:^2.0.0"
+ is-unicode-supported: "npm:^2.0.0"
+ log-symbols: "npm:^6.0.0"
+ stdin-discarder: "npm:^0.2.1"
+ string-width: "npm:^7.0.0"
+ strip-ansi: "npm:^7.1.0"
+ checksum: 3d37bb3f53e965e5176004af319f82feef7323ee0b2428db5ee6f689b9b9ba939d7b1e81691d4614333c4fb9e294790eb049db9c1e990b14b9bbe150c6f09993
+ languageName: node
+ linkType: hard
+
+"p-cancelable@npm:^3.0.0":
+ version: 3.0.0
+ resolution: "p-cancelable@npm:3.0.0"
+ checksum: a5eab7cf5ac5de83222a014eccdbfde65ecfb22005ee9bc242041f0b4441e07fac7629432c82f48868aa0f8413fe0df6c6067c16f76bf9217cd8dc651923c93d
languageName: node
linkType: hard
@@ -7047,6 +7959,13 @@ __metadata:
languageName: node
linkType: hard
+"parseurl@npm:~1.3.3":
+ version: 1.3.3
+ resolution: "parseurl@npm:1.3.3"
+ checksum: 407cee8e0a3a4c5cd472559bca8b6a45b82c124e9a4703302326e9ab60fc1081442ada4e02628efef1eb16197ddc7f8822f5a91fd7d7c86b51f530aedb17dfa2
+ languageName: node
+ linkType: hard
+
"path-exists@npm:^4.0.0":
version: 4.0.0
resolution: "path-exists@npm:4.0.0"
@@ -7099,6 +8018,13 @@ __metadata:
languageName: node
linkType: hard
+"path-to-regexp@npm:0.1.7":
+ version: 0.1.7
+ resolution: "path-to-regexp@npm:0.1.7"
+ checksum: 701c99e1f08e3400bea4d701cf6f03517474bb1b608da71c78b1eb261415b645c5670dfae49808c89e12cea2dccd113b069f040a80de012da0400191c6dbd1c8
+ languageName: node
+ linkType: hard
+
"path-to-regexp@npm:^6.2.0":
version: 6.2.1
resolution: "path-to-regexp@npm:6.2.1"
@@ -7150,6 +8076,80 @@ __metadata:
languageName: node
linkType: hard
+"pino-abstract-transport@npm:^1.0.0, pino-abstract-transport@npm:v1.1.0":
+ version: 1.1.0
+ resolution: "pino-abstract-transport@npm:1.1.0"
+ dependencies:
+ readable-stream: "npm:^4.0.0"
+ split2: "npm:^4.0.0"
+ checksum: 39b4496c9e4289e8d44a1d01adfa8dfeebb374e14b7a6451a4f3713561aeb9e181c64ff0272921667abcb95aceb312ab2761b82e253db23a456ab3dd35a42675
+ languageName: node
+ linkType: hard
+
+"pino-http@npm:^9.0.0":
+ version: 9.0.0
+ resolution: "pino-http@npm:9.0.0"
+ dependencies:
+ get-caller-file: "npm:^2.0.5"
+ pino: "npm:^8.17.1"
+ pino-std-serializers: "npm:^6.2.2"
+ process-warning: "npm:^3.0.0"
+ checksum: 12a0b016c62c1f4e7d506f6a690d10b7ccdbb5868fba8156a1bf515143e522087f87ce83a732615773e903271468c308d5f5eb908ca5446208803da119025fdf
+ languageName: node
+ linkType: hard
+
+"pino-pretty@npm:^10.3.1":
+ version: 10.3.1
+ resolution: "pino-pretty@npm:10.3.1"
+ dependencies:
+ colorette: "npm:^2.0.7"
+ dateformat: "npm:^4.6.3"
+ fast-copy: "npm:^3.0.0"
+ fast-safe-stringify: "npm:^2.1.1"
+ help-me: "npm:^5.0.0"
+ joycon: "npm:^3.1.1"
+ minimist: "npm:^1.2.6"
+ on-exit-leak-free: "npm:^2.1.0"
+ pino-abstract-transport: "npm:^1.0.0"
+ pump: "npm:^3.0.0"
+ readable-stream: "npm:^4.0.0"
+ secure-json-parse: "npm:^2.4.0"
+ sonic-boom: "npm:^3.0.0"
+ strip-json-comments: "npm:^3.1.1"
+ bin:
+ pino-pretty: bin.js
+ checksum: 4284f125f7e8a5a10e856c8fd591ba34c30c0a0071a0b265a9eda43c3e447ba11d40b06cc67108675586358a5d1213a6ac3a92f6abd2896abfbab9a5b4c17072
+ languageName: node
+ linkType: hard
+
+"pino-std-serializers@npm:^6.0.0, pino-std-serializers@npm:^6.2.2":
+ version: 6.2.2
+ resolution: "pino-std-serializers@npm:6.2.2"
+ checksum: a00cdff4e1fbc206da9bed047e6dc400b065f43e8b4cef1635b0192feab0e8f932cdeb0faaa38a5d93d2e777ba4cda939c2ed4c1a70f6839ff25f9aef97c27ff
+ languageName: node
+ linkType: hard
+
+"pino@npm:^8.17.1, pino@npm:^8.17.2":
+ version: 8.17.2
+ resolution: "pino@npm:8.17.2"
+ dependencies:
+ atomic-sleep: "npm:^1.0.0"
+ fast-redact: "npm:^3.1.1"
+ on-exit-leak-free: "npm:^2.1.0"
+ pino-abstract-transport: "npm:v1.1.0"
+ pino-std-serializers: "npm:^6.0.0"
+ process-warning: "npm:^3.0.0"
+ quick-format-unescaped: "npm:^4.0.3"
+ real-require: "npm:^0.2.0"
+ safe-stable-stringify: "npm:^2.3.1"
+ sonic-boom: "npm:^3.7.0"
+ thread-stream: "npm:^2.0.0"
+ bin:
+ pino: bin.js
+ checksum: 90b74e4db3b3f8664b13def3eb4e39585a842396fd7a000ef00f2329076c889126f91bdaff0f653b9b5dd5a612dddde6ff87599ad046b47a264e8f7bfabb0ea8
+ languageName: node
+ linkType: hard
+
"pkg-types@npm:^1.0.3":
version: 1.0.3
resolution: "pkg-types@npm:1.0.3"
@@ -7213,6 +8213,20 @@ __metadata:
languageName: node
linkType: hard
+"process-warning@npm:^3.0.0":
+ version: 3.0.0
+ resolution: "process-warning@npm:3.0.0"
+ checksum: 2d82fa641e50a5789eaf0f2b33453760996e373d4591aac576a22d696186ab7e240a0592db86c264d4f28a46c2abbe9b94689752017db7dadc90f169f12b0924
+ languageName: node
+ linkType: hard
+
+"process@npm:^0.11.10":
+ version: 0.11.10
+ resolution: "process@npm:0.11.10"
+ checksum: dbaa7e8d1d5cf375c36963ff43116772a989ef2bb47c9bdee20f38fd8fc061119cf38140631cf90c781aca4d3f0f0d2c834711952b728953f04fd7d238f59f5b
+ languageName: node
+ linkType: hard
+
"promise-retry@npm:^2.0.1":
version: 2.0.1
resolution: "promise-retry@npm:2.0.1"
@@ -7243,7 +8257,7 @@ __metadata:
languageName: node
linkType: hard
-"protobufjs@npm:7.2.5, protobufjs@npm:^7.0.0, protobufjs@npm:^7.2.4":
+"protobufjs@npm:7.2.5, protobufjs@npm:^7.0.0, protobufjs@npm:^7.2.4, protobufjs@npm:^7.2.5":
version: 7.2.5
resolution: "protobufjs@npm:7.2.5"
dependencies:
@@ -7263,6 +8277,16 @@ __metadata:
languageName: node
linkType: hard
+"proxy-addr@npm:~2.0.7":
+ version: 2.0.7
+ resolution: "proxy-addr@npm:2.0.7"
+ dependencies:
+ forwarded: "npm:0.2.0"
+ ipaddr.js: "npm:1.9.1"
+ checksum: f24a0c80af0e75d31e3451398670d73406ec642914da11a2965b80b1898ca6f66a0e3e091a11a4327079b2b268795f6fa06691923fef91887215c3d0e8ea3f68
+ languageName: node
+ linkType: hard
+
"ps-tree@npm:^1.2.0":
version: 1.2.0
resolution: "ps-tree@npm:1.2.0"
@@ -7274,6 +8298,16 @@ __metadata:
languageName: node
linkType: hard
+"pump@npm:^3.0.0":
+ version: 3.0.0
+ resolution: "pump@npm:3.0.0"
+ dependencies:
+ end-of-stream: "npm:^1.1.0"
+ once: "npm:^1.3.1"
+ checksum: e42e9229fba14732593a718b04cb5e1cfef8254544870997e0ecd9732b189a48e1256e4e5478148ecb47c8511dca2b09eae56b4d0aad8009e6fac8072923cfc9
+ languageName: node
+ linkType: hard
+
"punycode@npm:^2.1.0":
version: 2.3.1
resolution: "punycode@npm:2.3.1"
@@ -7281,6 +8315,24 @@ __metadata:
languageName: node
linkType: hard
+"qs@npm:6.11.0":
+ version: 6.11.0
+ resolution: "qs@npm:6.11.0"
+ dependencies:
+ side-channel: "npm:^1.0.4"
+ checksum: 5a3bfea3e2f359ede1bfa5d2f0dbe54001aa55e40e27dc3e60fab814362d83a9b30758db057c2011b6f53a2d4e4e5150194b5bac45372652aecb3e3c0d4b256e
+ languageName: node
+ linkType: hard
+
+"qs@npm:^6.11.0, qs@npm:^6.7.0":
+ version: 6.11.2
+ resolution: "qs@npm:6.11.2"
+ dependencies:
+ side-channel: "npm:^1.0.4"
+ checksum: f2321d0796664d0f94e92447ccd3bdfd6b6f3a50b6b762aa79d7f5b1ea3a7a9f94063ba896b82bc2a877ed6a7426d4081e4f16568fdb04f0ee188cca9d8505b4
+ languageName: node
+ linkType: hard
+
"queue-microtask@npm:^1.2.2":
version: 1.2.3
resolution: "queue-microtask@npm:1.2.3"
@@ -7288,6 +8340,13 @@ __metadata:
languageName: node
linkType: hard
+"quick-format-unescaped@npm:^4.0.3":
+ version: 4.0.4
+ resolution: "quick-format-unescaped@npm:4.0.4"
+ checksum: 591eca457509a99368b623db05248c1193aa3cedafc9a077d7acab09495db1231017ba3ad1b5386e5633271edd0a03b312d8640a59ee585b8516a42e15438aa7
+ languageName: node
+ linkType: hard
+
"quick-lru@npm:^5.1.1":
version: 5.1.1
resolution: "quick-lru@npm:5.1.1"
@@ -7295,6 +8354,25 @@ __metadata:
languageName: node
linkType: hard
+"range-parser@npm:~1.2.1":
+ version: 1.2.1
+ resolution: "range-parser@npm:1.2.1"
+ checksum: ce21ef2a2dd40506893157970dc76e835c78cf56437e26e19189c48d5291e7279314477b06ac38abd6a401b661a6840f7b03bd0b1249da9b691deeaa15872c26
+ languageName: node
+ linkType: hard
+
+"raw-body@npm:2.5.1":
+ version: 2.5.1
+ resolution: "raw-body@npm:2.5.1"
+ dependencies:
+ bytes: "npm:3.1.2"
+ http-errors: "npm:2.0.0"
+ iconv-lite: "npm:0.4.24"
+ unpipe: "npm:1.0.0"
+ checksum: 280bedc12db3490ecd06f740bdcf66093a07535374b51331242382c0e130bb273ebb611b7bc4cba1b4b4e016cc7b1f4b05a6df885a6af39c2bc3b94c02291c84
+ languageName: node
+ linkType: hard
+
"react-dom@npm:^18.2.0":
version: 18.2.0
resolution: "react-dom@npm:18.2.0"
@@ -7410,6 +8488,19 @@ __metadata:
languageName: node
linkType: hard
+"readable-stream@npm:^4.0.0":
+ version: 4.5.2
+ resolution: "readable-stream@npm:4.5.2"
+ dependencies:
+ abort-controller: "npm:^3.0.0"
+ buffer: "npm:^6.0.3"
+ events: "npm:^3.3.0"
+ process: "npm:^0.11.10"
+ string_decoder: "npm:^1.3.0"
+ checksum: 01b128a559c5fd76a898495f858cf0a8839f135e6a69e3409f986e88460134791657eb46a2ff16826f331682a3c4d0c5a75cef5e52ef259711021ba52b1c2e82
+ languageName: node
+ linkType: hard
+
"readdirp@npm:~3.6.0":
version: 3.6.0
resolution: "readdirp@npm:3.6.0"
@@ -7419,6 +8510,13 @@ __metadata:
languageName: node
linkType: hard
+"real-require@npm:^0.2.0":
+ version: 0.2.0
+ resolution: "real-require@npm:0.2.0"
+ checksum: ddf44ee76301c774e9c9f2826da8a3c5c9f8fc87310f4a364e803ef003aa1a43c378b4323051ced212097fff1af459070f4499338b36a7469df1d4f7e8c0ba4c
+ languageName: node
+ linkType: hard
+
"reflect.getprototypeof@npm:^1.0.4":
version: 1.0.4
resolution: "reflect.getprototypeof@npm:1.0.4"
@@ -7467,6 +8565,13 @@ __metadata:
languageName: node
linkType: hard
+"requires-port@npm:^1.0.0":
+ version: 1.0.0
+ resolution: "requires-port@npm:1.0.0"
+ checksum: 878880ee78ccdce372784f62f52a272048e2d0827c29ae31e7f99da18b62a2b9463ea03a75f277352f4697c100183debb0532371ad515a2d49d4bfe596dd4c20
+ languageName: node
+ linkType: hard
+
"resolve-alpn@npm:^1.2.0":
version: 1.2.1
resolution: "resolve-alpn@npm:1.2.1"
@@ -7563,6 +8668,16 @@ __metadata:
languageName: node
linkType: hard
+"restore-cursor@npm:^4.0.0":
+ version: 4.0.0
+ resolution: "restore-cursor@npm:4.0.0"
+ dependencies:
+ onetime: "npm:^5.1.0"
+ signal-exit: "npm:^3.0.2"
+ checksum: 5b675c5a59763bf26e604289eab35711525f11388d77f409453904e1e69c0d37ae5889295706b2c81d23bd780165084d040f9b68fffc32cc921519031c4fa4af
+ languageName: node
+ linkType: hard
+
"retry-request@npm:^7.0.0":
version: 7.0.1
resolution: "retry-request@npm:7.0.1"
@@ -7697,8 +8812,8 @@ __metadata:
"@emotion/babel-plugin": "npm:^11.11.0"
"@emotion/eslint-plugin": "npm:^11.11.0"
"@types/eslint": "npm:^8.56.1"
- "@typescript-eslint/eslint-plugin": "npm:^6.17.0"
- "@typescript-eslint/parser": "npm:^6.17.0"
+ "@typescript-eslint/eslint-plugin": "npm:^6.18.0"
+ "@typescript-eslint/parser": "npm:^6.18.0"
eslint: "npm:^8.56.0"
eslint-config-prettier: "npm:^9.1.0"
eslint-import-resolver-typescript: "npm:^3.6.1"
@@ -7739,7 +8854,7 @@ __metadata:
languageName: node
linkType: hard
-"safe-buffer@npm:>=5.1.0, safe-buffer@npm:^5.0.1, safe-buffer@npm:~5.2.0":
+"safe-buffer@npm:5.2.1, safe-buffer@npm:>=5.1.0, safe-buffer@npm:^5.0.1, safe-buffer@npm:~5.2.0":
version: 5.2.1
resolution: "safe-buffer@npm:5.2.1"
checksum: 32872cd0ff68a3ddade7a7617b8f4c2ae8764d8b7d884c651b74457967a9e0e886267d3ecc781220629c44a865167b61c375d2da6c720c840ecd73f45d5d9451
@@ -7757,7 +8872,14 @@ __metadata:
languageName: node
linkType: hard
-"safer-buffer@npm:>= 2.1.2 < 3.0.0":
+"safe-stable-stringify@npm:^2.3.1":
+ version: 2.4.3
+ resolution: "safe-stable-stringify@npm:2.4.3"
+ checksum: a6c192bbefe47770a11072b51b500ed29be7b1c15095371c1ee1dc13e45ce48ee3c80330214c56764d006c485b88bd0b24940d868948170dddc16eed312582d8
+ languageName: node
+ linkType: hard
+
+"safer-buffer@npm:>= 2.1.2 < 3, safer-buffer@npm:>= 2.1.2 < 3.0.0":
version: 2.1.2
resolution: "safer-buffer@npm:2.1.2"
checksum: 7eaf7a0cf37cc27b42fb3ef6a9b1df6e93a1c6d98c6c6702b02fe262d5fcbd89db63320793b99b21cb5348097d0a53de81bd5f4e8b86e20cc9412e3f1cfb4e83
@@ -7780,7 +8902,7 @@ __metadata:
dotenv: "npm:^16.3.1"
execa: "npm:^8.0.1"
get-port: "npm:^7.0.0"
- got: "npm:^14.0.0"
+ got: "npm:^13.0.0"
graphql: "npm:^16.8.1"
lodash-es: "npm:^4.17.21"
miniflare: "npm:^3.20231218.1"
@@ -7792,6 +8914,13 @@ __metadata:
languageName: unknown
linkType: soft
+"secure-json-parse@npm:^2.4.0":
+ version: 2.7.0
+ resolution: "secure-json-parse@npm:2.7.0"
+ checksum: 974386587060b6fc5b1ac06481b2f9dbbb0d63c860cc73dc7533f27835fdb67b0ef08762dbfef25625c15bc0a0c366899e00076cb0d556af06b71e22f1dede4c
+ languageName: node
+ linkType: hard
+
"selfsigned@npm:^2.0.1":
version: 2.4.1
resolution: "selfsigned@npm:2.4.1"
@@ -7811,7 +8940,7 @@ __metadata:
languageName: node
linkType: hard
-"semver@npm:^7.0.0, semver@npm:^7.3.5, semver@npm:^7.3.7, semver@npm:^7.5.4":
+"semver@npm:^7.0.0, semver@npm:^7.3.5, semver@npm:^7.3.7, semver@npm:^7.3.8, semver@npm:^7.5.4":
version: 7.5.4
resolution: "semver@npm:7.5.4"
dependencies:
@@ -7822,6 +8951,76 @@ __metadata:
languageName: node
linkType: hard
+"send@npm:0.18.0":
+ version: 0.18.0
+ resolution: "send@npm:0.18.0"
+ dependencies:
+ debug: "npm:2.6.9"
+ depd: "npm:2.0.0"
+ destroy: "npm:1.2.0"
+ encodeurl: "npm:~1.0.2"
+ escape-html: "npm:~1.0.3"
+ etag: "npm:~1.8.1"
+ fresh: "npm:0.5.2"
+ http-errors: "npm:2.0.0"
+ mime: "npm:1.6.0"
+ ms: "npm:2.1.3"
+ on-finished: "npm:2.4.1"
+ range-parser: "npm:~1.2.1"
+ statuses: "npm:2.0.1"
+ checksum: ec66c0ad109680ad8141d507677cfd8b4e40b9559de23191871803ed241718e99026faa46c398dcfb9250676076573bd6bfe5d0ec347f88f4b7b8533d1d391cb
+ languageName: node
+ linkType: hard
+
+"serve-static@npm:1.15.0":
+ version: 1.15.0
+ resolution: "serve-static@npm:1.15.0"
+ dependencies:
+ encodeurl: "npm:~1.0.2"
+ escape-html: "npm:~1.0.3"
+ parseurl: "npm:~1.3.3"
+ send: "npm:0.18.0"
+ checksum: 699b2d4c29807a51d9b5e0f24955346911437aebb0178b3c4833ad30d3eca93385ff9927254f5c16da345903cad39d9cd4a532198c95a5129cc4ed43911b15a4
+ languageName: node
+ linkType: hard
+
+"server@workspace:server":
+ version: 0.0.0-use.local
+ resolution: "server@workspace:server"
+ dependencies:
+ "@google-cloud/firestore": "npm:^7.1.0"
+ "@googleapis/identitytoolkit": "npm:^8.0.0"
+ "@trpc/server": "npm:^10.45.0"
+ "@types/express": "npm:^4.17.21"
+ "@types/http-errors": "npm:^2.0.4"
+ "@types/node": "npm:^20.10.7"
+ "@types/supertest": "npm:^6.0.2"
+ "@types/ws": "npm:^8.5.10"
+ db: "workspace:*"
+ envalid: "npm:^8.0.0"
+ envars: "npm:^1.0.2"
+ express: "npm:^4.18.2"
+ get-port: "npm:^7.0.0"
+ google-auth-library: "npm:^9.4.1"
+ got: "npm:^13.0.0"
+ http-errors: "npm:^2.0.0"
+ http-proxy-middleware: "npm:^2.0.6"
+ nanoid: "npm:^5.0.4"
+ openai: "npm:^4.24.1"
+ pino: "npm:^8.17.2"
+ pino-http: "npm:^9.0.0"
+ pino-pretty: "npm:^10.3.1"
+ supertest: "npm:^6.3.3"
+ type-fest: "npm:^4.9.0"
+ typescript: "npm:~5.3.3"
+ vite: "npm:~5.0.11"
+ vite-node: "npm:~1.1.3"
+ vitest: "npm:~1.1.3"
+ ws: "npm:^8.16.0"
+ zod: "npm:^3.22.4"
+ languageName: unknown
+ linkType: soft
+
"set-cookie-parser@npm:^2.4.8":
version: 2.6.0
resolution: "set-cookie-parser@npm:2.6.0"
@@ -7852,6 +9051,13 @@ __metadata:
languageName: node
linkType: hard
+"setprototypeof@npm:1.2.0":
+ version: 1.2.0
+ resolution: "setprototypeof@npm:1.2.0"
+ checksum: fde1630422502fbbc19e6844346778f99d449986b2f9cdcceb8326730d2f3d9964dbcb03c02aaadaefffecd0f2c063315ebea8b3ad895914bf1afc1747fc172e
+ languageName: node
+ linkType: hard
+
"shebang-command@npm:^2.0.0":
version: 2.0.0
resolution: "shebang-command@npm:2.0.0"
@@ -7886,7 +9092,7 @@ __metadata:
languageName: node
linkType: hard
-"signal-exit@npm:^3.0.7":
+"signal-exit@npm:^3.0.2, signal-exit@npm:^3.0.7":
version: 3.0.7
resolution: "signal-exit@npm:3.0.7"
checksum: a2f098f247adc367dffc27845853e9959b9e88b01cb301658cfe4194352d8d2bb32e18467c786a7fe15f1d44b233ea35633d076d5e737870b7139949d1ab6318
@@ -7942,6 +9148,15 @@ __metadata:
languageName: node
linkType: hard
+"sonic-boom@npm:^3.0.0, sonic-boom@npm:^3.7.0":
+ version: 3.8.0
+ resolution: "sonic-boom@npm:3.8.0"
+ dependencies:
+ atomic-sleep: "npm:^1.0.0"
+ checksum: 470a82cb1af3ab99fcd3003bbecb2ce79a6b243d0f6012c59e5f567f71cbe039c8cd810752748b5820ee20d72c8da81aa298e510eec9e41a4ca05c7f419825ff
+ languageName: node
+ linkType: hard
+
"source-map-js@npm:^1.0.2":
version: 1.0.2
resolution: "source-map-js@npm:1.0.2"
@@ -8004,6 +9219,13 @@ __metadata:
languageName: node
linkType: hard
+"split2@npm:^4.0.0":
+ version: 4.2.0
+ resolution: "split2@npm:4.2.0"
+ checksum: 09bbefc11bcf03f044584c9764cd31a252d8e52cea29130950b26161287c11f519807c5e54bd9e5804c713b79c02cefe6a98f4688630993386be353e03f534ab
+ languageName: node
+ linkType: hard
+
"split@npm:0.3":
version: 0.3.3
resolution: "split@npm:0.3.3"
@@ -8046,6 +9268,13 @@ __metadata:
languageName: node
linkType: hard
+"statuses@npm:2.0.1":
+ version: 2.0.1
+ resolution: "statuses@npm:2.0.1"
+ checksum: 18c7623fdb8f646fb213ca4051be4df7efb3484d4ab662937ca6fbef7ced9b9e12842709872eb3020cc3504b93bde88935c9f6417489627a7786f24f8031cbcb
+ languageName: node
+ linkType: hard
+
"std-env@npm:^3.5.0":
version: 3.7.0
resolution: "std-env@npm:3.7.0"
@@ -8053,6 +9282,13 @@ __metadata:
languageName: node
linkType: hard
+"stdin-discarder@npm:^0.2.1":
+ version: 0.2.2
+ resolution: "stdin-discarder@npm:0.2.2"
+ checksum: 642ffd05bd5b100819d6b24a613d83c6e3857c6de74eb02fc51506fa61dc1b0034665163831873868157c4538d71e31762bcf319be86cea04c3aba5336470478
+ languageName: node
+ linkType: hard
+
"stoppable@npm:^1.1.0":
version: 1.1.0
resolution: "stoppable@npm:1.1.0"
@@ -8114,6 +9350,17 @@ __metadata:
languageName: node
linkType: hard
+"string-width@npm:^7.0.0":
+ version: 7.0.0
+ resolution: "string-width@npm:7.0.0"
+ dependencies:
+ emoji-regex: "npm:^10.3.0"
+ get-east-asian-width: "npm:^1.0.0"
+ strip-ansi: "npm:^7.1.0"
+ checksum: bc0de5700a2690895169fce447ec4ed44bc62de80312c2093d5606bfd48319bb88e48a99e97f269dff2bc9577448b91c26b3804c16e7d9b389699795e4655c3b
+ languageName: node
+ linkType: hard
+
"string.prototype.matchall@npm:^4.0.8":
version: 4.0.10
resolution: "string.prototype.matchall@npm:4.0.10"
@@ -8164,7 +9411,7 @@ __metadata:
languageName: node
linkType: hard
-"string_decoder@npm:^1.1.1":
+"string_decoder@npm:^1.1.1, string_decoder@npm:^1.3.0":
version: 1.3.0
resolution: "string_decoder@npm:1.3.0"
dependencies:
@@ -8182,7 +9429,7 @@ __metadata:
languageName: node
linkType: hard
-"strip-ansi@npm:^7.0.1":
+"strip-ansi@npm:^7.0.1, strip-ansi@npm:^7.1.0":
version: 7.1.0
resolution: "strip-ansi@npm:7.1.0"
dependencies:
@@ -8235,6 +9482,34 @@ __metadata:
languageName: node
linkType: hard
+"superagent@npm:^8.0.5":
+ version: 8.1.2
+ resolution: "superagent@npm:8.1.2"
+ dependencies:
+ component-emitter: "npm:^1.3.0"
+ cookiejar: "npm:^2.1.4"
+ debug: "npm:^4.3.4"
+ fast-safe-stringify: "npm:^2.1.1"
+ form-data: "npm:^4.0.0"
+ formidable: "npm:^2.1.2"
+ methods: "npm:^1.1.2"
+ mime: "npm:2.6.0"
+ qs: "npm:^6.11.0"
+ semver: "npm:^7.3.8"
+ checksum: 33d0072e051baf91c7d68131c70682a0650dd1bd0b8dfb6f88e5bdfcb02e18cc2b42a66e44b32fd405ac6bcf5fd57c6e267bf80e2a8ce57a18166a9d3a78f57d
+ languageName: node
+ linkType: hard
+
+"supertest@npm:^6.3.3":
+ version: 6.3.3
+ resolution: "supertest@npm:6.3.3"
+ dependencies:
+ methods: "npm:^1.1.2"
+ superagent: "npm:^8.0.5"
+ checksum: 7a7da05bf0f43bc891fd377f6ee192c6fa115f01034f522d4f53c6305ab8866c760a5b0e7aedb3e422224e35be00a75b283613c446651cc1d9e35f80d9b461b3
+ languageName: node
+ linkType: hard
+
"supports-color@npm:^5.3.0":
version: 5.5.0
resolution: "supports-color@npm:5.5.0"
@@ -8301,6 +9576,15 @@ __metadata:
languageName: node
linkType: hard
+"thread-stream@npm:^2.0.0":
+ version: 2.4.1
+ resolution: "thread-stream@npm:2.4.1"
+ dependencies:
+ real-require: "npm:^0.2.0"
+ checksum: baac5bf555912f216a2494bf3f66377733a843306cddd233b1c5ad63084307266f61af35d6122e3936c657836d5db4a14da34300cd25930e013943b807a29c9b
+ languageName: node
+ linkType: hard
+
"through@npm:2, through@npm:~2.3, through@npm:~2.3.1":
version: 2.3.8
resolution: "through@npm:2.3.8"
@@ -8345,6 +9629,13 @@ __metadata:
languageName: node
linkType: hard
+"toidentifier@npm:1.0.1":
+ version: 1.0.1
+ resolution: "toidentifier@npm:1.0.1"
+ checksum: 952c29e2a85d7123239b5cfdd889a0dde47ab0497f0913d70588f19c53f7e0b5327c95f4651e413c74b785147f9637b17410ac8c846d5d4a20a5a33eb6dc3a45
+ languageName: node
+ linkType: hard
+
"toml@npm:^3.0.0":
version: 3.0.0
resolution: "toml@npm:3.0.0"
@@ -8380,7 +9671,7 @@ __metadata:
languageName: node
linkType: hard
-"tslib@npm:^2.1.0, tslib@npm:^2.2.0":
+"tslib@npm:2.6.2, tslib@npm:^2.1.0, tslib@npm:^2.2.0":
version: 2.6.2
resolution: "tslib@npm:2.6.2"
checksum: bd26c22d36736513980091a1e356378e8b662ded04204453d353a7f34a4c21ed0afc59b5f90719d4ba756e581a162ecbf93118dc9c6be5acf70aa309188166ca
@@ -8417,13 +9708,23 @@ __metadata:
languageName: node
linkType: hard
-"type-fest@npm:^4.2.0":
+"type-fest@npm:^4.2.0, type-fest@npm:^4.9.0":
version: 4.9.0
resolution: "type-fest@npm:4.9.0"
checksum: 49acfb67999566a24d5604435c8cff786dfc26ebea5a2a343e14d437d34f30a55248f8e597b8f64446c344bb68ce14af68899f562cf66ca66c1e1a856b393259
languageName: node
linkType: hard
+"type-is@npm:~1.6.18":
+ version: 1.6.18
+ resolution: "type-is@npm:1.6.18"
+ dependencies:
+ media-typer: "npm:0.3.0"
+ mime-types: "npm:~2.1.24"
+ checksum: 0bd9eeae5efd27d98fd63519f999908c009e148039d8e7179a074f105362d4fcc214c38b24f6cda79c87e563cbd12083a4691381ed28559220d4a10c2047bed4
+ languageName: node
+ linkType: hard
+
"typed-array-buffer@npm:^1.0.0":
version: 1.0.0
resolution: "typed-array-buffer@npm:1.0.0"
@@ -8569,6 +9870,13 @@ __metadata:
languageName: node
linkType: hard
+"unpipe@npm:1.0.0, unpipe@npm:~1.0.0":
+ version: 1.0.0
+ resolution: "unpipe@npm:1.0.0"
+ checksum: 4fa18d8d8d977c55cb09715385c203197105e10a6d220087ec819f50cb68870f02942244f1017565484237f1f8c5d3cd413631b1ae104d3096f24fdfde1b4aa2
+ languageName: node
+ linkType: hard
+
"update-browserslist-db@npm:^1.0.13":
version: 1.0.13
resolution: "update-browserslist-db@npm:1.0.13"
@@ -8592,6 +9900,13 @@ __metadata:
languageName: node
linkType: hard
+"url-template@npm:^2.0.8":
+ version: 2.0.8
+ resolution: "url-template@npm:2.0.8"
+ checksum: fc6a4cf6c3c3c3d7f0a0bb4405c41b81934e583b454e52ace7b2e5d7ed32ec9c2970ff1826d240c5823955fcb13531a1fc4ff6ba4569b1886a2976665353e952
+ languageName: node
+ linkType: hard
+
"urlpattern-polyfill@npm:^4.0.3":
version: 4.0.3
resolution: "urlpattern-polyfill@npm:4.0.3"
@@ -8606,6 +9921,13 @@ __metadata:
languageName: node
linkType: hard
+"utils-merge@npm:1.0.1":
+ version: 1.0.1
+ resolution: "utils-merge@npm:1.0.1"
+ checksum: 5d6949693d58cb2e636a84f3ee1c6e7b2f9c16cb1d42d0ecb386d8c025c69e327205aa1c69e2868cc06a01e5e20681fbba55a4e0ed0cce913d60334024eae798
+ languageName: node
+ linkType: hard
+
"uuid@npm:^9.0.0":
version: 9.0.1
resolution: "uuid@npm:9.0.1"
@@ -8634,7 +9956,14 @@ __metadata:
languageName: node
linkType: hard
-"vite-node@npm:1.1.3":
+"vary@npm:~1.1.2":
+ version: 1.1.2
+ resolution: "vary@npm:1.1.2"
+ checksum: 31389debef15a480849b8331b220782230b9815a8e0dbb7b9a8369559aed2e9a7800cd904d4371ea74f4c3527db456dc8e7ac5befce5f0d289014dbdf47b2242
+ languageName: node
+ linkType: hard
+
+"vite-node@npm:1.1.3, vite-node@npm:~1.1.3":
version: 1.1.3
resolution: "vite-node@npm:1.1.3"
dependencies:
@@ -8765,7 +10094,14 @@ __metadata:
languageName: node
linkType: hard
-"web-streams-polyfill@npm:^3.0.3":
+"web-streams-polyfill@npm:4.0.0-beta.3":
+ version: 4.0.0-beta.3
+ resolution: "web-streams-polyfill@npm:4.0.0-beta.3"
+ checksum: dcdef67de57d83008f9dc330662b65ba4497315555dd0e4e7bcacb132ffdf8a830eaab8f74ad40a4a44f542461f51223f406e2a446ece1cc29927859b1405853
+ languageName: node
+ linkType: hard
+
+"web-streams-polyfill@npm:^3.0.3, web-streams-polyfill@npm:^3.2.1":
version: 3.3.2
resolution: "web-streams-polyfill@npm:3.3.2"
checksum: 0466050f40c3487a96dedb07c60581b07c334daedd1955a7380b9405b77934a5bf9e178411133abca5365f3533543e1876fa5a5b7082382b866d1e5d07385466
@@ -9026,7 +10362,7 @@ __metadata:
languageName: node
linkType: hard
-"ws@npm:^8.11.0, ws@npm:^8.2.2":
+"ws@npm:^8.11.0, ws@npm:^8.16.0, ws@npm:^8.2.2":
version: 8.16.0
resolution: "ws@npm:8.16.0"
peerDependencies: