stackkit-cli 0.4.4 → 0.4.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +26 -2
- package/bin/stackkit.js +1 -1
- package/dist/commands/add.js +20 -39
- package/dist/commands/doctor.d.ts +11 -0
- package/dist/commands/doctor.js +483 -0
- package/dist/commands/list.d.ts +1 -1
- package/dist/commands/list.js +59 -38
- package/dist/index.js +11 -13
- package/dist/types/index.d.ts +15 -0
- package/dist/utils/config-utils.d.ts +2 -0
- package/dist/utils/config-utils.js +88 -0
- package/dist/utils/detect.js +12 -2
- package/dist/utils/env-editor.d.ts +0 -1
- package/dist/utils/env-editor.js +50 -25
- package/dist/utils/files.d.ts +8 -0
- package/dist/utils/files.js +51 -0
- package/dist/utils/js-conversion.d.ts +1 -0
- package/dist/utils/js-conversion.js +244 -0
- package/dist/utils/module-utils.d.ts +2 -0
- package/dist/utils/module-utils.js +461 -0
- package/dist/utils/package-manager.js +15 -31
- package/modules/auth/authjs/files/api/auth/[...nextauth]/route.ts +6 -0
- package/modules/auth/authjs/files/lib/auth-client.ts +11 -0
- package/modules/auth/authjs/files/lib/auth.ts +41 -0
- package/modules/auth/authjs/files/schemas/prisma-schema.prisma +45 -0
- package/modules/auth/authjs/module.json +95 -0
- package/modules/auth/better-auth/files/lib/auth-client.ts +7 -0
- package/modules/auth/better-auth/files/lib/auth.ts +62 -0
- package/modules/auth/better-auth/files/lib/email-service.ts +34 -0
- package/modules/auth/better-auth/files/lib/email-templates.ts +89 -0
- package/modules/auth/better-auth/files/schemas/prisma-schema.prisma +1 -1
- package/modules/auth/better-auth/module.json +164 -27
- package/modules/database/mongoose/files/lib/db.ts +68 -0
- package/modules/database/{mongoose-mongodb → mongoose}/files/models/User.ts +0 -5
- package/modules/database/{mongoose-mongodb → mongoose}/module.json +9 -24
- package/modules/database/prisma/files/lib/prisma.ts +1 -3
- package/modules/database/prisma/files/prisma/schema.prisma +1 -1
- package/modules/database/prisma/files/prisma.config.ts +2 -2
- package/modules/database/prisma/module.json +5 -23
- package/package.json +1 -1
- package/templates/express/.env.example +0 -1
- package/templates/express/package.json +4 -4
- package/templates/express/src/app.ts +2 -2
- package/templates/express/src/features/health/health.controller.ts +18 -0
- package/templates/express/src/features/health/health.route.ts +9 -0
- package/templates/express/src/features/health/health.service.ts +6 -0
- package/templates/nextjs/lib/env.ts +8 -0
- package/templates/nextjs/package.json +7 -7
- package/templates/react-vite/.env.example +1 -2
- package/templates/react-vite/.prettierignore +4 -0
- package/templates/react-vite/.prettierrc +9 -0
- package/templates/react-vite/README.md +22 -0
- package/templates/react-vite/package.json +16 -16
- package/templates/react-vite/src/router.tsx +0 -12
- package/templates/react-vite/vite.config.ts +0 -6
- package/dist/commands/init.d.ts +0 -10
- package/dist/commands/init.js +0 -157
- package/dist/utils/code-inject.d.ts +0 -14
- package/dist/utils/code-inject.js +0 -70
- package/dist/utils/json-editor.d.ts +0 -8
- package/dist/utils/json-editor.js +0 -45
- package/modules/auth/clerk/files/express/auth.ts +0 -7
- package/modules/auth/clerk/files/nextjs/auth-provider.tsx +0 -5
- package/modules/auth/clerk/files/nextjs/middleware.ts +0 -9
- package/modules/auth/clerk/files/react/auth-provider.tsx +0 -15
- package/modules/auth/clerk/module.json +0 -115
- package/modules/database/mongoose-mongodb/files/lib/db.ts +0 -78
- package/templates/express/src/features/auth/auth.controller.ts +0 -48
- package/templates/express/src/features/auth/auth.route.ts +0 -10
- package/templates/express/src/features/auth/auth.service.ts +0 -21
- package/templates/react-vite/src/api/services/user.service.ts +0 -18
- package/templates/react-vite/src/pages/UserProfile.tsx +0 -40
- package/templates/react-vite/src/types/user.d.ts +0 -6
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "authjs",
|
|
3
|
+
"displayName": "Auth.js",
|
|
4
|
+
"description": "Authentication with Auth.js (NextAuth.js v5)",
|
|
5
|
+
"category": "auth",
|
|
6
|
+
"provider": "authjs",
|
|
7
|
+
"supportedFrameworks": ["nextjs"],
|
|
8
|
+
"databaseAdapters": {
|
|
9
|
+
"common": {
|
|
10
|
+
"dependencies": {
|
|
11
|
+
"@auth/prisma-adapter": "^2.7.1"
|
|
12
|
+
},
|
|
13
|
+
"devDependencies": {}
|
|
14
|
+
},
|
|
15
|
+
"prisma-postgresql": {
|
|
16
|
+
"schema": "files/schemas/prisma-schema.prisma",
|
|
17
|
+
"schemaDestination": "prisma/schema.prisma",
|
|
18
|
+
"dependencies": {}
|
|
19
|
+
},
|
|
20
|
+
"prisma-mongodb": {
|
|
21
|
+
"schema": "files/schemas/prisma-schema.prisma",
|
|
22
|
+
"schemaDestination": "prisma/schema.prisma",
|
|
23
|
+
"dependencies": {}
|
|
24
|
+
},
|
|
25
|
+
"prisma-mysql": {
|
|
26
|
+
"schema": "files/schemas/prisma-schema.prisma",
|
|
27
|
+
"schemaDestination": "prisma/schema.prisma",
|
|
28
|
+
"dependencies": {}
|
|
29
|
+
},
|
|
30
|
+
"prisma-sqlite": {
|
|
31
|
+
"schema": "files/schemas/prisma-schema.prisma",
|
|
32
|
+
"schemaDestination": "prisma/schema.prisma",
|
|
33
|
+
"dependencies": {}
|
|
34
|
+
}
|
|
35
|
+
},
|
|
36
|
+
"frameworkConfigs": {
|
|
37
|
+
"nextjs": {
|
|
38
|
+
"dependencies": {
|
|
39
|
+
"next-auth": "^5.0.0"
|
|
40
|
+
},
|
|
41
|
+
"envVars": [
|
|
42
|
+
{
|
|
43
|
+
"key": "NEXTAUTH_SECRET",
|
|
44
|
+
"value": "",
|
|
45
|
+
"required": true
|
|
46
|
+
},
|
|
47
|
+
{
|
|
48
|
+
"key": "NEXTAUTH_URL",
|
|
49
|
+
"value": "http://localhost:3000",
|
|
50
|
+
"required": true
|
|
51
|
+
},
|
|
52
|
+
{
|
|
53
|
+
"key": "GOOGLE_CLIENT_ID",
|
|
54
|
+
"value": "",
|
|
55
|
+
"required": false
|
|
56
|
+
},
|
|
57
|
+
{
|
|
58
|
+
"key": "GOOGLE_CLIENT_SECRET",
|
|
59
|
+
"value": "",
|
|
60
|
+
"required": false
|
|
61
|
+
},
|
|
62
|
+
{
|
|
63
|
+
"key": "GITHUB_ID",
|
|
64
|
+
"value": "",
|
|
65
|
+
"required": false
|
|
66
|
+
},
|
|
67
|
+
{
|
|
68
|
+
"key": "GITHUB_SECRET",
|
|
69
|
+
"value": "",
|
|
70
|
+
"required": false
|
|
71
|
+
}
|
|
72
|
+
],
|
|
73
|
+
"patches": [
|
|
74
|
+
{
|
|
75
|
+
"type": "create-file",
|
|
76
|
+
"description": "{{authDescription}}",
|
|
77
|
+
"source": "lib/{{authFile}}",
|
|
78
|
+
"destination": "{{lib}}/{{authFile}}"
|
|
79
|
+
},
|
|
80
|
+
{
|
|
81
|
+
"type": "create-file",
|
|
82
|
+
"description": "Create Auth.js API routes",
|
|
83
|
+
"source": "api/auth/[...nextauth]/route.ts",
|
|
84
|
+
"destination": "app/api/auth/[...nextauth]/route.ts"
|
|
85
|
+
},
|
|
86
|
+
{
|
|
87
|
+
"type": "create-file",
|
|
88
|
+
"description": "Create Auth.js client utilities",
|
|
89
|
+
"source": "lib/auth-client.ts",
|
|
90
|
+
"destination": "{{lib}}/auth-client.ts"
|
|
91
|
+
}
|
|
92
|
+
]
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
@@ -1,8 +1,22 @@
|
|
|
1
1
|
import { betterAuth } from "better-auth";
|
|
2
|
+
import { sendEmail } from "./email/email-service";
|
|
3
|
+
import { getVerificationEmailTemplate, getPasswordResetEmailTemplate } from "./email/email-templates";
|
|
4
|
+
{{dbImport}}
|
|
2
5
|
|
|
6
|
+
export const auth = betterAuth({
|
|
3
7
|
{{databaseAdapter}}
|
|
8
|
+
user: {
|
|
9
|
+
additionalFields: {
|
|
10
|
+
role: {
|
|
11
|
+
type: "string",
|
|
12
|
+
defaultValue: "USER",
|
|
13
|
+
required: true,
|
|
14
|
+
},
|
|
15
|
+
},
|
|
16
|
+
},
|
|
4
17
|
emailAndPassword: {
|
|
5
18
|
enabled: true,
|
|
19
|
+
requireEmailVerification: true,
|
|
6
20
|
},
|
|
7
21
|
socialProviders: {
|
|
8
22
|
google: {
|
|
@@ -10,4 +24,52 @@ import { betterAuth } from "better-auth";
|
|
|
10
24
|
clientSecret: process.env.GOOGLE_CLIENT_SECRET!,
|
|
11
25
|
},
|
|
12
26
|
},
|
|
27
|
+
emailVerification: {
|
|
28
|
+
sendVerificationEmail: async ({ user, url }) => {
|
|
29
|
+
const { html, text } = getVerificationEmailTemplate(user, url);
|
|
30
|
+
await sendEmail({
|
|
31
|
+
to: user.email,
|
|
32
|
+
subject: "Verify Your Email Address",
|
|
33
|
+
text,
|
|
34
|
+
html,
|
|
35
|
+
});
|
|
36
|
+
},
|
|
37
|
+
sendOnSignIn: true,
|
|
38
|
+
},
|
|
39
|
+
password: {
|
|
40
|
+
reset: {
|
|
41
|
+
sendResetEmail: async ({ user, url }) => {
|
|
42
|
+
const { html, text } = getPasswordResetEmailTemplate(user, url);
|
|
43
|
+
await sendEmail({
|
|
44
|
+
to: user.email,
|
|
45
|
+
subject: "Reset Your Password",
|
|
46
|
+
text,
|
|
47
|
+
html,
|
|
48
|
+
});
|
|
49
|
+
},
|
|
50
|
+
},
|
|
51
|
+
},
|
|
52
|
+
// Additional production-ready options
|
|
53
|
+
rateLimit: {
|
|
54
|
+
window: 10, // 10 seconds
|
|
55
|
+
max: 100, // max requests per window
|
|
56
|
+
},
|
|
57
|
+
account: {
|
|
58
|
+
accountLinking: {
|
|
59
|
+
enabled: true,
|
|
60
|
+
trustedProviders: ["google"],
|
|
61
|
+
},
|
|
62
|
+
},
|
|
63
|
+
session: {
|
|
64
|
+
cookieCache: {
|
|
65
|
+
enabled: true,
|
|
66
|
+
},
|
|
67
|
+
expiresIn: 60 * 60 * 24 * 7, // 7 days
|
|
68
|
+
updateAge: 60 * 60 * 24, // 1 day
|
|
69
|
+
},
|
|
70
|
+
user: {
|
|
71
|
+
changeEmail: {
|
|
72
|
+
enabled: true,
|
|
73
|
+
},
|
|
74
|
+
},
|
|
13
75
|
});
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import nodemailer from "nodemailer";
|
|
2
|
+
|
|
3
|
+
// Create email transporter
|
|
4
|
+
const transporter = nodemailer.createTransporter({
|
|
5
|
+
host: process.env.EMAIL_HOST,
|
|
6
|
+
port: parseInt(process.env.EMAIL_PORT || "587"),
|
|
7
|
+
secure: process.env.EMAIL_PORT === "465",
|
|
8
|
+
auth: {
|
|
9
|
+
user: process.env.EMAIL_USER,
|
|
10
|
+
pass: process.env.EMAIL_PASS,
|
|
11
|
+
},
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
// Send email function
|
|
15
|
+
export const sendEmail = async ({ to, subject, text, html }: {
|
|
16
|
+
to: string;
|
|
17
|
+
subject: string;
|
|
18
|
+
text?: string;
|
|
19
|
+
html?: string;
|
|
20
|
+
}) => {
|
|
21
|
+
try {
|
|
22
|
+
await transporter.sendMail({
|
|
23
|
+
from: process.env.EMAIL_FROM,
|
|
24
|
+
to,
|
|
25
|
+
subject,
|
|
26
|
+
text,
|
|
27
|
+
html,
|
|
28
|
+
});
|
|
29
|
+
} catch (error) {
|
|
30
|
+
// eslint-disable-next-line no-console
|
|
31
|
+
console.error("Email sending failed:", error);
|
|
32
|
+
throw error;
|
|
33
|
+
}
|
|
34
|
+
};
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
export const getVerificationEmailTemplate = (user: { name?: string; email: string }, url: string) => {
|
|
2
|
+
const html = `
|
|
3
|
+
<!DOCTYPE html>
|
|
4
|
+
<html lang="en">
|
|
5
|
+
<head>
|
|
6
|
+
<meta charset="UTF-8">
|
|
7
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
8
|
+
<title>Verify Your Email</title>
|
|
9
|
+
<style>
|
|
10
|
+
body { font-family: Arial, sans-serif; line-height: 1.6; color: #000; max-width: 600px; margin: 0 auto; padding: 20px; background-color: #fff; }
|
|
11
|
+
.header { padding: 20px; text-align: center; border-bottom: 1px solid #000; }
|
|
12
|
+
.content { padding: 20px; }
|
|
13
|
+
.button { display: inline-block; background-color: #fff; color: #000; padding: 10px 20px; text-decoration: none; border: 1px solid #000; border-radius: 5px; margin: 20px 0; }
|
|
14
|
+
.footer { font-size: 12px; color: #000; text-align: center; margin-top: 20px; border-top: 1px solid #000; padding-top: 20px; }
|
|
15
|
+
</style>
|
|
16
|
+
</head>
|
|
17
|
+
<body>
|
|
18
|
+
<div class="header">
|
|
19
|
+
<h1>Verify Your Email Address</h1>
|
|
20
|
+
</div>
|
|
21
|
+
<div class="content">
|
|
22
|
+
<p>Hi ${user.name || user.email},</p>
|
|
23
|
+
<p>Thank you for signing up. Please verify your email address to complete your registration.</p>
|
|
24
|
+
<a href="${url}" class="button">Verify Email</a>
|
|
25
|
+
<p>If the button doesn't work, copy and paste this link: <a href="${url}">${url}</a></p>
|
|
26
|
+
<p>This link expires in 24 hours.</p>
|
|
27
|
+
</div>
|
|
28
|
+
<div class="footer">
|
|
29
|
+
<p>If you didn't create an account, ignore this email.</p>
|
|
30
|
+
</div>
|
|
31
|
+
</body>
|
|
32
|
+
</html>
|
|
33
|
+
`;
|
|
34
|
+
|
|
35
|
+
const text = `Hi ${user.name || user.email},
|
|
36
|
+
|
|
37
|
+
Thank you for signing up. Please verify your email address by clicking this link: ${url}
|
|
38
|
+
|
|
39
|
+
This link expires in 24 hours.
|
|
40
|
+
|
|
41
|
+
If you didn't create an account, ignore this email.`;
|
|
42
|
+
|
|
43
|
+
return { html, text };
|
|
44
|
+
};
|
|
45
|
+
|
|
46
|
+
export const getPasswordResetEmailTemplate = (user: { name?: string; email: string }, url: string) => {
|
|
47
|
+
const html = `
|
|
48
|
+
<!DOCTYPE html>
|
|
49
|
+
<html lang="en">
|
|
50
|
+
<head>
|
|
51
|
+
<meta charset="UTF-8">
|
|
52
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
53
|
+
<title>Reset Your Password</title>
|
|
54
|
+
<style>
|
|
55
|
+
body { font-family: Arial, sans-serif; line-height: 1.6; color: #000; max-width: 600px; margin: 0 auto; padding: 20px; background-color: #fff; }
|
|
56
|
+
.header { padding: 20px; text-align: center; border-bottom: 1px solid #000; }
|
|
57
|
+
.content { padding: 20px; }
|
|
58
|
+
.button { display: inline-block; background-color: #fff; color: #000; padding: 10px 20px; text-decoration: none; border: 1px solid #000; border-radius: 5px; margin: 20px 0; }
|
|
59
|
+
.footer { font-size: 12px; color: #000; text-align: center; margin-top: 20px; border-top: 1px solid #000; padding-top: 20px; }
|
|
60
|
+
</style>
|
|
61
|
+
</head>
|
|
62
|
+
<body>
|
|
63
|
+
<div class="header">
|
|
64
|
+
<h1>Reset Your Password</h1>
|
|
65
|
+
</div>
|
|
66
|
+
<div class="content">
|
|
67
|
+
<p>Hi ${user.name || user.email},</p>
|
|
68
|
+
<p>You requested a password reset. Click the link below to reset your password.</p>
|
|
69
|
+
<a href="${url}" class="button">Reset Password</a>
|
|
70
|
+
<p>If the button doesn't work, copy and paste this link: <a href="${url}">${url}</a></p>
|
|
71
|
+
<p>This link expires in 1 hour.</p>
|
|
72
|
+
</div>
|
|
73
|
+
<div class="footer">
|
|
74
|
+
<p>If you didn't request this, ignore this email.</p>
|
|
75
|
+
</div>
|
|
76
|
+
</body>
|
|
77
|
+
</html>
|
|
78
|
+
`;
|
|
79
|
+
|
|
80
|
+
const text = `Hi ${user.name || user.email},
|
|
81
|
+
|
|
82
|
+
You requested a password reset. Click this link to reset your password: ${url}
|
|
83
|
+
|
|
84
|
+
This link expires in 1 hour.
|
|
85
|
+
|
|
86
|
+
If you didn't request this, ignore this email.`;
|
|
87
|
+
|
|
88
|
+
return { html, text };
|
|
89
|
+
};
|
|
@@ -6,49 +6,186 @@
|
|
|
6
6
|
"provider": "better-auth",
|
|
7
7
|
"supportedFrameworks": ["nextjs", "express", "react-vite"],
|
|
8
8
|
"databaseAdapters": {
|
|
9
|
+
"common": {
|
|
10
|
+
"dependencies": {},
|
|
11
|
+
"devDependencies": {}
|
|
12
|
+
},
|
|
9
13
|
"prisma-postgresql": {
|
|
10
|
-
"adapterCode": "import { prisma } from \"{{dbImport}}\";\nimport { prismaAdapter } from \"@better-auth/prisma\";\n\nexport const auth = betterAuth({\n database: prismaAdapter(prisma, {\n provider: \"postgresql\",\n }),",
|
|
11
14
|
"schema": "files/schemas/prisma-schema.prisma",
|
|
12
15
|
"schemaDestination": "prisma/schema.prisma",
|
|
13
16
|
"dependencies": {
|
|
14
|
-
"
|
|
17
|
+
"@prisma/adapter-pg": "^5.0.0",
|
|
18
|
+
"pg": "^8.0.0"
|
|
15
19
|
}
|
|
16
20
|
},
|
|
17
21
|
"prisma-mongodb": {
|
|
18
|
-
"
|
|
22
|
+
"schema": "files/schemas/prisma-schema.prisma",
|
|
23
|
+
"schemaDestination": "prisma/schema.prisma",
|
|
24
|
+
"dependencies": {}
|
|
25
|
+
},
|
|
26
|
+
"prisma-mysql": {
|
|
19
27
|
"schema": "files/schemas/prisma-schema.prisma",
|
|
20
28
|
"schemaDestination": "prisma/schema.prisma",
|
|
21
29
|
"dependencies": {
|
|
22
|
-
"
|
|
30
|
+
"@prisma/adapter-mariadb": "^5.0.0",
|
|
31
|
+
"mysql2": "^3.0.0"
|
|
23
32
|
}
|
|
24
33
|
},
|
|
25
|
-
"
|
|
26
|
-
"
|
|
34
|
+
"prisma-sqlite": {
|
|
35
|
+
"schema": "files/schemas/prisma-schema.prisma",
|
|
36
|
+
"schemaDestination": "prisma/schema.prisma",
|
|
27
37
|
"dependencies": {
|
|
28
|
-
"better-
|
|
38
|
+
"@prisma/adapter-better-sqlite3": "^5.0.0",
|
|
39
|
+
"better-sqlite3": "^9.0.0"
|
|
29
40
|
}
|
|
30
41
|
}
|
|
31
42
|
},
|
|
32
|
-
"
|
|
33
|
-
{
|
|
34
|
-
"
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
43
|
+
"frameworkConfigs": {
|
|
44
|
+
"shared": {
|
|
45
|
+
"dependencies": {
|
|
46
|
+
"better-auth": "^1.4.12",
|
|
47
|
+
"nodemailer": "^7.0.12"
|
|
48
|
+
},
|
|
49
|
+
"devDependencies": {
|
|
50
|
+
"@types/nodemailer": "^6.4.17"
|
|
51
|
+
},
|
|
52
|
+
"envVars": [
|
|
53
|
+
{
|
|
54
|
+
"key": "BETTER_AUTH_SECRET",
|
|
55
|
+
"value": "",
|
|
56
|
+
"required": true
|
|
57
|
+
},
|
|
58
|
+
{
|
|
59
|
+
"key": "BETTER_AUTH_URL",
|
|
60
|
+
"value": "http://localhost:3000",
|
|
61
|
+
"required": true
|
|
62
|
+
},
|
|
63
|
+
{
|
|
64
|
+
"key": "EMAIL_HOST",
|
|
65
|
+
"value": "smtp.gmail.com",
|
|
66
|
+
"required": true
|
|
67
|
+
},
|
|
68
|
+
{
|
|
69
|
+
"key": "EMAIL_PORT",
|
|
70
|
+
"value": "587",
|
|
71
|
+
"required": true
|
|
72
|
+
},
|
|
73
|
+
{
|
|
74
|
+
"key": "EMAIL_USER",
|
|
75
|
+
"value": "",
|
|
76
|
+
"required": true
|
|
77
|
+
},
|
|
78
|
+
{
|
|
79
|
+
"key": "EMAIL_PASS",
|
|
80
|
+
"value": "",
|
|
81
|
+
"required": true
|
|
82
|
+
},
|
|
83
|
+
{
|
|
84
|
+
"key": "EMAIL_FROM",
|
|
85
|
+
"value": "noreply@yourapp.com",
|
|
86
|
+
"required": true
|
|
87
|
+
}
|
|
88
|
+
]
|
|
38
89
|
},
|
|
39
|
-
{
|
|
40
|
-
"
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
90
|
+
"nextjs": {
|
|
91
|
+
"patches": [
|
|
92
|
+
{
|
|
93
|
+
"type": "create-file",
|
|
94
|
+
"description": "{{authDescription}}",
|
|
95
|
+
"source": "lib/{{authFile}}",
|
|
96
|
+
"destination": "{{lib}}/{{authFile}}"
|
|
97
|
+
},
|
|
98
|
+
{
|
|
99
|
+
"type": "create-file",
|
|
100
|
+
"description": "Create Better Auth API routes",
|
|
101
|
+
"source": "api/auth/[...all]/route.ts",
|
|
102
|
+
"destination": "app/api/auth/[...all]/route.ts"
|
|
103
|
+
},
|
|
104
|
+
{
|
|
105
|
+
"type": "create-file",
|
|
106
|
+
"description": "Create Better Auth client for React",
|
|
107
|
+
"source": "lib/auth-client.ts",
|
|
108
|
+
"destination": "{{lib}}/auth-client.ts"
|
|
109
|
+
},
|
|
110
|
+
{
|
|
111
|
+
"type": "create-file",
|
|
112
|
+
"description": "Create email service for sending emails",
|
|
113
|
+
"source": "lib/email-service.ts",
|
|
114
|
+
"destination": "{{lib}}/email/email-service.ts"
|
|
115
|
+
},
|
|
116
|
+
{
|
|
117
|
+
"type": "create-file",
|
|
118
|
+
"description": "Create email templates",
|
|
119
|
+
"source": "lib/email-templates.ts",
|
|
120
|
+
"destination": "{{lib}}/email/email-templates.ts"
|
|
121
|
+
}
|
|
122
|
+
]
|
|
123
|
+
},
|
|
124
|
+
"express": {
|
|
125
|
+
"patches": [
|
|
126
|
+
{
|
|
127
|
+
"type": "create-file",
|
|
128
|
+
"description": "{{authDescription}}",
|
|
129
|
+
"source": "lib/{{authFile}}",
|
|
130
|
+
"destination": "{{lib}}/{{authFile}}"
|
|
131
|
+
},
|
|
132
|
+
{
|
|
133
|
+
"type": "patch-file",
|
|
134
|
+
"description": "Add Better Auth routes to Express app",
|
|
135
|
+
"file": "src/app.ts",
|
|
136
|
+
"operations": [
|
|
137
|
+
{
|
|
138
|
+
"type": "add-import",
|
|
139
|
+
"imports": [
|
|
140
|
+
"import { auth } from \"@/lib/auth\";",
|
|
141
|
+
"import { toNodeHandler } from \"better-auth/node\";"
|
|
142
|
+
]
|
|
143
|
+
},
|
|
144
|
+
{
|
|
145
|
+
"type": "add-code",
|
|
146
|
+
"after": "// routes",
|
|
147
|
+
"code": "\napp.all(\"/api/auth/*splat\", toNodeHandler(auth));"
|
|
148
|
+
}
|
|
149
|
+
]
|
|
150
|
+
},
|
|
151
|
+
{
|
|
152
|
+
"type": "create-file",
|
|
153
|
+
"description": "Create email service for sending emails",
|
|
154
|
+
"source": "lib/email-service.ts",
|
|
155
|
+
"destination": "{{lib}}/email/email-service.ts"
|
|
156
|
+
},
|
|
157
|
+
{
|
|
158
|
+
"type": "create-file",
|
|
159
|
+
"description": "Create email templates",
|
|
160
|
+
"source": "lib/email-templates.ts",
|
|
161
|
+
"destination": "{{lib}}/email/email-templates.ts"
|
|
162
|
+
}
|
|
163
|
+
]
|
|
164
|
+
},
|
|
165
|
+
"react-vite": {
|
|
166
|
+
"dependencies": {
|
|
167
|
+
"better-auth": "^1.4.12"
|
|
168
|
+
},
|
|
169
|
+
"envVars": [
|
|
170
|
+
{
|
|
171
|
+
"key": "BETTER_AUTH_SECRET",
|
|
172
|
+
"value": "",
|
|
173
|
+
"required": true
|
|
174
|
+
},
|
|
175
|
+
{
|
|
176
|
+
"key": "BETTER_AUTH_URL",
|
|
177
|
+
"value": "http://localhost:3000",
|
|
178
|
+
"required": true
|
|
179
|
+
}
|
|
180
|
+
],
|
|
181
|
+
"patches": [
|
|
182
|
+
{
|
|
183
|
+
"type": "create-file",
|
|
184
|
+
"description": "Create Better Auth client for React",
|
|
185
|
+
"source": "lib/auth-client.ts",
|
|
186
|
+
"destination": "{{lib}}/auth-client.ts"
|
|
187
|
+
}
|
|
188
|
+
]
|
|
52
189
|
}
|
|
53
|
-
|
|
190
|
+
}
|
|
54
191
|
}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import mongoose from "mongoose";
|
|
2
|
+
import { ServerApiVersion } from "mongodb";
|
|
3
|
+
|
|
4
|
+
type MongooseCache = {
|
|
5
|
+
conn: typeof mongoose | null;
|
|
6
|
+
promise: Promise<typeof mongoose> | null;
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
const globalWithMongoose = globalThis as unknown as {
|
|
10
|
+
mongoose: MongooseCache;
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
// Initialize the cache if not already present
|
|
14
|
+
const cached = globalWithMongoose.mongoose || {
|
|
15
|
+
conn: null,
|
|
16
|
+
promise: null,
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
if (!globalWithMongoose.mongoose) {
|
|
20
|
+
globalWithMongoose.mongoose = cached;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
async function dbConnect(): Promise<typeof mongoose> {
|
|
24
|
+
if (cached.conn) {
|
|
25
|
+
return cached.conn;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
const uri = process.env.DATABASE_URL as string;
|
|
29
|
+
|
|
30
|
+
if (!cached.promise) {
|
|
31
|
+
const opts = {
|
|
32
|
+
bufferCommands: false,
|
|
33
|
+
connectTimeoutMS: 10000,
|
|
34
|
+
serverSelectionTimeoutMS: 10000,
|
|
35
|
+
serverApi: {
|
|
36
|
+
version: ServerApiVersion.v1,
|
|
37
|
+
strict: true,
|
|
38
|
+
deprecationErrors: true,
|
|
39
|
+
},
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
cached.promise = mongoose
|
|
43
|
+
.connect(uri, opts)
|
|
44
|
+
.then(async (mongooseInstance: typeof mongoose) => {
|
|
45
|
+
console.info("MongoDB connected successfully");
|
|
46
|
+
if (mongoose.connection.db) {
|
|
47
|
+
await mongoose.connection.db.admin().command({ ping: 1 });
|
|
48
|
+
console.info("Pinged your deployment. You successfully connected to MongoDB!");
|
|
49
|
+
}
|
|
50
|
+
return mongooseInstance;
|
|
51
|
+
})
|
|
52
|
+
.catch((error: Error) => {
|
|
53
|
+
console.error("MongoDB connection failed", { error });
|
|
54
|
+
throw error;
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
try {
|
|
59
|
+
cached.conn = await cached.promise;
|
|
60
|
+
} catch (error) {
|
|
61
|
+
cached.promise = null;
|
|
62
|
+
throw error;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
return cached.conn;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
export default dbConnect;
|
|
@@ -3,7 +3,6 @@ import mongoose, { Document, Schema } from "mongoose";
|
|
|
3
3
|
export interface IUser extends Document {
|
|
4
4
|
name: string;
|
|
5
5
|
email: string;
|
|
6
|
-
password: string;
|
|
7
6
|
createdAt: Date;
|
|
8
7
|
updatedAt: Date;
|
|
9
8
|
}
|
|
@@ -22,10 +21,6 @@ const UserSchema: Schema = new Schema(
|
|
|
22
21
|
lowercase: true,
|
|
23
22
|
trim: true,
|
|
24
23
|
},
|
|
25
|
-
password: {
|
|
26
|
-
type: String,
|
|
27
|
-
required: true,
|
|
28
|
-
},
|
|
29
24
|
},
|
|
30
25
|
{
|
|
31
26
|
timestamps: true,
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
|
-
"name": "mongoose
|
|
3
|
-
"displayName": "Mongoose
|
|
2
|
+
"name": "mongoose",
|
|
3
|
+
"displayName": "Mongoose (MongoDB)",
|
|
4
4
|
"description": "Mongoose ODM for MongoDB database",
|
|
5
5
|
"category": "database",
|
|
6
6
|
"provider": "mongoose",
|
|
@@ -8,7 +8,7 @@
|
|
|
8
8
|
"supportedFrameworks": ["nextjs", "express"],
|
|
9
9
|
"dependencies": {
|
|
10
10
|
"mongoose": "^8.8.4",
|
|
11
|
-
"
|
|
11
|
+
"mongodb": "^6.10.0"
|
|
12
12
|
},
|
|
13
13
|
"devDependencies": {
|
|
14
14
|
"@types/mongoose": "^5.11.97"
|
|
@@ -17,8 +17,7 @@
|
|
|
17
17
|
"common": [
|
|
18
18
|
{
|
|
19
19
|
"key": "DATABASE_URL",
|
|
20
|
-
"value": "mongodb://
|
|
21
|
-
"description": "MongoDB connection string",
|
|
20
|
+
"value": "mongodb+srv://username:password@cluster.mongodb.net/mydb",
|
|
22
21
|
"required": true
|
|
23
22
|
}
|
|
24
23
|
]
|
|
@@ -30,35 +29,21 @@
|
|
|
30
29
|
"compilerOptions": {
|
|
31
30
|
"baseUrl": ".",
|
|
32
31
|
"paths": {
|
|
33
|
-
"@/*": ["
|
|
34
|
-
"@/models/*": ["./src/models/*"]
|
|
32
|
+
"@/*": ["./*"]
|
|
35
33
|
}
|
|
36
34
|
},
|
|
37
|
-
"include": ["src/**/*", "
|
|
35
|
+
"include": ["src/**/*", "lib/**/*"],
|
|
38
36
|
"exclude": ["node_modules", "dist"]
|
|
39
37
|
}
|
|
40
38
|
}
|
|
41
|
-
},
|
|
42
|
-
"nextjs": {
|
|
43
|
-
"tsconfig.json": {
|
|
44
|
-
"merge": {
|
|
45
|
-
"compilerOptions": {
|
|
46
|
-
"baseUrl": ".",
|
|
47
|
-
"paths": {
|
|
48
|
-
"@/models/*": ["./models/*"]
|
|
49
|
-
}
|
|
50
|
-
},
|
|
51
|
-
"exclude": ["node_modules", ".next", "out"]
|
|
52
|
-
}
|
|
53
|
-
}
|
|
54
39
|
}
|
|
55
40
|
},
|
|
56
41
|
"patches": [
|
|
57
42
|
{
|
|
58
43
|
"type": "create-file",
|
|
59
|
-
"description": "
|
|
60
|
-
"source": "lib/
|
|
61
|
-
"destination": "{{lib}}/
|
|
44
|
+
"description": "{{dbDescription}}",
|
|
45
|
+
"source": "lib/{{dbFile}}",
|
|
46
|
+
"destination": "{{lib}}/{{dbFile}}"
|
|
62
47
|
},
|
|
63
48
|
{
|
|
64
49
|
"type": "create-file",
|