diff --git a/ts/saas-nextforge-encore/.cursorrules.example b/ts/saas-nextforge-encore/.cursorrules.example new file mode 100644 index 00000000..21030ba2 --- /dev/null +++ b/ts/saas-nextforge-encore/.cursorrules.example @@ -0,0 +1,37 @@ +# [PROJECT NAME] + +## PROJECT DESCRIPTION +- [PROJECT DESCRIPTION - What is the goal of the project? What is the purpose of the project?] + +## AI AGENT ROLE +- [AI AGENT ROLE - What is the role of the AI agent? What is the goal of the AI agent? Example ↴] +- You are a senior software engineer with great experience in [PROJECT LANGUAGE] and [PROJECT TECHNOLOGY]. +- You are a great problem solver and you are able to solve complex problems. + +## CODING STYLE AND STRUCTURE +- [How do you want the agent to write the code? What is the coding style and structure?] +- Prefer iteration and modularization over code duplication +- Use descriptive variable names with auxiliary verbs +- Write concise, technical TypeScript code with accurate examples + +## Error Handling +- [How do you want the agent to handle errors?] +- Implement proper error boundaries +- Log errors appropriately for debugging +- Provide user-friendly error messages +- Handle network failures gracefully + +## Testing +- [How do you want the agent to handle testing?] +- Write unit tests for utilities and components +- Implement E2E tests for critical flows +- Test across different Chrome versions +- Test memory usage and performance + +## Security +- [How do you want the agent to handle security?] +- Implement Content Security Policy +- Sanitize user inputs +- Handle sensitive data properly +- Follow Chrome extension security best practices +- Implement proper CORS handling \ No newline at end of file diff --git a/ts/saas-nextforge-encore/.github/CODE_OF_CONDUCT.md b/ts/saas-nextforge-encore/.github/CODE_OF_CONDUCT.md new file mode 100644 index 00000000..18aeab32 --- /dev/null +++ b/ts/saas-nextforge-encore/.github/CODE_OF_CONDUCT.md @@ -0,0 +1,128 @@ +# Contributor Covenant Code of Conduct + +## Our Pledge + +We as members, contributors, and leaders pledge to make participation in our +community a harassment-free experience for everyone, regardless of age, body +size, visible or invisible disability, ethnicity, sex characteristics, gender +identity and expression, level of experience, education, socio-economic status, +nationality, personal appearance, race, religion, or sexual identity +and orientation. + +We pledge to act and interact in ways that contribute to an open, welcoming, +diverse, inclusive, and healthy community. + +## Our Standards + +Examples of behavior that contributes to a positive environment for our +community include: + +* Demonstrating empathy and kindness toward other people +* Being respectful of differing opinions, viewpoints, and experiences +* Giving and gracefully accepting constructive feedback +* Accepting responsibility and apologizing to those affected by our mistakes, + and learning from the experience +* Focusing on what is best not just for us as individuals, but for the + overall community + +Examples of unacceptable behavior include: + +* The use of sexualized language or imagery, and sexual attention or + advances of any kind +* Trolling, insulting or derogatory comments, and personal or political attacks +* Public or private harassment +* Publishing others' private information, such as a physical or email + address, without their explicit permission +* Other conduct which could reasonably be considered inappropriate in a + professional setting + +## Enforcement Responsibilities + +Community leaders are responsible for clarifying and enforcing our standards of +acceptable behavior and will take appropriate and fair corrective action in +response to any behavior that they deem inappropriate, threatening, offensive, +or harmful. + +Community leaders have the right and responsibility to remove, edit, or reject +comments, commits, code, wiki edits, issues, and other contributions that are +not aligned to this Code of Conduct, and will communicate reasons for moderation +decisions when appropriate. + +## Scope + +This Code of Conduct applies within all community spaces, and also applies when +an individual is officially representing the community in public spaces. +Examples of representing our community include using an official e-mail address, +posting via an official social media account, or acting as an appointed +representative at an online or offline event. + +## Enforcement + +Instances of abusive, harassing, or otherwise unacceptable behavior may be +reported to the community leaders responsible for enforcement at +haydenbleasel.com/contact. +All complaints will be reviewed and investigated promptly and fairly. + +All community leaders are obligated to respect the privacy and security of the +reporter of any incident. + +## Enforcement Guidelines + +Community leaders will follow these Community Impact Guidelines in determining +the consequences for any action they deem in violation of this Code of Conduct: + +### 1. Correction + +**Community Impact**: Use of inappropriate language or other behavior deemed +unprofessional or unwelcome in the community. + +**Consequence**: A private, written warning from community leaders, providing +clarity around the nature of the violation and an explanation of why the +behavior was inappropriate. A public apology may be requested. + +### 2. Warning + +**Community Impact**: A violation through a single incident or series +of actions. + +**Consequence**: A warning with consequences for continued behavior. No +interaction with the people involved, including unsolicited interaction with +those enforcing the Code of Conduct, for a specified period of time. This +includes avoiding interactions in community spaces as well as external channels +like social media. Violating these terms may lead to a temporary or +permanent ban. + +### 3. Temporary Ban + +**Community Impact**: A serious violation of community standards, including +sustained inappropriate behavior. + +**Consequence**: A temporary ban from any sort of interaction or public +communication with the community for a specified period of time. No public or +private interaction with the people involved, including unsolicited interaction +with those enforcing the Code of Conduct, is allowed during this period. +Violating these terms may lead to a permanent ban. + +### 4. Permanent Ban + +**Community Impact**: Demonstrating a pattern of violation of community +standards, including sustained inappropriate behavior, harassment of an +individual, or aggression toward or disparagement of classes of individuals. + +**Consequence**: A permanent ban from any sort of public interaction within +the community. + +## Attribution + +This Code of Conduct is adapted from the [Contributor Covenant][homepage], +version 2.0, available at +https://www.contributor-covenant.org/version/2/0/code_of_conduct.html. + +Community Impact Guidelines were inspired by [Mozilla's code of conduct +enforcement ladder](https://github.com/mozilla/diversity). + +[homepage]: https://www.contributor-covenant.org + +For answers to common questions about this code of conduct, see the FAQ at +https://www.contributor-covenant.org/faq. Translations are available at +https://www.contributor-covenant.org/translations. diff --git a/ts/saas-nextforge-encore/.github/ISSUE_TEMPLATE/bug_report.md b/ts/saas-nextforge-encore/.github/ISSUE_TEMPLATE/bug_report.md new file mode 100644 index 00000000..ec775d45 --- /dev/null +++ b/ts/saas-nextforge-encore/.github/ISSUE_TEMPLATE/bug_report.md @@ -0,0 +1,31 @@ +--- +name: Bug report +about: Create a report to help us improve +title: '' +labels: bug +assignees: '' + +--- + +**Describe the bug** +A clear and concise description of what the bug is. + +**next-forge version** +I am using version ... + +**To Reproduce** +Steps to reproduce the behavior: +1. Go to '...' +2. Click on '....' +3. Scroll down to '....' +4. See error + +**Expected behavior** +A clear and concise description of what you expected to happen. + +**Screenshots** +If applicable, add screenshots to help explain your problem. + +**Desktop (please complete the following information):** + - OS: [e.g. MacOS] + - Browser [e.g. chrome v130, safari] diff --git a/ts/saas-nextforge-encore/.github/ISSUE_TEMPLATE/feature_request.md b/ts/saas-nextforge-encore/.github/ISSUE_TEMPLATE/feature_request.md new file mode 100644 index 00000000..234ef254 --- /dev/null +++ b/ts/saas-nextforge-encore/.github/ISSUE_TEMPLATE/feature_request.md @@ -0,0 +1,17 @@ +--- +name: Feature request +about: Suggest an idea for this project +title: '' +labels: enhancement +assignees: '' + +--- + +**Is your feature request related to a problem? Please describe.** +A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] + +**Describe the solution you'd like** +A clear and concise description of what you want to happen. + +**Describe alternatives you've considered** +A clear and concise description of any alternative solutions or features you've considered. diff --git a/ts/saas-nextforge-encore/.github/copilot-instructions.md.example b/ts/saas-nextforge-encore/.github/copilot-instructions.md.example new file mode 100644 index 00000000..6a166377 --- /dev/null +++ b/ts/saas-nextforge-encore/.github/copilot-instructions.md.example @@ -0,0 +1,26 @@ +# Copilot Guidelines + +This project uses . + +## Project Structure +Structure of how project files are setup. Making changes to files should be in their respected file. +``` +| App | Description | +|-----------|-----------------------------------------------------------------------------| +| api | Contains serverless functions designed to run separately from the main app e.g. webhooks and cron jobs. | +| app | The main application, featuring a shadcn/ui template. | +| docs | The documentation, which contains the documentation for the app e.g. guides and tutorials. | +| email | The email preview server from react.email. | +| storybook | The storybook, which contains the storybook for the app. | +| studio | Prisma Studio, which is a graphical editor for the database. | +| web | The website, featuring a twblocks template. | +``` + +## Nesting +- Avoid deeply nested code. Break down logic into smaller functions. +- Opening curly braces should be on the same line as the statement. + +## Error Handling +- Always catch a specific error instead of a generic one. +- Log the error message and stack trace. + diff --git a/ts/saas-nextforge-encore/.github/dependabot.yml b/ts/saas-nextforge-encore/.github/dependabot.yml new file mode 100644 index 00000000..4bbf9212 --- /dev/null +++ b/ts/saas-nextforge-encore/.github/dependabot.yml @@ -0,0 +1,16 @@ +version: 2 +updates: + + # Maintain dependencies for GitHub Actions + - package-ecosystem: "github-actions" + directory: "/" + open-pull-requests-limit: 10 + schedule: + interval: "monthly" + + # Maintain dependencies for npm + - package-ecosystem: "npm" + directory: "/" + open-pull-requests-limit: 10 + schedule: + interval: "monthly" diff --git a/ts/saas-nextforge-encore/.github/pull_request_template.md b/ts/saas-nextforge-encore/.github/pull_request_template.md new file mode 100644 index 00000000..b20f2167 --- /dev/null +++ b/ts/saas-nextforge-encore/.github/pull_request_template.md @@ -0,0 +1,24 @@ +## Description + +Please provide a brief description of the changes introduced in this pull request. + +## Related Issues + +Closes # + +## Checklist + +- [ ] My code follows the code style of this project. +- [ ] I have performed a self-review of my code. +- [ ] I have commented my code, particularly in hard-to-understand areas. +- [ ] I have updated the documentation, if necessary. +- [ ] I have added tests that prove my fix is effective or my feature works. +- [ ] New and existing tests pass locally with my changes. + +## Screenshots (if applicable) + + + +## Additional Notes + + diff --git a/ts/saas-nextforge-encore/.gitignore b/ts/saas-nextforge-encore/.gitignore new file mode 100644 index 00000000..43518810 --- /dev/null +++ b/ts/saas-nextforge-encore/.gitignore @@ -0,0 +1,48 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# Dependencies +node_modules +.pnp +.pnp.js + +# Local env files +.env +.env.local +.env.development.local +.env.test.local +.env.production.local + +# Testing +coverage + +# Turbo +.turbo + +# Vercel +.vercel + +# Build Outputs +.next/ +out/ +build +dist + + +# Debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# Misc +.DS_Store +*.pem + +# Sentry Config File +.env.sentry-build-plugin + +# BaseHub +.basehub + +# AI Rules +.cursorrules +.github/copilot-instructions.md diff --git a/ts/saas-nextforge-encore/.vscode/extensions.json b/ts/saas-nextforge-encore/.vscode/extensions.json new file mode 100644 index 00000000..4b25b580 --- /dev/null +++ b/ts/saas-nextforge-encore/.vscode/extensions.json @@ -0,0 +1,10 @@ +{ + "recommendations": [ + "biomejs.biome", + "bradlc.vscode-tailwindcss", + "Prisma.prisma", + "unifiedjs.vscode-mdx", + "mikestead.dotenv", + "christian-kohler.npm-intellisense" + ] +} diff --git a/ts/saas-nextforge-encore/.vscode/launch.json b/ts/saas-nextforge-encore/.vscode/launch.json new file mode 100644 index 00000000..64381977 --- /dev/null +++ b/ts/saas-nextforge-encore/.vscode/launch.json @@ -0,0 +1,62 @@ +{ + "version": "0.2.0", + "configurations": [ + { + "name": "Next.js: debug server-side", + "type": "node-terminal", + "request": "launch", + "command": "pnpm dev" + }, + { + "name": "Next.js: debug client-side (app)", + "type": "chrome", + "request": "launch", + "url": "http://localhost:3000" + }, + { + "name": "Next.js: debug client-side (web)", + "type": "chrome", + "request": "launch", + "url": "http://localhost:3001" + }, + { + "name": "Next.js: debug client-side (api)", + "type": "chrome", + "request": "launch", + "url": "http://localhost:3002" + }, + { + "name": "Next.js: debug client-side (email)", + "type": "chrome", + "request": "launch", + "url": "http://localhost:3003" + }, + { + "name": "Next.js: debug client-side (app)", + "type": "chrome", + "request": "launch", + "url": "http://localhost:3004" + }, + { + "name": "Next.js: debug client-side (studio)", + "type": "chrome", + "request": "launch", + "url": "http://localhost:3005" + }, + { + "name": "Next.js: debug full stack", + "type": "node", + "request": "launch", + "program": "${workspaceFolder}/node_modules/.bin/next", + "runtimeArgs": ["--inspect"], + "skipFiles": ["/**"], + "serverReadyAction": { + "action": "debugWithEdge", + "killOnServerStop": true, + "pattern": "- Local:.+(https?://.+)", + "uriFormat": "%s", + "webRoot": "${workspaceFolder}" + } + } + ] +} diff --git a/ts/saas-nextforge-encore/.vscode/settings.json b/ts/saas-nextforge-encore/.vscode/settings.json new file mode 100644 index 00000000..e92051d0 --- /dev/null +++ b/ts/saas-nextforge-encore/.vscode/settings.json @@ -0,0 +1,28 @@ +{ + "[javascript]": { + "editor.defaultFormatter": "biomejs.biome" + }, + "[json]": { + "editor.defaultFormatter": "biomejs.biome" + }, + "[jsonc]": { + "editor.defaultFormatter": "biomejs.biome" + }, + "[typescript]": { + "editor.defaultFormatter": "biomejs.biome" + }, + "[typescriptreact]": { + "editor.defaultFormatter": "biomejs.biome" + }, + "editor.codeActionsOnSave": { + "quickfix.biome": "explicit", + "source.organizeImports.biome": "explicit" + }, + "editor.defaultFormatter": "biomejs.biome", + "editor.formatOnPaste": true, + "editor.formatOnSave": true, + "emmet.showExpandedAbbreviation": "never", + "prettier.enable": false, + "typescript.tsdk": "node_modules/typescript/lib", + "tailwindCSS.experimental.configFile": "frontend/packages/design-system/styles/globals.css" +} diff --git a/ts/saas-nextforge-encore/backend/.gitignore b/ts/saas-nextforge-encore/backend/.gitignore new file mode 100644 index 00000000..30a613ef --- /dev/null +++ b/ts/saas-nextforge-encore/backend/.gitignore @@ -0,0 +1,6 @@ +.encore +encore.gen.go +encore.gen.cue +/.encore +node_modules +/encore.gen diff --git a/ts/saas-nextforge-encore/backend/.yarnrc.yml b/ts/saas-nextforge-encore/backend/.yarnrc.yml new file mode 100644 index 00000000..3186f3f0 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/.yarnrc.yml @@ -0,0 +1 @@ +nodeLinker: node-modules diff --git a/ts/saas-nextforge-encore/backend/README.md b/ts/saas-nextforge-encore/backend/README.md new file mode 100644 index 00000000..e42dfb89 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/README.md @@ -0,0 +1,119 @@ +# URL Shortener Starter + +This is an Encore starter for a URL Shortener. It has two API endpoints and a PostgreSQL database to store the URL IDs +and retrieve the full URL given an ID. + +## Build from scratch with a tutorial + +If you prefer to built it yourself, check out the [tutorial](https://encore.dev/docs/ts/tutorials/rest-api) to learn how to build this application from scratch. + +## Prerequisites + +**Install Encore:** +- **macOS:** `brew install encoredev/tap/encore` +- **Linux:** `curl -L https://encore.dev/install.sh | bash` +- **Windows:** `iwr https://encore.dev/install.ps1 | iex` + +**Docker:** +1. [Install Docker](https://docker.com) +2. Start Docker + +## Create app + +Create a local app from this template: + +```bash +encore app create my-app-name --example=ts/url-shortener +``` + +## Run app locally + +Before running your application, make sure you have Docker installed and running. Then run this command from your application's root folder: + +```bash +encore run +``` + +## Using the API + +### url.shorten — Shortens a URL and adds it to the database + +```bash +curl 'http://127.0.0.1:4000/url' -d '{"url":"https://google.com"}' +``` + +### url.get — Gets a URL from the database using a short ID + +```bash +curl 'http://127.0.0.1:4000/url/:id' +``` + +### url.list — Lists all shortened URLs + +```bash +curl 'http://127.0.0.1:4000/url' +``` + +## Open the developer dashboard + +While `encore run` is running, open [http://localhost:9400](http://localhost:9400) to access Encore's [local developer dashboard](https://encore.dev/docs/ts/observability/dev-dash). + +Here you can see API docs, make requests in the API explorer, and view traces of the responses. + +## Using the API + +To see that your app is running, you can ping the API to shorten a url. + +```bash +curl 'http://localhost:4000/url' -d '{"url":"https://news.ycombinator.com"}' +``` + +When you ping the API, you will see traces and logs appearing in the local development dashboard: [http://localhost:9400](http://localhost:9400) + +## Connecting to databases + +You can connect to your databases via psql shell: + +```bash +encore db shell --env=local --superuser +``` + +Learn more in the [CLI docs](https://encore.dev/docs/ts/cli/cli-reference#database-management). + +## Deployment + +### Self-hosting + +See the [self-hosting instructions](https://encore.dev/docs/ts/self-host/build) for how to use `encore build docker` to create a Docker image and configure it. + +### Encore Cloud Platform + +Deploy your application to a free staging environment in Encore's development cloud using `git push encore`: + +```bash +git add -A . +git commit -m 'Commit message' +git push encore +``` + +You can also open your app in the [Cloud Dashboard](https://app.encore.dev) to integrate with GitHub, or connect your AWS/GCP account, enabling Encore to automatically handle cloud deployments for you. + +## Link to GitHub + +Follow these steps to link your app to GitHub: + +1. Create a GitHub repo, commit and push the app. +2. Open your app in the [Cloud Dashboard](https://app.encore.dev). +3. Go to **Settings ➔ GitHub** and click on **Link app to GitHub** to link your app to GitHub and select the repo you just created. +4. To configure Encore to automatically trigger deploys when you push to a specific branch name, go to the **Overview** page for your intended environment. Click on **Settings** and then in the section **Branch Push** configure the **Branch name** and hit **Save**. +5. Commit and push a change to GitHub to trigger a deploy. + +[Learn more in the docs](https://encore.dev/docs/platform/integrations/github) + +## Testing + +To run tests, configure the `test` command in your `package.json` to the test runner of your choice, and then use the command `encore test` from the CLI. The `encore test` command sets up all the necessary infrastructure in test mode before handing over to the test runner. [Learn more](https://encore.dev/docs/ts/develop/testing) + +```bash +encore test +``` diff --git a/ts/saas-nextforge-encore/backend/api/README.md b/ts/saas-nextforge-encore/backend/api/README.md new file mode 100644 index 00000000..a1ffc121 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/api/README.md @@ -0,0 +1,5 @@ +# API Service + +## Overview + +This service, named `api`, is designed to be the starting point for all API-related functionalities. As the service requirements grow larger, it can be split into separate services, each potentially using a clone of the same database. Encore makes such decisions very easy to implement. diff --git a/ts/saas-nextforge-encore/backend/api/database.ts b/ts/saas-nextforge-encore/backend/api/database.ts new file mode 100644 index 00000000..cc3c3e60 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/api/database.ts @@ -0,0 +1,22 @@ +import { PrismaClient } from '@prisma/client'; +import { SQLDatabase } from "encore.dev/storage/sqldb"; + +// Define a database named 'encore_prisma_test', using the database migrations +// in the "./prisma/migrations" folder (where prisma will generate their migrations). +// Set `source` to `prisma` to let Encore know that the migrations are generated by Prisma. +const DB = new SQLDatabase("encore_prisma_test", { + migrations: { + path: "./prisma/migrations", + source: "prisma", + }, +}); + +const prisma = new PrismaClient({ + datasources: { + db: { + url: DB.connectionString, + }, + }, +}); + +export { prisma }; diff --git a/ts/saas-nextforge-encore/backend/api/encore.service.ts b/ts/saas-nextforge-encore/backend/api/encore.service.ts new file mode 100644 index 00000000..e1d88c31 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/api/encore.service.ts @@ -0,0 +1,31 @@ +import { APIError, api } from 'encore.dev/api'; +import { Service } from 'encore.dev/service'; +import * as auth from '~encore/auth'; + +interface APIResponse { + session: any; + user: any; +} + +export const APITesting = api( + { + method: ['GET'], + path: '/api/me', + expose: true, + auth: true, + }, + async (): Promise<{ data: APIResponse }> => { + const authData = auth.getAuthData(); + if (!authData) { + throw APIError.unauthenticated('Unauthenticated'); + } + return { + data: { + session: authData.session, + user: authData.user, + }, + }; + } +); + +export default new Service('api'); diff --git a/ts/saas-nextforge-encore/backend/api/prisma/migrations/migration_lock.toml b/ts/saas-nextforge-encore/backend/api/prisma/migrations/migration_lock.toml new file mode 100644 index 00000000..fbffa92c --- /dev/null +++ b/ts/saas-nextforge-encore/backend/api/prisma/migrations/migration_lock.toml @@ -0,0 +1,3 @@ +# Please do not edit this file manually +# It should be added in your version-control system (i.e. Git) +provider = "postgresql" \ No newline at end of file diff --git a/ts/saas-nextforge-encore/backend/api/prisma/schema.prisma b/ts/saas-nextforge-encore/backend/api/prisma/schema.prisma new file mode 100644 index 00000000..acbbf103 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/api/prisma/schema.prisma @@ -0,0 +1,138 @@ +// This is your Prisma schema file, +// learn more about it in the docs: https://pris.ly/d/prisma-schema + +datasource db { + provider = "postgresql" + url = "postgresql://encore-example-amk2:shadow-cv0o572lu9j08ktm1jd0@127.0.0.1:9500/encore_prisma_test?sslmode=disable" + relationMode = "prisma" +} +// TODO: replace the url below with the output from running +// encore db conn-uri encore_prisma_test --shadow +generator client { + provider = "prisma-client-js" + binaryTargets = ["native", "debian-openssl-3.0.x"] +} + + +model Page { + id Int @id @default(autoincrement()) + email String @unique + name String? +} + +model User { + id String @id + name String + email String + emailVerified Boolean + image String? + createdAt DateTime + updatedAt DateTime + username String + displayUsername String + phoneNumber String? + phoneNumberVerified Boolean? + isAnonymous Boolean? + sessions Session[] + accounts Account[] + members Member[] + invitations Invitation[] + privateMetadata Json? + + @@unique([email]) + @@unique([username]) + @@unique([phoneNumber]) + @@map("user") +} + +model Session { + id String @id + expiresAt DateTime + token String + createdAt DateTime + updatedAt DateTime + ipAddress String? + userAgent String? + userId String + user User @relation(fields: [userId], references: [id], onDelete: Cascade) + activeOrganizationId String? + + @@unique([token]) + @@index([userId]) + @@map("session") +} + +model Account { + id String @id + accountId String + providerId String + userId String + user User @relation(fields: [userId], references: [id], onDelete: Cascade) + accessToken String? + refreshToken String? + idToken String? + accessTokenExpiresAt DateTime? + refreshTokenExpiresAt DateTime? + scope String? + password String? + createdAt DateTime + updatedAt DateTime + + @@index([userId]) + @@map("account") +} + +model Verification { + id String @id + identifier String + value String + expiresAt DateTime + createdAt DateTime? + updatedAt DateTime? + + @@map("verification") +} + +model Organization { + id String @id + name String + slug String? + logo String? + createdAt DateTime + metadata String? + members Member[] + invitations Invitation[] + + @@unique([slug]) + @@map("organization") +} + +model Member { + id String @id + organizationId String + organization Organization @relation(fields: [organizationId], references: [id], onDelete: Cascade) + userId String + user User @relation(fields: [userId], references: [id], onDelete: Cascade) + role String + createdAt DateTime + + @@index([organizationId]) + @@index([userId]) + @@map("member") +} + +model Invitation { + id String @id + organizationId String + organization Organization @relation(fields: [organizationId], references: [id], onDelete: Cascade) + email String + role String? + status String + expiresAt DateTime + inviterId String + user User @relation(fields: [inviterId], references: [id], onDelete: Cascade) + + @@index([organizationId]) + @@index([inviterId]) + @@map("invitation") +} diff --git a/ts/saas-nextforge-encore/backend/auth/.gitignore b/ts/saas-nextforge-encore/backend/auth/.gitignore new file mode 100644 index 00000000..11ddd8db --- /dev/null +++ b/ts/saas-nextforge-encore/backend/auth/.gitignore @@ -0,0 +1,3 @@ +node_modules +# Keep environment variables out of version control +.env diff --git a/ts/saas-nextforge-encore/backend/auth/better-auth.routes.ts b/ts/saas-nextforge-encore/backend/auth/better-auth.routes.ts new file mode 100644 index 00000000..0b20a0a7 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/auth/better-auth.routes.ts @@ -0,0 +1,533 @@ +/** + * WARNING: This file is generated automatically. Do not edit. use/create generator plugins to override. + */ +import { api } from "encore.dev/api"; +import { auth } from './encore.service.js'; + +interface SignInSocialParams { + callbackURL?: string + newUserCallbackURL?: string + errorCallbackURL?: string + provider: "apple" | "discord" | "facebook" | "github" | "microsoft" | "google" | "spotify" | "twitch" | "twitter" | "dropbox" | "linkedin" | "gitlab" | "tiktok" | "reddit" | "roblox" | "vk" + disableRedirect?: boolean + idToken?: any + scopes?: any[] + requestSignUp?: boolean +} +interface SignInSocialResponse { + session: string + user: any + url: string + redirect: boolean +} + + +// Sign in with a social provider +export const signInSocial = api( + { method: ["POST"], path: "/auth/sign-in/social", expose: true, tags: ["/sign-in/social"] }, + async (params: SignInSocialParams): Promise<{ data: SignInSocialResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.signInSocial(params) as { data: SignInSocialResponse }; + } +); + +interface CallbackOAuthParams { + id: string + state?: string + code?: string + device_id?: string + error?: string + error_description?: string +} + +// API endpoint +export const callbackOAuth = api( + { method: ["GET", "POST"], path: "/auth/callback/:id", expose: true, tags: ["/callback/:id"] }, + async (params: CallbackOAuthParams): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.callbackOAuth(params) as { data: any }; + } +); + +interface GetSessionResponse { + session?: any + user?: any +} + +// Get the current session +export const getSession = api( + { method: ["GET"], path: "/auth/get-session", expose: true, tags: ["/get-session"] }, + async (): Promise<{ data: GetSessionResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.getSession() as { data: GetSessionResponse }; + } +); + +interface SignOutResponse { + success?: boolean +} + +// Sign out the current user +export const signOut = api( + { method: ["POST"], path: "/auth/sign-out", expose: true, tags: ["/sign-out"] }, + async (): Promise<{ data: SignOutResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.signOut() as { data: SignOutResponse }; + } +); + +interface SignUpEmailParams { + name: string + email: string + password: string + username?: string + displayUsername?: string +} +interface SignUpEmailResponseUser { + id: string + email: string + name: string + image?: string | null | undefined + emailVerified: boolean + createdAt: Date + updatedAt: Date + username?: string + displayUsername?: string +} +interface SignUpEmailResponse { + token: null + user: SignUpEmailResponseUser +} + + +// Sign up a user using email and password +export const signUpEmail = api( + { method: ["POST"], path: "/auth/sign-up/email", expose: true, tags: ["/sign-up/email"] }, + async (params: SignUpEmailParams): Promise<{ data: SignUpEmailResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.signUpEmail(params) as { data: SignUpEmailResponse }; + } +); + +interface SignInEmailParams { + email: string + password: string + callbackURL?: string + rememberMe?: any +} +interface SignInEmailResponse { + user: any + url: string + redirect: boolean +} + + +// Sign in with email and password +export const signInEmail = api( + { method: ["POST"], path: "/auth/sign-in/email", expose: true, tags: ["/sign-in/email"] }, + async (params: SignInEmailParams): Promise<{ data: SignInEmailResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.signInEmail(params) as { data: SignInEmailResponse }; + } +); + +interface ForgetPasswordParams { + email: string + redirectTo?: string +} +interface ForgetPasswordResponse { + status?: boolean +} + + +// Send a password reset email to the user +export const forgetPassword = api( + { method: ["POST"], path: "/auth/forget-password", expose: true, tags: ["/forget-password"] }, + async (params: ForgetPasswordParams): Promise<{ data: ForgetPasswordResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.forgetPassword(params) as { data: ForgetPasswordResponse }; + } +); + +interface ResetPasswordParams { + newPassword: string + token?: string +} +interface ResetPasswordResponse { + status?: boolean +} + + +// Reset the password for a user +export const resetPassword = api( + { method: ["POST"], path: "/auth/reset-password", expose: true, tags: ["/reset-password"] }, + async (params: ResetPasswordParams): Promise<{ data: ResetPasswordResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.resetPassword(params) as { data: ResetPasswordResponse }; + } +); + +interface VerifyEmailParams { + token: string + callbackURL?: string +} +interface VerifyEmailResponse { + user: any + status: boolean +} + + +// Verify the email of the user +export const verifyEmail = api( + { method: ["GET"], path: "/auth/verify-email", expose: true, tags: ["/verify-email"] }, + async (_: VerifyEmailParams): Promise<{ data: VerifyEmailResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.verifyEmail() as { data: VerifyEmailResponse }; + } +); + +interface SendVerificationEmailParams { + email: string + callbackURL?: string +} +interface SendVerificationEmailResponse { + status?: boolean +} + + +// Send a verification email to the user +export const sendVerificationEmail = api( + { method: ["POST"], path: "/auth/send-verification-email", expose: true, tags: ["/send-verification-email"] }, + async (params: SendVerificationEmailParams): Promise<{ data: SendVerificationEmailResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.sendVerificationEmail(params) as { data: SendVerificationEmailResponse }; + } +); + +interface ChangeEmailParams { + newEmail: string + callbackURL?: string +} +interface ChangeEmailResponse { + user?: any + status?: boolean +} + + +// API endpoint +export const changeEmail = api( + { method: ["POST"], path: "/auth/change-email", expose: true, tags: ["/change-email"] }, + async (params: ChangeEmailParams): Promise<{ data: ChangeEmailResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.changeEmail(params) as { data: ChangeEmailResponse }; + } +); + +interface ChangePasswordParams { + newPassword: string + currentPassword: string + revokeOtherSessions?: boolean +} +interface ChangePasswordResponse { + user?: any +} + + +// Change the password of the user +export const changePassword = api( + { method: ["POST"], path: "/auth/change-password", expose: true, tags: ["/change-password"] }, + async (params: ChangePasswordParams): Promise<{ data: ChangePasswordResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.changePassword(params) as { data: ChangePasswordResponse }; + } +); + +interface UpdateUserParams { + name?: string + image?: string | null + username?: string + displayUsername?: string +} +interface UpdateUserResponse { + status: boolean +} + + +// Update the current user +export const updateUser = api( + { method: ["POST"], path: "/auth/update-user", expose: true, tags: ["/update-user"] }, + async (params: UpdateUserParams): Promise<{ data: UpdateUserResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.updateUser(params) as { data: UpdateUserResponse }; + } +); + +interface DeleteUserParams { + callbackURL?: string + password?: string + token?: string +} +interface DeleteUserResponse { + +} + + +// Delete the user +export const deleteUser = api( + { method: ["POST"], path: "/auth/delete-user", expose: true, tags: ["/delete-user"] }, + async (params: DeleteUserParams): Promise<{ data: DeleteUserResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.deleteUser(params) as { data: DeleteUserResponse }; + } +); + +interface ForgetPasswordCallbackParams { + token: string + callbackURL: string +} +interface ForgetPasswordCallbackResponse { + token?: string +} + + +// Redirects the user to the callback URL with the token +export const forgetPasswordCallback = api( + { method: ["GET"], path: "/auth/reset-password/:token", expose: true, tags: ["/reset-password/:token"] }, + async (_: ForgetPasswordCallbackParams): Promise<{ data: ForgetPasswordCallbackResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.forgetPasswordCallback() as { data: ForgetPasswordCallbackResponse }; + } +); + + +// List all active sessions for the user +export const listSessions = api( + { method: ["GET"], path: "/auth/list-sessions", expose: true, tags: ["/list-sessions"] }, + async (): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.listSessions() as { data: any }; + } +); + +interface RevokeSessionParams { + token: string +} + +// Revoke a single session +export const revokeSession = api( + { method: ["POST"], path: "/auth/revoke-session", expose: true, tags: ["/revoke-session"] }, + async (params: RevokeSessionParams): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.revokeSession(params) as { data: any }; + } +); + +interface RevokeSessionsResponse { + status: boolean +} + +// Revoke all sessions for the user +export const revokeSessions = api( + { method: ["POST"], path: "/auth/revoke-sessions", expose: true, tags: ["/revoke-sessions"] }, + async (): Promise<{ data: RevokeSessionsResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.revokeSessions() as { data: RevokeSessionsResponse }; + } +); + +interface RevokeOtherSessionsResponse { + status?: boolean +} + +// Revoke all other sessions for the user except the current one +export const revokeOtherSessions = api( + { method: ["POST"], path: "/auth/revoke-other-sessions", expose: true, tags: ["/revoke-other-sessions"] }, + async (): Promise<{ data: RevokeOtherSessionsResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.revokeOtherSessions() as { data: RevokeOtherSessionsResponse }; + } +); + +interface LinkSocialAccountParams { + callbackURL?: string + provider: "apple" | "discord" | "facebook" | "github" | "microsoft" | "google" | "spotify" | "twitch" | "twitter" | "dropbox" | "linkedin" | "gitlab" | "tiktok" | "reddit" | "roblox" | "vk" +} +interface LinkSocialAccountResponse { + url: string + redirect: boolean +} + + +// Link a social account to the user +export const linkSocialAccount = api( + { method: ["POST"], path: "/auth/link-social", expose: true, tags: ["/link-social"] }, + async (params: LinkSocialAccountParams): Promise<{ data: LinkSocialAccountResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.linkSocialAccount(params) as { data: LinkSocialAccountResponse }; + } +); + + +// List all accounts linked to the user +export const listUserAccounts = api( + { method: ["GET"], path: "/auth/list-accounts", expose: true, tags: ["/list-accounts"] }, + async (): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.listUserAccounts() as { data: any }; + } +); + +interface DeleteUserCallbackParams { + token: string + callbackURL?: string +} + +// API endpoint +export const deleteUserCallback = api( + { method: ["GET"], path: "/auth/delete-user/callback", expose: true, tags: ["/delete-user/callback"] }, + async (_: DeleteUserCallbackParams): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.deleteUserCallback() as { data: any }; + } +); + +interface UnlinkAccountParams { + providerId: string +} + +// API endpoint +export const unlinkAccount = api( + { method: ["POST"], path: "/auth/unlink-account", expose: true, tags: ["/unlink-account"] }, + async (params: UnlinkAccountParams): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.unlinkAccount(params) as { data: any }; + } +); + +interface SignInUsernameParams { + username: string + password: string + rememberMe?: boolean +} +interface SignInUsernameResponse { + user?: any + session?: any +} + + +// Sign in with username +export const signInUsername = api( + { method: ["POST"], path: "/auth/sign-in/username", expose: true, tags: ["/sign-in/username"] }, + async (params: SignInUsernameParams): Promise<{ data: SignInUsernameResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.signInUsername(params) as { data: SignInUsernameResponse }; + } +); + +interface CreateApiKeyParams { + name?: string + expiresIn: any + userId?: string + prefix?: string + remaining: any + metadata?: any + refillAmount?: number + refillInterval?: number + rateLimitTimeWindow?: number + rateLimitMax?: number + rateLimitEnabled?: boolean + permissions?: any +} + +// API endpoint +export const createApiKey = api( + { method: ["POST"], path: "/auth/api-key/create", expose: true, tags: ["/api-key/create"] }, + async (params: CreateApiKeyParams): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.createApiKey(params) as { data: any }; + } +); + +interface GetApiKeyParams { + id: string +} + +// API endpoint +export const getApiKey = api( + { method: ["GET"], path: "/auth/api-key/get", expose: true, tags: ["/api-key/get"] }, + async (_: GetApiKeyParams): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.getApiKey() as { data: any }; + } +); + +interface UpdateApiKeyParams { + keyId: string + userId?: string + name?: string + enabled?: boolean + remaining?: number + refillAmount?: number + refillInterval?: number + metadata?: any + expiresIn?: any + rateLimitEnabled?: boolean + rateLimitTimeWindow?: number + rateLimitMax?: number + permissions?: any +} + +// API endpoint +export const updateApiKey = api( + { method: ["POST"], path: "/auth/api-key/update", expose: true, tags: ["/api-key/update"] }, + async (params: UpdateApiKeyParams): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.updateApiKey(params) as { data: any }; + } +); + +interface DeleteApiKeyParams { + keyId: string +} + +// API endpoint +export const deleteApiKey = api( + { method: ["DELETE"], path: "/auth/api-key/delete", expose: true, tags: ["/api-key/delete"] }, + async (params: DeleteApiKeyParams): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.deleteApiKey(params) as { data: any }; + } +); + + +// API endpoint +export const listApiKeys = api( + { method: ["GET"], path: "/auth/api-key/list", expose: true, tags: ["/api-key/list"] }, + async (): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.listApiKeys() as { data: any }; + } +); + +interface OkResponse { + ok?: boolean +} + +// Check if the API is working +export const ok = api( + { method: ["GET"], path: "/auth/ok", expose: true, tags: ["/ok"] }, + async (): Promise<{ data: OkResponse }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.ok() as { data: OkResponse }; + } +); + + +// Displays an error page +export const error = api( + { method: ["GET"], path: "/auth/error", expose: true, tags: ["/error"] }, + async (): Promise<{ data: any }> => { + // Using "as" to ignore response inconsistency from OpenAPI, to be resolved with PR https://github.com/better-auth/better-auth/pull/1699 + return await auth.routeHandlers.error() as { data: any }; + } +); \ No newline at end of file diff --git a/ts/saas-nextforge-encore/backend/auth/database.ts b/ts/saas-nextforge-encore/backend/auth/database.ts new file mode 100644 index 00000000..cc3c3e60 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/auth/database.ts @@ -0,0 +1,22 @@ +import { PrismaClient } from '@prisma/client'; +import { SQLDatabase } from "encore.dev/storage/sqldb"; + +// Define a database named 'encore_prisma_test', using the database migrations +// in the "./prisma/migrations" folder (where prisma will generate their migrations). +// Set `source` to `prisma` to let Encore know that the migrations are generated by Prisma. +const DB = new SQLDatabase("encore_prisma_test", { + migrations: { + path: "./prisma/migrations", + source: "prisma", + }, +}); + +const prisma = new PrismaClient({ + datasources: { + db: { + url: DB.connectionString, + }, + }, +}); + +export { prisma }; diff --git a/ts/saas-nextforge-encore/backend/auth/encore.service.ts b/ts/saas-nextforge-encore/backend/auth/encore.service.ts new file mode 100644 index 00000000..9f359213 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/auth/encore.service.ts @@ -0,0 +1,91 @@ +import { prismaAdapter } from 'better-auth/adapters/prisma'; +import { apiKey, username } from 'better-auth/plugins'; +import { encoreBetterAuth } from 'encore-better-auth'; +import { currentRequest } from 'encore.dev'; +import { APIError, Gateway, type Header, api } from 'encore.dev/api'; +import { authHandler } from 'encore.dev/auth'; +import { Service } from 'encore.dev/service'; +import * as encoreAuth from '~encore/auth'; +import { prisma } from './database.js'; + +// ensure to export auth. +export const auth = encoreBetterAuth({ + currentRequest: currentRequest, + database: prismaAdapter(prisma, { + provider: 'sqlite', // or "mysql", "postgresql", ...etc + }), + emailAndPassword: { + enabled: true, + }, + basePath: '/auth', + plugins: [ + username(), + apiKey({ + rateLimit: { + enabled: true, + timeWindow: 1000 * 60 * 60 * 24, // 1 day + maxRequests: 10, // 10 requests per day + }, + }), + ], + // socialProviders: { + // facebook: { + // clientId: "", + // clientSecret: "", + // }, + // google: { + // clientId: "", + // clientSecret: "", + // }, + // }, + generateRoutes: true, + wrapResponse: true, // must be true. +}); + +// Encore will consider this directory and all its subdirectories as part of the "users" service. +// https://encore.dev/docs/ts/primitives/services +export default new Service('auth', { + middlewares: [...auth.middlewares], +}); + +interface AuthParams { + cookie: Header<'Cookie'>; +} + +interface AuthData { + userID: string; + user: any; + session: any; +} + +export const handler = authHandler(async (authdata) => { + return auth.getValidatedSession(authdata.cookie); +}); + +export const gateway = new Gateway({ authHandler: handler }); + +interface MeResponse { + session: any; + user: any; +} + +export const me = api( + { + method: ['GET'], + path: '/me', + expose: true, + auth: true, + }, + async (): Promise<{ data: MeResponse }> => { + const authData = encoreAuth.getAuthData(); + if (!authData) { + throw APIError.unauthenticated('Unauthenticated'); + } + return { + data: { + session: authData.session, + user: authData.user, + }, + }; + } +); diff --git a/ts/saas-nextforge-encore/backend/auth/prisma/migrations/migration_lock.toml b/ts/saas-nextforge-encore/backend/auth/prisma/migrations/migration_lock.toml new file mode 100644 index 00000000..fbffa92c --- /dev/null +++ b/ts/saas-nextforge-encore/backend/auth/prisma/migrations/migration_lock.toml @@ -0,0 +1,3 @@ +# Please do not edit this file manually +# It should be added in your version-control system (i.e. Git) +provider = "postgresql" \ No newline at end of file diff --git a/ts/saas-nextforge-encore/backend/auth/prisma/schema.prisma b/ts/saas-nextforge-encore/backend/auth/prisma/schema.prisma new file mode 100644 index 00000000..acbbf103 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/auth/prisma/schema.prisma @@ -0,0 +1,138 @@ +// This is your Prisma schema file, +// learn more about it in the docs: https://pris.ly/d/prisma-schema + +datasource db { + provider = "postgresql" + url = "postgresql://encore-example-amk2:shadow-cv0o572lu9j08ktm1jd0@127.0.0.1:9500/encore_prisma_test?sslmode=disable" + relationMode = "prisma" +} +// TODO: replace the url below with the output from running +// encore db conn-uri encore_prisma_test --shadow +generator client { + provider = "prisma-client-js" + binaryTargets = ["native", "debian-openssl-3.0.x"] +} + + +model Page { + id Int @id @default(autoincrement()) + email String @unique + name String? +} + +model User { + id String @id + name String + email String + emailVerified Boolean + image String? + createdAt DateTime + updatedAt DateTime + username String + displayUsername String + phoneNumber String? + phoneNumberVerified Boolean? + isAnonymous Boolean? + sessions Session[] + accounts Account[] + members Member[] + invitations Invitation[] + privateMetadata Json? + + @@unique([email]) + @@unique([username]) + @@unique([phoneNumber]) + @@map("user") +} + +model Session { + id String @id + expiresAt DateTime + token String + createdAt DateTime + updatedAt DateTime + ipAddress String? + userAgent String? + userId String + user User @relation(fields: [userId], references: [id], onDelete: Cascade) + activeOrganizationId String? + + @@unique([token]) + @@index([userId]) + @@map("session") +} + +model Account { + id String @id + accountId String + providerId String + userId String + user User @relation(fields: [userId], references: [id], onDelete: Cascade) + accessToken String? + refreshToken String? + idToken String? + accessTokenExpiresAt DateTime? + refreshTokenExpiresAt DateTime? + scope String? + password String? + createdAt DateTime + updatedAt DateTime + + @@index([userId]) + @@map("account") +} + +model Verification { + id String @id + identifier String + value String + expiresAt DateTime + createdAt DateTime? + updatedAt DateTime? + + @@map("verification") +} + +model Organization { + id String @id + name String + slug String? + logo String? + createdAt DateTime + metadata String? + members Member[] + invitations Invitation[] + + @@unique([slug]) + @@map("organization") +} + +model Member { + id String @id + organizationId String + organization Organization @relation(fields: [organizationId], references: [id], onDelete: Cascade) + userId String + user User @relation(fields: [userId], references: [id], onDelete: Cascade) + role String + createdAt DateTime + + @@index([organizationId]) + @@index([userId]) + @@map("member") +} + +model Invitation { + id String @id + organizationId String + organization Organization @relation(fields: [organizationId], references: [id], onDelete: Cascade) + email String + role String? + status String + expiresAt DateTime + inviterId String + user User @relation(fields: [inviterId], references: [id], onDelete: Cascade) + + @@index([organizationId]) + @@index([inviterId]) + @@map("invitation") +} diff --git a/ts/saas-nextforge-encore/backend/encore.app b/ts/saas-nextforge-encore/backend/encore.app new file mode 100644 index 00000000..b1796ef5 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/encore.app @@ -0,0 +1,4 @@ +{ + "id": "forge-next-encore-bnvi", + "lang": "typescript" +} diff --git a/ts/saas-nextforge-encore/backend/internal/analytics/posthog/index.ts b/ts/saas-nextforge-encore/backend/internal/analytics/posthog/index.ts new file mode 100644 index 00000000..0e25d445 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/internal/analytics/posthog/index.ts @@ -0,0 +1,9 @@ +import { PostHog } from 'posthog-node'; + +export const analytics = new PostHog('NEXT_PUBLIC_POSTHOG_KEY', { + host: 'NEXT_PUBLIC_POSTHOG_HOST', + + // Don't batch events and flush immediately - we're running in a serverless environment + flushAt: 1, + flushInterval: 0, +}); diff --git a/ts/saas-nextforge-encore/backend/package-lock.json b/ts/saas-nextforge-encore/backend/package-lock.json new file mode 100644 index 00000000..67dbaae2 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/package-lock.json @@ -0,0 +1,1361 @@ +{ + "name": "url-shortener", + "version": "0.0.1", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "url-shortener", + "version": "0.0.1", + "license": "MPL-2.0", + "dependencies": { + "encore.dev": "^1.46.6" + }, + "devDependencies": { + "@types/node": "^20.5.7", + "typescript": "^5.2.2", + "vitest": "^2.1.4" + }, + "optionalDependencies": { + "@rollup/rollup-linux-x64-gnu": "^4.13.0" + } + }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz", + "integrity": "sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.21.5.tgz", + "integrity": "sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz", + "integrity": "sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.21.5.tgz", + "integrity": "sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz", + "integrity": "sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz", + "integrity": "sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz", + "integrity": "sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz", + "integrity": "sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz", + "integrity": "sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz", + "integrity": "sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz", + "integrity": "sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz", + "integrity": "sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz", + "integrity": "sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz", + "integrity": "sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz", + "integrity": "sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz", + "integrity": "sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz", + "integrity": "sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz", + "integrity": "sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz", + "integrity": "sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz", + "integrity": "sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz", + "integrity": "sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz", + "integrity": "sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz", + "integrity": "sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz", + "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.27.2.tgz", + "integrity": "sha512-Tj+j7Pyzd15wAdSJswvs5CJzJNV+qqSUcr/aCD+jpQSBtXvGnV0pnrjoc8zFTe9fcKCatkpFpOO7yAzpO998HA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.27.2.tgz", + "integrity": "sha512-xsPeJgh2ThBpUqlLgRfiVYBEf/P1nWlWvReG+aBWfNv3XEBpa6ZCmxSVnxJgLgkNz4IbxpLy64h2gCmAAQLneQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.27.2.tgz", + "integrity": "sha512-KnXU4m9MywuZFedL35Z3PuwiTSn/yqRIhrEA9j+7OSkji39NzVkgxuxTYg5F8ryGysq4iFADaU5osSizMXhU2A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.27.2.tgz", + "integrity": "sha512-Hj77A3yTvUeCIx/Vi+4d4IbYhyTwtHj07lVzUgpUq9YpJSEiGJj4vXMKwzJ3w5zp5v3PFvpJNgc/J31smZey6g==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-freebsd-arm64": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.27.2.tgz", + "integrity": "sha512-RjgKf5C3xbn8gxvCm5VgKZ4nn0pRAIe90J0/fdHUsgztd3+Zesb2lm2+r6uX4prV2eUByuxJNdt647/1KPRq5g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-freebsd-x64": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.27.2.tgz", + "integrity": "sha512-duq21FoXwQtuws+V9H6UZ+eCBc7fxSpMK1GQINKn3fAyd9DFYKPJNcUhdIKOrMFjLEJgQskoMoiuizMt+dl20g==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.27.2.tgz", + "integrity": "sha512-6npqOKEPRZkLrMcvyC/32OzJ2srdPzCylJjiTJT2c0bwwSGm7nz2F9mNQ1WrAqCBZROcQn91Fno+khFhVijmFA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.27.2.tgz", + "integrity": "sha512-V9Xg6eXtgBtHq2jnuQwM/jr2mwe2EycnopO8cbOvpzFuySCGtKlPCI3Hj9xup/pJK5Q0388qfZZy2DqV2J8ftw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.27.2.tgz", + "integrity": "sha512-uCFX9gtZJoQl2xDTpRdseYuNqyKkuMDtH6zSrBTA28yTfKyjN9hQ2B04N5ynR8ILCoSDOrG/Eg+J2TtJ1e/CSA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.27.2.tgz", + "integrity": "sha512-/PU9P+7Rkz8JFYDHIi+xzHabOu9qEWR07L5nWLIUsvserrxegZExKCi2jhMZRd0ATdboKylu/K5yAXbp7fYFvA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.27.2.tgz", + "integrity": "sha512-eCHmol/dT5odMYi/N0R0HC8V8QE40rEpkyje/ZAXJYNNoSfrObOvG/Mn+s1F/FJyB7co7UQZZf6FuWnN6a7f4g==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.27.2.tgz", + "integrity": "sha512-DEP3Njr9/ADDln3kNi76PXonLMSSMiCir0VHXxmGSHxCxDfQ70oWjHcJGfiBugzaqmYdTC7Y+8Int6qbnxPBIQ==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.27.2.tgz", + "integrity": "sha512-NHGo5i6IE/PtEPh5m0yw5OmPMpesFnzMIS/lzvN5vknnC1sXM5Z/id5VgcNPgpD+wHmIcuYYgW+Q53v+9s96lQ==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.27.2.tgz", + "integrity": "sha512-PaW2DY5Tan+IFvNJGHDmUrORadbe/Ceh8tQxi8cmdQVCCYsLoQo2cuaSj+AU+YRX8M4ivS2vJ9UGaxfuNN7gmg==", + "cpu": [ + "x64" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.27.2.tgz", + "integrity": "sha512-dOlWEMg2gI91Qx5I/HYqOD6iqlJspxLcS4Zlg3vjk1srE67z5T2Uz91yg/qA8sY0XcwQrFzWWiZhMNERylLrpQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.27.2.tgz", + "integrity": "sha512-euMIv/4x5Y2/ImlbGl88mwKNXDsvzbWUlT7DFky76z2keajCtcbAsN9LUdmk31hAoVmJJYSThgdA0EsPeTr1+w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.27.2.tgz", + "integrity": "sha512-RsnE6LQkUHlkC10RKngtHNLxb7scFykEbEwOFDjr3CeCMG+Rr+cKqlkKc2/wJ1u4u990urRHCbjz31x84PBrSQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.27.2.tgz", + "integrity": "sha512-foJM5vv+z2KQmn7emYdDLyTbkoO5bkHZE1oth2tWbQNGW7mX32d46Hz6T0MqXdWS2vBZhaEtHqdy9WYwGfiliA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@types/estree": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.6.tgz", + "integrity": "sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/node": { + "version": "20.9.2", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.9.2.tgz", + "integrity": "sha512-WHZXKFCEyIUJzAwh3NyyTHYSR35SevJ6mZ1nWwJafKtiQbqRTIKSRcw3Ma3acqgsent3RRDqeVwpHntMk+9irg==", + "dev": true, + "dependencies": { + "undici-types": "~5.26.4" + } + }, + "node_modules/@vitest/expect": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@vitest/expect/-/expect-2.1.5.tgz", + "integrity": "sha512-nZSBTW1XIdpZvEJyoP/Sy8fUg0b8od7ZpGDkTUcfJ7wz/VoZAFzFfLyxVxGFhUjJzhYqSbIpfMtl/+k/dpWa3Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/spy": "2.1.5", + "@vitest/utils": "2.1.5", + "chai": "^5.1.2", + "tinyrainbow": "^1.2.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/mocker": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@vitest/mocker/-/mocker-2.1.5.tgz", + "integrity": "sha512-XYW6l3UuBmitWqSUXTNXcVBUCRytDogBsWuNXQijc00dtnU/9OqpXWp4OJroVrad/gLIomAq9aW8yWDBtMthhQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/spy": "2.1.5", + "estree-walker": "^3.0.3", + "magic-string": "^0.30.12" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "msw": "^2.4.9", + "vite": "^5.0.0" + }, + "peerDependenciesMeta": { + "msw": { + "optional": true + }, + "vite": { + "optional": true + } + } + }, + "node_modules/@vitest/pretty-format": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-2.1.5.tgz", + "integrity": "sha512-4ZOwtk2bqG5Y6xRGHcveZVr+6txkH7M2e+nPFd6guSoN638v/1XQ0K06eOpi0ptVU/2tW/pIU4IoPotY/GZ9fw==", + "dev": true, + "license": "MIT", + "dependencies": { + "tinyrainbow": "^1.2.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/runner": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@vitest/runner/-/runner-2.1.5.tgz", + "integrity": "sha512-pKHKy3uaUdh7X6p1pxOkgkVAFW7r2I818vHDthYLvUyjRfkKOU6P45PztOch4DZarWQne+VOaIMwA/erSSpB9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/utils": "2.1.5", + "pathe": "^1.1.2" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/snapshot": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@vitest/snapshot/-/snapshot-2.1.5.tgz", + "integrity": "sha512-zmYw47mhfdfnYbuhkQvkkzYroXUumrwWDGlMjpdUr4jBd3HZiV2w7CQHj+z7AAS4VOtWxI4Zt4bWt4/sKcoIjg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "2.1.5", + "magic-string": "^0.30.12", + "pathe": "^1.1.2" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/spy": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@vitest/spy/-/spy-2.1.5.tgz", + "integrity": "sha512-aWZF3P0r3w6DiYTVskOYuhBc7EMc3jvn1TkBg8ttylFFRqNN2XGD7V5a4aQdk6QiUzZQ4klNBSpCLJgWNdIiNw==", + "dev": true, + "license": "MIT", + "dependencies": { + "tinyspy": "^3.0.2" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/utils": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@vitest/utils/-/utils-2.1.5.tgz", + "integrity": "sha512-yfj6Yrp0Vesw2cwJbP+cl04OC+IHFsuQsrsJBL9pyGeQXE56v1UAOQco+SR55Vf1nQzfV0QJg1Qum7AaWUwwYg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "2.1.5", + "loupe": "^3.1.2", + "tinyrainbow": "^1.2.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/assertion-error": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz", + "integrity": "sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + } + }, + "node_modules/cac": { + "version": "6.7.14", + "resolved": "https://registry.npmjs.org/cac/-/cac-6.7.14.tgz", + "integrity": "sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/chai": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/chai/-/chai-5.1.2.tgz", + "integrity": "sha512-aGtmf24DW6MLHHG5gCx4zaI3uBq3KRtxeVs0DjFH6Z0rDNbsvTxFASFvdj79pxjxZ8/5u3PIiN3IwEIQkiiuPw==", + "dev": true, + "license": "MIT", + "dependencies": { + "assertion-error": "^2.0.1", + "check-error": "^2.1.1", + "deep-eql": "^5.0.1", + "loupe": "^3.1.0", + "pathval": "^2.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/check-error": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/check-error/-/check-error-2.1.1.tgz", + "integrity": "sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 16" + } + }, + "node_modules/debug": { + "version": "4.3.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz", + "integrity": "sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/deep-eql": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-5.0.2.tgz", + "integrity": "sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/encore.dev": { + "version": "1.46.6", + "resolved": "https://registry.npmjs.org/encore.dev/-/encore.dev-1.46.6.tgz", + "integrity": "sha512-LX2eZXCdiF1qV9vvchZlx5RXU4I8c1AtFPVStAT0bo4YBT1mTGB+JJkl1q7NMr4yzFW+gJtFJONyb3hDyZcE7w==", + "license": "MPL-2.0", + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/es-module-lexer": { + "version": "1.5.4", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.5.4.tgz", + "integrity": "sha512-MVNK56NiMrOwitFB7cqDwq0CQutbw+0BvLshJSse0MUNU+y1FC3bUS/AQg7oUng+/wKrrki7JfmwtVHkVfPLlw==", + "dev": true, + "license": "MIT" + }, + "node_modules/esbuild": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.5.tgz", + "integrity": "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=12" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.21.5", + "@esbuild/android-arm": "0.21.5", + "@esbuild/android-arm64": "0.21.5", + "@esbuild/android-x64": "0.21.5", + "@esbuild/darwin-arm64": "0.21.5", + "@esbuild/darwin-x64": "0.21.5", + "@esbuild/freebsd-arm64": "0.21.5", + "@esbuild/freebsd-x64": "0.21.5", + "@esbuild/linux-arm": "0.21.5", + "@esbuild/linux-arm64": "0.21.5", + "@esbuild/linux-ia32": "0.21.5", + "@esbuild/linux-loong64": "0.21.5", + "@esbuild/linux-mips64el": "0.21.5", + "@esbuild/linux-ppc64": "0.21.5", + "@esbuild/linux-riscv64": "0.21.5", + "@esbuild/linux-s390x": "0.21.5", + "@esbuild/linux-x64": "0.21.5", + "@esbuild/netbsd-x64": "0.21.5", + "@esbuild/openbsd-x64": "0.21.5", + "@esbuild/sunos-x64": "0.21.5", + "@esbuild/win32-arm64": "0.21.5", + "@esbuild/win32-ia32": "0.21.5", + "@esbuild/win32-x64": "0.21.5" + } + }, + "node_modules/estree-walker": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-3.0.3.tgz", + "integrity": "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "^1.0.0" + } + }, + "node_modules/expect-type": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/expect-type/-/expect-type-1.1.0.tgz", + "integrity": "sha512-bFi65yM+xZgk+u/KRIpekdSYkTB5W1pEf0Lt8Q8Msh7b+eQ7LXVtIB1Bkm4fvclDEL1b2CZkMhv2mOeF8tMdkA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=12.0.0" + } + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/loupe": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/loupe/-/loupe-3.1.2.tgz", + "integrity": "sha512-23I4pFZHmAemUnz8WZXbYRSKYj801VDaNv9ETuMh7IrMc7VuVVSo+Z9iLE3ni30+U48iDWfi30d3twAXBYmnCg==", + "dev": true, + "license": "MIT" + }, + "node_modules/magic-string": { + "version": "0.30.12", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.12.tgz", + "integrity": "sha512-Ea8I3sQMVXr8JhN4z+H/d8zwo+tYDgHE9+5G4Wnrwhs0gaK9fXTKx0Tw5Xwsd/bCPTTZNRAdpyzvoeORe9LYpw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/nanoid": { + "version": "3.3.7", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz", + "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "bin": { + "nanoid": "bin/nanoid.cjs" + }, + "engines": { + "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" + } + }, + "node_modules/pathe": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/pathe/-/pathe-1.1.2.tgz", + "integrity": "sha512-whLdWMYL2TwI08hn8/ZqAbrVemu0LNaNNJZX73O6qaIdCTfXutsLhMkjdENX0qhsQ9uIimo4/aQOmXkoon2nDQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/pathval": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/pathval/-/pathval-2.0.0.tgz", + "integrity": "sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 14.16" + } + }, + "node_modules/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", + "dev": true, + "license": "ISC" + }, + "node_modules/postcss": { + "version": "8.4.49", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.49.tgz", + "integrity": "sha512-OCVPnIObs4N29kxTjzLfUryOkvZEq+pf8jTF0lg8E7uETuWHA+v7j3c/xJmiqpX450191LlmZfUKkXxkTry7nA==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/postcss" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "nanoid": "^3.3.7", + "picocolors": "^1.1.1", + "source-map-js": "^1.2.1" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/rollup": { + "version": "4.27.2", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.27.2.tgz", + "integrity": "sha512-KreA+PzWmk2yaFmZVwe6GB2uBD86nXl86OsDkt1bJS9p3vqWuEQ6HnJJ+j/mZi/q0920P99/MVRlB4L3crpF5w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "1.0.6" + }, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=18.0.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "@rollup/rollup-android-arm-eabi": "4.27.2", + "@rollup/rollup-android-arm64": "4.27.2", + "@rollup/rollup-darwin-arm64": "4.27.2", + "@rollup/rollup-darwin-x64": "4.27.2", + "@rollup/rollup-freebsd-arm64": "4.27.2", + "@rollup/rollup-freebsd-x64": "4.27.2", + "@rollup/rollup-linux-arm-gnueabihf": "4.27.2", + "@rollup/rollup-linux-arm-musleabihf": "4.27.2", + "@rollup/rollup-linux-arm64-gnu": "4.27.2", + "@rollup/rollup-linux-arm64-musl": "4.27.2", + "@rollup/rollup-linux-powerpc64le-gnu": "4.27.2", + "@rollup/rollup-linux-riscv64-gnu": "4.27.2", + "@rollup/rollup-linux-s390x-gnu": "4.27.2", + "@rollup/rollup-linux-x64-gnu": "4.27.2", + "@rollup/rollup-linux-x64-musl": "4.27.2", + "@rollup/rollup-win32-arm64-msvc": "4.27.2", + "@rollup/rollup-win32-ia32-msvc": "4.27.2", + "@rollup/rollup-win32-x64-msvc": "4.27.2", + "fsevents": "~2.3.2" + } + }, + "node_modules/siginfo": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/siginfo/-/siginfo-2.0.0.tgz", + "integrity": "sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==", + "dev": true, + "license": "ISC" + }, + "node_modules/source-map-js": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", + "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/stackback": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/stackback/-/stackback-0.0.2.tgz", + "integrity": "sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==", + "dev": true, + "license": "MIT" + }, + "node_modules/std-env": { + "version": "3.8.0", + "resolved": "https://registry.npmjs.org/std-env/-/std-env-3.8.0.tgz", + "integrity": "sha512-Bc3YwwCB+OzldMxOXJIIvC6cPRWr/LxOp48CdQTOkPyk/t4JWWJbrilwBd7RJzKV8QW7tJkcgAmeuLLJugl5/w==", + "dev": true, + "license": "MIT" + }, + "node_modules/tinybench": { + "version": "2.9.0", + "resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.9.0.tgz", + "integrity": "sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==", + "dev": true, + "license": "MIT" + }, + "node_modules/tinyexec": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.1.tgz", + "integrity": "sha512-WiCJLEECkO18gwqIp6+hJg0//p23HXp4S+gGtAKu3mI2F2/sXC4FvHvXvB0zJVVaTPhx1/tOwdbRsa1sOBIKqQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/tinypool": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/tinypool/-/tinypool-1.0.2.tgz", + "integrity": "sha512-al6n+QEANGFOMf/dmUMsuS5/r9B06uwlyNjZZql/zv8J7ybHCgoihBNORZCY2mzUuAnomQa2JdhyHKzZxPCrFA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.0.0 || >=20.0.0" + } + }, + "node_modules/tinyrainbow": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/tinyrainbow/-/tinyrainbow-1.2.0.tgz", + "integrity": "sha512-weEDEq7Z5eTHPDh4xjX789+fHfF+P8boiFB+0vbWzpbnbsEr/GRaohi/uMKxg8RZMXnl1ItAi/IUHWMsjDV7kQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/tinyspy": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/tinyspy/-/tinyspy-3.0.2.tgz", + "integrity": "sha512-n1cw8k1k0x4pgA2+9XrOkFydTerNcJ1zWCO5Nn9scWHTD+5tp8dghT2x1uduQePZTZgd3Tupf+x9BxJjeJi77Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/typescript": { + "version": "5.2.2", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz", + "integrity": "sha512-mI4WrpHsbCIcwT9cF4FZvr80QUeKvsUsUvKDoR+X/7XHQH98xYD8YHZg7ANtz2GtZt/CBq2QJ0thkGJMHfqc1w==", + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/undici-types": { + "version": "5.26.5", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", + "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==", + "dev": true + }, + "node_modules/vite": { + "version": "5.4.11", + "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.11.tgz", + "integrity": "sha512-c7jFQRklXua0mTzneGW9QVyxFjUgwcihC4bXEtujIo2ouWCe1Ajt/amn2PCxYnhYfd5k09JX3SB7OYWFKYqj8Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "esbuild": "^0.21.3", + "postcss": "^8.4.43", + "rollup": "^4.20.0" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^18.0.0 || >=20.0.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^18.0.0 || >=20.0.0", + "less": "*", + "lightningcss": "^1.21.0", + "sass": "*", + "sass-embedded": "*", + "stylus": "*", + "sugarss": "*", + "terser": "^5.4.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + } + } + }, + "node_modules/vite-node": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/vite-node/-/vite-node-2.1.5.tgz", + "integrity": "sha512-rd0QIgx74q4S1Rd56XIiL2cYEdyWn13cunYBIuqh9mpmQr7gGS0IxXoP8R6OaZtNQQLyXSWbd4rXKYUbhFpK5w==", + "dev": true, + "license": "MIT", + "dependencies": { + "cac": "^6.7.14", + "debug": "^4.3.7", + "es-module-lexer": "^1.5.4", + "pathe": "^1.1.2", + "vite": "^5.0.0" + }, + "bin": { + "vite-node": "vite-node.mjs" + }, + "engines": { + "node": "^18.0.0 || >=20.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/vitest": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/vitest/-/vitest-2.1.5.tgz", + "integrity": "sha512-P4ljsdpuzRTPI/kbND2sDZ4VmieerR2c9szEZpjc+98Z9ebvnXmM5+0tHEKqYZumXqlvnmfWsjeFOjXVriDG7A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/expect": "2.1.5", + "@vitest/mocker": "2.1.5", + "@vitest/pretty-format": "^2.1.5", + "@vitest/runner": "2.1.5", + "@vitest/snapshot": "2.1.5", + "@vitest/spy": "2.1.5", + "@vitest/utils": "2.1.5", + "chai": "^5.1.2", + "debug": "^4.3.7", + "expect-type": "^1.1.0", + "magic-string": "^0.30.12", + "pathe": "^1.1.2", + "std-env": "^3.8.0", + "tinybench": "^2.9.0", + "tinyexec": "^0.3.1", + "tinypool": "^1.0.1", + "tinyrainbow": "^1.2.0", + "vite": "^5.0.0", + "vite-node": "2.1.5", + "why-is-node-running": "^2.3.0" + }, + "bin": { + "vitest": "vitest.mjs" + }, + "engines": { + "node": "^18.0.0 || >=20.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "@edge-runtime/vm": "*", + "@types/node": "^18.0.0 || >=20.0.0", + "@vitest/browser": "2.1.5", + "@vitest/ui": "2.1.5", + "happy-dom": "*", + "jsdom": "*" + }, + "peerDependenciesMeta": { + "@edge-runtime/vm": { + "optional": true + }, + "@types/node": { + "optional": true + }, + "@vitest/browser": { + "optional": true + }, + "@vitest/ui": { + "optional": true + }, + "happy-dom": { + "optional": true + }, + "jsdom": { + "optional": true + } + } + }, + "node_modules/why-is-node-running": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/why-is-node-running/-/why-is-node-running-2.3.0.tgz", + "integrity": "sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==", + "dev": true, + "license": "MIT", + "dependencies": { + "siginfo": "^2.0.0", + "stackback": "0.0.2" + }, + "bin": { + "why-is-node-running": "cli.js" + }, + "engines": { + "node": ">=8" + } + } + } +} diff --git a/ts/saas-nextforge-encore/backend/package.json b/ts/saas-nextforge-encore/backend/package.json new file mode 100644 index 00000000..d0c1a16b --- /dev/null +++ b/ts/saas-nextforge-encore/backend/package.json @@ -0,0 +1,28 @@ +{ + "name": "api", + "private": true, + "version": "0.0.1", + "description": "NextForge Encore Backend", + "license": "MIT", + "type": "module", + "scripts": { + "test": "vitest", + "postinstall": "npx prisma generate --schema=auth/prisma/schema.prisma" + }, + "devDependencies": { + "@types/node": "^20.5.7", + "typescript": "^5.2.2", + "prisma": "^5.22.0", + "vitest": "^2.1.4" + }, + "dependencies": { + "encore.dev": "^1.46.6", + "@prisma/client": "^5.22.0", + "better-auth": "^1.2.3", + "encore-better-auth": "0.2.0", + "posthog-node": "^4.9.0" + }, + "optionalDependencies": { + "@rollup/rollup-linux-x64-gnu": "^4.13.0" + } +} diff --git a/ts/saas-nextforge-encore/backend/pnpm-lock.yaml b/ts/saas-nextforge-encore/backend/pnpm-lock.yaml new file mode 100644 index 00000000..6c4a2a5e --- /dev/null +++ b/ts/saas-nextforge-encore/backend/pnpm-lock.yaml @@ -0,0 +1,1357 @@ +lockfileVersion: '9.0' + +settings: + autoInstallPeers: true + excludeLinksFromLockfile: false + +importers: + + .: + dependencies: + '@prisma/client': + specifier: ^5.22.0 + version: 5.22.0(prisma@5.22.0) + better-auth: + specifier: ^1.2.3 + version: 1.2.3(typescript@5.8.2) + encore-better-auth: + specifier: 0.2.0 + version: 0.2.0(better-auth@1.2.3(typescript@5.8.2)) + encore.dev: + specifier: ^1.46.6 + version: 1.46.7 + posthog-node: + specifier: ^4.9.0 + version: 4.10.1 + optionalDependencies: + '@rollup/rollup-linux-x64-gnu': + specifier: ^4.13.0 + version: 4.35.0 + devDependencies: + '@types/node': + specifier: ^20.5.7 + version: 20.17.24 + prisma: + specifier: ^5.22.0 + version: 5.22.0 + typescript: + specifier: ^5.2.2 + version: 5.8.2 + vitest: + specifier: ^2.1.4 + version: 2.1.9(@types/node@20.17.24) + +packages: + + '@better-auth/utils@0.2.3': + resolution: {integrity: sha512-Ap1GaSmo6JYhJhxJOpUB0HobkKPTNzfta+bLV89HfpyCAHN7p8ntCrmNFHNAVD0F6v0mywFVEUg1FUhNCc81Rw==} + + '@better-fetch/fetch@1.1.15': + resolution: {integrity: sha512-0Bl8YYj1f8qCTNHeSn5+1DWv2hy7rLBrQ8rS8Y9XYloiwZEfc3k4yspIG0llRxafxqhGCwlGRg+F8q1HZRCMXA==} + + '@esbuild/aix-ppc64@0.21.5': + resolution: {integrity: sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [aix] + + '@esbuild/android-arm64@0.21.5': + resolution: {integrity: sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==} + engines: {node: '>=12'} + cpu: [arm64] + os: [android] + + '@esbuild/android-arm@0.21.5': + resolution: {integrity: sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==} + engines: {node: '>=12'} + cpu: [arm] + os: [android] + + '@esbuild/android-x64@0.21.5': + resolution: {integrity: sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==} + engines: {node: '>=12'} + cpu: [x64] + os: [android] + + '@esbuild/darwin-arm64@0.21.5': + resolution: {integrity: sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==} + engines: {node: '>=12'} + cpu: [arm64] + os: [darwin] + + '@esbuild/darwin-x64@0.21.5': + resolution: {integrity: sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==} + engines: {node: '>=12'} + cpu: [x64] + os: [darwin] + + '@esbuild/freebsd-arm64@0.21.5': + resolution: {integrity: sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==} + engines: {node: '>=12'} + cpu: [arm64] + os: [freebsd] + + '@esbuild/freebsd-x64@0.21.5': + resolution: {integrity: sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [freebsd] + + '@esbuild/linux-arm64@0.21.5': + resolution: {integrity: sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==} + engines: {node: '>=12'} + cpu: [arm64] + os: [linux] + + '@esbuild/linux-arm@0.21.5': + resolution: {integrity: sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==} + engines: {node: '>=12'} + cpu: [arm] + os: [linux] + + '@esbuild/linux-ia32@0.21.5': + resolution: {integrity: sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==} + engines: {node: '>=12'} + cpu: [ia32] + os: [linux] + + '@esbuild/linux-loong64@0.21.5': + resolution: {integrity: sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==} + engines: {node: '>=12'} + cpu: [loong64] + os: [linux] + + '@esbuild/linux-mips64el@0.21.5': + resolution: {integrity: sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==} + engines: {node: '>=12'} + cpu: [mips64el] + os: [linux] + + '@esbuild/linux-ppc64@0.21.5': + resolution: {integrity: sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [linux] + + '@esbuild/linux-riscv64@0.21.5': + resolution: {integrity: sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==} + engines: {node: '>=12'} + cpu: [riscv64] + os: [linux] + + '@esbuild/linux-s390x@0.21.5': + resolution: {integrity: sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==} + engines: {node: '>=12'} + cpu: [s390x] + os: [linux] + + '@esbuild/linux-x64@0.21.5': + resolution: {integrity: sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [linux] + + '@esbuild/netbsd-x64@0.21.5': + resolution: {integrity: sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==} + engines: {node: '>=12'} + cpu: [x64] + os: [netbsd] + + '@esbuild/openbsd-x64@0.21.5': + resolution: {integrity: sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==} + engines: {node: '>=12'} + cpu: [x64] + os: [openbsd] + + '@esbuild/sunos-x64@0.21.5': + resolution: {integrity: sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==} + engines: {node: '>=12'} + cpu: [x64] + os: [sunos] + + '@esbuild/win32-arm64@0.21.5': + resolution: {integrity: sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==} + engines: {node: '>=12'} + cpu: [arm64] + os: [win32] + + '@esbuild/win32-ia32@0.21.5': + resolution: {integrity: sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==} + engines: {node: '>=12'} + cpu: [ia32] + os: [win32] + + '@esbuild/win32-x64@0.21.5': + resolution: {integrity: sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==} + engines: {node: '>=12'} + cpu: [x64] + os: [win32] + + '@hexagon/base64@1.1.28': + resolution: {integrity: sha512-lhqDEAvWixy3bZ+UOYbPwUbBkwBq5C1LAJ/xPC8Oi+lL54oyakv/npbA0aU2hgCsx/1NUd4IBvV03+aUBWxerw==} + + '@jridgewell/sourcemap-codec@1.5.0': + resolution: {integrity: sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==} + + '@levischuck/tiny-cbor@0.2.11': + resolution: {integrity: sha512-llBRm4dT4Z89aRsm6u2oEZ8tfwL/2l6BwpZ7JcyieouniDECM5AqNgr/y08zalEIvW3RSK4upYyybDcmjXqAow==} + + '@noble/ciphers@0.6.0': + resolution: {integrity: sha512-mIbq/R9QXk5/cTfESb1OKtyFnk7oc1Om/8onA1158K9/OZUQFDEVy55jVTato+xmp3XX6F6Qh0zz0Nc1AxAlRQ==} + + '@noble/hashes@1.7.1': + resolution: {integrity: sha512-B8XBPsn4vT/KJAGqDzbwztd+6Yte3P4V7iafm24bxgDe/mlRuK6xmWPuCNrKt2vDafZ8MfJLlchDG/vYafQEjQ==} + engines: {node: ^14.21.3 || >=16} + + '@peculiar/asn1-android@2.3.15': + resolution: {integrity: sha512-8U2TIj59cRlSXTX2d0mzUKP7whfWGFMzTeC3qPgAbccXFrPNZLaDhpNEdG5U2QZ/tBv/IHlCJ8s+KYXpJeop6w==} + + '@peculiar/asn1-ecc@2.3.15': + resolution: {integrity: sha512-/HtR91dvgog7z/WhCVdxZJ/jitJuIu8iTqiyWVgRE9Ac5imt2sT/E4obqIVGKQw7PIy+X6i8lVBoT6wC73XUgA==} + + '@peculiar/asn1-rsa@2.3.15': + resolution: {integrity: sha512-p6hsanvPhexRtYSOHihLvUUgrJ8y0FtOM97N5UEpC+VifFYyZa0iZ5cXjTkZoDwxJ/TTJ1IJo3HVTB2JJTpXvg==} + + '@peculiar/asn1-schema@2.3.15': + resolution: {integrity: sha512-QPeD8UA8axQREpgR5UTAfu2mqQmm97oUqahDtNdBcfj3qAnoXzFdQW+aNf/tD2WVXF8Fhmftxoj0eMIT++gX2w==} + + '@peculiar/asn1-x509@2.3.15': + resolution: {integrity: sha512-0dK5xqTqSLaxv1FHXIcd4Q/BZNuopg+u1l23hT9rOmQ1g4dNtw0g/RnEi+TboB0gOwGtrWn269v27cMgchFIIg==} + + '@prisma/client@5.22.0': + resolution: {integrity: sha512-M0SVXfyHnQREBKxCgyo7sffrKttwE6R8PMq330MIUF0pTwjUhLbW84pFDlf06B27XyCR++VtjugEnIHdr07SVA==} + engines: {node: '>=16.13'} + peerDependencies: + prisma: '*' + peerDependenciesMeta: + prisma: + optional: true + + '@prisma/debug@5.22.0': + resolution: {integrity: sha512-AUt44v3YJeggO2ZU5BkXI7M4hu9BF2zzH2iF2V5pyXT/lRTyWiElZ7It+bRH1EshoMRxHgpYg4VB6rCM+mG5jQ==} + + '@prisma/engines-version@5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2': + resolution: {integrity: sha512-2PTmxFR2yHW/eB3uqWtcgRcgAbG1rwG9ZriSvQw+nnb7c4uCr3RAcGMb6/zfE88SKlC1Nj2ziUvc96Z379mHgQ==} + + '@prisma/engines@5.22.0': + resolution: {integrity: sha512-UNjfslWhAt06kVL3CjkuYpHAWSO6L4kDCVPegV6itt7nD1kSJavd3vhgAEhjglLJJKEdJ7oIqDJ+yHk6qO8gPA==} + + '@prisma/fetch-engine@5.22.0': + resolution: {integrity: sha512-bkrD/Mc2fSvkQBV5EpoFcZ87AvOgDxbG99488a5cexp5Ccny+UM6MAe/UFkUC0wLYD9+9befNOqGiIJhhq+HbA==} + + '@prisma/get-platform@5.22.0': + resolution: {integrity: sha512-pHhpQdr1UPFpt+zFfnPazhulaZYCUqeIcPpJViYoq9R+D/yw4fjE+CtnsnKzPYm0ddUbeXUzjGVGIRVgPDCk4Q==} + + '@rollup/rollup-android-arm-eabi@4.35.0': + resolution: {integrity: sha512-uYQ2WfPaqz5QtVgMxfN6NpLD+no0MYHDBywl7itPYd3K5TjjSghNKmX8ic9S8NU8w81NVhJv/XojcHptRly7qQ==} + cpu: [arm] + os: [android] + + '@rollup/rollup-android-arm64@4.35.0': + resolution: {integrity: sha512-FtKddj9XZudurLhdJnBl9fl6BwCJ3ky8riCXjEw3/UIbjmIY58ppWwPEvU3fNu+W7FUsAsB1CdH+7EQE6CXAPA==} + cpu: [arm64] + os: [android] + + '@rollup/rollup-darwin-arm64@4.35.0': + resolution: {integrity: sha512-Uk+GjOJR6CY844/q6r5DR/6lkPFOw0hjfOIzVx22THJXMxktXG6CbejseJFznU8vHcEBLpiXKY3/6xc+cBm65Q==} + cpu: [arm64] + os: [darwin] + + '@rollup/rollup-darwin-x64@4.35.0': + resolution: {integrity: sha512-3IrHjfAS6Vkp+5bISNQnPogRAW5GAV1n+bNCrDwXmfMHbPl5EhTmWtfmwlJxFRUCBZ+tZ/OxDyU08aF6NI/N5Q==} + cpu: [x64] + os: [darwin] + + '@rollup/rollup-freebsd-arm64@4.35.0': + resolution: {integrity: sha512-sxjoD/6F9cDLSELuLNnY0fOrM9WA0KrM0vWm57XhrIMf5FGiN8D0l7fn+bpUeBSU7dCgPV2oX4zHAsAXyHFGcQ==} + cpu: [arm64] + os: [freebsd] + + '@rollup/rollup-freebsd-x64@4.35.0': + resolution: {integrity: sha512-2mpHCeRuD1u/2kruUiHSsnjWtHjqVbzhBkNVQ1aVD63CcexKVcQGwJ2g5VphOd84GvxfSvnnlEyBtQCE5hxVVw==} + cpu: [x64] + os: [freebsd] + + '@rollup/rollup-linux-arm-gnueabihf@4.35.0': + resolution: {integrity: sha512-mrA0v3QMy6ZSvEuLs0dMxcO2LnaCONs1Z73GUDBHWbY8tFFocM6yl7YyMu7rz4zS81NDSqhrUuolyZXGi8TEqg==} + cpu: [arm] + os: [linux] + + '@rollup/rollup-linux-arm-musleabihf@4.35.0': + resolution: {integrity: sha512-DnYhhzcvTAKNexIql8pFajr0PiDGrIsBYPRvCKlA5ixSS3uwo/CWNZxB09jhIapEIg945KOzcYEAGGSmTSpk7A==} + cpu: [arm] + os: [linux] + + '@rollup/rollup-linux-arm64-gnu@4.35.0': + resolution: {integrity: sha512-uagpnH2M2g2b5iLsCTZ35CL1FgyuzzJQ8L9VtlJ+FckBXroTwNOaD0z0/UF+k5K3aNQjbm8LIVpxykUOQt1m/A==} + cpu: [arm64] + os: [linux] + + '@rollup/rollup-linux-arm64-musl@4.35.0': + resolution: {integrity: sha512-XQxVOCd6VJeHQA/7YcqyV0/88N6ysSVzRjJ9I9UA/xXpEsjvAgDTgH3wQYz5bmr7SPtVK2TsP2fQ2N9L4ukoUg==} + cpu: [arm64] + os: [linux] + + '@rollup/rollup-linux-loongarch64-gnu@4.35.0': + resolution: {integrity: sha512-5pMT5PzfgwcXEwOaSrqVsz/LvjDZt+vQ8RT/70yhPU06PTuq8WaHhfT1LW+cdD7mW6i/J5/XIkX/1tCAkh1W6g==} + cpu: [loong64] + os: [linux] + + '@rollup/rollup-linux-powerpc64le-gnu@4.35.0': + resolution: {integrity: sha512-c+zkcvbhbXF98f4CtEIP1EBA/lCic5xB0lToneZYvMeKu5Kamq3O8gqrxiYYLzlZH6E3Aq+TSW86E4ay8iD8EA==} + cpu: [ppc64] + os: [linux] + + '@rollup/rollup-linux-riscv64-gnu@4.35.0': + resolution: {integrity: sha512-s91fuAHdOwH/Tad2tzTtPX7UZyytHIRR6V4+2IGlV0Cej5rkG0R61SX4l4y9sh0JBibMiploZx3oHKPnQBKe4g==} + cpu: [riscv64] + os: [linux] + + '@rollup/rollup-linux-s390x-gnu@4.35.0': + resolution: {integrity: sha512-hQRkPQPLYJZYGP+Hj4fR9dDBMIM7zrzJDWFEMPdTnTy95Ljnv0/4w/ixFw3pTBMEuuEuoqtBINYND4M7ujcuQw==} + cpu: [s390x] + os: [linux] + + '@rollup/rollup-linux-x64-gnu@4.35.0': + resolution: {integrity: sha512-Pim1T8rXOri+0HmV4CdKSGrqcBWX0d1HoPnQ0uw0bdp1aP5SdQVNBy8LjYncvnLgu3fnnCt17xjWGd4cqh8/hA==} + cpu: [x64] + os: [linux] + + '@rollup/rollup-linux-x64-musl@4.35.0': + resolution: {integrity: sha512-QysqXzYiDvQWfUiTm8XmJNO2zm9yC9P/2Gkrwg2dH9cxotQzunBHYr6jk4SujCTqnfGxduOmQcI7c2ryuW8XVg==} + cpu: [x64] + os: [linux] + + '@rollup/rollup-win32-arm64-msvc@4.35.0': + resolution: {integrity: sha512-OUOlGqPkVJCdJETKOCEf1mw848ZyJ5w50/rZ/3IBQVdLfR5jk/6Sr5m3iO2tdPgwo0x7VcncYuOvMhBWZq8ayg==} + cpu: [arm64] + os: [win32] + + '@rollup/rollup-win32-ia32-msvc@4.35.0': + resolution: {integrity: sha512-2/lsgejMrtwQe44glq7AFFHLfJBPafpsTa6JvP2NGef/ifOa4KBoglVf7AKN7EV9o32evBPRqfg96fEHzWo5kw==} + cpu: [ia32] + os: [win32] + + '@rollup/rollup-win32-x64-msvc@4.35.0': + resolution: {integrity: sha512-PIQeY5XDkrOysbQblSW7v3l1MDZzkTEzAfTPkj5VAu3FW8fS4ynyLg2sINp0fp3SjZ8xkRYpLqoKcYqAkhU1dw==} + cpu: [x64] + os: [win32] + + '@simplewebauthn/browser@13.1.0': + resolution: {integrity: sha512-WuHZ/PYvyPJ9nxSzgHtOEjogBhwJfC8xzYkPC+rR/+8chl/ft4ngjiK8kSU5HtRJfczupyOh33b25TjYbvwAcg==} + + '@simplewebauthn/server@13.1.1': + resolution: {integrity: sha512-1hsLpRHfSuMB9ee2aAdh0Htza/X3f4djhYISrggqGe3xopNjOcePiSDkDDoPzDYaaMCrbqGP1H2TYU7bgL9PmA==} + engines: {node: '>=20.0.0'} + + '@types/estree@1.0.6': + resolution: {integrity: sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==} + + '@types/node@20.17.24': + resolution: {integrity: sha512-d7fGCyB96w9BnWQrOsJtpyiSaBcAYYr75bnK6ZRjDbql2cGLj/3GsL5OYmLPNq76l7Gf2q4Rv9J2o6h5CrD9sA==} + + '@vitest/expect@2.1.9': + resolution: {integrity: sha512-UJCIkTBenHeKT1TTlKMJWy1laZewsRIzYighyYiJKZreqtdxSos/S1t+ktRMQWu2CKqaarrkeszJx1cgC5tGZw==} + + '@vitest/mocker@2.1.9': + resolution: {integrity: sha512-tVL6uJgoUdi6icpxmdrn5YNo3g3Dxv+IHJBr0GXHaEdTcw3F+cPKnsXFhli6nO+f/6SDKPHEK1UN+k+TQv0Ehg==} + peerDependencies: + msw: ^2.4.9 + vite: ^5.0.0 + peerDependenciesMeta: + msw: + optional: true + vite: + optional: true + + '@vitest/pretty-format@2.1.9': + resolution: {integrity: sha512-KhRIdGV2U9HOUzxfiHmY8IFHTdqtOhIzCpd8WRdJiE7D/HUcZVD0EgQCVjm+Q9gkUXWgBvMmTtZgIG48wq7sOQ==} + + '@vitest/runner@2.1.9': + resolution: {integrity: sha512-ZXSSqTFIrzduD63btIfEyOmNcBmQvgOVsPNPe0jYtESiXkhd8u2erDLnMxmGrDCwHCCHE7hxwRDCT3pt0esT4g==} + + '@vitest/snapshot@2.1.9': + resolution: {integrity: sha512-oBO82rEjsxLNJincVhLhaxxZdEtV0EFHMK5Kmx5sJ6H9L183dHECjiefOAdnqpIgT5eZwT04PoggUnW88vOBNQ==} + + '@vitest/spy@2.1.9': + resolution: {integrity: sha512-E1B35FwzXXTs9FHNK6bDszs7mtydNi5MIfUWpceJ8Xbfb1gBMscAnwLbEu+B44ed6W3XjL9/ehLPHR1fkf1KLQ==} + + '@vitest/utils@2.1.9': + resolution: {integrity: sha512-v0psaMSkNJ3A2NMrUEHFRzJtDPFn+/VWZ5WxImB21T9fjucJRmS7xCS3ppEnARb9y11OAzaD+P2Ps+b+BGX5iQ==} + + asn1js@3.0.5: + resolution: {integrity: sha512-FVnvrKJwpt9LP2lAMl8qZswRNm3T4q9CON+bxldk2iwk3FFpuwhx2FfinyitizWHsVYyaY+y5JzDR0rCMV5yTQ==} + engines: {node: '>=12.0.0'} + + assertion-error@2.0.1: + resolution: {integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==} + engines: {node: '>=12'} + + asynckit@0.4.0: + resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==} + + axios@1.8.2: + resolution: {integrity: sha512-ls4GYBm5aig9vWx8AWDSGLpnpDQRtWAfrjU+EuytuODrFBkqesN2RkOQCBzrA1RQNHw1SmRMSDDDSwzNAYQ6Rg==} + + better-auth@1.2.3: + resolution: {integrity: sha512-y97/ah2SOWaW81IRg36m7xMSMVl7ATaHie/nhQ0in/reVlEX/6juVPszNqq0gcTwQtFsB8oe15wQKgdf4yHP9Q==} + + better-call@1.0.4: + resolution: {integrity: sha512-NdAihYdkS0IOz1mtz8mw1gWacCxR9r921U8YqB+VB6++rt8edMG13vVL16Y4TBL4XkjMK/DUewEsOOFkw9LJYQ==} + + cac@6.7.14: + resolution: {integrity: sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==} + engines: {node: '>=8'} + + call-bind-apply-helpers@1.0.2: + resolution: {integrity: sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==} + engines: {node: '>= 0.4'} + + chai@5.2.0: + resolution: {integrity: sha512-mCuXncKXk5iCLhfhwTc0izo0gtEmpz5CtG2y8GiOINBlMVS6v8TMRc5TaLWKS6692m9+dVVfzgeVxR5UxWHTYw==} + engines: {node: '>=12'} + + check-error@2.1.1: + resolution: {integrity: sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw==} + engines: {node: '>= 16'} + + combined-stream@1.0.8: + resolution: {integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==} + engines: {node: '>= 0.8'} + + debug@4.4.0: + resolution: {integrity: sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==} + engines: {node: '>=6.0'} + peerDependencies: + supports-color: '*' + peerDependenciesMeta: + supports-color: + optional: true + + deep-eql@5.0.2: + resolution: {integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==} + engines: {node: '>=6'} + + defu@6.1.4: + resolution: {integrity: sha512-mEQCMmwJu317oSz8CwdIOdwf3xMif1ttiM8LTufzc3g6kR+9Pe236twL8j3IYT1F7GfRgGcW6MWxzZjLIkuHIg==} + + delayed-stream@1.0.0: + resolution: {integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==} + engines: {node: '>=0.4.0'} + + dunder-proto@1.0.1: + resolution: {integrity: sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==} + engines: {node: '>= 0.4'} + + encore-better-auth@0.2.0: + resolution: {integrity: sha512-senGatzJcS4DnUuvc78p+3pYlUd+1S9jGHnWfP62t6xQnEpYsit3nXdAGdogBpdhu/0hXZTmoAk0F8+YMGRANg==} + peerDependencies: + better-auth: ^1.2.3 + + encore.dev@1.46.7: + resolution: {integrity: sha512-LpnBcnyPCmxJtY5y9gc3zvSA2opRJPDqf1NEdjwjm9UuSKJcuRJGsjFSqyfI1aelctrhcGVkwIX0KTFu6UKFIQ==} + engines: {node: '>=18.0.0'} + + es-define-property@1.0.1: + resolution: {integrity: sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==} + engines: {node: '>= 0.4'} + + es-errors@1.3.0: + resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} + engines: {node: '>= 0.4'} + + es-module-lexer@1.6.0: + resolution: {integrity: sha512-qqnD1yMU6tk/jnaMosogGySTZP8YtUgAffA9nMN+E/rjxcfRQ6IEk7IiozUjgxKoFHBGjTLnrHB/YC45r/59EQ==} + + es-object-atoms@1.1.1: + resolution: {integrity: sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==} + engines: {node: '>= 0.4'} + + es-set-tostringtag@2.1.0: + resolution: {integrity: sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==} + engines: {node: '>= 0.4'} + + esbuild@0.21.5: + resolution: {integrity: sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==} + engines: {node: '>=12'} + hasBin: true + + estree-walker@3.0.3: + resolution: {integrity: sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==} + + expect-type@1.2.0: + resolution: {integrity: sha512-80F22aiJ3GLyVnS/B3HzgR6RelZVumzj9jkL0Rhz4h0xYbNW9PjlQz5h3J/SShErbXBc295vseR4/MIbVmUbeA==} + engines: {node: '>=12.0.0'} + + follow-redirects@1.15.9: + resolution: {integrity: sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==} + engines: {node: '>=4.0'} + peerDependencies: + debug: '*' + peerDependenciesMeta: + debug: + optional: true + + form-data@4.0.2: + resolution: {integrity: sha512-hGfm/slu0ZabnNt4oaRZ6uREyfCj6P4fT/n6A1rGV+Z0VdGXjfOhVUpkn6qVQONHGIFwmveGXyDs75+nr6FM8w==} + engines: {node: '>= 6'} + + fsevents@2.3.3: + resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} + engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} + os: [darwin] + + function-bind@1.1.2: + resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} + + get-intrinsic@1.3.0: + resolution: {integrity: sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==} + engines: {node: '>= 0.4'} + + get-proto@1.0.1: + resolution: {integrity: sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==} + engines: {node: '>= 0.4'} + + gopd@1.2.0: + resolution: {integrity: sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==} + engines: {node: '>= 0.4'} + + has-symbols@1.1.0: + resolution: {integrity: sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==} + engines: {node: '>= 0.4'} + + has-tostringtag@1.0.2: + resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==} + engines: {node: '>= 0.4'} + + hasown@2.0.2: + resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} + engines: {node: '>= 0.4'} + + jose@5.10.0: + resolution: {integrity: sha512-s+3Al/p9g32Iq+oqXxkW//7jk2Vig6FF1CFqzVXoTUXt2qz89YWbL+OwS17NFYEvxC35n0FKeGO2LGYSxeM2Gg==} + + kysely@0.27.6: + resolution: {integrity: sha512-FIyV/64EkKhJmjgC0g2hygpBv5RNWVPyNCqSAD7eTCv6eFWNIi4PN1UvdSJGicN/o35bnevgis4Y0UDC0qi8jQ==} + engines: {node: '>=14.0.0'} + + loupe@3.1.3: + resolution: {integrity: sha512-kkIp7XSkP78ZxJEsSxW3712C6teJVoeHHwgo9zJ380de7IYyJ2ISlxojcH2pC5OFLewESmnRi/+XCDIEEVyoug==} + + magic-string@0.30.17: + resolution: {integrity: sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA==} + + math-intrinsics@1.1.0: + resolution: {integrity: sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==} + engines: {node: '>= 0.4'} + + mime-db@1.52.0: + resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==} + engines: {node: '>= 0.6'} + + mime-types@2.1.35: + resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==} + engines: {node: '>= 0.6'} + + ms@2.1.3: + resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} + + nanoid@3.3.9: + resolution: {integrity: sha512-SppoicMGpZvbF1l3z4x7No3OlIjP7QJvC9XR7AhZr1kL133KHnKPztkKDc+Ir4aJ/1VhTySrtKhrsycmrMQfvg==} + engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} + hasBin: true + + nanostores@0.11.4: + resolution: {integrity: sha512-k1oiVNN4hDK8NcNERSZLQiMfRzEGtfnvZvdBvey3SQbgn8Dcrk0h1I6vpxApjb10PFUflZrgJ2WEZyJQ+5v7YQ==} + engines: {node: ^18.0.0 || >=20.0.0} + + pathe@1.1.2: + resolution: {integrity: sha512-whLdWMYL2TwI08hn8/ZqAbrVemu0LNaNNJZX73O6qaIdCTfXutsLhMkjdENX0qhsQ9uIimo4/aQOmXkoon2nDQ==} + + pathval@2.0.0: + resolution: {integrity: sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA==} + engines: {node: '>= 14.16'} + + picocolors@1.1.1: + resolution: {integrity: sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==} + + postcss@8.5.3: + resolution: {integrity: sha512-dle9A3yYxlBSrt8Fu+IpjGT8SY8hN0mlaA6GY8t0P5PjIOZemULz/E2Bnm/2dcUOena75OTNkHI76uZBNUUq3A==} + engines: {node: ^10 || ^12 || >=14} + + posthog-node@4.10.1: + resolution: {integrity: sha512-rEzVszfaOkUFTjEabDcRLJNRqMwTOeU1WpqKgQEDV3x82O4ODifkvkoCERaPxl/ossi1NtqKXBOZ+XnhQ19pNQ==} + engines: {node: '>=15.0.0'} + + prisma@5.22.0: + resolution: {integrity: sha512-vtpjW3XuYCSnMsNVBjLMNkTj6OZbudcPPTPYHqX0CJfpcdWciI1dM8uHETwmDxxiqEwCIE6WvXucWUetJgfu/A==} + engines: {node: '>=16.13'} + hasBin: true + + proxy-from-env@1.1.0: + resolution: {integrity: sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==} + + pvtsutils@1.3.6: + resolution: {integrity: sha512-PLgQXQ6H2FWCaeRak8vvk1GW462lMxB5s3Jm673N82zI4vqtVUPuZdffdZbPDFRoU8kAhItWFtPCWiPpp4/EDg==} + + pvutils@1.1.3: + resolution: {integrity: sha512-pMpnA0qRdFp32b1sJl1wOJNxZLQ2cbQx+k6tjNtZ8CpvVhNqEPRgivZ2WOUev2YMajecdH7ctUPDvEe87nariQ==} + engines: {node: '>=6.0.0'} + + rollup@4.35.0: + resolution: {integrity: sha512-kg6oI4g+vc41vePJyO6dHt/yl0Rz3Thv0kJeVQ3D1kS3E5XSuKbPc29G4IpT/Kv1KQwgHVcN+HtyS+HYLNSvQg==} + engines: {node: '>=18.0.0', npm: '>=8.0.0'} + hasBin: true + + rou3@0.5.1: + resolution: {integrity: sha512-OXMmJ3zRk2xeXFGfA3K+EOPHC5u7RDFG7lIOx0X1pdnhUkI8MdVrbV+sNsD80ElpUZ+MRHdyxPnFthq9VHs8uQ==} + + set-cookie-parser@2.7.1: + resolution: {integrity: sha512-IOc8uWeOZgnb3ptbCURJWNjWUPcO3ZnTTdzsurqERrP6nPyv+paC55vJM0LpOlT2ne+Ix+9+CRG1MNLlyZ4GjQ==} + + siginfo@2.0.0: + resolution: {integrity: sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==} + + source-map-js@1.2.1: + resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==} + engines: {node: '>=0.10.0'} + + stackback@0.0.2: + resolution: {integrity: sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==} + + std-env@3.8.1: + resolution: {integrity: sha512-vj5lIj3Mwf9D79hBkltk5qmkFI+biIKWS2IBxEyEU3AX1tUf7AoL8nSazCOiiqQsGKIq01SClsKEzweu34uwvA==} + + tinybench@2.9.0: + resolution: {integrity: sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==} + + tinyexec@0.3.2: + resolution: {integrity: sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==} + + tinypool@1.0.2: + resolution: {integrity: sha512-al6n+QEANGFOMf/dmUMsuS5/r9B06uwlyNjZZql/zv8J7ybHCgoihBNORZCY2mzUuAnomQa2JdhyHKzZxPCrFA==} + engines: {node: ^18.0.0 || >=20.0.0} + + tinyrainbow@1.2.0: + resolution: {integrity: sha512-weEDEq7Z5eTHPDh4xjX789+fHfF+P8boiFB+0vbWzpbnbsEr/GRaohi/uMKxg8RZMXnl1ItAi/IUHWMsjDV7kQ==} + engines: {node: '>=14.0.0'} + + tinyspy@3.0.2: + resolution: {integrity: sha512-n1cw8k1k0x4pgA2+9XrOkFydTerNcJ1zWCO5Nn9scWHTD+5tp8dghT2x1uduQePZTZgd3Tupf+x9BxJjeJi77Q==} + engines: {node: '>=14.0.0'} + + tslib@2.8.1: + resolution: {integrity: sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==} + + typescript@5.8.2: + resolution: {integrity: sha512-aJn6wq13/afZp/jT9QZmwEjDqqvSGp1VT5GVg+f/t6/oVyrgXM6BY1h9BRh/O5p3PlUPAe+WuiEZOmb/49RqoQ==} + engines: {node: '>=14.17'} + hasBin: true + + uncrypto@0.1.3: + resolution: {integrity: sha512-Ql87qFHB3s/De2ClA9e0gsnS6zXG27SkTiSJwjCc9MebbfapQfuPzumMIUMi38ezPZVNFcHI9sUIepeQfw8J8Q==} + + undici-types@6.19.8: + resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} + + valibot@1.0.0-beta.15: + resolution: {integrity: sha512-BKy8XosZkDHWmYC+cJG74LBzP++Gfntwi33pP3D3RKztz2XV9jmFWnkOi21GoqARP8wAWARwhV6eTr1JcWzjGw==} + peerDependencies: + typescript: '>=5' + peerDependenciesMeta: + typescript: + optional: true + + vite-node@2.1.9: + resolution: {integrity: sha512-AM9aQ/IPrW/6ENLQg3AGY4K1N2TGZdR5e4gu/MmmR2xR3Ll1+dib+nook92g4TV3PXVyeyxdWwtaCAiUL0hMxA==} + engines: {node: ^18.0.0 || >=20.0.0} + hasBin: true + + vite@5.4.14: + resolution: {integrity: sha512-EK5cY7Q1D8JNhSaPKVK4pwBFvaTmZxEnoKXLG/U9gmdDcihQGNzFlgIvaxezFR4glP1LsuiedwMBqCXH3wZccA==} + engines: {node: ^18.0.0 || >=20.0.0} + hasBin: true + peerDependencies: + '@types/node': ^18.0.0 || >=20.0.0 + less: '*' + lightningcss: ^1.21.0 + sass: '*' + sass-embedded: '*' + stylus: '*' + sugarss: '*' + terser: ^5.4.0 + peerDependenciesMeta: + '@types/node': + optional: true + less: + optional: true + lightningcss: + optional: true + sass: + optional: true + sass-embedded: + optional: true + stylus: + optional: true + sugarss: + optional: true + terser: + optional: true + + vitest@2.1.9: + resolution: {integrity: sha512-MSmPM9REYqDGBI8439mA4mWhV5sKmDlBKWIYbA3lRb2PTHACE0mgKwA8yQ2xq9vxDTuk4iPrECBAEW2aoFXY0Q==} + engines: {node: ^18.0.0 || >=20.0.0} + hasBin: true + peerDependencies: + '@edge-runtime/vm': '*' + '@types/node': ^18.0.0 || >=20.0.0 + '@vitest/browser': 2.1.9 + '@vitest/ui': 2.1.9 + happy-dom: '*' + jsdom: '*' + peerDependenciesMeta: + '@edge-runtime/vm': + optional: true + '@types/node': + optional: true + '@vitest/browser': + optional: true + '@vitest/ui': + optional: true + happy-dom: + optional: true + jsdom: + optional: true + + why-is-node-running@2.3.0: + resolution: {integrity: sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==} + engines: {node: '>=8'} + hasBin: true + + zod@3.24.2: + resolution: {integrity: sha512-lY7CDW43ECgW9u1TcT3IoXHflywfVqDYze4waEz812jR/bZ8FHDsl7pFQoSZTz5N+2NqRXs8GBwnAwo3ZNxqhQ==} + +snapshots: + + '@better-auth/utils@0.2.3': + dependencies: + uncrypto: 0.1.3 + + '@better-fetch/fetch@1.1.15': {} + + '@esbuild/aix-ppc64@0.21.5': + optional: true + + '@esbuild/android-arm64@0.21.5': + optional: true + + '@esbuild/android-arm@0.21.5': + optional: true + + '@esbuild/android-x64@0.21.5': + optional: true + + '@esbuild/darwin-arm64@0.21.5': + optional: true + + '@esbuild/darwin-x64@0.21.5': + optional: true + + '@esbuild/freebsd-arm64@0.21.5': + optional: true + + '@esbuild/freebsd-x64@0.21.5': + optional: true + + '@esbuild/linux-arm64@0.21.5': + optional: true + + '@esbuild/linux-arm@0.21.5': + optional: true + + '@esbuild/linux-ia32@0.21.5': + optional: true + + '@esbuild/linux-loong64@0.21.5': + optional: true + + '@esbuild/linux-mips64el@0.21.5': + optional: true + + '@esbuild/linux-ppc64@0.21.5': + optional: true + + '@esbuild/linux-riscv64@0.21.5': + optional: true + + '@esbuild/linux-s390x@0.21.5': + optional: true + + '@esbuild/linux-x64@0.21.5': + optional: true + + '@esbuild/netbsd-x64@0.21.5': + optional: true + + '@esbuild/openbsd-x64@0.21.5': + optional: true + + '@esbuild/sunos-x64@0.21.5': + optional: true + + '@esbuild/win32-arm64@0.21.5': + optional: true + + '@esbuild/win32-ia32@0.21.5': + optional: true + + '@esbuild/win32-x64@0.21.5': + optional: true + + '@hexagon/base64@1.1.28': {} + + '@jridgewell/sourcemap-codec@1.5.0': {} + + '@levischuck/tiny-cbor@0.2.11': {} + + '@noble/ciphers@0.6.0': {} + + '@noble/hashes@1.7.1': {} + + '@peculiar/asn1-android@2.3.15': + dependencies: + '@peculiar/asn1-schema': 2.3.15 + asn1js: 3.0.5 + tslib: 2.8.1 + + '@peculiar/asn1-ecc@2.3.15': + dependencies: + '@peculiar/asn1-schema': 2.3.15 + '@peculiar/asn1-x509': 2.3.15 + asn1js: 3.0.5 + tslib: 2.8.1 + + '@peculiar/asn1-rsa@2.3.15': + dependencies: + '@peculiar/asn1-schema': 2.3.15 + '@peculiar/asn1-x509': 2.3.15 + asn1js: 3.0.5 + tslib: 2.8.1 + + '@peculiar/asn1-schema@2.3.15': + dependencies: + asn1js: 3.0.5 + pvtsutils: 1.3.6 + tslib: 2.8.1 + + '@peculiar/asn1-x509@2.3.15': + dependencies: + '@peculiar/asn1-schema': 2.3.15 + asn1js: 3.0.5 + pvtsutils: 1.3.6 + tslib: 2.8.1 + + '@prisma/client@5.22.0(prisma@5.22.0)': + optionalDependencies: + prisma: 5.22.0 + + '@prisma/debug@5.22.0': {} + + '@prisma/engines-version@5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2': {} + + '@prisma/engines@5.22.0': + dependencies: + '@prisma/debug': 5.22.0 + '@prisma/engines-version': 5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2 + '@prisma/fetch-engine': 5.22.0 + '@prisma/get-platform': 5.22.0 + + '@prisma/fetch-engine@5.22.0': + dependencies: + '@prisma/debug': 5.22.0 + '@prisma/engines-version': 5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2 + '@prisma/get-platform': 5.22.0 + + '@prisma/get-platform@5.22.0': + dependencies: + '@prisma/debug': 5.22.0 + + '@rollup/rollup-android-arm-eabi@4.35.0': + optional: true + + '@rollup/rollup-android-arm64@4.35.0': + optional: true + + '@rollup/rollup-darwin-arm64@4.35.0': + optional: true + + '@rollup/rollup-darwin-x64@4.35.0': + optional: true + + '@rollup/rollup-freebsd-arm64@4.35.0': + optional: true + + '@rollup/rollup-freebsd-x64@4.35.0': + optional: true + + '@rollup/rollup-linux-arm-gnueabihf@4.35.0': + optional: true + + '@rollup/rollup-linux-arm-musleabihf@4.35.0': + optional: true + + '@rollup/rollup-linux-arm64-gnu@4.35.0': + optional: true + + '@rollup/rollup-linux-arm64-musl@4.35.0': + optional: true + + '@rollup/rollup-linux-loongarch64-gnu@4.35.0': + optional: true + + '@rollup/rollup-linux-powerpc64le-gnu@4.35.0': + optional: true + + '@rollup/rollup-linux-riscv64-gnu@4.35.0': + optional: true + + '@rollup/rollup-linux-s390x-gnu@4.35.0': + optional: true + + '@rollup/rollup-linux-x64-gnu@4.35.0': + optional: true + + '@rollup/rollup-linux-x64-musl@4.35.0': + optional: true + + '@rollup/rollup-win32-arm64-msvc@4.35.0': + optional: true + + '@rollup/rollup-win32-ia32-msvc@4.35.0': + optional: true + + '@rollup/rollup-win32-x64-msvc@4.35.0': + optional: true + + '@simplewebauthn/browser@13.1.0': {} + + '@simplewebauthn/server@13.1.1': + dependencies: + '@hexagon/base64': 1.1.28 + '@levischuck/tiny-cbor': 0.2.11 + '@peculiar/asn1-android': 2.3.15 + '@peculiar/asn1-ecc': 2.3.15 + '@peculiar/asn1-rsa': 2.3.15 + '@peculiar/asn1-schema': 2.3.15 + '@peculiar/asn1-x509': 2.3.15 + + '@types/estree@1.0.6': {} + + '@types/node@20.17.24': + dependencies: + undici-types: 6.19.8 + + '@vitest/expect@2.1.9': + dependencies: + '@vitest/spy': 2.1.9 + '@vitest/utils': 2.1.9 + chai: 5.2.0 + tinyrainbow: 1.2.0 + + '@vitest/mocker@2.1.9(vite@5.4.14(@types/node@20.17.24))': + dependencies: + '@vitest/spy': 2.1.9 + estree-walker: 3.0.3 + magic-string: 0.30.17 + optionalDependencies: + vite: 5.4.14(@types/node@20.17.24) + + '@vitest/pretty-format@2.1.9': + dependencies: + tinyrainbow: 1.2.0 + + '@vitest/runner@2.1.9': + dependencies: + '@vitest/utils': 2.1.9 + pathe: 1.1.2 + + '@vitest/snapshot@2.1.9': + dependencies: + '@vitest/pretty-format': 2.1.9 + magic-string: 0.30.17 + pathe: 1.1.2 + + '@vitest/spy@2.1.9': + dependencies: + tinyspy: 3.0.2 + + '@vitest/utils@2.1.9': + dependencies: + '@vitest/pretty-format': 2.1.9 + loupe: 3.1.3 + tinyrainbow: 1.2.0 + + asn1js@3.0.5: + dependencies: + pvtsutils: 1.3.6 + pvutils: 1.1.3 + tslib: 2.8.1 + + assertion-error@2.0.1: {} + + asynckit@0.4.0: {} + + axios@1.8.2: + dependencies: + follow-redirects: 1.15.9 + form-data: 4.0.2 + proxy-from-env: 1.1.0 + transitivePeerDependencies: + - debug + + better-auth@1.2.3(typescript@5.8.2): + dependencies: + '@better-auth/utils': 0.2.3 + '@better-fetch/fetch': 1.1.15 + '@noble/ciphers': 0.6.0 + '@noble/hashes': 1.7.1 + '@simplewebauthn/browser': 13.1.0 + '@simplewebauthn/server': 13.1.1 + better-call: 1.0.4 + defu: 6.1.4 + jose: 5.10.0 + kysely: 0.27.6 + nanostores: 0.11.4 + valibot: 1.0.0-beta.15(typescript@5.8.2) + zod: 3.24.2 + transitivePeerDependencies: + - typescript + + better-call@1.0.4: + dependencies: + '@better-fetch/fetch': 1.1.15 + rou3: 0.5.1 + set-cookie-parser: 2.7.1 + uncrypto: 0.1.3 + + cac@6.7.14: {} + + call-bind-apply-helpers@1.0.2: + dependencies: + es-errors: 1.3.0 + function-bind: 1.1.2 + + chai@5.2.0: + dependencies: + assertion-error: 2.0.1 + check-error: 2.1.1 + deep-eql: 5.0.2 + loupe: 3.1.3 + pathval: 2.0.0 + + check-error@2.1.1: {} + + combined-stream@1.0.8: + dependencies: + delayed-stream: 1.0.0 + + debug@4.4.0: + dependencies: + ms: 2.1.3 + + deep-eql@5.0.2: {} + + defu@6.1.4: {} + + delayed-stream@1.0.0: {} + + dunder-proto@1.0.1: + dependencies: + call-bind-apply-helpers: 1.0.2 + es-errors: 1.3.0 + gopd: 1.2.0 + + encore-better-auth@0.2.0(better-auth@1.2.3(typescript@5.8.2)): + dependencies: + better-auth: 1.2.3(typescript@5.8.2) + encore.dev: 1.46.7 + zod: 3.24.2 + + encore.dev@1.46.7: {} + + es-define-property@1.0.1: {} + + es-errors@1.3.0: {} + + es-module-lexer@1.6.0: {} + + es-object-atoms@1.1.1: + dependencies: + es-errors: 1.3.0 + + es-set-tostringtag@2.1.0: + dependencies: + es-errors: 1.3.0 + get-intrinsic: 1.3.0 + has-tostringtag: 1.0.2 + hasown: 2.0.2 + + esbuild@0.21.5: + optionalDependencies: + '@esbuild/aix-ppc64': 0.21.5 + '@esbuild/android-arm': 0.21.5 + '@esbuild/android-arm64': 0.21.5 + '@esbuild/android-x64': 0.21.5 + '@esbuild/darwin-arm64': 0.21.5 + '@esbuild/darwin-x64': 0.21.5 + '@esbuild/freebsd-arm64': 0.21.5 + '@esbuild/freebsd-x64': 0.21.5 + '@esbuild/linux-arm': 0.21.5 + '@esbuild/linux-arm64': 0.21.5 + '@esbuild/linux-ia32': 0.21.5 + '@esbuild/linux-loong64': 0.21.5 + '@esbuild/linux-mips64el': 0.21.5 + '@esbuild/linux-ppc64': 0.21.5 + '@esbuild/linux-riscv64': 0.21.5 + '@esbuild/linux-s390x': 0.21.5 + '@esbuild/linux-x64': 0.21.5 + '@esbuild/netbsd-x64': 0.21.5 + '@esbuild/openbsd-x64': 0.21.5 + '@esbuild/sunos-x64': 0.21.5 + '@esbuild/win32-arm64': 0.21.5 + '@esbuild/win32-ia32': 0.21.5 + '@esbuild/win32-x64': 0.21.5 + + estree-walker@3.0.3: + dependencies: + '@types/estree': 1.0.6 + + expect-type@1.2.0: {} + + follow-redirects@1.15.9: {} + + form-data@4.0.2: + dependencies: + asynckit: 0.4.0 + combined-stream: 1.0.8 + es-set-tostringtag: 2.1.0 + mime-types: 2.1.35 + + fsevents@2.3.3: + optional: true + + function-bind@1.1.2: {} + + get-intrinsic@1.3.0: + dependencies: + call-bind-apply-helpers: 1.0.2 + es-define-property: 1.0.1 + es-errors: 1.3.0 + es-object-atoms: 1.1.1 + function-bind: 1.1.2 + get-proto: 1.0.1 + gopd: 1.2.0 + has-symbols: 1.1.0 + hasown: 2.0.2 + math-intrinsics: 1.1.0 + + get-proto@1.0.1: + dependencies: + dunder-proto: 1.0.1 + es-object-atoms: 1.1.1 + + gopd@1.2.0: {} + + has-symbols@1.1.0: {} + + has-tostringtag@1.0.2: + dependencies: + has-symbols: 1.1.0 + + hasown@2.0.2: + dependencies: + function-bind: 1.1.2 + + jose@5.10.0: {} + + kysely@0.27.6: {} + + loupe@3.1.3: {} + + magic-string@0.30.17: + dependencies: + '@jridgewell/sourcemap-codec': 1.5.0 + + math-intrinsics@1.1.0: {} + + mime-db@1.52.0: {} + + mime-types@2.1.35: + dependencies: + mime-db: 1.52.0 + + ms@2.1.3: {} + + nanoid@3.3.9: {} + + nanostores@0.11.4: {} + + pathe@1.1.2: {} + + pathval@2.0.0: {} + + picocolors@1.1.1: {} + + postcss@8.5.3: + dependencies: + nanoid: 3.3.9 + picocolors: 1.1.1 + source-map-js: 1.2.1 + + posthog-node@4.10.1: + dependencies: + axios: 1.8.2 + transitivePeerDependencies: + - debug + + prisma@5.22.0: + dependencies: + '@prisma/engines': 5.22.0 + optionalDependencies: + fsevents: 2.3.3 + + proxy-from-env@1.1.0: {} + + pvtsutils@1.3.6: + dependencies: + tslib: 2.8.1 + + pvutils@1.1.3: {} + + rollup@4.35.0: + dependencies: + '@types/estree': 1.0.6 + optionalDependencies: + '@rollup/rollup-android-arm-eabi': 4.35.0 + '@rollup/rollup-android-arm64': 4.35.0 + '@rollup/rollup-darwin-arm64': 4.35.0 + '@rollup/rollup-darwin-x64': 4.35.0 + '@rollup/rollup-freebsd-arm64': 4.35.0 + '@rollup/rollup-freebsd-x64': 4.35.0 + '@rollup/rollup-linux-arm-gnueabihf': 4.35.0 + '@rollup/rollup-linux-arm-musleabihf': 4.35.0 + '@rollup/rollup-linux-arm64-gnu': 4.35.0 + '@rollup/rollup-linux-arm64-musl': 4.35.0 + '@rollup/rollup-linux-loongarch64-gnu': 4.35.0 + '@rollup/rollup-linux-powerpc64le-gnu': 4.35.0 + '@rollup/rollup-linux-riscv64-gnu': 4.35.0 + '@rollup/rollup-linux-s390x-gnu': 4.35.0 + '@rollup/rollup-linux-x64-gnu': 4.35.0 + '@rollup/rollup-linux-x64-musl': 4.35.0 + '@rollup/rollup-win32-arm64-msvc': 4.35.0 + '@rollup/rollup-win32-ia32-msvc': 4.35.0 + '@rollup/rollup-win32-x64-msvc': 4.35.0 + fsevents: 2.3.3 + + rou3@0.5.1: {} + + set-cookie-parser@2.7.1: {} + + siginfo@2.0.0: {} + + source-map-js@1.2.1: {} + + stackback@0.0.2: {} + + std-env@3.8.1: {} + + tinybench@2.9.0: {} + + tinyexec@0.3.2: {} + + tinypool@1.0.2: {} + + tinyrainbow@1.2.0: {} + + tinyspy@3.0.2: {} + + tslib@2.8.1: {} + + typescript@5.8.2: {} + + uncrypto@0.1.3: {} + + undici-types@6.19.8: {} + + valibot@1.0.0-beta.15(typescript@5.8.2): + optionalDependencies: + typescript: 5.8.2 + + vite-node@2.1.9(@types/node@20.17.24): + dependencies: + cac: 6.7.14 + debug: 4.4.0 + es-module-lexer: 1.6.0 + pathe: 1.1.2 + vite: 5.4.14(@types/node@20.17.24) + transitivePeerDependencies: + - '@types/node' + - less + - lightningcss + - sass + - sass-embedded + - stylus + - sugarss + - supports-color + - terser + + vite@5.4.14(@types/node@20.17.24): + dependencies: + esbuild: 0.21.5 + postcss: 8.5.3 + rollup: 4.35.0 + optionalDependencies: + '@types/node': 20.17.24 + fsevents: 2.3.3 + + vitest@2.1.9(@types/node@20.17.24): + dependencies: + '@vitest/expect': 2.1.9 + '@vitest/mocker': 2.1.9(vite@5.4.14(@types/node@20.17.24)) + '@vitest/pretty-format': 2.1.9 + '@vitest/runner': 2.1.9 + '@vitest/snapshot': 2.1.9 + '@vitest/spy': 2.1.9 + '@vitest/utils': 2.1.9 + chai: 5.2.0 + debug: 4.4.0 + expect-type: 1.2.0 + magic-string: 0.30.17 + pathe: 1.1.2 + std-env: 3.8.1 + tinybench: 2.9.0 + tinyexec: 0.3.2 + tinypool: 1.0.2 + tinyrainbow: 1.2.0 + vite: 5.4.14(@types/node@20.17.24) + vite-node: 2.1.9(@types/node@20.17.24) + why-is-node-running: 2.3.0 + optionalDependencies: + '@types/node': 20.17.24 + transitivePeerDependencies: + - less + - lightningcss + - msw + - sass + - sass-embedded + - stylus + - sugarss + - supports-color + - terser + + why-is-node-running@2.3.0: + dependencies: + siginfo: 2.0.0 + stackback: 0.0.2 + + zod@3.24.2: {} diff --git a/ts/saas-nextforge-encore/backend/tsconfig.json b/ts/saas-nextforge-encore/backend/tsconfig.json new file mode 100644 index 00000000..ffe650c7 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/tsconfig.json @@ -0,0 +1,31 @@ +{ + "$schema": "https://json.schemastore.org/tsconfig", + "compilerOptions": { + /* Basic Options */ + "lib": ["ES2022"], + "target": "ES2022", + "module": "ES2022", + "types": ["node"], + "paths": { + "~encore/*": ["./encore.gen/*"] + }, + + /* Workspace Settings */ + "composite": true, + + /* Strict Type-Checking Options */ + "strict": true, + + /* Module Resolution Options */ + "moduleResolution": "bundler", + "allowSyntheticDefaultImports": true, + "isolatedModules": true, + "sourceMap": true, + + "declaration": true, + + /* Advanced Options */ + "forceConsistentCasingInFileNames": true, + "skipLibCheck": true + } +} diff --git a/ts/saas-nextforge-encore/backend/url/encore.service.ts b/ts/saas-nextforge-encore/backend/url/encore.service.ts new file mode 100644 index 00000000..c1a7e91b --- /dev/null +++ b/ts/saas-nextforge-encore/backend/url/encore.service.ts @@ -0,0 +1,7 @@ +import { Service } from "encore.dev/service"; + +// Encore will consider this directory and all its subdirectories as part of the "url" service. +// https://encore.dev/docs/ts/primitives/services + +// The url service is used to shorten URLs. +export default new Service("url"); diff --git a/ts/saas-nextforge-encore/backend/url/migrations/1_create_tables.up.sql b/ts/saas-nextforge-encore/backend/url/migrations/1_create_tables.up.sql new file mode 100644 index 00000000..df73b019 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/url/migrations/1_create_tables.up.sql @@ -0,0 +1,4 @@ +CREATE TABLE url ( + id TEXT PRIMARY KEY, + original_url TEXT NOT NULL +); diff --git a/ts/saas-nextforge-encore/backend/url/url.test.ts b/ts/saas-nextforge-encore/backend/url/url.test.ts new file mode 100644 index 00000000..983146ef --- /dev/null +++ b/ts/saas-nextforge-encore/backend/url/url.test.ts @@ -0,0 +1,10 @@ +import { describe, expect, test } from "vitest"; +import { get, shorten } from "./url.js"; + +describe("shorten", () => { + test("getting a shortened url should give back the original", async () => { + const resp = await shorten({ url: "https://example.com" }); + const url = await get({ id: resp.id }); + expect(url.url).toBe("https://example.com"); + }); +}); diff --git a/ts/saas-nextforge-encore/backend/url/url.ts b/ts/saas-nextforge-encore/backend/url/url.ts new file mode 100644 index 00000000..78f5f0d6 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/url/url.ts @@ -0,0 +1,60 @@ +import { api, APIError } from "encore.dev/api"; +import { SQLDatabase } from "encore.dev/storage/sqldb"; +import { randomBytes } from "node:crypto"; + +// 'url' database is used to store the URLs that are being shortened. +const db = new SQLDatabase("url", { migrations: "./migrations" }); + +interface URL { + id: string; // short-form URL id + url: string; // complete URL, in long form +} + +interface ShortenParams { + url: string; // the URL to shorten +} + +// shorten shortens a URL. +export const shorten = api( + { expose: true, auth: false, method: "POST", path: "/url" }, + async ({ url }: ShortenParams): Promise => { + const id = randomBytes(6).toString("base64url"); + await db.exec` + INSERT INTO url (id, original_url) + VALUES (${id}, ${url}) + `; + return { id, url }; + } +); + +// Get retrieves the original URL for the id. +export const get = api( + { expose: true, auth: false, method: "GET", path: "/url/:id" }, + async ({ id }: { id: string }): Promise => { + const row = await db.queryRow` + SELECT original_url FROM url WHERE id = ${id} + `; + if (!row) throw APIError.notFound("url not found"); + return { id, url: row.original_url }; + } +); + +interface ListResponse { + urls: URL[]; +} + +// List retrieves all URLs. +export const list = api( + { expose: false, method: "GET", path: "/url" }, + async (): Promise => { + const rows = db.query` + SELECT id, original_url + FROM url + `; + const urls: URL[] = []; + for await (const row of rows) { + urls.push({ id: row.id, url: row.original_url }); + } + return { urls }; + } +); diff --git a/ts/saas-nextforge-encore/backend/vite.config.ts b/ts/saas-nextforge-encore/backend/vite.config.ts new file mode 100644 index 00000000..a9a3b4a8 --- /dev/null +++ b/ts/saas-nextforge-encore/backend/vite.config.ts @@ -0,0 +1,11 @@ +/// +import { defineConfig } from "vite"; +import path from "path"; + +export default defineConfig({ + resolve: { + alias: { + "~encore": path.resolve(__dirname, "./encore.gen"), + }, + }, +}); diff --git a/ts/saas-nextforge-encore/biome.json b/ts/saas-nextforge-encore/biome.json new file mode 100644 index 00000000..80c91def --- /dev/null +++ b/ts/saas-nextforge-encore/biome.json @@ -0,0 +1,18 @@ +{ + "$schema": "https://biomejs.dev/schemas/1.9.4/schema.json", + "extends": ["ultracite"], + "javascript": { + "globals": ["Liveblocks"] + }, + "files": { + "ignore": [ + "frontend/packages/design-system/components/ui/**", + "frontend/packages/design-system/lib/**", + "frontend/packages/design-system/hooks/**", + "frontend/packages/collaboration/config.ts", + "frontend/apps/docs/**/*.json", + "frontend/apps/email/.react-email/**", + "backend/**/better-auth.routes.ts" + ] + } +} diff --git a/ts/saas-nextforge-encore/frontend/README.md b/ts/saas-nextforge-encore/frontend/README.md new file mode 100644 index 00000000..f03d510a --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/README.md @@ -0,0 +1,25 @@ +# next-forge + +**Production-grade Turborepo template for Next.js apps.** + +
+ + + +
+ +[next-forge](https://github.com/haydenbleasel/next-forge) is a [Next.js](https://nextjs.org/) project boilerplate for modern web application. It is designed to be a comprehensive starting point for new apps, providing a solid, opinionated foundation with a minimal amount of configuration. + +Clone the repo using: + +```sh +npx next-forge@latest init +``` + +Then read the [docs](https://docs.next-forge.com) for more information. + + + + + +Made with [contrib.rocks](https://contrib.rocks). diff --git a/ts/saas-nextforge-encore/frontend/apps/app/.env.example b/ts/saas-nextforge-encore/frontend/apps/app/.env.example new file mode 100644 index 00000000..5a6f5e8c --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/.env.example @@ -0,0 +1,34 @@ +# Server +CLERK_SECRET_KEY="" +CLERK_WEBHOOK_SECRET="" +RESEND_FROM="" +DATABASE_URL="" +RESEND_TOKEN="" +STRIPE_SECRET_KEY="" +STRIPE_WEBHOOK_SECRET="" +BETTERSTACK_API_KEY="" +BETTERSTACK_URL="" +FLAGS_SECRET="" +ARCJET_KEY="" +SVIX_TOKEN="" +LIVEBLOCKS_SECRET="" +BASEHUB_TOKEN="" +VERCEL_PROJECT_PRODUCTION_URL="http://localhost:3000" +KNOCK_API_KEY="" +KNOCK_FEED_CHANNEL_ID="" +KNOCK_SECRET_API_KEY="" + +# Client +NEXT_PUBLIC_CLERK_PUBLISHABLE_KEY="" +NEXT_PUBLIC_CLERK_SIGN_IN_URL="/sign-in" +NEXT_PUBLIC_CLERK_SIGN_UP_URL="/sign-up" +NEXT_PUBLIC_CLERK_AFTER_SIGN_IN_URL="/" +NEXT_PUBLIC_CLERK_AFTER_SIGN_UP_URL="/" +NEXT_PUBLIC_GA_MEASUREMENT_ID="" +NEXT_PUBLIC_KNOCK_API_KEY="" +NEXT_PUBLIC_KNOCK_FEED_CHANNEL_ID="" +NEXT_PUBLIC_POSTHOG_KEY="" +NEXT_PUBLIC_POSTHOG_HOST="" +NEXT_PUBLIC_APP_URL="http://localhost:3000" +NEXT_PUBLIC_WEB_URL="http://localhost:3001" +NEXT_PUBLIC_DOCS_URL="http://localhost:3004" diff --git a/ts/saas-nextforge-encore/frontend/apps/app/.gitignore b/ts/saas-nextforge-encore/frontend/apps/app/.gitignore new file mode 100644 index 00000000..9045095d --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/.gitignore @@ -0,0 +1,45 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# dependencies +/node_modules +/.pnp +.pnp.js + +# testing +/coverage + +# next.js +/.next/ +/out/ + +# production +/build + +# misc +.DS_Store +*.pem + +# debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* +.pnpm-debug.log* + +# local env files +.env*.local + +# vercel +.vercel + +# typescript +*.tsbuildinfo +next-env.d.ts + +# prisma +.env + +# react.email +.react-email + +# Sentry +.sentryclirc diff --git a/ts/saas-nextforge-encore/frontend/apps/app/__tests__/sign-in.test.tsx b/ts/saas-nextforge-encore/frontend/apps/app/__tests__/sign-in.test.tsx new file mode 100644 index 00000000..5bd4c283 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/__tests__/sign-in.test.tsx @@ -0,0 +1,13 @@ +import { render, screen } from '@testing-library/react'; +import { expect, test } from 'vitest'; +import Page from '../app/(unauthenticated)/sign-in/[[...sign-in]]/page'; + +test('Sign In Page', () => { + render(); + expect( + screen.getByRole('heading', { + level: 1, + name: 'Welcome back', + }) + ).toBeDefined(); +}); diff --git a/ts/saas-nextforge-encore/frontend/apps/app/__tests__/sign-up.test.tsx b/ts/saas-nextforge-encore/frontend/apps/app/__tests__/sign-up.test.tsx new file mode 100644 index 00000000..9a4dcbeb --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/__tests__/sign-up.test.tsx @@ -0,0 +1,13 @@ +import { render, screen } from '@testing-library/react'; +import { expect, test } from 'vitest'; +import Page from '../app/(unauthenticated)/sign-up/[[...sign-up]]/page'; + +test('Sign Up Page', () => { + render(); + expect( + screen.getByRole('heading', { + level: 1, + name: 'Create an account', + }) + ).toBeDefined(); +}); diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/avatar-stack.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/avatar-stack.tsx new file mode 100644 index 00000000..9db0f9cc --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/avatar-stack.tsx @@ -0,0 +1,58 @@ +'use client'; + +import { useOthers, useSelf } from '@repo/collaboration/hooks'; +import { + Avatar, + AvatarFallback, + AvatarImage, +} from '@repo/design-system/components/ui/avatar'; +import { + Tooltip, + TooltipContent, + TooltipTrigger, +} from '@repo/design-system/components/ui/tooltip'; + +type PresenceAvatarProps = { + info?: Liveblocks['UserMeta']['info']; +}; + +const PresenceAvatar = ({ info }: PresenceAvatarProps) => ( + + + + + + {info?.name?.slice(0, 2)} + + + + +

{info?.name ?? 'Unknown'}

+
+
+); + +export const AvatarStack = () => { + const others = useOthers(); + const self = useSelf(); + const hasMoreUsers = others.length > 3; + + return ( +
+ {others.slice(0, 3).map(({ connectionId, info }) => ( + + ))} + + {hasMoreUsers && ( + + )} + + {self && } +
+ ); +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/collaboration-provider.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/collaboration-provider.tsx new file mode 100644 index 00000000..1ddc4b53 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/collaboration-provider.tsx @@ -0,0 +1,48 @@ +'use client'; + +import { getUsers } from '@/app/actions/users/get'; +import { searchUsers } from '@/app/actions/users/search'; +import { Room } from '@repo/collaboration/room'; +import type { ReactNode } from 'react'; + +export const CollaborationProvider = ({ + orgId, + children, +}: { + orgId: string; + children: ReactNode; +}) => { + const resolveUsers = async ({ userIds }: { userIds: string[] }) => { + const response = await getUsers(userIds); + + if ('error' in response) { + throw new Error('Problem resolving users'); + } + + return response.data; + }; + + const resolveMentionSuggestions = async ({ text }: { text: string }) => { + const response = await searchUsers(text); + + if ('error' in response) { + throw new Error('Problem resolving mention suggestions'); + } + + return response.data; + }; + + return ( + Loading... + } + resolveUsers={resolveUsers} + resolveMentionSuggestions={resolveMentionSuggestions} + > + {children} + + ); +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/cursors.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/cursors.tsx new file mode 100644 index 00000000..9be7e577 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/cursors.tsx @@ -0,0 +1,106 @@ +'use client'; + +import { useMyPresence, useOthers } from '@repo/collaboration/hooks'; +import { useEffect } from 'react'; + +const Cursor = ({ + name, + color, + x, + y, +}: { + name: string | undefined; + color: string; + x: number; + y: number; +}) => ( +
+ + Cursor + + +
+ {name} +
+
+); + +export const Cursors = () => { + /** + * useMyPresence returns the presence of the current user and a function to update it. + * updateMyPresence is different than the setState function returned by the useState hook from React. + * You don't need to pass the full presence object to update it. + * See https://liveblocks.io/docs/api-reference/liveblocks-react#useMyPresence for more information + */ + const [_cursor, updateMyPresence] = useMyPresence(); + + /** + * Return all the other users in the room and their presence (a cursor position in this case) + */ + const others = useOthers(); + + useEffect(() => { + const onPointerMove = (event: PointerEvent) => { + // Update the user cursor position on every pointer move + updateMyPresence({ + cursor: { + x: Math.round(event.clientX), + y: Math.round(event.clientY), + }, + }); + }; + + const onPointerLeave = () => { + // When the pointer goes out, set cursor to null + updateMyPresence({ + cursor: null, + }); + }; + + document.body.addEventListener('pointermove', onPointerMove); + document.body.addEventListener('pointerleave', onPointerLeave); + + return () => { + document.body.removeEventListener('pointermove', onPointerMove); + document.body.removeEventListener('pointerleave', onPointerLeave); + }; + }, [updateMyPresence]); + + return others.map(({ connectionId, presence, info }) => { + if (!presence.cursor) { + return null; + } + + return ( + + ); + }); +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/header.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/header.tsx new file mode 100644 index 00000000..3127d361 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/header.tsx @@ -0,0 +1,43 @@ +import { + Breadcrumb, + BreadcrumbItem, + BreadcrumbLink, + BreadcrumbList, + BreadcrumbPage, + BreadcrumbSeparator, +} from '@repo/design-system/components/ui/breadcrumb'; +import { Separator } from '@repo/design-system/components/ui/separator'; +import { SidebarTrigger } from '@repo/design-system/components/ui/sidebar'; +import { Fragment, type ReactNode } from 'react'; + +type HeaderProps = { + pages: string[]; + page: string; + children?: ReactNode; +}; + +export const Header = ({ pages, page, children }: HeaderProps) => ( +
+
+ + + + + {pages.map((page, index) => ( + + {index > 0 && } + + {page} + + + ))} + + + {page} + + + +
+ {children} +
+); diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/posthog-identifier.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/posthog-identifier.tsx new file mode 100644 index 00000000..75118a5d --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/posthog-identifier.tsx @@ -0,0 +1,46 @@ +'use client'; + +import { useAnalytics } from '@repo/analytics/posthog/client'; +import { useUser } from '@repo/auth/client'; +import { usePathname, useSearchParams } from 'next/navigation'; +import { useEffect, useRef } from 'react'; + +export const PostHogIdentifier = () => { + const { user } = useUser(); + const identified = useRef(false); + const pathname = usePathname(); + const searchParams = useSearchParams(); + const analytics = useAnalytics(); + + useEffect(() => { + // Track pageviews + if (pathname && analytics) { + let url = window.origin + pathname; + if (searchParams.toString()) { + url = `${url}?${searchParams.toString()}`; + } + analytics.capture('$pageview', { + $current_url: url, + }); + } + }, [pathname, searchParams, analytics]); + + useEffect(() => { + if (!user || identified.current) { + return; + } + + analytics.identify(user.id, { + email: user.emailAddresses.at(0)?.emailAddress, + firstName: user.firstName, + lastName: user.lastName, + createdAt: user.createdAt, + avatar: user.imageUrl, + phoneNumber: user.phoneNumbers.at(0)?.phoneNumber, + }); + + identified.current = true; + }, [user, analytics]); + + return null; +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/search.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/search.tsx new file mode 100644 index 00000000..ccf4c1ae --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/search.tsx @@ -0,0 +1,26 @@ +import { Button } from '@repo/design-system/components/ui/button'; +import { Input } from '@repo/design-system/components/ui/input'; +import { ArrowRightIcon, SearchIcon } from 'lucide-react'; + +export const Search = () => ( +
+
+
+ +
+ + +
+
+); diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/sidebar.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/sidebar.tsx new file mode 100644 index 00000000..bf07a618 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/components/sidebar.tsx @@ -0,0 +1,359 @@ +'use client'; + +import { OrganizationSwitcher, UserButton } from '@repo/auth/client'; +import { ModeToggle } from '@repo/design-system/components/mode-toggle'; +import { Button } from '@repo/design-system/components/ui/button'; +import { + Collapsible, + CollapsibleContent, + CollapsibleTrigger, +} from '@repo/design-system/components/ui/collapsible'; +import { + DropdownMenu, + DropdownMenuContent, + DropdownMenuItem, + DropdownMenuSeparator, + DropdownMenuTrigger, +} from '@repo/design-system/components/ui/dropdown-menu'; +import { + Sidebar, + SidebarContent, + SidebarFooter, + SidebarGroup, + SidebarGroupContent, + SidebarGroupLabel, + SidebarHeader, + SidebarInset, + SidebarMenu, + SidebarMenuAction, + SidebarMenuButton, + SidebarMenuItem, + SidebarMenuSub, + SidebarMenuSubButton, + SidebarMenuSubItem, + useSidebar, +} from '@repo/design-system/components/ui/sidebar'; +import { cn } from '@repo/design-system/lib/utils'; +import { NotificationsTrigger } from '@repo/notifications/components/trigger'; +import { + AnchorIcon, + BookOpenIcon, + BotIcon, + ChevronRightIcon, + FolderIcon, + FrameIcon, + LifeBuoyIcon, + MapIcon, + MoreHorizontalIcon, + PieChartIcon, + SendIcon, + Settings2Icon, + ShareIcon, + SquareTerminalIcon, + Trash2Icon, +} from 'lucide-react'; +import Link from 'next/link'; +import type { ReactNode } from 'react'; +import { Search } from './search'; + +type GlobalSidebarProperties = { + readonly children: ReactNode; +}; + +const data = { + user: { + name: 'shadcn', + email: 'm@example.com', + avatar: '/avatars/shadcn.jpg', + }, + navMain: [ + { + title: 'Playground', + url: '#', + icon: SquareTerminalIcon, + isActive: true, + items: [ + { + title: 'History', + url: '#', + }, + { + title: 'Starred', + url: '#', + }, + { + title: 'Settings', + url: '#', + }, + ], + }, + { + title: 'Models', + url: '#', + icon: BotIcon, + items: [ + { + title: 'Genesis', + url: '#', + }, + { + title: 'Explorer', + url: '#', + }, + { + title: 'Quantum', + url: '#', + }, + ], + }, + { + title: 'Documentation', + url: '#', + icon: BookOpenIcon, + items: [ + { + title: 'Introduction', + url: '#', + }, + { + title: 'Get Started', + url: '#', + }, + { + title: 'Tutorials', + url: '#', + }, + { + title: 'Changelog', + url: '#', + }, + ], + }, + { + title: 'Settings', + url: '#', + icon: Settings2Icon, + items: [ + { + title: 'General', + url: '#', + }, + { + title: 'Team', + url: '#', + }, + { + title: 'Billing', + url: '#', + }, + { + title: 'Limits', + url: '#', + }, + ], + }, + ], + navSecondary: [ + { + title: 'Webhooks', + url: '/webhooks', + icon: AnchorIcon, + }, + { + title: 'Support', + url: '#', + icon: LifeBuoyIcon, + }, + { + title: 'Feedback', + url: '#', + icon: SendIcon, + }, + ], + projects: [ + { + name: 'Design Engineering', + url: '#', + icon: FrameIcon, + }, + { + name: 'Sales & Marketing', + url: '#', + icon: PieChartIcon, + }, + { + name: 'Travel', + url: '#', + icon: MapIcon, + }, + ], +}; + +export const GlobalSidebar = ({ children }: GlobalSidebarProperties) => { + const sidebar = useSidebar(); + + return ( + <> + + + + +
div]:w-full', + sidebar.open ? '' : '-mx-1' + )} + > + +
+
+
+
+ + + + Platform + + {data.navMain.map((item) => ( + + + + + + {item.title} + + + {item.items?.length ? ( + <> + + + + Toggle + + + + + {item.items?.map((subItem) => ( + + + + {subItem.title} + + + + ))} + + + + ) : null} + + + ))} + + + + Projects + + {data.projects.map((item) => ( + + + + + {item.name} + + + + + + + More + + + + + + View Project + + + + Share Project + + + + + Delete Project + + + + + ))} + + + + More + + + + + + + + {data.navSecondary.map((item) => ( + + + + + {item.title} + + + + ))} + + + + + + + + +
+ + +
+
+
+
+
+ {children} + + ); +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/layout.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/layout.tsx new file mode 100644 index 00000000..860dc97e --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/layout.tsx @@ -0,0 +1,49 @@ +import { env } from '@/env'; +import { currentUser } from '@repo/client/client/auth/server'; +import { SidebarProvider } from '@repo/design-system/components/ui/sidebar'; +import { showBetaFeature } from '@repo/feature-flags'; +import { NotificationsProvider } from '@repo/notifications/components/provider'; +import { secure } from '@repo/security'; +import type { ReactNode } from 'react'; +import { PostHogIdentifier } from './components/posthog-identifier'; +import { GlobalSidebar } from './components/sidebar'; + +type AppLayoutProperties = { + readonly children: ReactNode; +}; + +const AppLayout = async ({ children }: AppLayoutProperties) => { + if (env.ARCJET_KEY) { + await secure(['CATEGORY:PREVIEW']); + } + + const user = await currentUser(); + const betaFeature = await showBetaFeature(); + + if (!user) { + return { + redirect: { + destination: '/sign-in', + permanent: false, + }, + }; + } + + return ( + + + + {betaFeature && ( +
+ Beta feature now available +
+ )} + {children} +
+ +
+
+ ); +}; + +export default AppLayout; diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/page.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/page.tsx new file mode 100644 index 00000000..507027f1 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/page.tsx @@ -0,0 +1,57 @@ +import { env } from '@/env'; +import { auth } from '@repo/client/auth/server'; +import { database } from '@repo/database'; +import type { Metadata } from 'next'; +import dynamic from 'next/dynamic'; +import { notFound } from 'next/navigation'; +import { AvatarStack } from './components/avatar-stack'; +import { Cursors } from './components/cursors'; +import { Header } from './components/header'; + +const title = 'Acme Inc'; +const description = 'My application.'; + +const CollaborationProvider = dynamic(() => + import('./components/collaboration-provider').then( + (mod) => mod.CollaborationProvider + ) +); + +export const metadata: Metadata = { + title, + description, +}; + +const App = async () => { + const pages = await database.page.findMany(); + const { orgId } = await auth(); + + if (!orgId) { + notFound(); + } + + return ( + <> +
+ {env.LIVEBLOCKS_SECRET && ( + + + + + )} +
+
+
+ {pages.map((page) => ( +
+ {page.name} +
+ ))} +
+
+
+ + ); +}; + +export default App; diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/search/page.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/search/page.tsx new file mode 100644 index 00000000..7d53cbdb --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/search/page.tsx @@ -0,0 +1,59 @@ +import { auth } from '@repo/client/auth/server'; +import { database } from '@repo/database'; +import { notFound, redirect } from 'next/navigation'; +import { Header } from '../components/header'; + +type SearchPageProperties = { + searchParams: Promise<{ + q: string; + }>; +}; + +export const generateMetadata = async ({ + searchParams, +}: SearchPageProperties) => { + const { q } = await searchParams; + + return { + title: `${q} - Search results`, + description: `Search results for ${q}`, + }; +}; + +const SearchPage = async ({ searchParams }: SearchPageProperties) => { + const { q } = await searchParams; + const pages = await database.page.findMany({ + where: { + name: { + contains: q, + }, + }, + }); + const { orgId } = await auth(); + + if (!orgId) { + notFound(); + } + + if (!q) { + redirect('/'); + } + + return ( + <> +
+
+
+ {pages.map((page) => ( +
+ {page.name} +
+ ))} +
+
+
+ + ); +}; + +export default SearchPage; diff --git a/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/webhooks/page.tsx b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/webhooks/page.tsx new file mode 100644 index 00000000..360ff79c --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/app/app/(authenticated)/webhooks/page.tsx @@ -0,0 +1,29 @@ +import { webhooks } from '@repo/webhooks'; +import { notFound } from 'next/navigation'; + +export const metadata = { + title: 'Webhooks', + description: 'Send webhooks to your users.', +}; + +const WebhooksPage = async () => { + const response = await webhooks.getAppPortal(); + + if (!response?.url) { + notFound(); + } + + return ( +
+ + +
+ + + +Mintlify supports [HTML tags in Markdown](https://www.markdownguide.org/basic-syntax/#html). This is helpful if you prefer HTML tags to Markdown syntax, and lets you create documentation with infinite flexibility. + + + +### iFrames + +Loads another HTML page within the document. Most commonly used for embedding videos. + +```html + +``` diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/essentials/markdown.mdx b/ts/saas-nextforge-encore/frontend/apps/docs/essentials/markdown.mdx new file mode 100644 index 00000000..c8ad9c1f --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/essentials/markdown.mdx @@ -0,0 +1,88 @@ +--- +title: 'Markdown Syntax' +description: 'Text, title, and styling in standard markdown' +icon: 'text-size' +--- + +## Titles + +Best used for section headers. + +```md +## Titles +``` + +### Subtitles + +Best use to subsection headers. + +```md +### Subtitles +``` + + + +Each **title** and **subtitle** creates an anchor and also shows up on the table of contents on the right. + + + +## Text Formatting + +We support most markdown formatting. Simply add `**`, `_`, or `~` around text to format it. + +| Style | How to write it | Result | +| ------------- | ----------------- | --------------- | +| Bold | `**bold**` | **bold** | +| Italic | `_italic_` | _italic_ | +| Strikethrough | `~strikethrough~` | ~strikethrough~ | + +You can combine these. For example, write `**_bold and italic_**` to get **_bold and italic_** text. + +You need to use HTML to write superscript and subscript text. That is, add `` or `` around your text. + +| Text Size | How to write it | Result | +| ----------- | ------------------------ | ---------------------- | +| Superscript | `superscript` | superscript | +| Subscript | `subscript` | subscript | + +## Linking to Pages + +You can add a link by wrapping text in `[]()`. You would write `[link to google](https://google.com)` to [link to google](https://google.com). + +Links to pages in your docs need to be root-relative. Basically, you should include the entire folder path. For example, `[link to text](/writing-content/text)` links to the page "Text" in our components section. + +Relative links like `[link to text](../text)` will open slower because we cannot optimize them as easily. + +## Blockquotes + +### Singleline + +To create a blockquote, add a `>` in front of a paragraph. + +> Dorothy followed her through many of the beautiful rooms in her castle. + +```md +> Dorothy followed her through many of the beautiful rooms in her castle. +``` + +### Multiline + +> Dorothy followed her through many of the beautiful rooms in her castle. +> +> The Witch bade her clean the pots and kettles and sweep the floor and keep the fire fed with wood. + +```md +> Dorothy followed her through many of the beautiful rooms in her castle. +> +> The Witch bade her clean the pots and kettles and sweep the floor and keep the fire fed with wood. +``` + +### LaTeX + +Mintlify supports [LaTeX](https://www.latex-project.org) through the Latex component. + +8 x (vk x H1 - H2) = (0,1) + +```md +8 x (vk x H1 - H2) = (0,1) +``` diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/essentials/navigation.mdx b/ts/saas-nextforge-encore/frontend/apps/docs/essentials/navigation.mdx new file mode 100644 index 00000000..ca44bb64 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/essentials/navigation.mdx @@ -0,0 +1,66 @@ +--- +title: 'Navigation' +description: 'The navigation field in mint.json defines the pages that go in the navigation menu' +icon: 'map' +--- + +The navigation menu is the list of links on every website. + +You will likely update `mint.json` every time you add a new page. Pages do not show up automatically. + +## Navigation syntax + +Our navigation syntax is recursive which means you can make nested navigation groups. You don't need to include `.mdx` in page names. + + + +```json Regular Navigation +"navigation": [ + { + "group": "Getting Started", + "pages": ["quickstart"] + } +] +``` + +```json Nested Navigation +"navigation": [ + { + "group": "Getting Started", + "pages": [ + "quickstart", + { + "group": "Nested Reference Pages", + "pages": ["nested-reference-page"] + } + ] + } +] +``` + + + +## Folders + +Simply put your MDX files in folders and update the paths in `mint.json`. + +For example, to have a page at `https://yoursite.com/your-folder/your-page` you would make a folder called `your-folder` containing an MDX file called `your-page.mdx`. + + + +You cannot use `api` for the name of a folder unless you nest it inside another folder. Mintlify uses Next.js which reserves the top-level `api` folder for internal server calls. A folder name such as `api-reference` would be accepted. + + + +```json Navigation With Folder +"navigation": [ + { + "group": "Group Name", + "pages": ["your-folder/your-page"] + } +] +``` + +## Hidden Pages + +MDX files not included in `mint.json` will not show up in the sidebar but are accessible through the search bar and by linking directly to them. diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/essentials/reusable-snippets.mdx b/ts/saas-nextforge-encore/frontend/apps/docs/essentials/reusable-snippets.mdx new file mode 100644 index 00000000..a0a55297 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/essentials/reusable-snippets.mdx @@ -0,0 +1,110 @@ +--- +title: Reusable Snippets +description: Reusable, custom snippets to keep content in sync +icon: 'recycle' +--- + +import SnippetIntro from '/snippets/snippet-intro.mdx'; + + + +## Creating a custom snippet + +**Pre-condition**: You must create your snippet file in the `snippets` directory. + + + Any page in the `snippets` directory will be treated as a snippet and will not + be rendered into a standalone page. If you want to create a standalone page + from the snippet, import the snippet into another file and call it as a + component. + + +### Default export + +1. Add content to your snippet file that you want to re-use across multiple + locations. Optionally, you can add variables that can be filled in via props + when you import the snippet. + +```mdx snippets/my-snippet.mdx +Hello world! This is my content I want to reuse across pages. My keyword of the +day is {word}. +``` + + + The content that you want to reuse must be inside the `snippets` directory in + order for the import to work. + + +2. Import the snippet into your destination file. + +```mdx destination-file.mdx +--- +title: My title +description: My Description +--- + +import MySnippet from '/snippets/path/to/my-snippet.mdx'; + +## Header + +Lorem impsum dolor sit amet. + + +``` + +### Reusable variables + +1. Export a variable from your snippet file: + +```mdx snippets/path/to/custom-variables.mdx +export const myName = 'my name'; + +export const myObject = { fruit: 'strawberries' }; +``` + +2. Import the snippet from your destination file and use the variable: + +```mdx destination-file.mdx +--- +title: My title +description: My Description +--- + +import { myName, myObject } from '/snippets/path/to/custom-variables.mdx'; + +Hello, my name is {myName} and I like {myObject.fruit}. +``` + +### Reusable components + +1. Inside your snippet file, create a component that takes in props by exporting + your component in the form of an arrow function. + +```mdx snippets/custom-component.mdx +export const MyComponent = ({ title }) => ( +
+

{title}

+

... snippet content ...

+
+); +``` + + + MDX does not compile inside the body of an arrow function. Stick to HTML + syntax when you can or use a default export if you need to use MDX. + + +2. Import the snippet into your destination file and pass in the props + +```mdx destination-file.mdx +--- +title: My title +description: My Description +--- + +import { MyComponent } from '/snippets/custom-component.mdx'; + +Lorem ipsum dolor sit amet. + + +``` diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/essentials/settings.mdx b/ts/saas-nextforge-encore/frontend/apps/docs/essentials/settings.mdx new file mode 100644 index 00000000..d9dd2d7e --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/essentials/settings.mdx @@ -0,0 +1,318 @@ +--- +title: 'Global Settings' +description: 'Mintlify gives you complete control over the look and feel of your documentation using the mint.json file' +icon: 'gear' +--- + +Every Mintlify site needs a `mint.json` file with the core configuration settings. Learn more about the [properties](#properties) below. + +## Properties + + +Name of your project. Used for the global title. + +Example: `mintlify` + + + + + An array of groups with all the pages within that group + + + The name of the group. + + Example: `Settings` + + + + The relative paths to the markdown files that will serve as pages. + + Example: `["customization", "page"]` + + + + + + + + Path to logo image or object with path to "light" and "dark" mode logo images + + + Path to the logo in light mode + + + Path to the logo in dark mode + + + Where clicking on the logo links you to + + + + + + Path to the favicon image + + + + Hex color codes for your global theme + + + The primary color. Used for most often for highlighted content, section + headers, accents, in light mode + + + The primary color for dark mode. Used for most often for highlighted + content, section headers, accents, in dark mode + + + The primary color for important buttons + + + The color of the background in both light and dark mode + + + The hex color code of the background in light mode + + + The hex color code of the background in dark mode + + + + + + + + Array of `name`s and `url`s of links you want to include in the topbar + + + The name of the button. + + Example: `Contact us` + + + The url once you click on the button. Example: `https://mintlify.com/contact` + + + + + + + + + Link shows a button. GitHub shows the repo information at the url provided including the number of GitHub stars. + + + If `link`: What the button links to. + + If `github`: Link to the repository to load GitHub information from. + + + Text inside the button. Only required if `type` is a `link`. + + + + + + + Array of version names. Only use this if you want to show different versions + of docs with a dropdown in the navigation bar. + + + + An array of the anchors, includes the `icon`, `color`, and `url`. + + + The [Font Awesome](https://fontawesome.com/search?s=brands%2Cduotone) icon used to feature the anchor. + + Example: `comments` + + + The name of the anchor label. + + Example: `Community` + + + The start of the URL that marks what pages go in the anchor. Generally, this is the name of the folder you put your pages in. + + + The hex color of the anchor icon background. Can also be a gradient if you pass an object with the properties `from` and `to` that are each a hex color. + + + Used if you want to hide an anchor until the correct docs version is selected. + + + Pass `true` if you want to hide the anchor until you directly link someone to docs inside it. + + + One of: "brands", "duotone", "light", "sharp-solid", "solid", or "thin" + + + + + + + Override the default configurations for the top-most anchor. + + + The name of the top-most anchor + + + Font Awesome icon. + + + One of: "brands", "duotone", "light", "sharp-solid", "solid", or "thin" + + + + + + An array of navigational tabs. + + + The name of the tab label. + + + The start of the URL that marks what pages go in the tab. Generally, this + is the name of the folder you put your pages in. + + + + + + Configuration for API settings. Learn more about API pages at [API Components](/api-playground/demo). + + + The base url for all API endpoints. If `baseUrl` is an array, it will enable for multiple base url + options that the user can toggle. + + + + + + The authentication strategy used for all API endpoints. + + + The name of the authentication parameter used in the API playground. + + If method is `basic`, the format should be `[usernameName]:[passwordName]` + + + The default value that's designed to be a prefix for the authentication input field. + + E.g. If an `inputPrefix` of `AuthKey` would inherit the default input result of the authentication field as `AuthKey`. + + + + + + Configurations for the API playground + + + + Whether the playground is showing, hidden, or only displaying the endpoint with no added user interactivity `simple` + + Learn more at the [playground guides](/api-playground/demo) + + + + + + Enabling this flag ensures that key ordering in OpenAPI pages matches the key ordering defined in the OpenAPI file. + + This behavior will soon be enabled by default, at which point this field will be deprecated. + + + + + + + A string or an array of strings of URL(s) or relative path(s) pointing to your + OpenAPI file. + + Examples: + + ```json Absolute + "openapi": "https://example.com/openapi.json" + ``` + ```json Relative + "openapi": "/openapi.json" + ``` + ```json Multiple + "openapi": ["https://example.com/openapi1.json", "/openapi2.json", "/openapi3.json"] + ``` + + + + + + An object of social media accounts where the key:property pair represents the social media platform and the account url. + + Example: + ```json + { + "x": "https://x.com/mintlify", + "website": "https://mintlify.com" + } + ``` + + + One of the following values `website`, `facebook`, `x`, `discord`, `slack`, `github`, `linkedin`, `instagram`, `hacker-news` + + Example: `x` + + + The URL to the social platform. + + Example: `https://x.com/mintlify` + + + + + + Configurations to enable feedback buttons + + + + Enables a button to allow users to suggest edits via pull requests + + + Enables a button to allow users to raise an issue about the documentation + + + + + + Customize the dark mode toggle. + + + Set if you always want to show light or dark mode for new users. When not + set, we default to the same mode as the user's operating system. + + + Set to true to hide the dark/light mode toggle. You can combine `isHidden` with `default` to force your docs to only use light or dark mode. For example: + + + ```json Only Dark Mode + "modeToggle": { + "default": "dark", + "isHidden": true + } + ``` + + ```json Only Light Mode + "modeToggle": { + "default": "light", + "isHidden": true + } + ``` + + + + + + + + + A background image to be displayed behind every page. See example with + [Infisical](https://infisical.com/docs) and [FRPC](https://frpc.io). + diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/favicon.svg b/ts/saas-nextforge-encore/frontend/apps/docs/favicon.svg new file mode 100644 index 00000000..6a323326 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/favicon.svg @@ -0,0 +1,49 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/images/checks-passed.png b/ts/saas-nextforge-encore/frontend/apps/docs/images/checks-passed.png new file mode 100644 index 00000000..3303c773 Binary files /dev/null and b/ts/saas-nextforge-encore/frontend/apps/docs/images/checks-passed.png differ diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/images/hero-dark.svg b/ts/saas-nextforge-encore/frontend/apps/docs/images/hero-dark.svg new file mode 100644 index 00000000..c6a30e88 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/images/hero-dark.svg @@ -0,0 +1,161 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/images/hero-light.svg b/ts/saas-nextforge-encore/frontend/apps/docs/images/hero-light.svg new file mode 100644 index 00000000..297d68fb --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/images/hero-light.svg @@ -0,0 +1,155 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/introduction.mdx b/ts/saas-nextforge-encore/frontend/apps/docs/introduction.mdx new file mode 100644 index 00000000..2589c0b9 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/introduction.mdx @@ -0,0 +1,71 @@ +--- +title: Introduction +description: 'Welcome to the home of your new documentation' +--- + +Hero Light +Hero Dark + +## Setting up + +The first step to world-class documentation is setting up your editing environments. + + + + Get your docs set up locally for easy development + + + Preview your changes before you push to make sure they're perfect + + + +## Make it yours + +Update your docs to your brand and add valuable content for the best user conversion. + + + + Customize your docs to your company's colors and brands + + + Automatically generate endpoints from an OpenAPI spec + + + Build interactive features and designs to guide your users + + + Check out our showcase of our favorite documentation + + diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/logo/dark.svg b/ts/saas-nextforge-encore/frontend/apps/docs/logo/dark.svg new file mode 100644 index 00000000..a6283786 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/logo/dark.svg @@ -0,0 +1,55 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/logo/light.svg b/ts/saas-nextforge-encore/frontend/apps/docs/logo/light.svg new file mode 100644 index 00000000..582b3b95 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/logo/light.svg @@ -0,0 +1,51 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/mint.json b/ts/saas-nextforge-encore/frontend/apps/docs/mint.json new file mode 100644 index 00000000..f32f7637 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/mint.json @@ -0,0 +1,85 @@ +{ + "$schema": "https://mintlify.com/schema.json", + "name": "Starter Kit", + "logo": { + "dark": "/logo/dark.svg", + "light": "/logo/light.svg" + }, + "favicon": "/favicon.svg", + "colors": { + "primary": "#0D9373", + "light": "#07C983", + "dark": "#0D9373", + "anchors": { + "from": "#0D9373", + "to": "#07C983" + } + }, + "topbarLinks": [ + { + "name": "Support", + "url": "mailto:support@mintlify.com" + } + ], + "topbarCtaButton": { + "name": "Dashboard", + "url": "https://dashboard.mintlify.com" + }, + "tabs": [ + { + "name": "API Reference", + "url": "api-reference" + } + ], + "anchors": [ + { + "name": "Documentation", + "icon": "book-open-cover", + "url": "https://mintlify.com/docs" + }, + { + "name": "Community", + "icon": "slack", + "url": "https://mintlify.com/community" + }, + { + "name": "Blog", + "icon": "newspaper", + "url": "https://mintlify.com/blog" + } + ], + "navigation": [ + { + "group": "Get Started", + "pages": ["introduction", "quickstart", "development"] + }, + { + "group": "Essentials", + "pages": [ + "essentials/markdown", + "essentials/code", + "essentials/images", + "essentials/settings", + "essentials/navigation", + "essentials/reusable-snippets" + ] + }, + { + "group": "API Documentation", + "pages": ["api-reference/introduction"] + }, + { + "group": "Endpoint Examples", + "pages": [ + "api-reference/endpoint/get", + "api-reference/endpoint/create", + "api-reference/endpoint/delete" + ] + } + ], + "footerSocials": { + "x": "https://x.com/mintlify", + "github": "https://github.com/mintlify", + "linkedin": "https://www.linkedin.com/company/mintlify" + } +} diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/package.json b/ts/saas-nextforge-encore/frontend/apps/docs/package.json new file mode 100644 index 00000000..1e87718a --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/package.json @@ -0,0 +1,11 @@ +{ + "name": "docs", + "private": true, + "scripts": { + "dev": "mintlify dev --port 3004", + "lint": "mintlify broken-links" + }, + "devDependencies": { + "typescript": "^5.8.2" + } +} diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/quickstart.mdx b/ts/saas-nextforge-encore/frontend/apps/docs/quickstart.mdx new file mode 100644 index 00000000..d7f34867 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/quickstart.mdx @@ -0,0 +1,86 @@ +--- +title: 'Quickstart' +description: 'Start building awesome documentation in under 5 minutes' +--- + +## Setup your development + +Learn how to update your docs locally and and deploy them to the public. + +### Edit and preview + + + + During the onboarding process, we created a repository on your Github with + your docs content. You can find this repository on our + [dashboard](https://dashboard.mintlify.com). To clone the repository + locally, follow these + [instructions](https://docs.github.com/en/repositories/creating-and-managing-repositories/cloning-a-repository) + in your terminal. + + + Previewing helps you make sure your changes look as intended. We built a + command line interface to render these changes locally. 1. Install the + [Mintlify CLI](https://www.npmjs.com/package/mintlify) to preview the + documentation changes locally with this command: ``` npm i -g mintlify ``` + 2. Run the following command at the root of your documentation (where + `mint.json` is): ``` mintlify dev ``` + + + +### Deploy your changes + + + + + Our Github app automatically deploys your changes to your docs site, so you + don't need to manage deployments yourself. You can find the link to install on + your [dashboard](https://dashboard.mintlify.com). Once the bot has been + successfully installed, there should be a check mark next to the commit hash + of the repo. + + + [Commit and push your changes to + Git](https://docs.github.com/en/get-started/using-git/pushing-commits-to-a-remote-repository#about-git-push) + for your changes to update in your docs site. If you push and don't see that + the Github app successfully deployed your changes, you can also manually + update your docs through our [dashboard](https://dashboard.mintlify.com). + + + + +## Update your docs + +Add content directly in your files with MDX syntax and React components. You can use any of our components, or even build your own. + + + + + Add flair to your docs with personalized branding. + + + + Implement your OpenAPI spec and enable API user interaction. + + + + Draw insights from user interactions with your documentation. + + + + Keep your docs on your own website's subdomain. + + + diff --git a/ts/saas-nextforge-encore/frontend/apps/docs/snippets/snippet-intro.mdx b/ts/saas-nextforge-encore/frontend/apps/docs/snippets/snippet-intro.mdx new file mode 100644 index 00000000..c57e7c75 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/docs/snippets/snippet-intro.mdx @@ -0,0 +1,4 @@ +One of the core principles of software development is DRY (Don't Repeat +Yourself). This is a principle that apply to documentation as +well. If you find yourself repeating the same content in multiple places, you +should consider creating a custom snippet to keep your content in sync. diff --git a/ts/saas-nextforge-encore/frontend/apps/email/.gitignore b/ts/saas-nextforge-encore/frontend/apps/email/.gitignore new file mode 100644 index 00000000..9e9e8799 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/email/.gitignore @@ -0,0 +1 @@ +.react-email \ No newline at end of file diff --git a/ts/saas-nextforge-encore/frontend/apps/email/emails/contact.tsx b/ts/saas-nextforge-encore/frontend/apps/email/emails/contact.tsx new file mode 100644 index 00000000..f47b7357 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/email/emails/contact.tsx @@ -0,0 +1,11 @@ +import { ContactTemplate } from '@repo/email/templates/contact'; + +const ExampleContactEmail = () => ( + +); + +export default ExampleContactEmail; diff --git a/ts/saas-nextforge-encore/frontend/apps/email/package.json b/ts/saas-nextforge-encore/frontend/apps/email/package.json new file mode 100644 index 00000000..fe6610eb --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/email/package.json @@ -0,0 +1,25 @@ +{ + "name": "email", + "version": "0.0.0", + "private": true, + "scripts": { + "build": "email build", + "dev": "email dev --port 3003", + "export": "email export", + "clean": "git clean -xdf .cache .turbo dist node_modules", + "typecheck": "tsc --noEmit --emitDeclarationOnly false" + }, + "dependencies": { + "@react-email/components": "0.0.33", + "@repo/email": "workspace:*", + "react": "19.0.0", + "react-email": "3.0.7" + }, + "devDependencies": { + "@repo/typescript-config": "workspace:*", + "@types/node": "22.13.9", + "@types/react": "19.0.10", + "next": "15.1.7", + "typescript": "^5.8.2" + } +} diff --git a/ts/saas-nextforge-encore/frontend/apps/email/tsconfig.json b/ts/saas-nextforge-encore/frontend/apps/email/tsconfig.json new file mode 100644 index 00000000..9352a57d --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/email/tsconfig.json @@ -0,0 +1,5 @@ +{ + "extends": "@repo/typescript-config/nextjs.json", + "include": ["**/*.ts", "**/*.tsx"], + "exclude": ["node_modules"] +} diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/.gitignore b/ts/saas-nextforge-encore/frontend/apps/storybook/.gitignore new file mode 100644 index 00000000..d85f7c1c --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/.gitignore @@ -0,0 +1,45 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# dependencies +/node_modules +/.pnp +.pnp.* +.yarn/* +!.yarn/patches +!.yarn/plugins +!.yarn/releases +!.yarn/versions + +# testing +/coverage + +# next.js +/.next/ +/out/ + +# production +/build + +# misc +.DS_Store +*.pem + +# debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# env files (can opt-in for commiting if needed) +.env* + +# vercel +.vercel + +# typescript +*.tsbuildinfo +next-env.d.ts + +*storybook.log + +# storybook +storybook-static/ \ No newline at end of file diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/.storybook/main.ts b/ts/saas-nextforge-encore/frontend/apps/storybook/.storybook/main.ts new file mode 100644 index 00000000..0592005d --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/.storybook/main.ts @@ -0,0 +1,30 @@ +import { dirname, join } from 'node:path'; +import type { StorybookConfig } from '@storybook/nextjs'; + +/** + * This function is used to resolve the absolute path of a package. + * It is needed in projects that use Yarn PnP or are set up within a monorepo. + */ +const getAbsolutePath = (value: string) => + dirname(require.resolve(join(value, 'package.json'))); + +const config: StorybookConfig = { + stories: [ + '../stories/**/*.mdx', + '../stories/**/*.stories.@(js|jsx|mjs|ts|tsx)', + ], + addons: [ + getAbsolutePath('@storybook/addon-onboarding'), + getAbsolutePath('@storybook/addon-essentials'), + getAbsolutePath('@chromatic-com/storybook'), + getAbsolutePath('@storybook/addon-interactions'), + getAbsolutePath('@storybook/addon-themes'), + ], + framework: { + name: getAbsolutePath('@storybook/nextjs'), + options: {}, + }, + staticDirs: ['../public'], +}; + +export default config; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/.storybook/preview-head.html b/ts/saas-nextforge-encore/frontend/apps/storybook/.storybook/preview-head.html new file mode 100644 index 00000000..5873c0b6 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/.storybook/preview-head.html @@ -0,0 +1,17 @@ + + + + + + + diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/.storybook/preview.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/.storybook/preview.tsx new file mode 100644 index 00000000..a7bd0f60 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/.storybook/preview.tsx @@ -0,0 +1,53 @@ +import { Toaster } from '@repo/design-system/components/ui/sonner'; +import { TooltipProvider } from '@repo/design-system/components/ui/tooltip'; +import { ThemeProvider } from '@repo/design-system/providers/theme'; +import { withThemeByClassName } from '@storybook/addon-themes'; +import type { Preview } from '@storybook/react'; + +import '@repo/design-system/styles/globals.css'; + +const preview: Preview = { + parameters: { + controls: { + matchers: { + color: /(background|color)$/i, + date: /Date$/i, + }, + }, + chromatic: { + modes: { + light: { + theme: 'light', + className: 'light', + }, + dark: { + theme: 'dark', + className: 'dark', + }, + }, + }, + }, + decorators: [ + withThemeByClassName({ + themes: { + light: 'light', + dark: 'dark', + }, + defaultTheme: 'light', + }), + (Story) => { + return ( +
+ + + + + + +
+ ); + }, + ], +}; + +export default preview; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/README.md b/ts/saas-nextforge-encore/frontend/apps/storybook/README.md new file mode 100644 index 00000000..ef0e47e3 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/README.md @@ -0,0 +1,40 @@ +This is a [Next.js](https://nextjs.org) project bootstrapped with [`create-next-app`](https://nextjs.org/docs/pages/api-reference/create-next-app). + +## Getting Started + +First, run the development server: + +```bash +npm run dev +# or +yarn dev +# or +pnpm dev +# or +bun dev +``` + +Open [http://localhost:3000](http://localhost:3000) with your browser to see the result. + +You can start editing the page by modifying `pages/index.tsx`. The page auto-updates as you edit the file. + +[API routes](https://nextjs.org/docs/pages/building-your-application/routing/api-routes) can be accessed on [http://localhost:3000/api/hello](http://localhost:3000/api/hello). This endpoint can be edited in `pages/api/hello.ts`. + +The `pages/api` directory is mapped to `/api/*`. Files in this directory are treated as [API routes](https://nextjs.org/docs/pages/building-your-application/routing/api-routes) instead of React pages. + +This project uses [`next/font`](https://nextjs.org/docs/pages/building-your-application/optimizing/fonts) to automatically optimize and load [Geist](https://vercel.com/font), a new font family for Vercel. + +## Learn More + +To learn more about Next.js, take a look at the following resources: + +- [Next.js Documentation](https://nextjs.org/docs) - learn about Next.js features and API. +- [Learn Next.js](https://nextjs.org/learn-pages-router) - an interactive Next.js tutorial. + +You can check out [the Next.js GitHub repository](https://github.com/vercel/next.js) - your feedback and contributions are welcome! + +## Deploy on Vercel + +The easiest way to deploy your Next.js app is to use the [Vercel Platform](https://vercel.com/new?utm_medium=default-template&filter=next.js&utm_source=create-next-app&utm_campaign=create-next-app-readme) from the creators of Next.js. + +Check out our [Next.js deployment documentation](https://nextjs.org/docs/pages/building-your-application/deploying) for more details. diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/next.config.ts b/ts/saas-nextforge-encore/frontend/apps/storybook/next.config.ts new file mode 100644 index 00000000..b08f02be --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/next.config.ts @@ -0,0 +1,7 @@ +import type { NextConfig } from 'next'; + +const nextConfig: NextConfig = { + reactStrictMode: true, +}; + +export default nextConfig; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/package.json b/ts/saas-nextforge-encore/frontend/apps/storybook/package.json new file mode 100644 index 00000000..cb25ac9a --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/package.json @@ -0,0 +1,47 @@ +{ + "name": "storybook", + "version": "0.1.0", + "private": true, + "scripts": { + "dev": "storybook dev -p 6006", + "build": "storybook build", + "chromatic": "chromatic --exit-zero-on-changes", + "clean": "git clean -xdf .cache .turbo dist node_modules", + "typecheck": "tsc --noEmit --emitDeclarationOnly false" + }, + "dependencies": { + "@repo/design-system": "workspace:*", + "@storybook/addon-actions": "^8.6.4", + "cmdk": "^1.0.0", + "date-fns": "^4.1.0", + "input-otp": "^1.4.2", + "lucide-react": "^0.477.0", + "next": "15.1.7", + "react": "19.0.0", + "react-dom": "19.0.0", + "react-hook-form": "^7.54.2", + "recharts": "^2.15.1", + "sonner": "^2.0.1", + "zod": "^3.24.2" + }, + "devDependencies": { + "@chromatic-com/storybook": "^3.2.5", + "@repo/typescript-config": "workspace:*", + "@storybook/addon-essentials": "^8.6.3", + "@storybook/addon-interactions": "^8.6.3", + "@storybook/addon-onboarding": "^8.6.3", + "@storybook/addon-themes": "^8.6.3", + "@storybook/blocks": "^8.6.3", + "@storybook/nextjs": "^8.6.3", + "@storybook/react": "^8.6.3", + "@storybook/test": "^8.6.3", + "@types/node": "^22", + "@types/react": "^19", + "@types/react-dom": "^19", + "chromatic": "^11.27.0", + "postcss": "^8", + "storybook": "^8.6.3", + "tailwindcss": "^4.0.12", + "typescript": "^5" + } +} diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/postcss.config.mjs b/ts/saas-nextforge-encore/frontend/apps/storybook/postcss.config.mjs new file mode 100644 index 00000000..1a69fd2a --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/postcss.config.mjs @@ -0,0 +1,8 @@ +/** @type {import('postcss-load-config').Config} */ +const config = { + plugins: { + tailwindcss: {}, + }, +}; + +export default config; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/public/favicon.ico b/ts/saas-nextforge-encore/frontend/apps/storybook/public/favicon.ico new file mode 100644 index 00000000..718d6fea Binary files /dev/null and b/ts/saas-nextforge-encore/frontend/apps/storybook/public/favicon.ico differ diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/accordion.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/accordion.stories.tsx new file mode 100644 index 00000000..b59e7f67 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/accordion.stories.tsx @@ -0,0 +1,60 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Accordion, + AccordionContent, + AccordionItem, + AccordionTrigger, +} from '@repo/design-system/components/ui/accordion'; + +/** + * A vertically stacked set of interactive headings that each reveal a section + * of content. + */ +const meta = { + title: 'ui/Accordion', + component: Accordion, + tags: ['autodocs'], + argTypes: { + type: { + options: ['single', 'multiple'], + control: { type: 'radio' }, + }, + }, + args: { + type: 'single', + collapsible: true, + }, + render: (args) => ( + + + Is it accessible? + + Yes. It adheres to the WAI-ARIA design pattern. + + + + Is it styled? + + Yes. It comes with default styles that matches the other components' + aesthetic. + + + + Is it animated? + + Yes. It's animated by default, but you can disable it if you prefer. + + + + ), +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default behavior of the accordion allows only one item to be open. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/alert-dialog.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/alert-dialog.stories.tsx new file mode 100644 index 00000000..7f39030c --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/alert-dialog.stories.tsx @@ -0,0 +1,54 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + AlertDialog, + AlertDialogAction, + AlertDialogCancel, + AlertDialogContent, + AlertDialogDescription, + AlertDialogFooter, + AlertDialogHeader, + AlertDialogTitle, + AlertDialogTrigger, +} from '@repo/design-system/components/ui/alert-dialog'; + +/** + * A modal dialog that interrupts the user with important content and expects + * a response. + */ +const meta = { + title: 'ui/AlertDialog', + component: AlertDialog, + tags: ['autodocs'], + argTypes: {}, + render: (args) => ( + + Open + + + Are you sure absolutely sure? + + This action cannot be undone. This will permanently delete your + account and remove your data from our servers. + + + + Cancel + Continue + + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the alert dialog. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/alert.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/alert.stories.tsx new file mode 100644 index 00000000..b6aac2fa --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/alert.stories.tsx @@ -0,0 +1,60 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { AlertCircle } from 'lucide-react'; + +import { + Alert, + AlertDescription, + AlertTitle, +} from '@repo/design-system/components/ui/alert'; + +/** + * Displays a callout for user attention. + */ +const meta = { + title: 'ui/Alert', + component: Alert, + tags: ['autodocs'], + argTypes: { + variant: { + options: ['default', 'destructive'], + control: { type: 'radio' }, + }, + }, + args: { + variant: 'default', + }, + render: (args) => ( + + Heads up! + + You can add components to your app using the cli. + + + ), +} satisfies Meta; + +export default meta; + +type Story = StoryObj; +/** + * The default form of the alert. + */ +export const Default: Story = {}; + +/** + * Use the `destructive` alert to indicate a destructive action. + */ +export const Destructive: Story = { + render: (args) => ( + + + Error + + Your session has expired. Please log in again. + + + ), + args: { + variant: 'destructive', + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/aspect-ratio.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/aspect-ratio.stories.tsx new file mode 100644 index 00000000..0115dd7b --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/aspect-ratio.stories.tsx @@ -0,0 +1,71 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import Image from 'next/image'; + +import { AspectRatio } from '@repo/design-system/components/ui/aspect-ratio'; + +/** + * Displays content within a desired ratio. + */ +const meta: Meta = { + title: 'ui/AspectRatio', + component: AspectRatio, + tags: ['autodocs'], + argTypes: {}, + render: (args) => ( + + Photo by Alvaro Pinot + + ), + decorators: [ + (Story) => ( +
+ +
+ ), + ], +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the aspect ratio. + */ +export const Default: Story = { + args: { + ratio: 16 / 9, + }, +}; + +/** + * Use the `1:1` aspect ratio to display a square image. + */ +export const Square: Story = { + args: { + ratio: 1, + }, +}; + +/** + * Use the `4:3` aspect ratio to display a landscape image. + */ +export const Landscape: Story = { + args: { + ratio: 4 / 3, + }, +}; + +/** + * Use the `2.35:1` aspect ratio to display a cinemascope image. + */ +export const Cinemascope: Story = { + args: { + ratio: 2.35 / 1, + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/avatar.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/avatar.stories.tsx new file mode 100644 index 00000000..5d15a6fb --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/avatar.stories.tsx @@ -0,0 +1,35 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Avatar, + AvatarFallback, + AvatarImage, +} from '@repo/design-system/components/ui/avatar'; + +/** + * An image element with a fallback for representing the user. + */ +const meta = { + title: 'ui/Avatar', + component: Avatar, + tags: ['autodocs'], + argTypes: {}, + render: (args) => ( + + + CN + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the avatar. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/badge.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/badge.stories.tsx new file mode 100644 index 00000000..481def33 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/badge.stories.tsx @@ -0,0 +1,62 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { Badge } from '@repo/design-system/components/ui/badge'; + +/** + * Displays a badge or a component that looks like a badge. + */ +const meta = { + title: 'ui/Badge', + component: Badge, + tags: ['autodocs'], + argTypes: { + children: { + control: 'text', + }, + }, + args: { + children: 'Badge', + }, + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the badge. + */ +export const Default: Story = {}; + +/** + * Use the `secondary` badge to call for less urgent information, blending + * into the interface while still signaling minor updates or statuses. + */ +export const Secondary: Story = { + args: { + variant: 'secondary', + }, +}; + +/** + * Use the `destructive` badge to indicate errors, alerts, or the need for + * immediate attention. + */ +export const Destructive: Story = { + args: { + variant: 'destructive', + }, +}; + +/** + * Use the `outline` badge for overlaying without obscuring interface details, + * emphasizing clarity and subtlety.. + */ +export const Outline: Story = { + args: { + variant: 'outline', + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/breadcrumb.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/breadcrumb.stories.tsx new file mode 100644 index 00000000..89ae396a --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/breadcrumb.stories.tsx @@ -0,0 +1,78 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { ArrowRightSquare } from 'lucide-react'; + +import { + Breadcrumb, + BreadcrumbItem, + BreadcrumbLink, + BreadcrumbList, + BreadcrumbPage, + BreadcrumbSeparator, +} from '@repo/design-system/components/ui/breadcrumb'; + +/** + * Displays the path to the current resource using a hierarchy of links. + */ +const meta = { + title: 'ui/Breadcrumb', + component: Breadcrumb, + tags: ['autodocs'], + argTypes: {}, + args: {}, + render: (args) => ( + + + + Home + + + + Components + + + + Breadcrumb + + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * Displays the path of links to the current resource. + */ +export const Default: Story = {}; + +/** + * Displays the path with a custom icon for the separator. + */ +export const WithCustomSeparator: Story = { + render: (args) => ( + + + + Home + + + + + + Components + + + + + + Breadcrumb + + + + ), +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/button.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/button.stories.tsx new file mode 100644 index 00000000..31dbaf8f --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/button.stories.tsx @@ -0,0 +1,157 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { Loader2, Mail } from 'lucide-react'; + +import { Button } from '@repo/design-system/components/ui/button'; + +/** + * Displays a button or a component that looks like a button. + */ +const meta = { + title: 'ui/Button', + component: Button, + tags: ['autodocs'], + argTypes: { + children: { + control: 'text', + }, + }, + parameters: { + layout: 'centered', + }, + args: { + variant: 'default', + size: 'default', + children: 'Button', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the button, used for primary actions and commands. + */ +export const Default: Story = {}; + +/** + * Use the `outline` button to reduce emphasis on secondary actions, such as + * canceling or dismissing a dialog. + */ +export const Outline: Story = { + args: { + variant: 'outline', + }, +}; + +/** + * Use the `ghost` button is minimalistic and subtle, for less intrusive + * actions. + */ +export const Ghost: Story = { + args: { + variant: 'ghost', + }, +}; + +/** + * Use the `secondary` button to call for less emphasized actions, styled to + * complement the primary button while being less conspicuous. + */ +export const Secondary: Story = { + args: { + variant: 'secondary', + }, +}; + +/** + * Use the `destructive` button to indicate errors, alerts, or the need for + * immediate attention. + */ +export const Destructive: Story = { + args: { + variant: 'destructive', + }, +}; + +/** + * Use the `link` button to reduce emphasis on tertiary actions, such as + * hyperlink or navigation, providing a text-only interactive element. + */ +export const Link: Story = { + args: { + variant: 'link', + }, +}; + +/** + * Add the `disabled` prop to a button to prevent interactions and add a + * loading indicator, such as a spinner, to signify an in-progress action. + */ +export const Loading: Story = { + render: (args) => ( + + ), + args: { + ...Outline.args, + disabled: true, + }, +}; + +/** + * Add an icon element to a button to enhance visual communication and + * providing additional context for the action. + */ +export const WithIcon: Story = { + render: (args) => ( + + ), + args: { + ...Secondary.args, + }, +}; + +/** + * Use the `sm` size for a smaller button, suitable for interfaces needing + * compact elements without sacrificing usability. + */ +export const Small: Story = { + args: { + size: 'sm', + }, +}; + +/** + * Use the `lg` size for a larger button, offering better visibility and + * easier interaction for users. + */ +export const Large: Story = { + args: { + size: 'lg', + }, +}; + +/** + * Use the "icon" size for a button with only an icon. + */ +export const Icon: Story = { + args: { + ...Secondary.args, + size: 'icon', + children: , + }, +}; + +/** + * Add the `disabled` prop to prevent interactions with the button. + */ +export const Disabled: Story = { + args: { + disabled: true, + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/calendar.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/calendar.stories.tsx new file mode 100644 index 00000000..2a7a25be --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/calendar.stories.tsx @@ -0,0 +1,81 @@ +import { action } from '@storybook/addon-actions'; +import type { Meta, StoryObj } from '@storybook/react'; +import { addDays } from 'date-fns'; + +import { Calendar } from '@repo/design-system/components/ui/calendar'; + +/** + * A date field component that allows users to enter and edit date. + */ +const meta = { + title: 'ui/Calendar', + component: Calendar, + tags: ['autodocs'], + argTypes: {}, + args: { + mode: 'single', + selected: new Date(), + onSelect: action('onDayClick'), + className: 'rounded-md border w-fit', + }, + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the calendar. + */ +export const Default: Story = {}; + +/** + * Use the `multiple` mode to select multiple dates. + */ +export const Multiple: Story = { + args: { + min: 1, + selected: [new Date(), addDays(new Date(), 2), addDays(new Date(), 8)], + mode: 'multiple', + }, +}; + +/** + * Use the `range` mode to select a range of dates. + */ +export const Range: Story = { + args: { + selected: { + from: new Date(), + to: addDays(new Date(), 7), + }, + mode: 'range', + }, +}; + +/** + * Use the `disabled` prop to disable specific dates. + */ +export const Disabled: Story = { + args: { + disabled: [ + addDays(new Date(), 1), + addDays(new Date(), 2), + addDays(new Date(), 3), + addDays(new Date(), 5), + ], + }, +}; + +/** + * Use the `numberOfMonths` prop to display multiple months. + */ +export const MultipleMonths: Story = { + args: { + numberOfMonths: 2, + showOutsideDays: false, + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/card.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/card.stories.tsx new file mode 100644 index 00000000..d491e0c5 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/card.stories.tsx @@ -0,0 +1,75 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { BellRing } from 'lucide-react'; + +import { + Card, + CardContent, + CardDescription, + CardFooter, + CardHeader, + CardTitle, +} from '@repo/design-system/components/ui/card'; + +const notifications = [ + { + title: 'Your call has been confirmed.', + description: '1 hour ago', + }, + { + title: 'You have a new message!', + description: '1 hour ago', + }, + { + title: 'Your subscription is expiring soon!', + description: '2 hours ago', + }, +]; + +/** + * Displays a card with header, content, and footer. + */ +const meta = { + title: 'ui/Card', + component: Card, + tags: ['autodocs'], + argTypes: {}, + args: { + className: 'w-96', + }, + render: (args) => ( + + + Notifications + You have 3 unread messages. + + + {notifications.map((notification, index) => ( +
+ +
+

{notification.title}

+

{notification.description}

+
+
+ ))} +
+ + + +
+ ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the card. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/carousel.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/carousel.stories.tsx new file mode 100644 index 00000000..91185a49 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/carousel.stories.tsx @@ -0,0 +1,73 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Carousel, + CarouselContent, + CarouselItem, + CarouselNext, + CarouselPrevious, +} from '@repo/design-system/components/ui/carousel'; + +/** + * A carousel with motion and swipe built using Embla. + */ +const meta: Meta = { + title: 'ui/Carousel', + component: Carousel, + tags: ['autodocs'], + argTypes: {}, + args: { + className: 'w-full max-w-xs', + }, + render: (args) => ( + + + {Array.from({ length: 5 }).map((_, index) => ( + +
+ {index + 1} +
+
+ ))} +
+ + +
+ ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the carousel. + */ +export const Default: Story = {}; + +/** + * Use the `basis` utility class to change the size of the carousel. + */ +export const Size: Story = { + render: (args) => ( + + + {Array.from({ length: 5 }).map((_, index) => ( + +
+ {index + 1} +
+
+ ))} +
+ + +
+ ), + args: { + className: 'mx-12 w-full max-w-xs', + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/chart.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/chart.stories.tsx new file mode 100644 index 00000000..d10131b9 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/chart.stories.tsx @@ -0,0 +1,271 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { useMemo } from 'react'; +import { + Area, + AreaChart, + Bar, + BarChart, + CartesianGrid, + Label, + Line, + LineChart, + Pie, + PieChart, + XAxis, +} from 'recharts'; + +import { + type ChartConfig, + ChartContainer, + ChartTooltip, + ChartTooltipContent, +} from '@repo/design-system/components/ui/chart'; + +const multiSeriesData = [ + { month: 'January', desktop: 186, mobile: 80 }, + { month: 'February', desktop: 305, mobile: 200 }, + { month: 'March', desktop: 237, mobile: 120 }, + { month: 'April', desktop: 73, mobile: 190 }, + { month: 'May', desktop: 209, mobile: 130 }, + { month: 'June', desktop: 214, mobile: 140 }, +]; + +const multiSeriesConfig = { + desktop: { + label: 'Desktop', + color: 'hsl(var(--chart-1))', + }, + mobile: { + label: 'Mobile', + color: 'hsl(var(--chart-2))', + }, +} satisfies ChartConfig; + +const singleSeriesData = [ + { browser: 'chrome', visitors: 275, fill: 'var(--color-chrome)' }, + { browser: 'safari', visitors: 200, fill: 'var(--color-safari)' }, + { browser: 'other', visitors: 190, fill: 'var(--color-other)' }, +]; + +const singleSeriesConfig = { + visitors: { + label: 'Visitors', + }, + chrome: { + label: 'Chrome', + color: 'hsl(var(--chart-1))', + }, + safari: { + label: 'Safari', + color: 'hsl(var(--chart-2))', + }, + other: { + label: 'Other', + color: 'hsl(var(--chart-5))', + }, +} satisfies ChartConfig; + +/** + * Beautiful charts. Built using Recharts. Copy and paste into your apps. + */ +const meta = { + title: 'ui/Chart', + component: ChartContainer, + tags: ['autodocs'], + argTypes: {}, + args: { + children:
, + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * Combine multiple Area components to create a stacked area chart. + */ +export const StackedAreaChart: Story = { + args: { + config: multiSeriesConfig, + }, + render: (args) => ( + + + + value.slice(0, 3)} + /> + } + /> + + + + + ), +}; + +/** + * Combine multiple Bar components to create a stacked bar chart. + */ +export const StackedBarChart: Story = { + args: { + config: multiSeriesConfig, + }, + render: (args) => ( + + + + value.slice(0, 3)} + /> + } + /> + + + + + ), +}; + +/** + * Combine multiple Line components to create a single line chart. + */ +export const MultiLineChart: Story = { + args: { + config: multiSeriesConfig, + }, + render: (args) => ( + + + + value.slice(0, 3)} + /> + } + /> + + + + + ), +}; + +/** + * Combine Pie and Label components to create a doughnut chart. + */ +export const DoughnutChart: Story = { + args: { + config: singleSeriesConfig, + }, + render: (args) => { + const totalVisitors = useMemo(() => { + return singleSeriesData.reduce((acc, curr) => acc + curr.visitors, 0); + }, []); + return ( + + + } + /> + + + + + ); + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/checkbox.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/checkbox.stories.tsx new file mode 100644 index 00000000..c644c84e --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/checkbox.stories.tsx @@ -0,0 +1,50 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { Checkbox } from '@repo/design-system/components/ui/checkbox'; + +/** + * A control that allows the user to toggle between checked and not checked. + */ +const meta: Meta = { + title: 'ui/Checkbox', + component: Checkbox, + tags: ['autodocs'], + argTypes: {}, + args: { + id: 'terms', + disabled: false, + }, + render: (args) => ( +
+ + +
+ ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the checkbox. + */ +export const Default: Story = {}; + +/** + * Use the `disabled` prop to disable the checkbox. + */ +export const Disabled: Story = { + args: { + id: 'disabled-terms', + disabled: true, + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/collapsible.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/collapsible.stories.tsx new file mode 100644 index 00000000..acdc2701 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/collapsible.stories.tsx @@ -0,0 +1,55 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { Info } from 'lucide-react'; + +import { + Collapsible, + CollapsibleContent, + CollapsibleTrigger, +} from '@repo/design-system/components/ui/collapsible'; + +/** + * An interactive component which expands/collapses a panel. + */ +const meta = { + title: 'ui/Collapsible', + component: Collapsible, + tags: ['autodocs'], + argTypes: {}, + args: { + className: 'w-96', + disabled: false, + }, + render: (args) => ( + + +

Can I use this in my project?

+ +
+ + Yes. Free to use for personal and commercial projects. No attribution + required. + +
+ ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the collapsible. + */ +export const Default: Story = {}; + +/** + * Use the `disabled` prop to disable the interaction. + */ +export const Disabled: Story = { + args: { + disabled: true, + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/command.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/command.stories.tsx new file mode 100644 index 00000000..5b82920e --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/command.stories.tsx @@ -0,0 +1,55 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { CommandSeparator } from 'cmdk'; + +import { + Command, + CommandEmpty, + CommandGroup, + CommandInput, + CommandItem, + CommandList, +} from '@repo/design-system/components/ui/command'; + +/** + * Fast, composable, unstyled command menu for React. + */ +const meta = { + title: 'ui/Command', + component: Command, + tags: ['autodocs'], + argTypes: {}, + args: { + className: 'rounded-lg w-96 border shadow-md', + }, + render: (args) => ( + + + + No results found. + + Calendar + Search Emoji + Calculator + + + + Profile + Billing + Settings + + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the command. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/context-menu.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/context-menu.stories.tsx new file mode 100644 index 00000000..0bbf7109 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/context-menu.stories.tsx @@ -0,0 +1,153 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + ContextMenu, + ContextMenuCheckboxItem, + ContextMenuContent, + ContextMenuItem, + ContextMenuLabel, + ContextMenuRadioGroup, + ContextMenuRadioItem, + ContextMenuSeparator, + ContextMenuShortcut, + ContextMenuSub, + ContextMenuSubContent, + ContextMenuSubTrigger, + ContextMenuTrigger, +} from '@repo/design-system/components/ui/context-menu'; + +/** + * Displays a menu to the user — such as a set of actions or functions — + * triggered by a button. + */ +const meta = { + title: 'ui/ContextMenu', + component: ContextMenu, + tags: ['autodocs'], + argTypes: {}, + args: {}, + render: (args) => ( + + + Right click here + + + Profile + Billing + Team + Subscription + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the context menu. + */ +export const Default: Story = {}; + +/** + * A context menu with shortcuts. + */ +export const WithShortcuts: Story = { + render: (args) => ( + + + Right click here + + + + Back + ⌘[ + + + Forward + ⌘] + + + Reload + ⌘R + + + + ), +}; + +/** + * A context menu with a submenu. + */ +export const WithSubmenu: Story = { + render: (args) => ( + + + Right click here + + + + New Tab + ⌘N + + + More Tools + + + Save Page As... + ⇧⌘S + + Create Shortcut... + Name Window... + + Developer Tools + + + + + ), +}; + +/** + * A context menu with checkboxes. + */ +export const WithCheckboxes: Story = { + render: (args) => ( + + + Right click here + + + + Show Comments + ⌘⇧C + + Show Preview + + + ), +}; + +/** + * A context menu with a radio group. + */ +export const WithRadioGroup: Story = { + render: (args) => ( + + + Right click here + + + + Theme + Light + Dark + + + + ), +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/dialog.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/dialog.stories.tsx new file mode 100644 index 00000000..0f518459 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/dialog.stories.tsx @@ -0,0 +1,62 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Dialog, + DialogClose, + DialogContent, + DialogDescription, + DialogFooter, + DialogHeader, + DialogTitle, + DialogTrigger, +} from '@repo/design-system/components/ui/dialog'; + +/** + * A window overlaid on either the primary window or another dialog window, + * rendering the content underneath inert. + */ +const meta = { + title: 'ui/Dialog', + component: Dialog, + tags: ['autodocs'], + argTypes: {}, + render: (args) => ( + + Open + + + Are you absolutely sure? + + This action cannot be undone. This will permanently delete your + account and remove your data from our servers. + + + + + + + + + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the dialog. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/drawer.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/drawer.stories.tsx new file mode 100644 index 00000000..f7397317 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/drawer.stories.tsx @@ -0,0 +1,58 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Drawer, + DrawerClose, + DrawerContent, + DrawerDescription, + DrawerFooter, + DrawerHeader, + DrawerTitle, + DrawerTrigger, +} from '@repo/design-system/components/ui/drawer'; + +/** + * A drawer component for React. + */ +const meta: Meta = { + title: 'ui/Drawer', + component: Drawer, + tags: ['autodocs'], + argTypes: {}, + render: (args) => ( + + Open + + + Are you sure absolutely sure? + This action cannot be undone. + + + + + + + + + + ), + parameters: { + layout: 'centered', + }, +}; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the drawer. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/dropdown-menu.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/dropdown-menu.stories.tsx new file mode 100644 index 00000000..76bb662b --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/dropdown-menu.stories.tsx @@ -0,0 +1,159 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { Mail, Plus, PlusCircle, Search, UserPlus } from 'lucide-react'; + +import { + DropdownMenu, + DropdownMenuCheckboxItem, + DropdownMenuContent, + DropdownMenuGroup, + DropdownMenuItem, + DropdownMenuLabel, + DropdownMenuPortal, + DropdownMenuRadioGroup, + DropdownMenuRadioItem, + DropdownMenuSeparator, + DropdownMenuShortcut, + DropdownMenuSub, + DropdownMenuSubContent, + DropdownMenuSubTrigger, + DropdownMenuTrigger, +} from '@repo/design-system/components/ui/dropdown-menu'; + +/** + * Displays a menu to the user — such as a set of actions or functions — + * triggered by a button. + */ +const meta = { + title: 'ui/DropdownMenu', + component: DropdownMenu, + tags: ['autodocs'], + argTypes: {}, + render: (args) => ( + + Open + + My Account + + Profile + Billing + Team + Subscription + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the dropdown menu. + */ +export const Default: Story = {}; + +/** + * A dropdown menu with shortcuts. + */ +export const WithShortcuts: Story = { + render: (args) => ( + + Open + + Controls + + Back + ⌘[ + + + Forward + ⌘] + + + + ), +}; + +/** + * A dropdown menu with submenus. + */ +export const WithSubmenus: Story = { + render: (args) => ( + + Open + + + + Search + + + + + + New Team + ⌘+T + + + + + Invite users + + + + + + Email + + + + + More... + + + + + + + + ), +}; + +/** + * A dropdown menu with radio items. + */ +export const WithRadioItems: Story = { + render: (args) => ( + + Open + + Status + + Info + Warning + Error + + + + ), +}; + +/** + * A dropdown menu with checkboxes. + */ +export const WithCheckboxes: Story = { + render: (args) => ( + + Open + + + Autosave + ⌘S + + Show Comments + + + ), +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/form.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/form.stories.tsx new file mode 100644 index 00000000..50236be6 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/form.stories.tsx @@ -0,0 +1,85 @@ +import { zodResolver } from '@hookform/resolvers/zod'; +import { action } from '@storybook/addon-actions'; +import type { Meta, StoryObj } from '@storybook/react'; +import { useForm } from 'react-hook-form'; +import * as z from 'zod'; + +import { + Form, + FormControl, + FormDescription, + FormField, + FormItem, + FormLabel, + FormMessage, +} from '@repo/design-system/components/ui/form'; + +/** + * Building forms with React Hook Form and Zod. + */ +const meta: Meta = { + title: 'ui/Form', + component: Form, + tags: ['autodocs'], + argTypes: {}, + render: (args) => , +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +const formSchema = z.object({ + username: z.string().min(2, { + message: 'Username must be at least 2 characters.', + }), +}); + +const ProfileForm = (args: Story['args']) => { + const form = useForm>({ + resolver: zodResolver(formSchema), + defaultValues: { + username: '', + }, + }); + function onSubmit(values: z.infer) { + action('onSubmit')(values); + } + return ( +
+ + ( + + Username + + + + + This is your public display name. + + + + )} + /> + + + + ); +}; + +/** + * The default form of the form. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/hover-card.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/hover-card.stories.tsx new file mode 100644 index 00000000..2154158c --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/hover-card.stories.tsx @@ -0,0 +1,49 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + HoverCard, + HoverCardContent, + HoverCardTrigger, +} from '@repo/design-system/components/ui/hover-card'; + +/** + * For sighted users to preview content available behind a link. + */ +const meta = { + title: 'ui/HoverCard', + component: HoverCard, + tags: ['autodocs'], + argTypes: {}, + args: {}, + render: (args) => ( + + Hover + + The React Framework - created and maintained by @vercel. + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the hover card. + */ +export const Default: Story = {}; + +/** + * Use the `openDelay` and `closeDelay` props to control the delay before the + * hover card opens and closes. + */ +export const Instant: Story = { + args: { + openDelay: 0, + closeDelay: 0, + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/input-otp.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/input-otp.stories.tsx new file mode 100644 index 00000000..a8e574f0 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/input-otp.stories.tsx @@ -0,0 +1,70 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { REGEXP_ONLY_DIGITS_AND_CHARS } from 'input-otp'; + +import { + InputOTP, + InputOTPGroup, + InputOTPSeparator, + InputOTPSlot, +} from '@repo/design-system/components/ui/input-otp'; + +/** + * Accessible one-time password component with copy paste functionality. + */ +const meta = { + title: 'ui/InputOTP', + component: InputOTP, + tags: ['autodocs'], + argTypes: {}, + args: { + maxLength: 6, + pattern: REGEXP_ONLY_DIGITS_AND_CHARS, + children: null, + }, + + render: (args) => ( + + + + + + + + + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the InputOTP field. + */ +export const Default: Story = {}; + +/** + * Use multiple groups to separate the input slots. + */ +export const SeparatedGroup: Story = { + render: (args) => ( + + + + + + + + + + + + + + ), +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/input.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/input.stories.tsx new file mode 100644 index 00000000..5d2120da --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/input.stories.tsx @@ -0,0 +1,84 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { Input } from '@repo/design-system/components/ui/input'; + +/** + * Displays a form input field or a component that looks like an input field. + */ +const meta = { + title: 'ui/Input', + component: Input, + tags: ['autodocs'], + argTypes: {}, + args: { + className: 'w-96', + type: 'email', + placeholder: 'Email', + disabled: false, + }, + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the input field. + */ +export const Default: Story = {}; + +/** + * Use the `disabled` prop to make the input non-interactive and appears faded, + * indicating that input is not currently accepted. + */ +export const Disabled: Story = { + args: { disabled: true }, +}; + +/** + * Use the `Label` component to includes a clear, descriptive label above or + * alongside the input area to guide users. + */ +export const WithLabel: Story = { + render: (args) => ( +
+ + +
+ ), +}; + +/** + * Use a text element below the input field to provide additional instructions + * or information to users. + */ +export const WithHelperText: Story = { + render: (args) => ( +
+ + +

Enter your email address.

+
+ ), +}; + +/** + * Use the `Button` component to indicate that the input field can be submitted + * or used to trigger an action. + */ +export const WithButton: Story = { + render: (args) => ( +
+ + +
+ ), +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/label.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/label.stories.tsx new file mode 100644 index 00000000..bc735c30 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/label.stories.tsx @@ -0,0 +1,30 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { Label } from '@repo/design-system/components/ui/label'; + +/** + * Renders an accessible label associated with controls. + */ +const meta = { + title: 'ui/Label', + component: Label, + tags: ['autodocs'], + argTypes: { + children: { + control: { type: 'text' }, + }, + }, + args: { + children: 'Your email address', + htmlFor: 'email', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the label. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/menubar.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/menubar.stories.tsx new file mode 100644 index 00000000..be9d140d --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/menubar.stories.tsx @@ -0,0 +1,126 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Menubar, + MenubarCheckboxItem, + MenubarContent, + MenubarGroup, + MenubarItem, + MenubarLabel, + MenubarMenu, + MenubarRadioGroup, + MenubarRadioItem, + MenubarSeparator, + MenubarShortcut, + MenubarSub, + MenubarSubContent, + MenubarSubTrigger, + MenubarTrigger, +} from '@repo/design-system/components/ui/menubar'; + +/** + * A visually persistent menu common in desktop applications that provides + * quick access to a consistent set of commands. + */ +const meta = { + title: 'ui/Menubar', + component: Menubar, + tags: ['autodocs'], + argTypes: {}, + + render: (args) => ( + + + File + + + New Tab ⌘T + + New Window + + Share + + Print + + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the menubar. + */ +export const Default: Story = {}; + +/** + * A menubar with a submenu. + */ +export const WithSubmenu: Story = { + render: (args) => ( + + + Actions + + Download + + Share + + Email link + Messages + Notes + + + + + + ), +}; + +/** + * A menubar with radio items. + */ +export const WithRadioItems: Story = { + render: (args) => ( + + + View + + Device Size + + Small + Medium + Large + + + + + ), +}; + +/** + * A menubar with checkbox items. + */ +export const WithCheckboxItems: Story = { + render: (args) => ( + + + Filters + + Show All + + Unread + Important + Flagged + + + + + ), +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/navigation-menu.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/navigation-menu.stories.tsx new file mode 100644 index 00000000..f077875d --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/navigation-menu.stories.tsx @@ -0,0 +1,79 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + NavigationMenu, + NavigationMenuContent, + NavigationMenuItem, + NavigationMenuLink, + NavigationMenuList, + NavigationMenuTrigger, + navigationMenuTriggerStyle, +} from '@repo/design-system/components/ui/navigation-menu'; + +/** + * A collection of links for navigating websites. + */ +const meta = { + title: 'ui/NavigationMenu', + component: NavigationMenu, + tags: ['autodocs'], + argTypes: {}, + render: (args) => ( + + + + + Overview + + + + + + Documentation + + +
    +
  • + + API Reference + +
  • +
  • + + Getting Started + +
  • +
  • + + Guides + +
  • +
+
+
+
+ + + External + + +
+
+ ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the navigation menu. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/pagination.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/pagination.stories.tsx new file mode 100644 index 00000000..2cee5980 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/pagination.stories.tsx @@ -0,0 +1,57 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Pagination, + PaginationContent, + PaginationEllipsis, + PaginationItem, + PaginationLink, + PaginationNext, + PaginationPrevious, +} from '@repo/design-system/components/ui/pagination'; + +/** + * Pagination with page navigation, next and previous links. + */ +const meta = { + title: 'ui/Pagination', + component: Pagination, + tags: ['autodocs'], + argTypes: {}, + render: (args) => ( + + + + + + + 1 + + + 2 + + + 3 + + + + + + + + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the pagination. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/popover.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/popover.stories.tsx new file mode 100644 index 00000000..50ff47e8 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/popover.stories.tsx @@ -0,0 +1,36 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Popover, + PopoverContent, + PopoverTrigger, +} from '@repo/design-system/components/ui/popover'; + +/** + * Displays rich content in a portal, triggered by a button. + */ +const meta = { + title: 'ui/Popover', + component: Popover, + tags: ['autodocs'], + argTypes: {}, + + render: (args) => ( + + Open + Place content for the popover here. + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the popover. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/progress.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/progress.stories.tsx new file mode 100644 index 00000000..a325324e --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/progress.stories.tsx @@ -0,0 +1,45 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { Progress } from '@repo/design-system/components/ui/progress'; + +/** + * Displays an indicator showing the completion progress of a task, typically + * displayed as a progress bar. + */ +const meta = { + title: 'ui/Progress', + component: Progress, + tags: ['autodocs'], + argTypes: {}, + args: { + value: 30, + max: 100, + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the progress. + */ +export const Default: Story = {}; + +/** + * When the progress is indeterminate. + */ +export const Indeterminate: Story = { + args: { + value: undefined, + }, +}; + +/** + * When the progress is completed. + */ +export const Completed: Story = { + args: { + value: 100, + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/radio-group.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/radio-group.stories.tsx new file mode 100644 index 00000000..05b476df --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/radio-group.stories.tsx @@ -0,0 +1,40 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + RadioGroup, + RadioGroupItem, +} from '@repo/design-system/components/ui/radio-group'; + +/** + * A set of checkable buttons—known as radio buttons—where no more than one of + * the buttons can be checked at a time. + */ +const meta = { + title: 'ui/RadioGroup', + component: RadioGroup, + tags: ['autodocs'], + argTypes: {}, + args: { + defaultValue: 'comfortable', + className: 'grid gap-2 grid-cols-[1rem_1fr] items-center', + }, + render: (args) => ( + + + + + + + + + ), +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the radio group. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/resizable.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/resizable.stories.tsx new file mode 100644 index 00000000..dd664555 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/resizable.stories.tsx @@ -0,0 +1,59 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + ResizableHandle, + ResizablePanel, + ResizablePanelGroup, +} from '@repo/design-system/components/ui/resizable'; + +/** + * Accessible resizable panel groups and layouts with keyboard support. + */ +const meta: Meta = { + title: 'ui/ResizablePanelGroup', + component: ResizablePanelGroup, + tags: ['autodocs'], + argTypes: { + onLayout: { + control: false, + }, + }, + args: { + className: 'max-w-96 rounded-lg border', + direction: 'horizontal', + }, + render: (args) => ( + + +
+ One +
+
+ + + + +
+ Two +
+
+ + +
+ Three +
+
+
+
+
+ ), +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the resizable panel group. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/scroll-area.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/scroll-area.stories.tsx new file mode 100644 index 00000000..fe285521 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/scroll-area.stories.tsx @@ -0,0 +1,62 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { ScrollArea } from '@repo/design-system/components/ui/scroll-area'; + +/** + * Augments native scroll functionality for custom, cross-browser styling. + */ +const meta = { + title: 'ui/ScrollArea', + component: ScrollArea, + tags: ['autodocs'], + argTypes: { + children: { + control: 'text', + }, + }, + args: { + className: 'h-32 w-80 rounded-md border p-4', + type: 'auto', + children: + "Jokester began sneaking into the castle in the middle of the night and leaving jokes all over the place: under the king's pillow, in his soup, even in the royal toilet. The king was furious, but he couldn't seem to stop Jokester. And then, one day, the people of the kingdom discovered that the jokes left by Jokester were so funny that they couldn't help but laugh. And once they started laughing, they couldn't stop. The king was so angry that he banished Jokester from the kingdom, but the people still laughed, and they laughed, and they laughed. And they all lived happily ever after.", + }, + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the scroll area. + */ +export const Default: Story = {}; + +/** + * Use the `type` prop with `always` to always show the scroll area. + */ +export const Always: Story = { + args: { + type: 'always', + }, +}; + +/** + * Use the `type` prop with `hover` to show the scroll area on hover. + */ +export const Hover: Story = { + args: { + type: 'hover', + }, +}; + +/** + * Use the `type` prop with `scroll` to show the scroll area when scrolling. + */ +export const Scroll: Story = { + args: { + type: 'scroll', + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/select.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/select.stories.tsx new file mode 100644 index 00000000..b2ee1d08 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/select.stories.tsx @@ -0,0 +1,70 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Select, + SelectContent, + SelectGroup, + SelectItem, + SelectLabel, + SelectSeparator, + SelectTrigger, + SelectValue, +} from '@repo/design-system/components/ui/select'; + +/** + * Displays a list of options for the user to pick from—triggered by a button. + */ +const meta: Meta = { + title: 'ui/Select', + component: Select, + tags: ['autodocs'], + argTypes: {}, + render: (args) => ( + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the select. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/separator.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/separator.stories.tsx new file mode 100644 index 00000000..bb802931 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/separator.stories.tsx @@ -0,0 +1,43 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { Separator } from '@repo/design-system/components/ui/separator'; + +/** + * Visually or semantically separates content. + */ +const meta = { + title: 'ui/Separator', + component: Separator, + tags: ['autodocs'], + argTypes: {}, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the separator. + */ +export const Horizontal: Story = { + render: () => ( +
+
Left
+ +
Right
+
+ ), +}; + +/** + * A vertical separator. + */ +export const Vertical: Story = { + render: () => ( +
+
Top
+ +
Bottom
+
+ ), +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/sheet.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/sheet.stories.tsx new file mode 100644 index 00000000..81fbe29e --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/sheet.stories.tsx @@ -0,0 +1,72 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Sheet, + SheetClose, + SheetContent, + SheetDescription, + SheetFooter, + SheetHeader, + SheetTitle, + SheetTrigger, +} from '@repo/design-system/components/ui/sheet'; + +/** + * Extends the Dialog component to display content that complements the main + * content of the screen. + */ +const meta: Meta = { + title: 'ui/Sheet', + component: Sheet, + tags: ['autodocs'], + argTypes: { + side: { + options: ['top', 'bottom', 'left', 'right'], + control: { + type: 'radio', + }, + }, + }, + args: { + side: 'right', + }, + render: (args) => ( + + Open + + + Are you absolutely sure? + + This action cannot be undone. This will permanently delete your + account and remove your data from our servers. + + + + + + + + + + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the sheet. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/sidebar.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/sidebar.stories.tsx new file mode 100644 index 00000000..98c0d64c --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/sidebar.stories.tsx @@ -0,0 +1,494 @@ +import type { Meta, StoryObj } from '@storybook/react'; +import { + AudioWaveform, + BadgeCheck, + Bell, + BookOpen, + Bot, + ChevronRight, + ChevronsUpDown, + Command, + CreditCard, + Folder, + Forward, + Frame, + GalleryVerticalEnd, + LogOut, + // biome-ignore lint/suspicious/noShadowRestrictedNames: "icon name" + Map, + MoreHorizontal, + PieChart, + Plus, + Settings2, + Sparkles, + SquareTerminal, + Trash2, +} from 'lucide-react'; + +import { + Avatar, + AvatarFallback, + AvatarImage, +} from '@repo/design-system/components/ui/avatar'; +import { + Breadcrumb, + BreadcrumbItem, + BreadcrumbLink, + BreadcrumbList, + BreadcrumbPage, + BreadcrumbSeparator, +} from '@repo/design-system/components/ui/breadcrumb'; +import { + Collapsible, + CollapsibleContent, + CollapsibleTrigger, +} from '@repo/design-system/components/ui/collapsible'; +import { + DropdownMenu, + DropdownMenuContent, + DropdownMenuGroup, + DropdownMenuItem, + DropdownMenuLabel, + DropdownMenuSeparator, + DropdownMenuShortcut, + DropdownMenuTrigger, +} from '@repo/design-system/components/ui/dropdown-menu'; +import { Separator } from '@repo/design-system/components/ui/separator'; +import { + Sidebar, + SidebarContent, + SidebarFooter, + SidebarGroup, + SidebarGroupLabel, + SidebarHeader, + SidebarInset, + SidebarMenu, + SidebarMenuAction, + SidebarMenuButton, + SidebarMenuItem, + SidebarMenuSub, + SidebarMenuSubButton, + SidebarMenuSubItem, + SidebarProvider, + SidebarRail, + SidebarTrigger, +} from '@repo/design-system/components/ui/sidebar'; +import { useState } from 'react'; + +const meta: Meta = { + title: 'ui/Sidebar', + component: Sidebar, + tags: ['autodocs'], + argTypes: {}, +}; +export default meta; + +type Story = StoryObj; + +const data = { + user: { + name: 'shadcn', + email: 'm@example.com', + avatar: '/avatars/shadcn.jpg', + }, + teams: [ + { + name: 'Acme Inc', + logo: GalleryVerticalEnd, + plan: 'Enterprise', + }, + { + name: 'Acme Corp.', + logo: AudioWaveform, + plan: 'Startup', + }, + { + name: 'Evil Corp.', + logo: Command, + plan: 'Free', + }, + ], + navMain: [ + { + title: 'Playground', + url: '#', + icon: SquareTerminal, + isActive: true, + items: [ + { + title: 'History', + url: '#', + }, + { + title: 'Starred', + url: '#', + }, + { + title: 'Settings', + url: '#', + }, + ], + }, + { + title: 'Models', + url: '#', + icon: Bot, + items: [ + { + title: 'Genesis', + url: '#', + }, + { + title: 'Explorer', + url: '#', + }, + { + title: 'Quantum', + url: '#', + }, + ], + }, + { + title: 'Documentation', + url: '#', + icon: BookOpen, + items: [ + { + title: 'Introduction', + url: '#', + }, + { + title: 'Get Started', + url: '#', + }, + { + title: 'Tutorials', + url: '#', + }, + { + title: 'Changelog', + url: '#', + }, + ], + }, + { + title: 'Settings', + url: '#', + icon: Settings2, + items: [ + { + title: 'General', + url: '#', + }, + { + title: 'Team', + url: '#', + }, + { + title: 'Billing', + url: '#', + }, + { + title: 'Limits', + url: '#', + }, + ], + }, + ], + projects: [ + { + name: 'Design Engineering', + url: '#', + icon: Frame, + }, + { + name: 'Sales & Marketing', + url: '#', + icon: PieChart, + }, + { + name: 'Travel', + url: '#', + icon: Map, + }, + ], +}; + +export const Base: Story = { + render: () => { + const [activeTeam, setActiveTeam] = useState(data.teams[0]); + + return ( + + + + + + + + +
+ +
+
+ + {activeTeam.name} + + + {activeTeam.plan} + +
+ +
+
+ + + Teams + + {data.teams.map((team, index) => ( + setActiveTeam(team)} + className="gap-2 p-2" + > +
+ +
+ {team.name} + + ⌘{index + 1} + +
+ ))} + + +
+ +
+
+ Add team +
+
+
+
+
+
+
+ + + Platform + + {data.navMain.map((item) => ( + + + + + {item.icon && } + {item.title} + + + + + + {item.items?.map((subItem) => ( + + + + {subItem.title} + + + + ))} + + + + + ))} + + + + Projects + + {data.projects.map((item) => ( + + + + + {item.name} + + + + + + + More + + + + + + View Project + + + + Share Project + + + + + Delete Project + + + + + ))} + + + + More + + + + + + + + + + + + + + + CN + + +
+ + {data.user.name} + + + {data.user.email} + +
+ +
+
+ + +
+ + + + CN + + +
+ + {data.user.name} + + + {data.user.email} + +
+
+
+ + + + + Upgrade to Pro + + + + + + + Account + + + + Billing + + + + Notifications + + + + + + Log out + +
+
+
+
+
+ +
+ +
+
+ + + + + + + Building Your Application + + + + + Data Fetching + + + +
+
+
+
+
+
+
+
+
+
+ + + ); + }, + args: {}, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/skeleton.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/skeleton.stories.tsx new file mode 100644 index 00000000..7de496b0 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/skeleton.stories.tsx @@ -0,0 +1,35 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { Skeleton } from '@repo/design-system/components/ui/skeleton'; + +/** + * Use to show a placeholder while content is loading. + */ +const meta = { + title: 'ui/Skeleton', + component: Skeleton, + tags: ['autodocs'], + argTypes: {}, + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the skeleton. + */ +export const Default: Story = { + render: (args) => ( +
+ +
+ + +
+
+ ), +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/slider.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/slider.stories.tsx new file mode 100644 index 00000000..f33add05 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/slider.stories.tsx @@ -0,0 +1,45 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { Slider } from '@repo/design-system/components/ui/slider'; + +/** + * An input where the user selects a value from within a given range. + */ +const meta = { + title: 'ui/Slider', + component: Slider, + tags: ['autodocs'], + argTypes: {}, + args: { + defaultValue: [33], + max: 100, + step: 1, + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the slider. + */ +export const Default: Story = {}; + +/** + * Use the `inverted` prop to have the slider fill from right to left. + */ +export const Inverted: Story = { + args: { + inverted: true, + }, +}; + +/** + * Use the `disabled` prop to disable the slider. + */ +export const Disabled: Story = { + args: { + disabled: true, + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/sonner.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/sonner.stories.tsx new file mode 100644 index 00000000..b23559d7 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/sonner.stories.tsx @@ -0,0 +1,50 @@ +import { action } from '@storybook/addon-actions'; +import type { Meta, StoryObj } from '@storybook/react'; +import { toast } from 'sonner'; + +import { Toaster } from '@repo/design-system/components/ui/sonner'; + +/** + * An opinionated toast component for React. + */ +const meta: Meta = { + title: 'ui/Sonner', + component: Toaster, + tags: ['autodocs'], + argTypes: {}, + args: { + position: 'bottom-right', + }, + parameters: { + layout: 'fullscreen', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the toaster. + */ +export const Default: Story = { + render: (args) => ( +
+ + +
+ ), +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/switch.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/switch.stories.tsx new file mode 100644 index 00000000..9ad00b3e --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/switch.stories.tsx @@ -0,0 +1,47 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { Switch } from '@repo/design-system/components/ui/switch'; + +/** + * A control that allows the user to toggle between checked and not checked. + */ +const meta = { + title: 'ui/Switch', + component: Switch, + tags: ['autodocs'], + argTypes: {}, + parameters: { + layout: 'centered', + }, + render: (args) => ( +
+ + +
+ ), +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the switch. + */ +export const Default: Story = { + args: { + id: 'default-switch', + }, +}; + +/** + * Use the `disabled` prop to disable the switch. + */ +export const Disabled: Story = { + args: { + id: 'disabled-switch', + disabled: true, + }, +}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/table.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/table.stories.tsx new file mode 100644 index 00000000..29cd8e2f --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/table.stories.tsx @@ -0,0 +1,80 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Table, + TableBody, + TableCaption, + TableCell, + TableHead, + TableHeader, + TableRow, +} from '@repo/design-system/components/ui/table'; + +const invoices = [ + { + invoice: 'INV001', + paymentStatus: 'Paid', + totalAmount: '$250.00', + paymentMethod: 'Credit Card', + }, + { + invoice: 'INV002', + paymentStatus: 'Pending', + totalAmount: '$150.00', + paymentMethod: 'PayPal', + }, + { + invoice: 'INV003', + paymentStatus: 'Unpaid', + totalAmount: '$350.00', + paymentMethod: 'Bank Transfer', + }, + { + invoice: 'INV004', + paymentStatus: 'Paid', + totalAmount: '$450.00', + paymentMethod: 'Credit Card', + }, +]; + +/** + * Powerful table and datagrids built using TanStack Table. + */ +const meta = { + title: 'ui/Table', + component: Table, + tags: ['autodocs'], + argTypes: {}, + render: (args) => ( + + A list of your recent invoices. + + + Invoice + Status + Method + Amount + + + + {invoices.map((invoice) => ( + + {invoice.invoice} + {invoice.paymentStatus} + {invoice.paymentMethod} + {invoice.totalAmount} + + ))} + +
+ ), +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the table. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/tabs.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/tabs.stories.tsx new file mode 100644 index 00000000..8e4748a1 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/tabs.stories.tsx @@ -0,0 +1,47 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { + Tabs, + TabsContent, + TabsList, + TabsTrigger, +} from '@repo/design-system/components/ui/tabs'; + +/** + * A set of layered sections of content—known as tab panels—that are displayed + * one at a time. + */ +const meta = { + title: 'ui/Tabs', + component: Tabs, + tags: ['autodocs'], + argTypes: {}, + args: { + defaultValue: 'account', + className: 'w-96', + }, + render: (args) => ( + + + Account + Password + + + Make changes to your account here. + + Change your password here. + + ), + parameters: { + layout: 'centered', + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the tabs. + */ +export const Default: Story = {}; diff --git a/ts/saas-nextforge-encore/frontend/apps/storybook/stories/textarea.stories.tsx b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/textarea.stories.tsx new file mode 100644 index 00000000..bf3a32a6 --- /dev/null +++ b/ts/saas-nextforge-encore/frontend/apps/storybook/stories/textarea.stories.tsx @@ -0,0 +1,82 @@ +import type { Meta, StoryObj } from '@storybook/react'; + +import { Textarea } from '@repo/design-system/components/ui/textarea'; + +/** + * Displays a form textarea or a component that looks like a textarea. + */ +const meta = { + title: 'ui/Textarea', + component: Textarea, + tags: ['autodocs'], + argTypes: {}, + args: { + placeholder: 'Type your message here.', + disabled: false, + }, +} satisfies Meta; + +export default meta; + +type Story = StoryObj; + +/** + * The default form of the textarea. + */ +export const Default: Story = {}; + +/** + * Use the `disabled` prop to disable the textarea. + */ +export const Disabled: Story = { + args: { + disabled: true, + }, +}; + +/** + * Use the `Label` component to includes a clear, descriptive label above or + * alongside the text area to guide users. + */ +export const WithLabel: Story = { + render: (args) => ( +
+ +