Merge pull request 'Feature/Include linters' (#2) from feature/include-linters into main
All checks were successful
Lint and Check Types / Validations (push) Successful in 45s

Reviewed-on: #2
This commit is contained in:
Juan Sebastián Montoya 2025-05-06 20:46:47 -05:00
commit 5bd08c5a36
30 changed files with 260 additions and 126 deletions

View file

@ -11,6 +11,7 @@ MINIO_BUCKET_NAME=your-bucket-name
MINIO_ENDPOINT=your-endpoint MINIO_ENDPOINT=your-endpoint
MINIO_SECRET_KEY=your-secret-key MINIO_SECRET_KEY=your-secret-key
MINIO_USE_SSL=true MINIO_USE_SSL=true
MINIO_PORT=9000
NODE_ENV=production NODE_ENV=production
TOKEN_SECRET=your-secret-key TOKEN_SECRET=your-secret-key

View file

@ -1,12 +1,21 @@
name: CI name: Lint and Check Types
on: [push] on: [push]
jobs: jobs:
test: validations:
name: Validations
runs-on: docker runs-on: docker
steps: steps:
- name: Checkout - name: Checkout
uses: actions/checkout@v4 uses: actions/checkout@v4
- name: Status - name: Setup Node
run: git status uses: actions/setup-node@v4
- name: Host with:
run: cat /etc/hosts node-version: 22
- name: Install
run: npm install
- name: Prisma Generate
run: npm run prisma:generate
- name: Lint
run: npm run lint
- name: Check Types
run: npm run check-types

3
apps/api/.gitignore vendored
View file

@ -56,3 +56,6 @@ profile-*
profile* profile*
*clinic* *clinic*
*flamegraph* *flamegraph*
# prisma
prisma/client

View file

@ -7,6 +7,8 @@
"test": "test" "test": "test"
}, },
"scripts": { "scripts": {
"lint": "eslint . --max-warnings 0",
"check-types": "tsc --noEmit",
"test": "ts-node --test test/**/*.test.ts", "test": "ts-node --test test/**/*.test.ts",
"start": "node dist/index.js", "start": "node dist/index.js",
"dev": "nodemon --delay 2000ms src/index.ts", "dev": "nodemon --delay 2000ms src/index.ts",

View file

@ -6,6 +6,7 @@
generator client { generator client {
provider = "prisma-client-js" provider = "prisma-client-js"
output = "./client"
} }
datasource db { datasource db {

View file

@ -3,7 +3,7 @@ import dotenv from 'dotenv';
import path from 'path'; import path from 'path';
const rootDir = path.resolve(process.cwd(), '../../'); const rootDir = path.resolve(process.cwd(), '../../');
dotenv.config({ path: `${rootDir}/.env.local` }); dotenv.config({ path: `${rootDir}/.env` });
const schema = z const schema = z
.object({ .object({

View file

@ -4,7 +4,7 @@ import fastifyCookie from '@fastify/cookie';
import fastifyCors from '@fastify/cors'; import fastifyCors from '@fastify/cors';
import mercurius from 'mercurius'; import mercurius from 'mercurius';
import mercuriusCodegen from 'mercurius-codegen'; import mercuriusCodegen from 'mercurius-codegen';
import { PrismaClient } from '@prisma/client'; import { PrismaClient } from '../prisma/client';
import config from './config'; import config from './config';
import { resolvers } from './resolvers'; import { resolvers } from './resolvers';
import schema from './schema'; import schema from './schema';

View file

@ -91,7 +91,7 @@ export class MinioService {
try { try {
await this.client.statObject(this.bucketName, objectName); await this.client.statObject(this.bucketName, objectName);
return true; return true;
} catch (error) { } catch (_error) {
return false; return false;
} }
} }

View file

@ -2,7 +2,7 @@ import jwt from 'jsonwebtoken';
import { v4 as uuidv4 } from 'uuid'; import { v4 as uuidv4 } from 'uuid';
import crypto from 'crypto'; import crypto from 'crypto';
import { TokenConfig } from '../config'; import { TokenConfig } from '../config';
import { PrismaClient } from '@prisma/client'; import { PrismaClient } from '../../prisma/client';
import { MemcService } from './memc.service'; import { MemcService } from './memc.service';
type TransactionClient = Omit< type TransactionClient = Omit<
@ -90,7 +90,7 @@ export class TokenService {
); );
const tx = txn const tx = txn
? (callback: Function) => callback(txn) ? (callback: (tx: TransactionClient) => Promise<void>) => callback(txn)
: this.prisma.$transaction.bind(this.prisma); : this.prisma.$transaction.bind(this.prisma);
await tx(async (tx) => { await tx(async (tx) => {

View file

@ -1,4 +1,4 @@
import { PrismaClient } from '@prisma/client'; import type { PrismaClient } from '../prisma/client';
import { import {
MinioService, MinioService,
TokenService, TokenService,

View file

@ -1,3 +1,4 @@
/* eslint-disable @typescript-eslint/no-unsafe-function-type */
import { MercuriusContext } from 'mercurius'; import { MercuriusContext } from 'mercurius';
const isCallback = ( const isCallback = (

View file

@ -2,8 +2,9 @@
"name": "web", "name": "web",
"private": true, "private": true,
"version": "0.0.0", "version": "0.0.0",
"type": "module",
"scripts": { "scripts": {
"lint": "eslint . --max-warnings 0",
"check-types": "tsc --noEmit",
"dev": "vite", "dev": "vite",
"build": "tsc -b && vite build", "build": "tsc -b && vite build",
"preview": "vite preview" "preview": "vite preview"

View file

@ -27,8 +27,8 @@ function App() {
// Check if user is already authenticated // Check if user is already authenticated
const checkAuth = () => { const checkAuth = () => {
const token = localStorage.getItem('token'); const token = window.localStorage.getItem('token');
const storedUserId = localStorage.getItem('userId'); const storedUserId = window.localStorage.getItem('userId');
if (token && storedUserId) { if (token && storedUserId) {
setIsAuthenticated(true); setIsAuthenticated(true);
@ -52,8 +52,8 @@ function App() {
} catch (error) { } catch (error) {
console.error('Logout failed:', error); console.error('Logout failed:', error);
} finally { } finally {
localStorage.removeItem('token'); window.localStorage.removeItem('token');
localStorage.removeItem('userId'); window.localStorage.removeItem('userId');
setIsAuthenticated(false); setIsAuthenticated(false);
setUserId(''); setUserId('');
setSelectedRoomId(null); setSelectedRoomId(null);

View file

@ -75,10 +75,7 @@ interface ChatRoomProps {
} }
export function ChatRoom(props: ChatRoomProps) { export function ChatRoom(props: ChatRoomProps) {
const [variables, setVariables] = createSignal({ const [variables, setVariables] = createSignal({});
id: props.roomId,
roomId: props.roomId,
});
const [message, setMessage] = createSignal(''); const [message, setMessage] = createSignal('');
const [messages, setMessages] = createSignal<Message[]>([]); const [messages, setMessages] = createSignal<Message[]>([]);
const [confirmLeave, setConfirmLeave] = createSignal(false); const [confirmLeave, setConfirmLeave] = createSignal(false);
@ -240,9 +237,9 @@ export function ChatRoom(props: ChatRoomProps) {
} }
}; };
document.addEventListener('mousedown', handleClickOutside); window.document.addEventListener('mousedown', handleClickOutside);
return () => { return () => {
document.removeEventListener('mousedown', handleClickOutside); window.document.removeEventListener('mousedown', handleClickOutside);
}; };
}); });
@ -345,8 +342,8 @@ export function ChatRoom(props: ChatRoomProps) {
{showEmojiPicker() && ( {showEmojiPicker() && (
<div ref={emojiPickerContainer} class='emoji-picker-container'> <div ref={emojiPickerContainer} class='emoji-picker-container'>
{/* @ts-ignore */} {/* @ts-expect-error - emoji-picker-element is not typed */}
<emoji-picker ref={(el: Picker) => setPickerRef(el)}></emoji-picker> <emoji-picker ref={(el: Picker) => setPickerRef(el)} />
</div> </div>
)} )}
</form> </form>

View file

@ -48,8 +48,8 @@ export function LoginForm(props: LoginFormProps) {
if (result.data?.login) { if (result.data?.login) {
const { accessToken, user } = result.data.login; const { accessToken, user } = result.data.login;
localStorage.setItem('token', accessToken); window.localStorage.setItem('token', accessToken);
localStorage.setItem('userId', user.id); window.localStorage.setItem('userId', user.id);
props.onLoginSuccess(accessToken, user.id); props.onLoginSuccess(accessToken, user.id);
} else { } else {
setError('Login failed: No data received from server'); setError('Login failed: No data received from server');

View file

@ -56,8 +56,8 @@ export function RegisterForm(props: RegisterFormProps) {
if (result.data?.register) { if (result.data?.register) {
const { accessToken, user } = result.data.register; const { accessToken, user } = result.data.register;
localStorage.setItem('token', accessToken); window.localStorage.setItem('token', accessToken);
localStorage.setItem('userId', user.id); window.localStorage.setItem('userId', user.id);
props.onRegisterSuccess(accessToken, user.id); props.onRegisterSuccess(accessToken, user.id);
} else { } else {
setError('Registration failed: No data received from server'); setError('Registration failed: No data received from server');

View file

@ -172,7 +172,7 @@ export function RoomList(props: RoomListProps) {
// Check if the current user is a member of a room // Check if the current user is a member of a room
const isMember = (room: Room) => { const isMember = (room: Room) => {
const userId = localStorage.getItem('userId'); const userId = window.localStorage.getItem('userId');
return room.members?.some((member) => member.id === userId); return room.members?.some((member) => member.id === userId);
}; };

View file

@ -37,7 +37,7 @@ export const client = createClient({
], ],
// For development, we'll add a simple header-based authentication // For development, we'll add a simple header-based authentication
fetchOptions: () => { fetchOptions: () => {
const token = localStorage.getItem('token'); const token = window.localStorage.getItem('token');
return { return {
credentials: 'include', credentials: 'include',
headers: { headers: {

219
package-lock.json generated
View file

@ -158,19 +158,25 @@
} }
}, },
"node_modules/@babel/code-frame": { "node_modules/@babel/code-frame": {
"version": "7.26.2", "version": "7.27.1",
"resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.26.2.tgz", "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.27.1.tgz",
"integrity": "sha512-RJlIHRueQgwWitWgF8OdFYGZX328Ax5BCemNGlqHfplnRT9ESi8JkFlvaVYbS+UubVY6dpv87Fs2u5M29iNFVQ==", "integrity": "sha512-cjQ7ZlQ0Mv3b47hABuTevyTuYN4i+loJKGeV9flcCgIK37cCXRh+L1bd3iBHlynerhQ7BhCkn2BPbQUL+rGqFg==",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@babel/helper-validator-identifier": "^7.25.9", "@babel/helper-validator-identifier": "^7.27.1",
"js-tokens": "^4.0.0", "js-tokens": "^4.0.0",
"picocolors": "^1.0.0" "picocolors": "^1.1.1"
}, },
"engines": { "engines": {
"node": ">=6.9.0" "node": ">=6.9.0"
} }
}, },
"node_modules/@babel/code-frame/node_modules/picocolors": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
"integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==",
"license": "ISC"
},
"node_modules/@babel/compat-data": { "node_modules/@babel/compat-data": {
"version": "7.26.8", "version": "7.26.8",
"resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.26.8.tgz", "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.26.8.tgz",
@ -407,18 +413,18 @@
} }
}, },
"node_modules/@babel/helper-string-parser": { "node_modules/@babel/helper-string-parser": {
"version": "7.25.9", "version": "7.27.1",
"resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.25.9.tgz", "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz",
"integrity": "sha512-4A/SCr/2KLd5jrtOMFzaKjVtAei3+2r/NChoBNoZ3EyP/+GlhoaEGoWOZUmFmoITP7zOJyHIMm+DYRd8o3PvHA==", "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==",
"license": "MIT", "license": "MIT",
"engines": { "engines": {
"node": ">=6.9.0" "node": ">=6.9.0"
} }
}, },
"node_modules/@babel/helper-validator-identifier": { "node_modules/@babel/helper-validator-identifier": {
"version": "7.25.9", "version": "7.27.1",
"resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.25.9.tgz", "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.27.1.tgz",
"integrity": "sha512-Ed61U6XJc3CVRfkERJWDz4dJwKe7iLmmJsbOGu9wSloNSFttHV0I8g6UAgb7qnK5ly5bGLPd4oXZlxCdANBOWQ==", "integrity": "sha512-D2hP9eA+Sqx1kBZgzxZh0y1trbuU+JoDkiEwqhQ36nodYqJwyEIhPSdMNd7lOm/4io72luTPWH20Yda0xOuUow==",
"license": "MIT", "license": "MIT",
"engines": { "engines": {
"node": ">=6.9.0" "node": ">=6.9.0"
@ -434,25 +440,25 @@
} }
}, },
"node_modules/@babel/helpers": { "node_modules/@babel/helpers": {
"version": "7.26.9", "version": "7.27.1",
"resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.26.9.tgz", "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.27.1.tgz",
"integrity": "sha512-Mz/4+y8udxBKdmzt/UjPACs4G3j5SshJJEFFKxlCGPydG4JAHXxjWjAwjd09tf6oINvl1VfMJo+nB7H2YKQ0dA==", "integrity": "sha512-FCvFTm0sWV8Fxhpp2McP5/W53GPllQ9QeQ7SiqGWjMf/LVG07lFa5+pgK05IRhVwtvafT22KF+ZSnM9I545CvQ==",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@babel/template": "^7.26.9", "@babel/template": "^7.27.1",
"@babel/types": "^7.26.9" "@babel/types": "^7.27.1"
}, },
"engines": { "engines": {
"node": ">=6.9.0" "node": ">=6.9.0"
} }
}, },
"node_modules/@babel/parser": { "node_modules/@babel/parser": {
"version": "7.26.9", "version": "7.27.2",
"resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.26.9.tgz", "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.27.2.tgz",
"integrity": "sha512-81NWa1njQblgZbQHxWHpxxCzNsa3ZwvFqpUg7P+NNUU6f3UU2jBEg4OlF/J6rl8+PQGh1q6/zWScd001YwcA5A==", "integrity": "sha512-QYLs8299NA7WM/bZAdp+CviYYkVoYXlDW2rzliy3chxd1PQjej7JORuMJDJXJUb9g0TT+B99EwaVLKmX+sPXWw==",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@babel/types": "^7.26.9" "@babel/types": "^7.27.1"
}, },
"bin": { "bin": {
"parser": "bin/babel-parser.js" "parser": "bin/babel-parser.js"
@ -879,40 +885,36 @@
} }
}, },
"node_modules/@babel/runtime": { "node_modules/@babel/runtime": {
"version": "7.26.9", "version": "7.27.1",
"resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.26.9.tgz", "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.27.1.tgz",
"integrity": "sha512-aA63XwOkcl4xxQa3HjPMqOP6LiK0ZDv3mUPYEFXkpHbaFjtGggE1A61FjFzJnB+p7/oy2gA8E+rcBNl/zC1tMg==", "integrity": "sha512-1x3D2xEk2fRo3PAhwQwu5UubzgiVWSXTBfWpVd2Mx2AzRqJuDJCsgaDVZ7HB5iGzDW1Hl1sWN2mFyKjmR9uAog==",
"license": "MIT", "license": "MIT",
"dependencies": {
"regenerator-runtime": "^0.14.0"
},
"engines": { "engines": {
"node": ">=6.9.0" "node": ">=6.9.0"
} }
}, },
"node_modules/@babel/runtime-corejs3": { "node_modules/@babel/runtime-corejs3": {
"version": "7.26.9", "version": "7.27.1",
"resolved": "https://registry.npmjs.org/@babel/runtime-corejs3/-/runtime-corejs3-7.26.9.tgz", "resolved": "https://registry.npmjs.org/@babel/runtime-corejs3/-/runtime-corejs3-7.27.1.tgz",
"integrity": "sha512-5EVjbTegqN7RSJle6hMWYxO4voo4rI+9krITk+DWR+diJgGrjZjrIBnJhjrHYYQsFgI7j1w1QnrvV7YSKBfYGg==", "integrity": "sha512-909rVuj3phpjW6y0MCXAZ5iNeORePa6ldJvp2baWGcTjwqbBDDz6xoS5JHJ7lS88NlwLYj07ImL/8IUMtDZzTA==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"core-js-pure": "^3.30.2", "core-js-pure": "^3.30.2"
"regenerator-runtime": "^0.14.0"
}, },
"engines": { "engines": {
"node": ">=6.9.0" "node": ">=6.9.0"
} }
}, },
"node_modules/@babel/template": { "node_modules/@babel/template": {
"version": "7.26.9", "version": "7.27.2",
"resolved": "https://registry.npmjs.org/@babel/template/-/template-7.26.9.tgz", "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.27.2.tgz",
"integrity": "sha512-qyRplbeIpNZhmzOysF/wFMuP9sctmh2cFzRAZOn1YapxBsE1i9bJIY586R/WBLfLcmcBlM8ROBiQURnnNy+zfA==", "integrity": "sha512-LPDZ85aEJyYSd18/DkjNh4/y1ntkE5KwUHWTiqgRxruuZL2F1yuHligVHLvcHY2vMHXttKFpJn6LwfI7cw7ODw==",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@babel/code-frame": "^7.26.2", "@babel/code-frame": "^7.27.1",
"@babel/parser": "^7.26.9", "@babel/parser": "^7.27.2",
"@babel/types": "^7.26.9" "@babel/types": "^7.27.1"
}, },
"engines": { "engines": {
"node": ">=6.9.0" "node": ">=6.9.0"
@ -946,13 +948,13 @@
} }
}, },
"node_modules/@babel/types": { "node_modules/@babel/types": {
"version": "7.26.9", "version": "7.27.1",
"resolved": "https://registry.npmjs.org/@babel/types/-/types-7.26.9.tgz", "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.27.1.tgz",
"integrity": "sha512-Y3IR1cRnOxOCDvMmNiym7XpXQ93iGDDPHx+Zj+NM+rg0fBaShfQLkg+hKPaZCEvg5N/LeCo4+Rj/i3FuJsIQaw==", "integrity": "sha512-+EzkxvLNfiUeKMgy/3luqfsCWFRXLb7U6wNQTk60tovuckwB15B191tJWvpp4HjiQWdJkCxO3Wbvc6jlk3Xb2Q==",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@babel/helper-string-parser": "^7.25.9", "@babel/helper-string-parser": "^7.27.1",
"@babel/helper-validator-identifier": "^7.25.9" "@babel/helper-validator-identifier": "^7.27.1"
}, },
"engines": { "engines": {
"node": ">=6.9.0" "node": ">=6.9.0"
@ -5231,9 +5233,9 @@
} }
}, },
"node_modules/fastify": { "node_modules/fastify": {
"version": "5.2.1", "version": "5.3.2",
"resolved": "https://registry.npmjs.org/fastify/-/fastify-5.2.1.tgz", "resolved": "https://registry.npmjs.org/fastify/-/fastify-5.3.2.tgz",
"integrity": "sha512-rslrNBF67eg8/Gyn7P2URV8/6pz8kSAscFL4EThZJ8JBMaXacVdVE4hmUcnPNKERl5o/xTiBSLfdowBRhVF1WA==", "integrity": "sha512-AIPqBgtqBAwkOkrnwesEE+dOyU30dQ4kh7udxeGVR05CRGwubZx+p2H8P0C4cRnQT0+EPK4VGea2DTL2RtWttg==",
"funding": [ "funding": [
{ {
"type": "github", "type": "github",
@ -5256,9 +5258,9 @@
"find-my-way": "^9.0.0", "find-my-way": "^9.0.0",
"light-my-request": "^6.0.0", "light-my-request": "^6.0.0",
"pino": "^9.0.0", "pino": "^9.0.0",
"process-warning": "^4.0.0", "process-warning": "^5.0.0",
"rfdc": "^1.3.1", "rfdc": "^1.3.1",
"secure-json-parse": "^3.0.1", "secure-json-parse": "^4.0.0",
"semver": "^7.6.0", "semver": "^7.6.0",
"toad-cache": "^3.7.0" "toad-cache": "^3.7.0"
} }
@ -5269,6 +5271,38 @@
"integrity": "sha512-HCxs+YnRaWzCl+cWRYFnHmeRFyR5GVnJTAaCJQiYzQSDwK9MgJdyAsuL3nh0EWRCYMgQ5MeziymvmAhUHYHDUQ==", "integrity": "sha512-HCxs+YnRaWzCl+cWRYFnHmeRFyR5GVnJTAaCJQiYzQSDwK9MgJdyAsuL3nh0EWRCYMgQ5MeziymvmAhUHYHDUQ==",
"license": "MIT" "license": "MIT"
}, },
"node_modules/fastify/node_modules/process-warning": {
"version": "5.0.0",
"resolved": "https://registry.npmjs.org/process-warning/-/process-warning-5.0.0.tgz",
"integrity": "sha512-a39t9ApHNx2L4+HBnQKqxxHNs1r7KF+Intd8Q/g1bUh6q0WIp9voPXJ/x0j+ZL45KF1pJd9+q2jLIRMfvEshkA==",
"funding": [
{
"type": "github",
"url": "https://github.com/sponsors/fastify"
},
{
"type": "opencollective",
"url": "https://opencollective.com/fastify"
}
],
"license": "MIT"
},
"node_modules/fastify/node_modules/secure-json-parse": {
"version": "4.0.0",
"resolved": "https://registry.npmjs.org/secure-json-parse/-/secure-json-parse-4.0.0.tgz",
"integrity": "sha512-dxtLJO6sc35jWidmLxo7ij+Eg48PM/kleBsxpC8QJE0qJICe+KawkDQmvCMZUr9u7WKVHgMW6vy3fQ7zMiFZMA==",
"funding": [
{
"type": "github",
"url": "https://github.com/sponsors/fastify"
},
{
"type": "opencollective",
"url": "https://opencollective.com/fastify"
}
],
"license": "BSD-3-Clause"
},
"node_modules/fastparallel": { "node_modules/fastparallel": {
"version": "2.4.1", "version": "2.4.1",
"resolved": "https://registry.npmjs.org/fastparallel/-/fastparallel-2.4.1.tgz", "resolved": "https://registry.npmjs.org/fastparallel/-/fastparallel-2.4.1.tgz",
@ -8170,6 +8204,7 @@
"version": "1.0.1", "version": "1.0.1",
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.1.tgz", "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.1.tgz",
"integrity": "sha512-anP1Z8qwhkbmu7MFP5iTt+wQKXgwzf7zTyGlcdzabySa9vd0Xt392U0rVmz9poOaBj0uHJKyyo9/upk0HrEQew==", "integrity": "sha512-anP1Z8qwhkbmu7MFP5iTt+wQKXgwzf7zTyGlcdzabySa9vd0Xt392U0rVmz9poOaBj0uHJKyyo9/upk0HrEQew==",
"dev": true,
"license": "ISC" "license": "ISC"
}, },
"node_modules/picomatch": { "node_modules/picomatch": {
@ -8537,12 +8572,6 @@
"node": ">= 12.13.0" "node": ">= 12.13.0"
} }
}, },
"node_modules/regenerator-runtime": {
"version": "0.14.1",
"resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.14.1.tgz",
"integrity": "sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==",
"license": "MIT"
},
"node_modules/registry-auth-token": { "node_modules/registry-auth-token": {
"version": "3.3.2", "version": "3.3.2",
"resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-3.3.2.tgz", "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-3.3.2.tgz",
@ -9420,6 +9449,51 @@
"dev": true, "dev": true,
"license": "MIT" "license": "MIT"
}, },
"node_modules/tinyglobby": {
"version": "0.2.13",
"resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.13.tgz",
"integrity": "sha512-mEwzpUgrLySlveBwEVDMKk5B57bhLPYovRfPAXD5gA/98Opn0rCDj3GtLwFvCvH5RK9uPCExUROW5NjDwvqkxw==",
"dev": true,
"license": "MIT",
"dependencies": {
"fdir": "^6.4.4",
"picomatch": "^4.0.2"
},
"engines": {
"node": ">=12.0.0"
},
"funding": {
"url": "https://github.com/sponsors/SuperchupuDev"
}
},
"node_modules/tinyglobby/node_modules/fdir": {
"version": "6.4.4",
"resolved": "https://registry.npmjs.org/fdir/-/fdir-6.4.4.tgz",
"integrity": "sha512-1NZP+GK4GfuAv3PqKvxQRDMjdSRZjnkq7KfhlNrCNNlZ0ygQFpebfrnfnq/W7fpUnAv9aGWmY1zKx7FYL3gwhg==",
"dev": true,
"license": "MIT",
"peerDependencies": {
"picomatch": "^3 || ^4"
},
"peerDependenciesMeta": {
"picomatch": {
"optional": true
}
}
},
"node_modules/tinyglobby/node_modules/picomatch": {
"version": "4.0.2",
"resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.2.tgz",
"integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==",
"dev": true,
"license": "MIT",
"engines": {
"node": ">=12"
},
"funding": {
"url": "https://github.com/sponsors/jonschlinkert"
}
},
"node_modules/tinygradient": { "node_modules/tinygradient": {
"version": "1.1.5", "version": "1.1.5",
"resolved": "https://registry.npmjs.org/tinygradient/-/tinygradient-1.1.5.tgz", "resolved": "https://registry.npmjs.org/tinygradient/-/tinygradient-1.1.5.tgz",
@ -9965,15 +10039,18 @@
} }
}, },
"node_modules/vite": { "node_modules/vite": {
"version": "6.2.1", "version": "6.3.5",
"resolved": "https://registry.npmjs.org/vite/-/vite-6.2.1.tgz", "resolved": "https://registry.npmjs.org/vite/-/vite-6.3.5.tgz",
"integrity": "sha512-n2GnqDb6XPhlt9B8olZPrgMD/es/Nd1RdChF6CBD/fHW6pUyUTt2sQW2fPRX5GiD9XEa6+8A6A4f2vT6pSsE7Q==", "integrity": "sha512-cZn6NDFE7wdTpINgs++ZJ4N49W2vRp8LCKrn3Ob1kYNtOo21vfDoaV5GzBfLU4MovSAB8uNRm4jgzVQZ+mBzPQ==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"esbuild": "^0.25.0", "esbuild": "^0.25.0",
"fdir": "^6.4.4",
"picomatch": "^4.0.2",
"postcss": "^8.5.3", "postcss": "^8.5.3",
"rollup": "^4.30.1" "rollup": "^4.34.9",
"tinyglobby": "^0.2.13"
}, },
"bin": { "bin": {
"vite": "bin/vite.js" "vite": "bin/vite.js"
@ -10061,6 +10138,34 @@
} }
} }
}, },
"node_modules/vite/node_modules/fdir": {
"version": "6.4.4",
"resolved": "https://registry.npmjs.org/fdir/-/fdir-6.4.4.tgz",
"integrity": "sha512-1NZP+GK4GfuAv3PqKvxQRDMjdSRZjnkq7KfhlNrCNNlZ0ygQFpebfrnfnq/W7fpUnAv9aGWmY1zKx7FYL3gwhg==",
"dev": true,
"license": "MIT",
"peerDependencies": {
"picomatch": "^3 || ^4"
},
"peerDependenciesMeta": {
"picomatch": {
"optional": true
}
}
},
"node_modules/vite/node_modules/picomatch": {
"version": "4.0.2",
"resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.2.tgz",
"integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==",
"dev": true,
"license": "MIT",
"engines": {
"node": ">=12"
},
"funding": {
"url": "https://github.com/sponsors/jonschlinkert"
}
},
"node_modules/vitefu": { "node_modules/vitefu": {
"version": "1.0.6", "version": "1.0.6",
"resolved": "https://registry.npmjs.org/vitefu/-/vitefu-1.0.6.tgz", "resolved": "https://registry.npmjs.org/vitefu/-/vitefu-1.0.6.tgz",

View file

@ -7,7 +7,8 @@
"start:api": "turbo run start --filter=api", "start:api": "turbo run start --filter=api",
"lint": "turbo run lint", "lint": "turbo run lint",
"format": "prettier --write \"**/*.{ts,tsx,md}\"", "format": "prettier --write \"**/*.{ts,tsx,md}\"",
"check-types": "turbo run check-types" "check-types": "turbo run check-types",
"prisma:generate": "turbo run prisma:generate"
}, },
"devDependencies": { "devDependencies": {
"@types/node": "^22.13.9", "@types/node": "^22.13.9",

View file

@ -1,4 +1,4 @@
import js from '@eslint/js'; import eslint from '@eslint/js';
import turboPlugin from 'eslint-plugin-turbo'; import turboPlugin from 'eslint-plugin-turbo';
import tseslint from 'typescript-eslint'; import tseslint from 'typescript-eslint';
@ -7,18 +7,30 @@ import tseslint from 'typescript-eslint';
* *
* @type {import("eslint").Linter.Config[]} * @type {import("eslint").Linter.Config[]}
* */ * */
export const config = [ export const config = tseslint.config(
js.configs.recommended, eslint.configs.recommended,
...tseslint.configs.recommended, tseslint.configs.recommended,
{ [
plugins: { {
turbo: turboPlugin, plugins: {
turbo: turboPlugin,
},
rules: {
'turbo/no-undeclared-env-vars': 'warn',
'@typescript-eslint/no-explicit-any': 'off',
'@typescript-eslint/no-empty-object-type': 'off',
'@typescript-eslint/no-unused-vars': [
'error',
{
argsIgnorePattern: '^_',
varsIgnorePattern: '^_',
caughtErrorsIgnorePattern: '^_',
},
],
},
}, },
rules: { {
'turbo/no-undeclared-env-vars': 'warn', ignores: ['dist/**', 'prisma/client/**'],
}, },
}, ]
{ );
ignores: ['dist/**'],
},
];

View file

@ -1,4 +1,3 @@
import js from '@eslint/js';
import solid from 'eslint-plugin-solid/configs/typescript'; import solid from 'eslint-plugin-solid/configs/typescript';
import * as tsParser from '@typescript-eslint/parser'; import * as tsParser from '@typescript-eslint/parser';
import { config as baseConfig } from './base.js'; import { config as baseConfig } from './base.js';
@ -9,15 +8,19 @@ import { config as baseConfig } from './base.js';
* @type {import("eslint").Linter.Config[]} */ * @type {import("eslint").Linter.Config[]} */
export const config = [ export const config = [
...baseConfig, ...baseConfig,
js.configs.recommended,
{ {
files: ['**/*.{ts,tsx}'], files: ['**/*.{ts,tsx}'],
...solid, ...solid,
ignores: ['turbo/generators/**', 'vite.config.ts'],
languageOptions: { languageOptions: {
parser: tsParser, parser: tsParser,
parserOptions: { parserOptions: {
project: 'tsconfig.json', project: 'tsconfig.json',
}, },
globals: {
window: true,
console: true,
},
}, },
}, },
]; ];

View file

@ -5,6 +5,7 @@
"exports": { "exports": {
"./*": "./src/*.tsx" "./*": "./src/*.tsx"
}, },
"type": "module",
"scripts": { "scripts": {
"lint": "eslint . --max-warnings 0", "lint": "eslint . --max-warnings 0",
"generate:component": "turbo gen react-component", "generate:component": "turbo gen react-component",

View file

@ -8,13 +8,13 @@ interface ButtonProps {
appName: string; appName: string;
} }
export const Button = ({ children, className, appName }: ButtonProps) => { export const Button = (props: ButtonProps) => {
return ( return (
<button <button
class={className} class={props.className}
onClick={() => alert(`Hello from your ${appName} app!`)} onClick={() => window.alert(`Hello from your ${props.appName} app!`)}
> >
{children} {props.children}
</button> </button>
); );
}; };

View file

@ -1,11 +1,6 @@
import { type JSX } from 'solid-js/jsx-runtime'; import { type JSX } from 'solid-js/jsx-runtime';
export function Card({ export function Card(props: {
className,
title,
children,
href,
}: {
className?: string; className?: string;
title: string; title: string;
children: JSX.Element; children: JSX.Element;
@ -13,15 +8,15 @@ export function Card({
}): JSX.Element { }): JSX.Element {
return ( return (
<a <a
class={className} class={props.className}
href={`${href}?utm_source=create-turbo&utm_medium=basic&utm_campaign=create-turbo"`} href={`${props.href}?utm_source=create-turbo&utm_medium=basic&utm_campaign=create-turbo"`}
rel='noopener noreferrer' rel='noopener noreferrer'
target='_blank' target='_blank'
> >
<h2> <h2>
{title} <span>-&gt;</span> {props.title} <span>-&gt;</span>
</h2> </h2>
<p>{children}</p> <p>{props.children}</p>
</a> </a>
); );
} }

View file

@ -1,11 +1,8 @@
import { type JSX } from 'solid-js/jsx-runtime'; import { type JSX } from 'solid-js/jsx-runtime';
export function Code({ export function Code(props: {
children,
className,
}: {
children: JSX.Element; children: JSX.Element;
className?: string; className?: string;
}): JSX.Element { }): JSX.Element {
return <code class={className}>{children}</code>; return <code class={props.className}>{props.children}</code>;
} }

View file

@ -3,5 +3,6 @@
"compilerOptions": { "compilerOptions": {
"outDir": "dist" "outDir": "dist"
}, },
"include": ["src"] "include": ["src"],
"exclude": ["turbo/generators"]
} }

View file

@ -23,7 +23,7 @@
], ],
"tasks": { "tasks": {
"build": { "build": {
"dependsOn": ["^build"], "dependsOn": ["^build", "^prisma:generate"],
"inputs": [".env*"], "inputs": [".env*"],
"outputs": ["dist/**"] "outputs": ["dist/**"]
}, },
@ -34,9 +34,13 @@
"dependsOn": ["^check-types"] "dependsOn": ["^check-types"]
}, },
"dev": { "dev": {
"dependsOn": ["^prisma:generate"],
"cache": false, "cache": false,
"persistent": true "persistent": true
}, },
"prisma:generate": {
"cache": false
},
"start": { "start": {
"cache": false, "cache": false,
"persistent": true "persistent": true