diff --git a/app/components/auth/Login.tsx b/app/components/auth/Login.tsx
new file mode 100644
index 0000000..49ad241
--- /dev/null
+++ b/app/components/auth/Login.tsx
@@ -0,0 +1,48 @@
+import React, { useState } from 'react';
+import { useNavigate } from '@remix-run/react';
+
+export function Login() {
+ const [email, setEmail] = useState('');
+ const [password, setPassword] = useState('');
+ const navigate = useNavigate();
+
+ const handleSubmit = async (e: React.FormEvent) => {
+ e.preventDefault();
+ try {
+ const response = await fetch('/api/auth/login', {
+ method: 'POST',
+ headers: { 'Content-Type': 'application/json' },
+ body: JSON.stringify({ email, password }),
+ });
+ if (response.ok) {
+ const data = (await response.json()) as { token: string };
+ localStorage.setItem('token', data.token);
+ navigate('/dashboard');
+ } else {
+ // 处理错误
+ }
+ } catch (error) {
+ console.error('Login failed:', error);
+ }
+ };
+
+ return (
+
+ );
+}
diff --git a/app/components/auth/Register.tsx b/app/components/auth/Register.tsx
new file mode 100644
index 0000000..2c9046f
--- /dev/null
+++ b/app/components/auth/Register.tsx
@@ -0,0 +1,54 @@
+import React, { useState } from 'react';
+import { useNavigate } from '@remix-run/react';
+
+export function Register() {
+ const [username, setUsername] = useState('');
+ const [email, setEmail] = useState('');
+ const [password, setPassword] = useState('');
+ const navigate = useNavigate();
+
+ const handleSubmit = async (e: React.FormEvent) => {
+ e.preventDefault();
+ try {
+ const response = await fetch('/api/auth/register', {
+ method: 'POST',
+ headers: { 'Content-Type': 'application/json' },
+ body: JSON.stringify({ username, email, password }),
+ });
+ if (response.ok) {
+ navigate('/login');
+ } else {
+ // 处理错误
+ }
+ } catch (error) {
+ console.error('Registration failed:', error);
+ }
+ };
+
+ return (
+
+ );
+}
diff --git a/app/hooks/useAuth.ts b/app/hooks/useAuth.ts
new file mode 100644
index 0000000..7670ef0
--- /dev/null
+++ b/app/hooks/useAuth.ts
@@ -0,0 +1,25 @@
+import { useState, useEffect } from 'react';
+import { useNavigate } from '@remix-run/react';
+
+export function useAuth() {
+ const [isAuthenticated, setIsAuthenticated] = useState(false);
+ const navigate = useNavigate();
+
+ useEffect(() => {
+ const token = localStorage.getItem('token');
+ setIsAuthenticated(!!token);
+ }, []);
+
+ const login = (token: string) => {
+ localStorage.setItem('token', token);
+ setIsAuthenticated(true);
+ };
+
+ const logout = () => {
+ localStorage.removeItem('token');
+ setIsAuthenticated(false);
+ navigate('/login');
+ };
+
+ return { isAuthenticated, login, logout };
+}
diff --git a/app/root.tsx b/app/root.tsx
index 31eb387..af84aec 100644
--- a/app/root.tsx
+++ b/app/root.tsx
@@ -77,7 +77,7 @@ export function Layout({ children }: { children: React.ReactNode }) {
>
);
}
-
export default function App() {
return ;
}
+
diff --git a/app/routes/api.auth.login.ts b/app/routes/api.auth.login.ts
new file mode 100644
index 0000000..336c13d
--- /dev/null
+++ b/app/routes/api.auth.login.ts
@@ -0,0 +1,15 @@
+import { json } from '@remix-run/cloudflare';
+import { type ActionFunction } from '@remix-run/node';
+import { verifyLogin, createToken } from '~/utils/auth.server';
+
+export const action: ActionFunction = async ({ request }) => {
+ const { email, password } = (await request.json()) as { email: string; password: string };
+
+ const user = await verifyLogin(email, password);
+ if (!user) {
+ return json({ success: false, error: 'Invalid credentials' }, { status: 400 });
+ }
+
+ const token = createToken(user._id.toString());
+ return json({ success: true, token });
+};
diff --git a/app/routes/api.auth.register.ts b/app/routes/api.auth.register.ts
new file mode 100644
index 0000000..f82aa3e
--- /dev/null
+++ b/app/routes/api.auth.register.ts
@@ -0,0 +1,20 @@
+import { type ActionFunction, json } from '@remix-run/cloudflare';
+import { db } from '~/utils/db.server';
+import { hashPassword } from '~/utils/auth.server';
+
+export const action: ActionFunction = async ({ request }) => {
+ const { username, email, password } = (await request.json()) as { username: string; email: string; password: string };
+
+ try {
+ const hashedPassword = await hashPassword(password);
+ const [userId] = await db('users').insert({
+ username,
+ email,
+ password: hashedPassword
+ });
+ return json({ success: true, userId });
+ } catch (error) {
+ console.error('Registration error:', error);
+ return json({ success: false, error: 'Registration failed' }, { status: 400 });
+ }
+};
diff --git a/app/routes/login.tsx b/app/routes/login.tsx
new file mode 100644
index 0000000..cb09d42
--- /dev/null
+++ b/app/routes/login.tsx
@@ -0,0 +1,5 @@
+import { Login } from '~/components/auth/Login';
+
+export default function LoginPage() {
+ return ;
+}
diff --git a/app/routes/register.tsx b/app/routes/register.tsx
new file mode 100644
index 0000000..16b108b
--- /dev/null
+++ b/app/routes/register.tsx
@@ -0,0 +1,5 @@
+import { Register } from '~/components/auth/Register';
+
+export default function RegisterPage() {
+ return ;
+}
diff --git a/app/utils/api.ts b/app/utils/api.ts
new file mode 100644
index 0000000..dade1e3
--- /dev/null
+++ b/app/utils/api.ts
@@ -0,0 +1,17 @@
+export async function authenticatedFetch(url: string, options: RequestInit = {}) {
+ const token = localStorage.getItem('token');
+ const headers = {
+ ...options.headers,
+ 'Authorization': `Bearer ${token}`,
+ };
+
+ const response = await fetch(url, { ...options, headers });
+
+ if (response.status === 401) {
+ // Token is invalid or expired
+ localStorage.removeItem('token');
+ window.location.href = '/login';
+ }
+
+ return response;
+}
diff --git a/app/utils/auth.server.ts b/app/utils/auth.server.ts
new file mode 100644
index 0000000..629e6f3
--- /dev/null
+++ b/app/utils/auth.server.ts
@@ -0,0 +1,32 @@
+import bcrypt from 'bcryptjs';
+import jwt from 'jsonwebtoken';
+import { db } from './db.server';
+
+const JWT_SECRET = process.env.JWT_SECRET || 'your-secret-key';
+
+export async function hashPassword(password: string) {
+ return bcrypt.hash(password, 10);
+}
+
+export async function verifyLogin(email: string, password: string) {
+ const user = await db('users').where({ email }).first();
+ if (!user) return null;
+
+ const isValid = await bcrypt.compare(password, user.password);
+ if (!isValid) return null;
+
+ return user;
+}
+
+export function createToken(userId: string) {
+ return jwt.sign({ userId }, JWT_SECRET, { expiresIn: '30d' });
+}
+
+export function verifyToken(token: string) {
+ try {
+ const decoded = jwt.verify(token, JWT_SECRET);
+ return decoded as { userId: string };
+ } catch (error) {
+ return null;
+ }
+}
diff --git a/app/utils/db.server.ts b/app/utils/db.server.ts
new file mode 100644
index 0000000..615669a
--- /dev/null
+++ b/app/utils/db.server.ts
@@ -0,0 +1,37 @@
+import knex from 'knex';
+import type { Knex } from 'knex';
+
+let db: Knex;
+
+declare global {
+ var __db: Knex | undefined;
+}
+
+// 这个检查可以防止在开发模式下多次创建连接
+if (process.env.NODE_ENV === 'production') {
+ db = getDb();
+} else {
+ if (!global.__db) {
+ global.__db = getDb();
+ }
+ db = global.__db;
+}
+
+function getDb() {
+ return knex({
+ client: 'mysql2',
+ connection: {
+ host: process.env.DB_HOST,
+ user: process.env.DB_USER,
+ password: process.env.DB_PASSWORD,
+ database: process.env.DB_NAME,
+ port: Number(process.env.DB_PORT) || 3306,
+ },
+ pool: {
+ min: 2,
+ max: 10
+ }
+ });
+}
+
+export { db };
diff --git a/db/migrations/20241022093005_insert_initial_token_reload.js b/db/migrations/20241022093005_insert_initial_token_reload.js
index c18ec88..3131cae 100644
--- a/db/migrations/20241022093005_insert_initial_token_reload.js
+++ b/db/migrations/20241022093005_insert_initial_token_reload.js
@@ -1,7 +1,7 @@
export function up(knex) {
return knex('token_reloads').insert([
{
- name: '代币充值',
+ name: 'Token Reload',
tokens: 10000000,
price: 20.00,
description: '非订阅代币购买,每次充值10,000,000代币',
diff --git a/package.json b/package.json
index fe5f3df..1898f3c 100644
--- a/package.json
+++ b/package.json
@@ -52,6 +52,7 @@
"@radix-ui/react-dropdown-menu": "^2.1.1",
"@remix-run/cloudflare": "^2.10.2",
"@remix-run/cloudflare-pages": "^2.10.2",
+ "@remix-run/node": "^2.13.1",
"@remix-run/react": "^2.10.2",
"@uiw/codemirror-theme-vscode": "^4.23.0",
"@unocss/reset": "^0.61.0",
@@ -60,12 +61,14 @@
"@xterm/addon-web-links": "^0.11.0",
"@xterm/xterm": "^5.5.0",
"ai": "^3.3.4",
+ "bcryptjs": "^2.4.3",
"date-fns": "^3.6.0",
"diff": "^5.2.0",
"framer-motion": "^11.2.12",
"isbot": "^4.1.0",
"istextorbinary": "^9.5.0",
"jose": "^5.6.3",
+ "jsonwebtoken": "^9.0.2",
"knex": "^3.1.0",
"mysql2": "^3.11.3",
"nanostores": "^0.10.3",
@@ -88,7 +91,9 @@
"@blitz/eslint-plugin": "0.1.0",
"@cloudflare/workers-types": "^4.20240620.0",
"@remix-run/dev": "^2.10.0",
+ "@types/bcryptjs": "^2.4.6",
"@types/diff": "^5.2.1",
+ "@types/jsonwebtoken": "^9.0.7",
"@types/react": "^18.2.20",
"@types/react-dom": "^18.2.7",
"fast-glob": "^3.3.2",
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index b5dee02..57f4fc9 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -83,6 +83,9 @@ dependencies:
'@remix-run/cloudflare-pages':
specifier: ^2.10.2
version: 2.13.1(@cloudflare/workers-types@4.20241018.0)(typescript@5.6.3)
+ '@remix-run/node':
+ specifier: ^2.13.1
+ version: 2.13.1(typescript@5.6.3)
'@remix-run/react':
specifier: ^2.10.2
version: 2.13.1(react-dom@18.3.1)(react@18.3.1)(typescript@5.6.3)
@@ -107,6 +110,9 @@ dependencies:
ai:
specifier: ^3.3.4
version: 3.4.16(react@18.3.1)(svelte@5.0.4)(vue@3.5.12)(zod@3.23.8)
+ bcryptjs:
+ specifier: ^2.4.3
+ version: 2.4.3
date-fns:
specifier: ^3.6.0
version: 3.6.0
@@ -125,6 +131,9 @@ dependencies:
jose:
specifier: ^5.6.3
version: 5.9.6
+ jsonwebtoken:
+ specifier: ^9.0.2
+ version: 9.0.2
knex:
specifier: ^3.1.0
version: 3.1.0(mysql2@3.11.3)
@@ -166,7 +175,7 @@ dependencies:
version: 0.2.0(@remix-run/react@2.13.1)(@remix-run/server-runtime@2.13.1)(react-dom@18.3.1)(react@18.3.1)
remix-utils:
specifier: ^7.6.0
- version: 7.7.0(@remix-run/cloudflare@2.13.1)(@remix-run/react@2.13.1)(react@18.3.1)(zod@3.23.8)
+ version: 7.7.0(@remix-run/cloudflare@2.13.1)(@remix-run/node@2.13.1)(@remix-run/react@2.13.1)(react@18.3.1)(zod@3.23.8)
set-cookie-parser:
specifier: 2.4.8
version: 2.4.8
@@ -187,9 +196,15 @@ devDependencies:
'@remix-run/dev':
specifier: ^2.10.0
version: 2.13.1(@remix-run/react@2.13.1)(sass-embedded@1.80.3)(typescript@5.6.3)(vite@5.4.9)(wrangler@3.81.0)
+ '@types/bcryptjs':
+ specifier: ^2.4.6
+ version: 2.4.6
'@types/diff':
specifier: ^5.2.1
version: 5.2.3
+ '@types/jsonwebtoken':
+ specifier: ^9.0.7
+ version: 9.0.7
'@types/react':
specifier: ^18.2.20
version: 18.3.11
@@ -2862,7 +2877,6 @@ packages:
stream-slice: 0.1.2
typescript: 5.6.3
undici: 6.20.1
- dev: true
/@remix-run/react@2.13.1(react-dom@18.3.1)(react@18.3.1)(typescript@5.6.3):
resolution: {integrity: sha512-kZevCoKMz0ZDOOzTnG95yfM7M9ju38FkWNY1wtxCy+NnUJYrmTerGQtiBsJgMzYD6i29+w4EwoQsdqys7DmMSg==}
@@ -2911,7 +2925,6 @@ packages:
dependencies:
'@remix-run/web-stream': 1.1.0
web-encoding: 1.1.5
- dev: true
/@remix-run/web-fetch@4.4.2:
resolution: {integrity: sha512-jgKfzA713/4kAW/oZ4bC3MoLWyjModOVDjFPNseVqcJKSafgIscrYL9G50SurEYLswPuoU3HzSbO0jQCMYWHhA==}
@@ -2925,25 +2938,21 @@ packages:
abort-controller: 3.0.0
data-uri-to-buffer: 3.0.1
mrmime: 1.0.1
- dev: true
/@remix-run/web-file@3.1.0:
resolution: {integrity: sha512-dW2MNGwoiEYhlspOAXFBasmLeYshyAyhIdrlXBi06Duex5tDr3ut2LFKVj7tyHLmn8nnNwFf1BjNbkQpygC2aQ==}
dependencies:
'@remix-run/web-blob': 3.1.0
- dev: true
/@remix-run/web-form-data@3.1.0:
resolution: {integrity: sha512-NdeohLMdrb+pHxMQ/Geuzdp0eqPbea+Ieo8M8Jx2lGC6TBHsgHzYcBvr0LyPdPVycNRDEpWpiDdCOdCryo3f9A==}
dependencies:
web-encoding: 1.1.5
- dev: true
/@remix-run/web-stream@1.1.0:
resolution: {integrity: sha512-KRJtwrjRV5Bb+pM7zxcTJkhIqWWSy+MYsIxHK+0m5atcznsf15YwUBWHWulZerV2+vvHH1Lp1DD7pw6qKW8SgA==}
dependencies:
web-streams-polyfill: 3.3.3
- dev: true
/@rollup/plugin-inject@5.0.5:
resolution: {integrity: sha512-2+DEJbNBoPROPkgTDNe8/1YXWcqxbN5DTjASVIOx8HS+pITXushyNiBV56RB08zuptzz8gT3YfkqriTBVycepg==}
@@ -3159,6 +3168,10 @@ packages:
'@types/estree': 1.0.6
dev: true
+ /@types/bcryptjs@2.4.6:
+ resolution: {integrity: sha512-9xlo6R2qDs5uixm0bcIqCeMCE6HiQsIyel9KQySStiyqNl2tnj2mP3DX1Nf56MD6KMenNNlBBsy3LJ7gUEQPXQ==}
+ dev: true
+
/@types/cookie@0.6.0:
resolution: {integrity: sha512-4Kh9a6B2bQciAhf7FSuMRRkUWecJgJu9nPnx3yzpsfXX/c50REIqpHY4C82bXP90qrLtXtkDxTZosYO3UpOwlA==}
@@ -3199,6 +3212,12 @@ packages:
resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==}
dev: true
+ /@types/jsonwebtoken@9.0.7:
+ resolution: {integrity: sha512-ugo316mmTYBl2g81zDFnZ7cfxlut3o+/EQdaP7J8QN2kY6lJ22hmQYCK5EHcJHbrW+dkCGSCPgbG8JtYj6qSrg==}
+ dependencies:
+ '@types/node': 22.7.7
+ dev: true
+
/@types/mdast@3.0.15:
resolution: {integrity: sha512-LnwD+mUEfxWMa1QpDraczIn6k0Ee3SMicuYSSzS6ZYl2gKS09EClnJYGd8Du6rfc5r/GZEk5o1mRb8TaTj03sQ==}
dependencies:
@@ -3848,7 +3867,6 @@ packages:
/@zxing/text-encoding@0.9.0:
resolution: {integrity: sha512-U/4aVJ2mxI0aDNI8Uq0wEhMgY+u4CNtEb0om3+y3+niDAsoTCOB33UF0sxpzqzdqXLqmvc+vZyAt4O8pPdfkwA==}
requiresBuild: true
- dev: true
optional: true
/abort-controller@3.0.0:
@@ -3856,7 +3874,6 @@ packages:
engines: {node: '>=6.5'}
dependencies:
event-target-shim: 5.0.1
- dev: true
/accepts@1.3.8:
resolution: {integrity: sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==}
@@ -4054,7 +4071,6 @@ packages:
engines: {node: '>= 0.4'}
dependencies:
possible-typed-array-names: 1.0.0
- dev: true
/aws-ssl-profiles@1.1.2:
resolution: {integrity: sha512-NZKeq9AfyQvEeNlN0zSYAaWrmBffJh3IELMZfRpJVWgrpEbtEpnjvzqBPf+mxoI287JohRDoa+/nsfqqiZmF6g==}
@@ -4077,6 +4093,10 @@ packages:
resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==}
dev: true
+ /bcryptjs@2.4.3:
+ resolution: {integrity: sha512-V/Hy/X9Vt7f3BbPJEi8BdVFMByHi+jNXrYkW3huaybV/kQ0KJg0Y6PkEMbn+zeT+i+SiKZ/HMqJGIIt4LZDqNQ==}
+ dev: false
+
/binary-extensions@2.3.0:
resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==}
engines: {node: '>=8'}
@@ -4239,9 +4259,12 @@ packages:
resolution: {integrity: sha512-7VPMEPuYznPSoR21NE1zvd2Xna6c/CloiZCfcMXR1Jny6PjX0N4Nsa38zcBFo/FMK+BlA+FLKbJCQ0i2yxp+Xg==}
dev: true
+ /buffer-equal-constant-time@1.0.1:
+ resolution: {integrity: sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==}
+ dev: false
+
/buffer-from@1.1.2:
resolution: {integrity: sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==}
- dev: true
/buffer-xor@1.0.3:
resolution: {integrity: sha512-571s0T7nZWK6vB67HI5dyUF7wXiNcfaPPPTl6zYCNApANjIvYJTg7hlud/+cJpdAhS7dVzqMLmfhfHR3rAcOjQ==}
@@ -4305,7 +4328,6 @@ packages:
function-bind: 1.1.2
get-intrinsic: 1.2.4
set-function-length: 1.2.2
- dev: true
/callsites@3.1.0:
resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==}
@@ -4535,7 +4557,6 @@ packages:
/cookie-signature@1.2.1:
resolution: {integrity: sha512-78KWk9T26NhzXtuL26cIJ8/qNHANyJ/ZYrmEXFzUmhZdjpBv+DlWlOANRTGBt48YcyslsLrj0bMLFTmXvLRCOw==}
engines: {node: '>=6.6.0'}
- dev: true
/cookie@0.6.0:
resolution: {integrity: sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==}
@@ -4645,7 +4666,6 @@ packages:
/data-uri-to-buffer@3.0.1:
resolution: {integrity: sha512-WboRycPNsVw3B3TL559F7kuBUM4d8CgMEvk6xEJlOp7OBPjt6G7z8WMWlD2rOFZLk6OYfFIUGsCOWzcQH9K2og==}
engines: {node: '>= 6'}
- dev: true
/data-uri-to-buffer@4.0.1:
resolution: {integrity: sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==}
@@ -4735,7 +4755,6 @@ packages:
es-define-property: 1.0.0
es-errors: 1.3.0
gopd: 1.0.1
- dev: true
/define-properties@1.2.1:
resolution: {integrity: sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==}
@@ -4832,6 +4851,12 @@ packages:
resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==}
dev: true
+ /ecdsa-sig-formatter@1.0.11:
+ resolution: {integrity: sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==}
+ dependencies:
+ safe-buffer: 5.2.1
+ dev: false
+
/editions@6.21.0:
resolution: {integrity: sha512-ofkXJtn7z0urokN62DI3SBo/5xAtF0rR7tn+S/bSYV79Ka8pTajIIl+fFQ1q88DQEImymmo97M4azY3WX/nUdg==}
engines: {node: '>=4'}
@@ -4897,12 +4922,10 @@ packages:
engines: {node: '>= 0.4'}
dependencies:
get-intrinsic: 1.2.4
- dev: true
/es-errors@1.3.0:
resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==}
engines: {node: '>= 0.4'}
- dev: true
/es-module-lexer@1.5.4:
resolution: {integrity: sha512-MVNK56NiMrOwitFB7cqDwq0CQutbw+0BvLshJSse0MUNU+y1FC3bUS/AQg7oUng+/wKrrki7JfmwtVHkVfPLlw==}
@@ -5324,7 +5347,6 @@ packages:
/event-target-shim@5.0.1:
resolution: {integrity: sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==}
engines: {node: '>=6'}
- dev: true
/events@3.3.0:
resolution: {integrity: sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==}
@@ -5505,7 +5527,6 @@ packages:
resolution: {integrity: sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==}
dependencies:
is-callable: 1.2.7
- dev: true
/foreground-child@3.3.0:
resolution: {integrity: sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==}
@@ -5620,7 +5641,6 @@ packages:
has-proto: 1.0.3
has-symbols: 1.0.3
hasown: 2.0.2
- dev: true
/get-nonce@1.0.1:
resolution: {integrity: sha512-FJhYRoDaiatfEkUK8HKlicmu/3SGFD51q3itKDGoSTysQJBnfOcxU5GxnhE1E6soB76MbT0MBtnKJuXyAx+96Q==}
@@ -5712,7 +5732,6 @@ packages:
resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==}
dependencies:
get-intrinsic: 1.2.4
- dev: true
/graceful-fs@4.2.11:
resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==}
@@ -5755,24 +5774,20 @@ packages:
resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==}
dependencies:
es-define-property: 1.0.0
- dev: true
/has-proto@1.0.3:
resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==}
engines: {node: '>= 0.4'}
- dev: true
/has-symbols@1.0.3:
resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==}
engines: {node: '>= 0.4'}
- dev: true
/has-tostringtag@1.0.2:
resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==}
engines: {node: '>= 0.4'}
dependencies:
has-symbols: 1.0.3
- dev: true
/hash-base@3.0.4:
resolution: {integrity: sha512-EeeoJKjTyt868liAlVmcv2ZsUfGHlE3Q+BICOXcZiwN3osr5Q/zFGYmTJpoIzuaSTAwndFy+GqhEwlU4L3j4Ow==}
@@ -6054,7 +6069,6 @@ packages:
/inherits@2.0.4:
resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==}
- dev: true
/inline-style-parser@0.1.1:
resolution: {integrity: sha512-7NXolsK4CAS5+xvdj5OMMbI962hU/wvwoxk+LWR9Ek9bVtyuuYScDN6eS0rUm6TxApFpw7CX1o4uJzcd4AyD3Q==}
@@ -6095,7 +6109,6 @@ packages:
dependencies:
call-bind: 1.0.7
has-tostringtag: 1.0.2
- dev: true
/is-binary-path@2.1.0:
resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==}
@@ -6112,7 +6125,6 @@ packages:
/is-callable@1.2.7:
resolution: {integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==}
engines: {node: '>= 0.4'}
- dev: true
/is-ci@3.0.1:
resolution: {integrity: sha512-ZYvCgrefwqoQ6yTyYUbQu64HsITZ3NfKX1lzaEYdkTDcfKzzCI/wthRRYKkdjHKFVgNiXKAKm65Zo1pk2as/QQ==}
@@ -6149,7 +6161,6 @@ packages:
engines: {node: '>= 0.4'}
dependencies:
has-tostringtag: 1.0.2
- dev: true
/is-glob@4.0.3:
resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==}
@@ -6212,7 +6223,6 @@ packages:
engines: {node: '>= 0.4'}
dependencies:
which-typed-array: 1.1.15
- dev: true
/is-unicode-supported@0.1.0:
resolution: {integrity: sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==}
@@ -6343,6 +6353,37 @@ packages:
graceful-fs: 4.2.11
dev: true
+ /jsonwebtoken@9.0.2:
+ resolution: {integrity: sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==}
+ engines: {node: '>=12', npm: '>=6'}
+ dependencies:
+ jws: 3.2.2
+ lodash.includes: 4.3.0
+ lodash.isboolean: 3.0.3
+ lodash.isinteger: 4.0.4
+ lodash.isnumber: 3.0.3
+ lodash.isplainobject: 4.0.6
+ lodash.isstring: 4.0.1
+ lodash.once: 4.1.1
+ ms: 2.1.3
+ semver: 7.6.3
+ dev: false
+
+ /jwa@1.4.1:
+ resolution: {integrity: sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==}
+ dependencies:
+ buffer-equal-constant-time: 1.0.1
+ ecdsa-sig-formatter: 1.0.11
+ safe-buffer: 5.2.1
+ dev: false
+
+ /jws@3.2.2:
+ resolution: {integrity: sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==}
+ dependencies:
+ jwa: 1.4.1
+ safe-buffer: 5.2.1
+ dev: false
+
/keyv@4.5.4:
resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==}
dependencies:
@@ -6455,10 +6496,38 @@ packages:
resolution: {integrity: sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==}
dev: true
+ /lodash.includes@4.3.0:
+ resolution: {integrity: sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w==}
+ dev: false
+
+ /lodash.isboolean@3.0.3:
+ resolution: {integrity: sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg==}
+ dev: false
+
+ /lodash.isinteger@4.0.4:
+ resolution: {integrity: sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA==}
+ dev: false
+
+ /lodash.isnumber@3.0.3:
+ resolution: {integrity: sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw==}
+ dev: false
+
+ /lodash.isplainobject@4.0.6:
+ resolution: {integrity: sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==}
+ dev: false
+
+ /lodash.isstring@4.0.1:
+ resolution: {integrity: sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw==}
+ dev: false
+
/lodash.merge@4.6.2:
resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==}
dev: true
+ /lodash.once@4.1.1:
+ resolution: {integrity: sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==}
+ dev: false
+
/lodash@4.17.21:
resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==}
@@ -7572,7 +7641,6 @@ packages:
/mrmime@1.0.1:
resolution: {integrity: sha512-hzzEagAgDyoU1Q6yg5uI+AorQgdvMCur3FcKf7NhMKWsaYg+RnbTyHRa/9IlLF9rf455MOCtcqqrQQ83pPP7Uw==}
engines: {node: '>=10'}
- dev: true
/mrmime@2.0.0:
resolution: {integrity: sha512-eu38+hdgojoyq63s+yTpN4XMBdt5l8HhMhc4VKLO9KM5caLIBvUm4thi7fFaxyTmCKeNnXZ5pAlBwCUnhA09uw==}
@@ -8050,7 +8118,6 @@ packages:
/possible-typed-array-names@1.0.0:
resolution: {integrity: sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==}
engines: {node: '>= 0.4'}
- dev: true
/postcss-discard-duplicates@5.1.0(postcss@8.4.47):
resolution: {integrity: sha512-zmX3IoSI2aoenxHV6C7plngHWWhUOV3sP1T8y2ifzxzbtnuhk1EdPwm0S1bIUNaJ2eNbWeGLEwzw8huPD67aQw==}
@@ -8625,7 +8692,7 @@ packages:
react-dom: 18.3.1(react@18.3.1)
dev: false
- /remix-utils@7.7.0(@remix-run/cloudflare@2.13.1)(@remix-run/react@2.13.1)(react@18.3.1)(zod@3.23.8):
+ /remix-utils@7.7.0(@remix-run/cloudflare@2.13.1)(@remix-run/node@2.13.1)(@remix-run/react@2.13.1)(react@18.3.1)(zod@3.23.8):
resolution: {integrity: sha512-J8NhP044nrNIam/xOT1L9a4RQ9FSaA2wyrUwmN8ZT+c/+CdAAf70yfaLnvMyKcV5U+8BcURQ/aVbth77sT6jGA==}
engines: {node: '>=18.0.0'}
peerDependencies:
@@ -8659,6 +8726,7 @@ packages:
optional: true
dependencies:
'@remix-run/cloudflare': 2.13.1(@cloudflare/workers-types@4.20241018.0)(typescript@5.6.3)
+ '@remix-run/node': 2.13.1(typescript@5.6.3)
'@remix-run/react': 2.13.1(react-dom@18.3.1)(react@18.3.1)(typescript@5.6.3)
react: 18.3.1
type-fest: 4.26.1
@@ -8793,7 +8861,6 @@ packages:
/safe-buffer@5.2.1:
resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==}
- dev: true
/safer-buffer@2.1.2:
resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==}
@@ -9039,7 +9106,6 @@ packages:
resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==}
engines: {node: '>=10'}
hasBin: true
- dev: true
/send@0.19.0:
resolution: {integrity: sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw==}
@@ -9095,7 +9161,6 @@ packages:
get-intrinsic: 1.2.4
gopd: 1.0.1
has-property-descriptors: 1.0.2
- dev: true
/setimmediate@1.0.5:
resolution: {integrity: sha512-MATJdZp8sLqDl/68LfQmbP8zKPLQNV6BIZoIgrscFDQ+RsvK/BxeDQOgyxKKoh0y/8h3BqVFnCqQ/gd+reiIXA==}
@@ -9177,12 +9242,10 @@ packages:
dependencies:
buffer-from: 1.1.2
source-map: 0.6.1
- dev: true
/source-map@0.6.1:
resolution: {integrity: sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==}
engines: {node: '>=0.10.0'}
- dev: true
/source-map@0.7.4:
resolution: {integrity: sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==}
@@ -9286,7 +9349,6 @@ packages:
/stream-slice@0.1.2:
resolution: {integrity: sha512-QzQxpoacatkreL6jsxnVb7X5R/pGw9OUv2qWTYWnmLpg4NdN31snPy/f3TdQE1ZUXaThRvj1Zw4/OGg0ZkaLMA==}
- dev: true
/string-hash@1.1.3:
resolution: {integrity: sha512-kJUvRUFK49aub+a7T1nNE66EJbZBMnBgoC1UbCZ5n6bsZKBRga4KgBRTMn/pFkeCZSYtNeSyMxPDM0AXWELk2A==}
@@ -9698,7 +9760,6 @@ packages:
/undici@6.20.1:
resolution: {integrity: sha512-AjQF1QsmqfJys+LXfGTNum+qw4S88CojRInG/6t31W/1fk6G59s92bnAvGz5Cmur+kQv2SURXEvvudLmbrE8QA==}
engines: {node: '>=18.17'}
- dev: true
/unenv-nightly@2.0.0-20241009-125958-e8ea22f:
resolution: {integrity: sha512-hRxmKz1iSVRmuFx/vBdPsx7rX4o7Cas9vdjDNeUeWpQTK2LzU3Xy3Jz0zbo7MJX0bpqo/LEFCA+GPwsbl6zKEQ==}
@@ -9953,7 +10014,6 @@ packages:
is-generator-function: 1.0.10
is-typed-array: 1.1.13
which-typed-array: 1.1.15
- dev: true
/utils-merge@1.0.1:
resolution: {integrity: sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==}
@@ -10243,7 +10303,6 @@ packages:
util: 0.12.5
optionalDependencies:
'@zxing/text-encoding': 0.9.0
- dev: true
/web-namespaces@2.0.1:
resolution: {integrity: sha512-bKr1DkiNa2krS7qxNtdrtHAmzuYGFQLiQ13TsorsdT6ULTkPLKuu5+GsFpDlg6JFjUTwX2DyhMPG2be8uPrqsQ==}
@@ -10252,7 +10311,6 @@ packages:
/web-streams-polyfill@3.3.3:
resolution: {integrity: sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==}
engines: {node: '>= 8'}
- dev: true
/which-typed-array@1.1.15:
resolution: {integrity: sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==}
@@ -10263,7 +10321,6 @@ packages:
for-each: 0.3.3
gopd: 1.0.1
has-tostringtag: 1.0.2
- dev: true
/which@2.0.2:
resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==}