modelence 0.8.0 → 0.9.0-dev.0
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/dist/app/authConfig.d.ts +68 -0
- package/dist/app/authConfig.d.ts.map +1 -0
- package/dist/app/authConfig.js +8 -0
- package/dist/app/authConfig.js.map +1 -0
- package/dist/app/backendApi.d.ts +29 -0
- package/dist/app/backendApi.d.ts.map +1 -0
- package/dist/app/backendApi.js +69 -0
- package/dist/app/backendApi.js.map +1 -0
- package/dist/app/email.d.ts +3 -0
- package/dist/app/email.d.ts.map +1 -0
- package/dist/app/email.js +8 -0
- package/dist/app/email.js.map +1 -0
- package/dist/app/emailConfig.d.ts +27 -0
- package/dist/app/emailConfig.d.ts.map +1 -0
- package/dist/app/emailConfig.js +8 -0
- package/dist/app/emailConfig.js.map +1 -0
- package/dist/app/index.d.ts +20 -0
- package/dist/app/index.d.ts.map +1 -0
- package/dist/app/index.js +258 -0
- package/dist/app/index.js.map +1 -0
- package/dist/app/loggerProcess.d.ts +5 -0
- package/dist/app/loggerProcess.d.ts.map +1 -0
- package/dist/app/loggerProcess.js +90 -0
- package/dist/app/loggerProcess.js.map +1 -0
- package/dist/app/metrics.d.ts +6 -0
- package/dist/app/metrics.d.ts.map +1 -0
- package/dist/app/metrics.js +92 -0
- package/dist/app/metrics.js.map +1 -0
- package/dist/app/module.d.ts +77 -0
- package/dist/app/module.d.ts.map +1 -0
- package/dist/app/module.js +46 -0
- package/dist/app/module.js.map +1 -0
- package/dist/app/server.d.ts +37 -0
- package/dist/app/server.d.ts.map +1 -0
- package/dist/app/server.js +198 -0
- package/dist/app/server.js.map +1 -0
- package/dist/app/state.d.ts +19 -0
- package/dist/app/state.d.ts.map +1 -0
- package/dist/app/state.js +27 -0
- package/dist/app/state.js.map +1 -0
- package/dist/app/websocketConfig.d.ts +7 -0
- package/dist/app/websocketConfig.d.ts.map +1 -0
- package/dist/app/websocketConfig.js +8 -0
- package/dist/app/websocketConfig.js.map +1 -0
- package/dist/auth/client/index.d.ts +69 -0
- package/dist/auth/client/index.d.ts.map +1 -0
- package/dist/auth/client/index.js +79 -0
- package/dist/auth/client/index.js.map +1 -0
- package/dist/auth/db.d.ts +176 -0
- package/dist/auth/db.d.ts.map +1 -0
- package/dist/auth/db.js +117 -0
- package/dist/auth/db.js.map +1 -0
- package/dist/auth/deleteUser.d.ts +5 -0
- package/dist/auth/deleteUser.d.ts.map +1 -0
- package/dist/auth/deleteUser.js +34 -0
- package/dist/auth/deleteUser.js.map +1 -0
- package/dist/auth/disposableEmails.d.ts +6 -0
- package/dist/auth/disposableEmails.d.ts.map +1 -0
- package/dist/auth/disposableEmails.js +47 -0
- package/dist/auth/disposableEmails.js.map +1 -0
- package/dist/auth/index.d.ts +7 -0
- package/dist/auth/index.d.ts.map +1 -0
- package/dist/auth/index.js +33 -0
- package/dist/auth/index.js.map +1 -0
- package/dist/auth/login.d.ts +9 -0
- package/dist/auth/login.d.ts.map +1 -0
- package/dist/auth/login.js +100 -0
- package/dist/auth/login.js.map +1 -0
- package/dist/auth/password.d.ts +14 -0
- package/dist/auth/password.d.ts.map +1 -0
- package/dist/auth/password.js +26 -0
- package/dist/auth/password.js.map +1 -0
- package/dist/auth/profile.d.ts +10 -0
- package/dist/auth/profile.d.ts.map +1 -0
- package/dist/auth/profile.js +13 -0
- package/dist/auth/profile.js.map +1 -0
- package/dist/auth/providers/github.d.ts +3 -0
- package/dist/auth/providers/github.d.ts.map +1 -0
- package/dist/auth/providers/github.js +122 -0
- package/dist/auth/providers/github.js.map +1 -0
- package/dist/auth/providers/google.d.ts +3 -0
- package/dist/auth/providers/google.d.ts.map +1 -0
- package/dist/auth/providers/google.js +108 -0
- package/dist/auth/providers/google.js.map +1 -0
- package/dist/auth/providers/oauth-common.d.ts +13 -0
- package/dist/auth/providers/oauth-common.d.ts.map +1 -0
- package/dist/auth/providers/oauth-common.js +109 -0
- package/dist/auth/providers/oauth-common.js.map +1 -0
- package/dist/auth/resetPassword.d.ts +10 -0
- package/dist/auth/resetPassword.d.ts.map +1 -0
- package/dist/auth/resetPassword.js +108 -0
- package/dist/auth/resetPassword.js.map +1 -0
- package/dist/auth/role.d.ts +8 -0
- package/dist/auth/role.d.ts.map +1 -0
- package/dist/auth/role.js +37 -0
- package/dist/auth/role.js.map +1 -0
- package/dist/auth/session.d.ts +24 -0
- package/dist/auth/session.d.ts.map +1 -0
- package/dist/auth/session.js +84 -0
- package/dist/auth/session.js.map +1 -0
- package/dist/auth/signup.d.ts +3 -0
- package/dist/auth/signup.d.ts.map +1 -0
- package/dist/auth/signup.js +89 -0
- package/dist/auth/signup.js.map +1 -0
- package/dist/auth/templates/emailVerficationTemplate.d.ts +6 -0
- package/dist/auth/templates/emailVerficationTemplate.d.ts.map +1 -0
- package/dist/auth/templates/emailVerficationTemplate.js +9 -0
- package/dist/auth/templates/emailVerficationTemplate.js.map +1 -0
- package/dist/auth/types.d.ts +22 -0
- package/dist/auth/types.d.ts.map +1 -0
- package/dist/auth/types.js +2 -0
- package/dist/auth/types.js.map +1 -0
- package/dist/auth/user.d.ts +5 -0
- package/dist/auth/user.d.ts.map +1 -0
- package/dist/auth/user.js +152 -0
- package/dist/auth/user.js.map +1 -0
- package/dist/auth/validators.d.ts +3 -0
- package/dist/auth/validators.d.ts.map +1 -0
- package/dist/auth/validators.js +8 -0
- package/dist/auth/validators.js.map +1 -0
- package/dist/auth/verification.d.ts +8 -0
- package/dist/auth/verification.d.ts.map +1 -0
- package/dist/auth/verification.js +94 -0
- package/dist/auth/verification.js.map +1 -0
- package/dist/bin/auth.d.ts +4 -0
- package/dist/bin/auth.d.ts.map +1 -0
- package/dist/bin/auth.js +48 -0
- package/dist/bin/auth.js.map +1 -0
- package/dist/bin/build.d.ts +2 -0
- package/dist/bin/build.d.ts.map +1 -0
- package/dist/bin/build.js +78 -0
- package/dist/bin/build.js.map +1 -0
- package/dist/bin/config.d.ts +11 -0
- package/dist/bin/config.d.ts.map +1 -0
- package/dist/bin/config.js +91 -0
- package/dist/bin/config.js.map +1 -0
- package/dist/bin/deploy.d.ts +6 -0
- package/dist/bin/deploy.d.ts.map +1 -0
- package/dist/bin/deploy.js +129 -0
- package/dist/bin/deploy.js.map +1 -0
- package/dist/bin/dev.d.ts +2 -0
- package/dist/bin/dev.d.ts.map +1 -0
- package/dist/bin/dev.js +14 -0
- package/dist/bin/dev.js.map +1 -0
- package/dist/bin/modelence.d.ts +3 -0
- package/dist/bin/modelence.d.ts.map +1 -0
- package/dist/bin/modelence.js +57 -3
- package/dist/bin/modelence.js.map +1 -1
- package/dist/bin/setup.d.ts +5 -0
- package/dist/bin/setup.d.ts.map +1 -0
- package/dist/bin/setup.js +91 -0
- package/dist/bin/setup.js.map +1 -0
- package/dist/bin/start.d.ts +2 -0
- package/dist/bin/start.d.ts.map +1 -0
- package/dist/bin/start.js +15 -0
- package/dist/bin/start.js.map +1 -0
- package/dist/client/AppProvider.d.ts +8 -0
- package/dist/client/AppProvider.d.ts.map +1 -0
- package/dist/client/AppProvider.js +31 -0
- package/dist/client/AppProvider.js.map +1 -0
- package/dist/client/errorHandler.d.ts +4 -0
- package/dist/client/errorHandler.d.ts.map +1 -0
- package/dist/client/errorHandler.js +10 -0
- package/dist/client/errorHandler.js.map +1 -0
- package/dist/client/localStorage.d.ts +3 -0
- package/dist/client/localStorage.d.ts.map +1 -0
- package/dist/client/localStorage.js +14 -0
- package/dist/client/localStorage.js.map +1 -0
- package/dist/client/method.d.ts +6 -0
- package/dist/client/method.d.ts.map +1 -0
- package/dist/client/method.js +52 -0
- package/dist/client/method.js.map +1 -0
- package/dist/client/renderApp.d.ts +9 -0
- package/dist/client/renderApp.d.ts.map +1 -0
- package/dist/client/renderApp.js +28 -0
- package/dist/client/renderApp.js.map +1 -0
- package/dist/client/session.d.ts +34 -0
- package/dist/client/session.d.ts.map +1 -0
- package/dist/client/session.js +83 -0
- package/dist/client/session.js.map +1 -0
- package/dist/client.d.ts +10 -133
- package/dist/client.d.ts.map +1 -0
- package/dist/client.js +14 -1
- package/dist/client.js.map +1 -1
- package/dist/config/client.d.ts +10 -0
- package/dist/config/client.d.ts.map +1 -0
- package/dist/config/client.js +14 -0
- package/dist/config/client.js.map +1 -0
- package/dist/config/server.d.ts +48 -0
- package/dist/config/server.d.ts.map +1 -0
- package/dist/config/server.js +85 -0
- package/dist/config/server.js.map +1 -0
- package/dist/config/sync.d.ts +2 -0
- package/dist/config/sync.d.ts.map +1 -0
- package/dist/config/sync.js +32 -0
- package/dist/config/sync.js.map +1 -0
- package/dist/config/types.d.ts +19 -0
- package/dist/config/types.d.ts.map +1 -0
- package/dist/config/types.js +2 -0
- package/dist/config/types.js.map +1 -0
- package/dist/cron/jobs.d.ts +13 -0
- package/dist/cron/jobs.d.ts.map +1 -0
- package/dist/cron/jobs.js +155 -0
- package/dist/cron/jobs.js.map +1 -0
- package/dist/cron/types.d.ts +29 -0
- package/dist/cron/types.d.ts.map +1 -0
- package/dist/cron/types.js +2 -0
- package/dist/cron/types.js.map +1 -0
- package/dist/data/schemaSerializer.d.ts +37 -0
- package/dist/data/schemaSerializer.d.ts.map +1 -0
- package/dist/data/schemaSerializer.js +104 -0
- package/dist/data/schemaSerializer.js.map +1 -0
- package/dist/{types-WgRbQ-tj.d.ts → data/store.d.ts} +13 -231
- package/dist/data/store.d.ts.map +1 -0
- package/dist/data/store.js +501 -0
- package/dist/data/store.js.map +1 -0
- package/dist/data/types.d.ts +47 -0
- package/dist/data/types.d.ts.map +1 -0
- package/dist/data/types.js +35 -0
- package/dist/data/types.js.map +1 -0
- package/dist/db/client.d.ts +5 -0
- package/dist/db/client.d.ts.map +1 -0
- package/dist/db/client.js +41 -0
- package/dist/db/client.js.map +1 -0
- package/dist/error.d.ts +16 -0
- package/dist/error.d.ts.map +1 -0
- package/dist/error.js +24 -0
- package/dist/error.js.map +1 -0
- package/dist/index.d.ts +6 -31
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +2 -1
- package/dist/index.js.map +1 -1
- package/dist/lock/db.d.ts +19 -0
- package/dist/lock/db.d.ts.map +1 -0
- package/dist/lock/db.js +28 -0
- package/dist/lock/db.js.map +1 -0
- package/dist/lock/helpers.d.ts +28 -0
- package/dist/lock/helpers.d.ts.map +1 -0
- package/dist/lock/helpers.js +107 -0
- package/dist/lock/helpers.js.map +1 -0
- package/dist/lock/index.d.ts +4 -0
- package/dist/lock/index.d.ts.map +1 -0
- package/dist/lock/index.js +4 -0
- package/dist/lock/index.js.map +1 -0
- package/dist/lock/module.d.ts +7 -0
- package/dist/lock/module.d.ts.map +1 -0
- package/dist/lock/module.js +9 -0
- package/dist/lock/module.js.map +1 -0
- package/dist/methods/index.d.ts +7 -0
- package/dist/methods/index.d.ts.map +1 -0
- package/dist/methods/index.js +65 -0
- package/dist/methods/index.js.map +1 -0
- package/dist/methods/serialize.d.ts +15 -0
- package/dist/methods/serialize.d.ts.map +1 -0
- package/dist/methods/serialize.js +56 -0
- package/dist/methods/serialize.js.map +1 -0
- package/dist/methods/types.d.ts +37 -0
- package/dist/methods/types.d.ts.map +1 -0
- package/dist/methods/types.js +2 -0
- package/dist/methods/types.js.map +1 -0
- package/dist/migration/db.d.ts +17 -0
- package/dist/migration/db.d.ts.map +1 -0
- package/dist/migration/db.js +13 -0
- package/dist/migration/db.js.map +1 -0
- package/dist/migration/index.d.ts +11 -0
- package/dist/migration/index.d.ts.map +1 -0
- package/dist/migration/index.js +85 -0
- package/dist/migration/index.js.map +1 -0
- package/dist/mongo.d.ts +2 -3
- package/dist/mongo.d.ts.map +1 -0
- package/dist/mongo.js +2 -1
- package/dist/mongo.js.map +1 -1
- package/dist/rate-limit/db.d.ts +29 -0
- package/dist/rate-limit/db.d.ts.map +1 -0
- package/dist/rate-limit/db.js +25 -0
- package/dist/rate-limit/db.js.map +1 -0
- package/dist/rate-limit/index.d.ts +4 -0
- package/dist/rate-limit/index.d.ts.map +1 -0
- package/dist/rate-limit/index.js +6 -0
- package/dist/rate-limit/index.js.map +1 -0
- package/dist/rate-limit/rules.d.ts +22 -0
- package/dist/rate-limit/rules.d.ts.map +1 -0
- package/dist/rate-limit/rules.js +112 -0
- package/dist/rate-limit/rules.js.map +1 -0
- package/dist/rate-limit/types.d.ts +12 -0
- package/dist/rate-limit/types.d.ts.map +1 -0
- package/dist/rate-limit/types.js +2 -0
- package/dist/rate-limit/types.js.map +1 -0
- package/dist/routes/handler.d.ts +4 -0
- package/dist/routes/handler.d.ts.map +1 -0
- package/dist/routes/handler.js +66 -0
- package/dist/routes/handler.js.map +1 -0
- package/dist/routes/types.d.ts +45 -0
- package/dist/routes/types.d.ts.map +1 -0
- package/dist/routes/types.js +2 -0
- package/dist/routes/types.js.map +1 -0
- package/dist/server.d.ts +19 -551
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +15 -18
- package/dist/server.js.map +1 -1
- package/dist/system/index.d.ts +4 -0
- package/dist/system/index.d.ts.map +1 -0
- package/dist/system/index.js +21 -0
- package/dist/system/index.js.map +1 -0
- package/dist/telemetry/index.d.ts +11 -0
- package/dist/telemetry/index.d.ts.map +1 -0
- package/dist/telemetry/index.js +79 -0
- package/dist/telemetry/index.js.map +1 -0
- package/dist/telemetry.d.ts +2 -11
- package/dist/telemetry.d.ts.map +1 -0
- package/dist/telemetry.js +1 -1
- package/dist/telemetry.js.map +1 -1
- package/dist/time.d.ts +8 -0
- package/dist/time.d.ts.map +1 -0
- package/dist/time.js +13 -0
- package/dist/time.js.map +1 -0
- package/dist/types/email.d.ts +25 -0
- package/dist/types/email.d.ts.map +1 -0
- package/dist/types/email.js +2 -0
- package/dist/types/email.js.map +1 -0
- package/dist/types/index.d.ts +14 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +2 -0
- package/dist/types/index.js.map +1 -0
- package/dist/types.d.ts +10 -7
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +1 -1
- package/dist/types.js.map +1 -1
- package/dist/utils/index.d.ts +4 -0
- package/dist/utils/index.d.ts.map +1 -0
- package/dist/utils/index.js +15 -0
- package/dist/utils/index.js.map +1 -0
- package/dist/viteServer.d.ts +13 -0
- package/dist/viteServer.d.ts.map +1 -0
- package/dist/viteServer.js +132 -0
- package/dist/viteServer.js.map +1 -0
- package/dist/websocket/client.d.ts +9 -0
- package/dist/websocket/client.d.ts.map +1 -0
- package/dist/websocket/client.js +16 -0
- package/dist/websocket/client.js.map +1 -0
- package/dist/websocket/clientChannel.d.ts +9 -0
- package/dist/websocket/clientChannel.d.ts.map +1 -0
- package/dist/websocket/clientChannel.js +26 -0
- package/dist/websocket/clientChannel.js.map +1 -0
- package/dist/websocket/serverChannel.d.ts +14 -0
- package/dist/websocket/serverChannel.d.ts.map +1 -0
- package/dist/websocket/serverChannel.js +21 -0
- package/dist/websocket/serverChannel.js.map +1 -0
- package/dist/websocket/socketio/client.d.ts +4 -0
- package/dist/websocket/socketio/client.d.ts.map +1 -0
- package/dist/websocket/socketio/client.js +48 -0
- package/dist/websocket/socketio/client.js.map +1 -0
- package/dist/websocket/socketio/server.d.ts +10 -0
- package/dist/websocket/socketio/server.d.ts.map +1 -0
- package/dist/websocket/socketio/server.js +80 -0
- package/dist/websocket/socketio/server.js.map +1 -0
- package/dist/websocket/types.d.ts +45 -0
- package/dist/websocket/types.d.ts.map +1 -0
- package/dist/websocket/types.js +2 -0
- package/dist/websocket/types.js.map +1 -0
- package/package.json +5 -6
- package/dist/chunk-3S2FFBNS.js +0 -2
- package/dist/chunk-3S2FFBNS.js.map +0 -1
- package/dist/chunk-3YAV3UUU.js +0 -3
- package/dist/chunk-3YAV3UUU.js.map +0 -1
- package/dist/chunk-55J6XMHW.js +0 -2
- package/dist/chunk-55J6XMHW.js.map +0 -1
- package/dist/chunk-C3UESBRX.js +0 -2
- package/dist/chunk-C3UESBRX.js.map +0 -1
- package/dist/chunk-DO5TZLF5.js +0 -2
- package/dist/chunk-DO5TZLF5.js.map +0 -1
- package/dist/chunk-PB6WQQ4L.js +0 -3
- package/dist/chunk-PB6WQQ4L.js.map +0 -1
- package/dist/index-CwdohC5n.d.ts +0 -15
- package/dist/package-3YQBVIVQ.js +0 -2
- package/dist/package-3YQBVIVQ.js.map +0 -1
- package/dist/types-Ds1ESQSs.d.ts +0 -106
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import { z } from 'zod';
|
|
2
|
+
import { randomBytes } from 'crypto';
|
|
3
|
+
import { usersCollection, resetPasswordTokensCollection } from './db';
|
|
4
|
+
import { getEmailConfig } from '../app/emailConfig';
|
|
5
|
+
import { time } from '../time';
|
|
6
|
+
import { htmlToText } from '../utils';
|
|
7
|
+
import { validateEmail, validatePassword } from './validators';
|
|
8
|
+
import { hashPassword } from './password';
|
|
9
|
+
function resolveUrl(baseUrl, configuredUrl) {
|
|
10
|
+
if (!configuredUrl) {
|
|
11
|
+
return baseUrl;
|
|
12
|
+
}
|
|
13
|
+
if (configuredUrl.startsWith('http://') || configuredUrl.startsWith('https://')) {
|
|
14
|
+
return configuredUrl;
|
|
15
|
+
}
|
|
16
|
+
// Handle relative URL
|
|
17
|
+
return `${baseUrl}${configuredUrl.startsWith('/') ? '' : '/'}${configuredUrl}`;
|
|
18
|
+
}
|
|
19
|
+
function defaultPasswordResetTemplate({ email, resetUrl }) {
|
|
20
|
+
return `
|
|
21
|
+
<p>Hi,</p>
|
|
22
|
+
<p>We received a request to reset your password for ${email}.</p>
|
|
23
|
+
<p>Click the link below to reset your password:</p>
|
|
24
|
+
<p><a href="${resetUrl}">${resetUrl}</a></p>
|
|
25
|
+
<p>This link will expire in 1 hour.</p>
|
|
26
|
+
<p>If you did not request this password reset, please ignore this email.</p>
|
|
27
|
+
`;
|
|
28
|
+
}
|
|
29
|
+
const passwordResetSent = {
|
|
30
|
+
success: true,
|
|
31
|
+
message: 'If an account with that email exists, a password reset link has been sent',
|
|
32
|
+
};
|
|
33
|
+
export async function handleSendResetPasswordToken(args, { connectionInfo }) {
|
|
34
|
+
const email = validateEmail(args.email);
|
|
35
|
+
// Find user by email
|
|
36
|
+
const userDoc = await usersCollection.findOne({ 'emails.address': email, status: { $nin: ['deleted', 'disabled'] } }, { collation: { locale: 'en', strength: 2 } });
|
|
37
|
+
if (!userDoc) {
|
|
38
|
+
// For security, don't reveal if email exists or not
|
|
39
|
+
return passwordResetSent;
|
|
40
|
+
}
|
|
41
|
+
// Check if user has password auth method
|
|
42
|
+
if (!userDoc.authMethods?.password) {
|
|
43
|
+
return passwordResetSent;
|
|
44
|
+
}
|
|
45
|
+
const emailProvider = getEmailConfig().provider;
|
|
46
|
+
if (!emailProvider) {
|
|
47
|
+
throw new Error('Email provider is not configured');
|
|
48
|
+
}
|
|
49
|
+
// Generate reset token
|
|
50
|
+
const resetToken = randomBytes(32).toString('hex');
|
|
51
|
+
const now = Date.now();
|
|
52
|
+
const createdAt = new Date(now);
|
|
53
|
+
const expiresAt = new Date(now + time.hours(1)); // 1 hour expiry
|
|
54
|
+
// Store reset token
|
|
55
|
+
await resetPasswordTokensCollection.insertOne({
|
|
56
|
+
userId: userDoc._id,
|
|
57
|
+
token: resetToken,
|
|
58
|
+
createdAt,
|
|
59
|
+
expiresAt,
|
|
60
|
+
});
|
|
61
|
+
// Build reset URL
|
|
62
|
+
const baseUrl = process.env.MODELENCE_SITE_URL || connectionInfo?.baseUrl;
|
|
63
|
+
const resetPasswordUrl = resolveUrl(baseUrl, getEmailConfig().passwordReset?.redirectUrl);
|
|
64
|
+
const resetUrl = `${resetPasswordUrl}?token=${resetToken}`;
|
|
65
|
+
// Send email
|
|
66
|
+
const template = getEmailConfig()?.passwordReset?.template || defaultPasswordResetTemplate;
|
|
67
|
+
const htmlTemplate = template({ email, resetUrl, name: '' });
|
|
68
|
+
const textContent = htmlToText(htmlTemplate);
|
|
69
|
+
await emailProvider.sendEmail({
|
|
70
|
+
to: email,
|
|
71
|
+
from: getEmailConfig()?.from || 'noreply@modelence.com',
|
|
72
|
+
subject: getEmailConfig()?.passwordReset?.subject || 'Reset your password',
|
|
73
|
+
text: textContent,
|
|
74
|
+
html: htmlTemplate,
|
|
75
|
+
});
|
|
76
|
+
return passwordResetSent;
|
|
77
|
+
}
|
|
78
|
+
export async function handleResetPassword(args, {}) {
|
|
79
|
+
const token = z.string().parse(args.token);
|
|
80
|
+
const password = validatePassword(args.password);
|
|
81
|
+
// Find the reset token
|
|
82
|
+
const resetTokenDoc = await resetPasswordTokensCollection.findOne({ token });
|
|
83
|
+
if (!resetTokenDoc) {
|
|
84
|
+
throw new Error('Invalid or expired reset token');
|
|
85
|
+
}
|
|
86
|
+
// Check if token is expired
|
|
87
|
+
if (resetTokenDoc.expiresAt < new Date()) {
|
|
88
|
+
await resetPasswordTokensCollection.deleteOne({ token });
|
|
89
|
+
throw new Error('Reset token has expired');
|
|
90
|
+
}
|
|
91
|
+
// Find the user
|
|
92
|
+
const userDoc = await usersCollection.findOne({ _id: resetTokenDoc.userId });
|
|
93
|
+
if (!userDoc) {
|
|
94
|
+
throw new Error('User not found');
|
|
95
|
+
}
|
|
96
|
+
// Hash the new password
|
|
97
|
+
const hash = await hashPassword(password);
|
|
98
|
+
// Update user's password
|
|
99
|
+
await usersCollection.updateOne({ _id: userDoc._id }, {
|
|
100
|
+
$set: {
|
|
101
|
+
'authMethods.password.hash': hash,
|
|
102
|
+
},
|
|
103
|
+
});
|
|
104
|
+
// Delete the used reset token
|
|
105
|
+
await resetPasswordTokensCollection.deleteOne({ token });
|
|
106
|
+
return { success: true, message: 'Password has been reset successfully' };
|
|
107
|
+
}
|
|
108
|
+
//# sourceMappingURL=resetPassword.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"resetPassword.js","sourceRoot":"","sources":["../../src/auth/resetPassword.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import { RoleDefinition, Role, Permission } from './types';
|
|
2
|
+
export declare function initRoles(roles: Record<Role, RoleDefinition>, _defaultRoles: Record<string, Role>): void;
|
|
3
|
+
export declare function getUnauthenticatedRoles(): string[];
|
|
4
|
+
export declare function getDefaultAuthenticatedRoles(): string[];
|
|
5
|
+
export declare function hasAccess(roles: Role[], requiredPermissions: Permission[]): boolean;
|
|
6
|
+
export declare function requireAccess(roles: Role[], requiredPermissions: Permission[]): void;
|
|
7
|
+
export declare function hasPermission(roles: Role[], permission: Permission): boolean;
|
|
8
|
+
//# sourceMappingURL=role.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"role.d.ts","sourceRoot":"","sources":["../../src/auth/role.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,IAAI,EAAgB,UAAU,EAAE,MAAM,SAAS,CAAC;AAQzE,wBAAgB,SAAS,CACvB,KAAK,EAAE,MAAM,CAAC,IAAI,EAAE,cAAc,CAAC,EACnC,aAAa,EAAE,MAAM,CAAC,MAAM,EAAE,IAAI,CAAC,QAQpC;AAED,wBAAgB,uBAAuB,aAEtC;AAED,wBAAgB,4BAA4B,aAE3C;AAED,wBAAgB,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,mBAAmB,EAAE,UAAU,EAAE,WAEzE;AAED,wBAAgB,aAAa,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,mBAAmB,EAAE,UAAU,EAAE,QAQ7E;AAED,wBAAgB,aAAa,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,UAAU,EAAE,UAAU,WAUlE"}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
const roleMap = new Map();
|
|
2
|
+
const defaultRoles = {
|
|
3
|
+
authenticated: null,
|
|
4
|
+
unauthenticated: null,
|
|
5
|
+
};
|
|
6
|
+
export function initRoles(roles, _defaultRoles) {
|
|
7
|
+
defaultRoles.authenticated = _defaultRoles.authenticated;
|
|
8
|
+
defaultRoles.unauthenticated = _defaultRoles.unauthenticated;
|
|
9
|
+
for (const [name, definition] of Object.entries(roles)) {
|
|
10
|
+
roleMap.set(name, definition);
|
|
11
|
+
}
|
|
12
|
+
}
|
|
13
|
+
export function getUnauthenticatedRoles() {
|
|
14
|
+
return defaultRoles.unauthenticated ? [defaultRoles.unauthenticated] : [];
|
|
15
|
+
}
|
|
16
|
+
export function getDefaultAuthenticatedRoles() {
|
|
17
|
+
return defaultRoles.authenticated ? [defaultRoles.authenticated] : [];
|
|
18
|
+
}
|
|
19
|
+
export function hasAccess(roles, requiredPermissions) {
|
|
20
|
+
return requiredPermissions.every((permission) => hasPermission(roles, permission));
|
|
21
|
+
}
|
|
22
|
+
export function requireAccess(roles, requiredPermissions) {
|
|
23
|
+
const missingPermission = requiredPermissions.find((permission) => !hasPermission(roles, permission));
|
|
24
|
+
if (missingPermission) {
|
|
25
|
+
throw new Error(`Access denied - missing permission: '${missingPermission}'`);
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
export function hasPermission(roles, permission) {
|
|
29
|
+
for (const role of roles) {
|
|
30
|
+
const definition = roleMap.get(role);
|
|
31
|
+
if (definition && definition.permissions.includes(permission)) {
|
|
32
|
+
return true;
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
return false;
|
|
36
|
+
}
|
|
37
|
+
//# sourceMappingURL=role.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"role.js","sourceRoot":"","sources":["../../src/auth/role.ts"],"names":[],"mappings":"AAEA,MAAM,OAAO,GAAG,IAAI,GAAG,EAAwB,CAAC;AAChD,MAAM,YAAY,GAAiB;IACjC,aAAa,EAAE,IAAI;IACnB,eAAe,EAAE,IAAI;CACtB,CAAC;AAEF,MAAM,UAAU,SAAS,CACvB,KAAmC,EACnC,aAAmC;IAEnC,YAAY,CAAC,aAAa,GAAG,aAAa,CAAC,aAAa,CAAC;IACzD,YAAY,CAAC,eAAe,GAAG,aAAa,CAAC,eAAe,CAAC;IAE7D,KAAK,MAAM,CAAC,IAAI,EAAE,UAAU,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QACvD,OAAO,CAAC,GAAG,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;IAChC,CAAC;AACH,CAAC;AAED,MAAM,UAAU,uBAAuB;IACrC,OAAO,YAAY,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;AAC5E,CAAC;AAED,MAAM,UAAU,4BAA4B;IAC1C,OAAO,YAAY,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;AACxE,CAAC;AAED,MAAM,UAAU,SAAS,CAAC,KAAa,EAAE,mBAAiC;IACxE,OAAO,mBAAmB,CAAC,KAAK,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,aAAa,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC;AACrF,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,KAAa,EAAE,mBAAiC;IAC5E,MAAM,iBAAiB,GAAG,mBAAmB,CAAC,IAAI,CAChD,CAAC,UAAU,EAAE,EAAE,CAAC,CAAC,aAAa,CAAC,KAAK,EAAE,UAAU,CAAC,CAClD,CAAC;IAEF,IAAI,iBAAiB,EAAE,CAAC;QACtB,MAAM,IAAI,KAAK,CAAC,wCAAwC,iBAAiB,GAAG,CAAC,CAAC;IAChF,CAAC;AACH,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,KAAa,EAAE,UAAsB;IACjE,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,MAAM,UAAU,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAErC,IAAI,UAAU,IAAI,UAAU,CAAC,WAAW,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE,CAAC;YAC9D,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { ObjectId } from 'mongodb';
|
|
2
|
+
import { Module } from '../app/module';
|
|
3
|
+
import { Store } from '../data/store';
|
|
4
|
+
import { Session } from './types';
|
|
5
|
+
export declare const sessionsCollection: Store<{
|
|
6
|
+
authToken: import("zod").ZodString;
|
|
7
|
+
createdAt: import("zod").ZodDate;
|
|
8
|
+
expiresAt: import("zod").ZodDate;
|
|
9
|
+
userId: import("zod").ZodNullable<import("zod").ZodType<ObjectId, import("zod").ZodTypeDef, ObjectId>>;
|
|
10
|
+
}, Record<string, (this: Pick<import("../types").InferDocumentType<{
|
|
11
|
+
authToken: import("zod").ZodString;
|
|
12
|
+
createdAt: import("zod").ZodDate;
|
|
13
|
+
expiresAt: import("zod").ZodDate;
|
|
14
|
+
userId: import("zod").ZodNullable<import("zod").ZodType<ObjectId, import("zod").ZodTypeDef, ObjectId>>;
|
|
15
|
+
}>, "authToken" | "createdAt" | "expiresAt" | "userId"> & {
|
|
16
|
+
_id: ObjectId;
|
|
17
|
+
}, ...args: any[]) => any>>;
|
|
18
|
+
export declare function obtainSession(authToken: string | null): Promise<Session>;
|
|
19
|
+
export declare function setSessionUser(authToken: string, userId: ObjectId): Promise<void>;
|
|
20
|
+
export declare function clearSessionUser(authToken: string): Promise<void>;
|
|
21
|
+
export declare function createSession(userId?: ObjectId | null): Promise<Session>;
|
|
22
|
+
declare const _default: Module;
|
|
23
|
+
export default _default;
|
|
24
|
+
//# sourceMappingURL=session.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session.d.ts","sourceRoot":"","sources":["../../src/auth/session.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AACnC,OAAO,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAEvC,OAAO,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;AAGtC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAElC,eAAO,MAAM,kBAAkB;;;;;;;;;;;;2BAS7B,CAAC;AAEH,wBAAsB,aAAa,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI,GAAG,OAAO,CAAC,OAAO,CAAC,CAY9E;AAED,wBAAsB,cAAc,CAAC,SAAS,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,iBAOvE;AAED,wBAAsB,gBAAgB,CAAC,SAAS,EAAE,MAAM,iBAOvD;AAED,wBAAsB,aAAa,CAAC,MAAM,GAAE,QAAQ,GAAG,IAAW,GAAG,OAAO,CAAC,OAAO,CAAC,CAmBpF;;AAiBD,wBAmBG"}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import { randomBytes } from 'crypto';
|
|
2
|
+
import { Module } from '../app/module';
|
|
3
|
+
import { getPublicConfigs } from '../config/server';
|
|
4
|
+
import { Store } from '../data/store';
|
|
5
|
+
import { schema } from '../data/types';
|
|
6
|
+
import { time } from '../time';
|
|
7
|
+
export const sessionsCollection = new Store('_modelenceSessions', {
|
|
8
|
+
schema: {
|
|
9
|
+
authToken: schema.string(),
|
|
10
|
+
createdAt: schema.date(),
|
|
11
|
+
expiresAt: schema.date(),
|
|
12
|
+
userId: schema.userId().nullable(),
|
|
13
|
+
},
|
|
14
|
+
indexes: [{ key: { authToken: 1 }, unique: true }, { key: { expiresAt: 1 } }],
|
|
15
|
+
// TODO: add TTL index on expiresAt
|
|
16
|
+
});
|
|
17
|
+
export async function obtainSession(authToken) {
|
|
18
|
+
const existingSession = authToken ? await sessionsCollection.findOne({ authToken }) : null;
|
|
19
|
+
if (existingSession) {
|
|
20
|
+
return {
|
|
21
|
+
authToken: String(existingSession.authToken),
|
|
22
|
+
expiresAt: new Date(existingSession.expiresAt),
|
|
23
|
+
userId: existingSession.userId ?? null,
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
return await createSession();
|
|
27
|
+
}
|
|
28
|
+
export async function setSessionUser(authToken, userId) {
|
|
29
|
+
await sessionsCollection.updateOne({ authToken }, {
|
|
30
|
+
$set: { userId },
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
export async function clearSessionUser(authToken) {
|
|
34
|
+
await sessionsCollection.updateOne({ authToken }, {
|
|
35
|
+
$set: { userId: null },
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
export async function createSession(userId = null) {
|
|
39
|
+
// TODO: add rate-limiting and captcha handling
|
|
40
|
+
const authToken = randomBytes(32).toString('base64url');
|
|
41
|
+
const now = Date.now();
|
|
42
|
+
const expiresAt = new Date(now + time.days(7));
|
|
43
|
+
await sessionsCollection.insertOne({
|
|
44
|
+
authToken,
|
|
45
|
+
createdAt: new Date(now),
|
|
46
|
+
expiresAt,
|
|
47
|
+
userId,
|
|
48
|
+
});
|
|
49
|
+
return {
|
|
50
|
+
authToken,
|
|
51
|
+
expiresAt,
|
|
52
|
+
userId,
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
async function processSessionHeartbeat(session) {
|
|
56
|
+
const now = Date.now();
|
|
57
|
+
const newExpiresAt = new Date(now + time.days(7));
|
|
58
|
+
await sessionsCollection.updateOne({ authToken: session.authToken }, {
|
|
59
|
+
$set: {
|
|
60
|
+
lastActiveDate: new Date(now),
|
|
61
|
+
expiresAt: newExpiresAt,
|
|
62
|
+
},
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
export default new Module('_system.session', {
|
|
66
|
+
stores: [sessionsCollection],
|
|
67
|
+
mutations: {
|
|
68
|
+
init: async function (args, { session, user }) {
|
|
69
|
+
// TODO: mark or track app load somewhere
|
|
70
|
+
return {
|
|
71
|
+
session,
|
|
72
|
+
user,
|
|
73
|
+
configs: getPublicConfigs(),
|
|
74
|
+
};
|
|
75
|
+
},
|
|
76
|
+
heartbeat: async function (args, { session }) {
|
|
77
|
+
// Session might not exist if there is no database/authentication setup
|
|
78
|
+
if (session) {
|
|
79
|
+
await processSessionHeartbeat(session);
|
|
80
|
+
}
|
|
81
|
+
},
|
|
82
|
+
},
|
|
83
|
+
});
|
|
84
|
+
//# sourceMappingURL=session.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session.js","sourceRoot":"","sources":["../../src/auth/session.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"signup.d.ts","sourceRoot":"","sources":["../../src/auth/signup.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AASjD,wBAAsB,wBAAwB,CAC5C,IAAI,EAAE,IAAI,EACV,EAAE,IAAI,EAAE,OAAO,EAAE,cAAc,EAAE,EAAE,OAAO,oCAwG3C"}
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import { usersCollection } from './db';
|
|
2
|
+
import { isDisposableEmail } from './disposableEmails';
|
|
3
|
+
import { consumeRateLimit } from '../rate-limit/rules';
|
|
4
|
+
import { sendVerificationEmail } from './verification';
|
|
5
|
+
import { validateEmail, validatePassword } from './validators';
|
|
6
|
+
import { getAuthConfig } from '../app/authConfig';
|
|
7
|
+
import { hashPassword } from './password';
|
|
8
|
+
export async function handleSignupWithPassword(args, { user, session, connectionInfo }) {
|
|
9
|
+
try {
|
|
10
|
+
const email = validateEmail(args.email);
|
|
11
|
+
const password = validatePassword(args.password);
|
|
12
|
+
const ip = connectionInfo?.ip;
|
|
13
|
+
if (ip) {
|
|
14
|
+
await consumeRateLimit({
|
|
15
|
+
bucket: 'signupAttempt',
|
|
16
|
+
type: 'ip',
|
|
17
|
+
value: ip,
|
|
18
|
+
});
|
|
19
|
+
}
|
|
20
|
+
if (await isDisposableEmail(email)) {
|
|
21
|
+
throw new Error('Please use a permanent email address');
|
|
22
|
+
}
|
|
23
|
+
// TODO: captcha check
|
|
24
|
+
if (user) {
|
|
25
|
+
// TODO: handle cases where a user is already logged in
|
|
26
|
+
}
|
|
27
|
+
const existingUser = await usersCollection.findOne({ 'emails.address': email }, { collation: { locale: 'en', strength: 2 } });
|
|
28
|
+
if (existingUser) {
|
|
29
|
+
const existingEmail = existingUser.emails?.find((e) => e.address === email);
|
|
30
|
+
if (existingUser.status === 'disabled') {
|
|
31
|
+
throw new Error(`User is marked for deletion, please contact support if you want to restore the account.`);
|
|
32
|
+
}
|
|
33
|
+
throw new Error(`User with email already exists: ${existingEmail?.address}`);
|
|
34
|
+
}
|
|
35
|
+
if (ip) {
|
|
36
|
+
await consumeRateLimit({
|
|
37
|
+
bucket: 'signup',
|
|
38
|
+
type: 'ip',
|
|
39
|
+
value: ip,
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
// Hash password with native crypto scrypt (salt is automatically generated)
|
|
43
|
+
const hash = await hashPassword(password);
|
|
44
|
+
const result = await usersCollection.insertOne({
|
|
45
|
+
handle: email,
|
|
46
|
+
status: 'active',
|
|
47
|
+
emails: [
|
|
48
|
+
{
|
|
49
|
+
address: email,
|
|
50
|
+
verified: false,
|
|
51
|
+
},
|
|
52
|
+
],
|
|
53
|
+
createdAt: new Date(),
|
|
54
|
+
authMethods: {
|
|
55
|
+
password: {
|
|
56
|
+
hash,
|
|
57
|
+
},
|
|
58
|
+
},
|
|
59
|
+
});
|
|
60
|
+
const userDocument = await usersCollection.findOne({ _id: result.insertedId }, { readPreference: 'primary' });
|
|
61
|
+
if (!userDocument) {
|
|
62
|
+
throw new Error('User not found');
|
|
63
|
+
}
|
|
64
|
+
await sendVerificationEmail({
|
|
65
|
+
userId: result?.insertedId,
|
|
66
|
+
email,
|
|
67
|
+
baseUrl: connectionInfo?.baseUrl,
|
|
68
|
+
});
|
|
69
|
+
getAuthConfig().onAfterSignup?.({
|
|
70
|
+
user: userDocument,
|
|
71
|
+
session,
|
|
72
|
+
connectionInfo,
|
|
73
|
+
});
|
|
74
|
+
getAuthConfig().signup?.onSuccess?.(userDocument);
|
|
75
|
+
return result.insertedId;
|
|
76
|
+
}
|
|
77
|
+
catch (error) {
|
|
78
|
+
if (error instanceof Error) {
|
|
79
|
+
getAuthConfig().onSignupError?.({
|
|
80
|
+
error,
|
|
81
|
+
session,
|
|
82
|
+
connectionInfo,
|
|
83
|
+
});
|
|
84
|
+
getAuthConfig().signup?.onError?.(error);
|
|
85
|
+
}
|
|
86
|
+
throw error;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
//# sourceMappingURL=signup.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"signup.js","sourceRoot":"","sources":["../../src/auth/signup.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"emailVerficationTemplate.d.ts","sourceRoot":"","sources":["../../../src/auth/templates/emailVerficationTemplate.ts"],"names":[],"mappings":"AAAA,wBAAgB,yBAAyB,CAAC,EACxC,IAAI,EACJ,KAAK,EACL,eAAe,GAChB,EAAE;IACD,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;IACd,eAAe,EAAE,MAAM,CAAC;CACzB,UAOA"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
export function emailVerificationTemplate({ name, email, verificationUrl, }) {
|
|
2
|
+
return `
|
|
3
|
+
<p>Hi${name ? ` ${name}` : ''},</p>
|
|
4
|
+
<p>Please verify your email address ${email} by clicking the link below:</p>
|
|
5
|
+
<p><a href="${verificationUrl}">${verificationUrl}</a></p>
|
|
6
|
+
<p>If you did not request this, please ignore this email.</p>
|
|
7
|
+
`;
|
|
8
|
+
}
|
|
9
|
+
//# sourceMappingURL=emailVerficationTemplate.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"emailVerficationTemplate.js","sourceRoot":"","sources":["../../../src/auth/templates/emailVerficationTemplate.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,yBAAyB,CAAC,EACxC,IAAI,EACJ,KAAK,EACL,eAAe,GAKhB;IACC,OAAO;WACE,IAAI,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE;0CACS,KAAK;kBAC7B,eAAe,KAAK,eAAe;;GAElD,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { Document, ObjectId } from 'mongodb';
|
|
2
|
+
export type User = Document;
|
|
3
|
+
export type UserInfo = {
|
|
4
|
+
id: string;
|
|
5
|
+
handle: string;
|
|
6
|
+
roles: string[];
|
|
7
|
+
hasRole: (role: string) => boolean;
|
|
8
|
+
requireRole: (role: string) => void;
|
|
9
|
+
};
|
|
10
|
+
export type Role = string;
|
|
11
|
+
export type DefaultRoles = Record<'authenticated' | 'unauthenticated', Role | null>;
|
|
12
|
+
export type Session = {
|
|
13
|
+
authToken: string;
|
|
14
|
+
expiresAt: Date;
|
|
15
|
+
userId: ObjectId | null;
|
|
16
|
+
};
|
|
17
|
+
export type Permission = string;
|
|
18
|
+
export type RoleDefinition = {
|
|
19
|
+
description?: string;
|
|
20
|
+
permissions: Permission[];
|
|
21
|
+
};
|
|
22
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/auth/types.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAE7C,MAAM,MAAM,IAAI,GAAG,QAAQ,CAAC;AAE5B,MAAM,MAAM,QAAQ,GAAG;IACrB,EAAE,EAAE,MAAM,CAAC;IACX,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,OAAO,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,OAAO,CAAC;IACnC,WAAW,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;CACrC,CAAC;AAEF,MAAM,MAAM,IAAI,GAAG,MAAM,CAAC;AAE1B,MAAM,MAAM,YAAY,GAAG,MAAM,CAAC,eAAe,GAAG,iBAAiB,EAAE,IAAI,GAAG,IAAI,CAAC,CAAC;AAEpF,MAAM,MAAM,OAAO,GAAG;IACpB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,IAAI,CAAC;IAChB,MAAM,EAAE,QAAQ,GAAG,IAAI,CAAC;CACzB,CAAC;AAEF,MAAM,MAAM,UAAU,GAAG,MAAM,CAAC;AAEhC,MAAM,MAAM,cAAc,GAAG;IAC3B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,WAAW,EAAE,UAAU,EAAE,CAAC;CAC3B,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/auth/types.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"user.d.ts","sourceRoot":"","sources":["../../src/auth/user.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAevC,wBAAsB,eAAe,qCAkBpC;;AAED,wBA6HG"}
|
|
@@ -0,0 +1,152 @@
|
|
|
1
|
+
import { randomBytes } from 'crypto';
|
|
2
|
+
import { Module } from '../app/module';
|
|
3
|
+
import { time } from '../time';
|
|
4
|
+
import { dbDisposableEmailDomains, emailVerificationTokensCollection, resetPasswordTokensCollection, usersCollection, } from './db';
|
|
5
|
+
import { updateDisposableEmailListCron } from './disposableEmails';
|
|
6
|
+
import { handleLoginWithPassword, handleLogout } from './login';
|
|
7
|
+
import { getOwnProfile } from './profile';
|
|
8
|
+
import { handleSignupWithPassword } from './signup';
|
|
9
|
+
import { handleVerifyEmail } from './verification';
|
|
10
|
+
import { handleResetPassword, handleSendResetPasswordToken } from './resetPassword';
|
|
11
|
+
export async function createGuestUser() {
|
|
12
|
+
// TODO: add rate-limiting and captcha handling
|
|
13
|
+
const guestId = randomBytes(9)
|
|
14
|
+
.toString('base64')
|
|
15
|
+
.replace(/[+/]/g, (c) => (c === '+' ? 'a' : 'b'));
|
|
16
|
+
const handle = `guest_${guestId}`;
|
|
17
|
+
// TODO: re-try on handle collision
|
|
18
|
+
const result = await usersCollection.insertOne({
|
|
19
|
+
handle,
|
|
20
|
+
status: 'active',
|
|
21
|
+
createdAt: new Date(),
|
|
22
|
+
authMethods: {},
|
|
23
|
+
});
|
|
24
|
+
return result.insertedId;
|
|
25
|
+
}
|
|
26
|
+
export default new Module('_system.user', {
|
|
27
|
+
stores: [
|
|
28
|
+
usersCollection,
|
|
29
|
+
dbDisposableEmailDomains,
|
|
30
|
+
emailVerificationTokensCollection,
|
|
31
|
+
resetPasswordTokensCollection,
|
|
32
|
+
],
|
|
33
|
+
queries: {
|
|
34
|
+
getOwnProfile,
|
|
35
|
+
},
|
|
36
|
+
mutations: {
|
|
37
|
+
signupWithPassword: handleSignupWithPassword,
|
|
38
|
+
loginWithPassword: handleLoginWithPassword,
|
|
39
|
+
logout: handleLogout,
|
|
40
|
+
sendResetPasswordToken: handleSendResetPasswordToken,
|
|
41
|
+
resetPassword: handleResetPassword,
|
|
42
|
+
},
|
|
43
|
+
cronJobs: {
|
|
44
|
+
updateDisposableEmailList: updateDisposableEmailListCron,
|
|
45
|
+
},
|
|
46
|
+
rateLimits: [
|
|
47
|
+
{
|
|
48
|
+
bucket: 'signup',
|
|
49
|
+
type: 'ip',
|
|
50
|
+
window: time.minutes(15),
|
|
51
|
+
limit: 20,
|
|
52
|
+
},
|
|
53
|
+
{
|
|
54
|
+
bucket: 'signup',
|
|
55
|
+
type: 'ip',
|
|
56
|
+
window: time.days(1),
|
|
57
|
+
limit: 200,
|
|
58
|
+
},
|
|
59
|
+
{
|
|
60
|
+
bucket: 'signupAttempt',
|
|
61
|
+
type: 'ip',
|
|
62
|
+
window: time.minutes(15),
|
|
63
|
+
limit: 50,
|
|
64
|
+
},
|
|
65
|
+
{
|
|
66
|
+
bucket: 'signupAttempt',
|
|
67
|
+
type: 'ip',
|
|
68
|
+
window: time.days(1),
|
|
69
|
+
limit: 500,
|
|
70
|
+
},
|
|
71
|
+
{
|
|
72
|
+
bucket: 'signin',
|
|
73
|
+
type: 'ip',
|
|
74
|
+
window: time.minutes(15),
|
|
75
|
+
limit: 50,
|
|
76
|
+
},
|
|
77
|
+
{
|
|
78
|
+
bucket: 'signin',
|
|
79
|
+
type: 'ip',
|
|
80
|
+
window: time.days(1),
|
|
81
|
+
limit: 500,
|
|
82
|
+
},
|
|
83
|
+
{
|
|
84
|
+
bucket: 'verification',
|
|
85
|
+
type: 'user',
|
|
86
|
+
window: time.minutes(15),
|
|
87
|
+
limit: 3,
|
|
88
|
+
},
|
|
89
|
+
{
|
|
90
|
+
bucket: 'verification',
|
|
91
|
+
type: 'user',
|
|
92
|
+
window: time.days(1),
|
|
93
|
+
limit: 10,
|
|
94
|
+
},
|
|
95
|
+
],
|
|
96
|
+
configSchema: {
|
|
97
|
+
'auth.email.enabled': {
|
|
98
|
+
type: 'boolean',
|
|
99
|
+
isPublic: true,
|
|
100
|
+
default: true,
|
|
101
|
+
},
|
|
102
|
+
'auth.email.from': {
|
|
103
|
+
type: 'string',
|
|
104
|
+
isPublic: false,
|
|
105
|
+
default: '',
|
|
106
|
+
},
|
|
107
|
+
'auth.email.verification': {
|
|
108
|
+
type: 'boolean',
|
|
109
|
+
isPublic: true,
|
|
110
|
+
default: false,
|
|
111
|
+
},
|
|
112
|
+
'auth.google.enabled': {
|
|
113
|
+
type: 'boolean',
|
|
114
|
+
isPublic: true,
|
|
115
|
+
default: false,
|
|
116
|
+
},
|
|
117
|
+
'auth.google.clientId': {
|
|
118
|
+
type: 'string',
|
|
119
|
+
isPublic: false,
|
|
120
|
+
default: '',
|
|
121
|
+
},
|
|
122
|
+
'auth.google.clientSecret': {
|
|
123
|
+
type: 'secret',
|
|
124
|
+
isPublic: false,
|
|
125
|
+
default: '',
|
|
126
|
+
},
|
|
127
|
+
'auth.github.enabled': {
|
|
128
|
+
type: 'boolean',
|
|
129
|
+
isPublic: true,
|
|
130
|
+
default: false,
|
|
131
|
+
},
|
|
132
|
+
'auth.github.clientId': {
|
|
133
|
+
type: 'string',
|
|
134
|
+
isPublic: false,
|
|
135
|
+
default: '',
|
|
136
|
+
},
|
|
137
|
+
'auth.github.clientSecret': {
|
|
138
|
+
type: 'secret',
|
|
139
|
+
isPublic: false,
|
|
140
|
+
default: '',
|
|
141
|
+
},
|
|
142
|
+
},
|
|
143
|
+
routes: [
|
|
144
|
+
{
|
|
145
|
+
path: '/api/_internal/auth/verify-email',
|
|
146
|
+
handlers: {
|
|
147
|
+
get: handleVerifyEmail,
|
|
148
|
+
},
|
|
149
|
+
},
|
|
150
|
+
],
|
|
151
|
+
});
|
|
152
|
+
//# sourceMappingURL=user.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"user.js","sourceRoot":"","sources":["../../src/auth/user.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,QAAQ,CAAC;AAErC,OAAO,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACvC,OAAO,EAAE,IAAI,EAAE,MAAM,SAAS,CAAC;AAC/B,OAAO,EACL,wBAAwB,EACxB,iCAAiC,EACjC,6BAA6B,EAC7B,eAAe,GAChB,MAAM,MAAM,CAAC;AACd,OAAO,EAAE,6BAA6B,EAAE,MAAM,oBAAoB,CAAC;AACnE,OAAO,EAAE,uBAAuB,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AAChE,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,wBAAwB,EAAE,MAAM,UAAU,CAAC;AACpD,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,mBAAmB,EAAE,4BAA4B,EAAE,MAAM,iBAAiB,CAAC;AAEpF,MAAM,CAAC,KAAK,UAAU,eAAe;IACnC,+CAA+C;IAE/C,MAAM,OAAO,GAAG,WAAW,CAAC,CAAC,CAAC;SAC3B,QAAQ,CAAC,QAAQ,CAAC;SAClB,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;IAEpD,MAAM,MAAM,GAAG,SAAS,OAAO,EAAE,CAAC;IAClC,mCAAmC;IAEnC,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,SAAS,CAAC;QAC7C,MAAM;QACN,MAAM,EAAE,QAAQ;QAChB,SAAS,EAAE,IAAI,IAAI,EAAE;QACrB,WAAW,EAAE,EAAE;KAChB,CAAC,CAAC;IAEH,OAAO,MAAM,CAAC,UAAU,CAAC;AAC3B,CAAC;AAED,eAAe,IAAI,MAAM,CAAC,cAAc,EAAE;IACxC,MAAM,EAAE;QACN,eAAe;QACf,wBAAwB;QACxB,iCAAiC;QACjC,6BAA6B;KAC9B;IACD,OAAO,EAAE;QACP,aAAa;KACd;IACD,SAAS,EAAE;QACT,kBAAkB,EAAE,wBAAwB;QAC5C,iBAAiB,EAAE,uBAAuB;QAC1C,MAAM,EAAE,YAAY;QACpB,sBAAsB,EAAE,4BAA4B;QACpD,aAAa,EAAE,mBAAmB;KACnC;IACD,QAAQ,EAAE;QACR,yBAAyB,EAAE,6BAA6B;KACzD;IACD,UAAU,EAAE;QACV;YACE,MAAM,EAAE,QAAQ;YAChB,IAAI,EAAE,IAAI;YACV,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YACxB,KAAK,EAAE,EAAE;SACV;QACD;YACE,MAAM,EAAE,QAAQ;YAChB,IAAI,EAAE,IAAI;YACV,MAAM,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;YACpB,KAAK,EAAE,GAAG;SACX;QACD;YACE,MAAM,EAAE,eAAe;YACvB,IAAI,EAAE,IAAI;YACV,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YACxB,KAAK,EAAE,EAAE;SACV;QACD;YACE,MAAM,EAAE,eAAe;YACvB,IAAI,EAAE,IAAI;YACV,MAAM,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;YACpB,KAAK,EAAE,GAAG;SACX;QACD;YACE,MAAM,EAAE,QAAQ;YAChB,IAAI,EAAE,IAAI;YACV,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YACxB,KAAK,EAAE,EAAE;SACV;QACD;YACE,MAAM,EAAE,QAAQ;YAChB,IAAI,EAAE,IAAI;YACV,MAAM,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;YACpB,KAAK,EAAE,GAAG;SACX;QACD;YACE,MAAM,EAAE,cAAc;YACtB,IAAI,EAAE,MAAM;YACZ,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YACxB,KAAK,EAAE,CAAC;SACT;QACD;YACE,MAAM,EAAE,cAAc;YACtB,IAAI,EAAE,MAAM;YACZ,MAAM,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;YACpB,KAAK,EAAE,EAAE;SACV;KACF;IACD,YAAY,EAAE;QACZ,oBAAoB,EAAE;YACpB,IAAI,EAAE,SAAS;YACf,QAAQ,EAAE,IAAI;YACd,OAAO,EAAE,IAAI;SACd;QACD,iBAAiB,EAAE;YACjB,IAAI,EAAE,QAAQ;YACd,QAAQ,EAAE,KAAK;YACf,OAAO,EAAE,EAAE;SACZ;QACD,yBAAyB,EAAE;YACzB,IAAI,EAAE,SAAS;YACf,QAAQ,EAAE,IAAI;YACd,OAAO,EAAE,KAAK;SACf;QACD,qBAAqB,EAAE;YACrB,IAAI,EAAE,SAAS;YACf,QAAQ,EAAE,IAAI;YACd,OAAO,EAAE,KAAK;SACf;QACD,sBAAsB,EAAE;YACtB,IAAI,EAAE,QAAQ;YACd,QAAQ,EAAE,KAAK;YACf,OAAO,EAAE,EAAE;SACZ;QACD,0BAA0B,EAAE;YAC1B,IAAI,EAAE,QAAQ;YACd,QAAQ,EAAE,KAAK;YACf,OAAO,EAAE,EAAE;SACZ;QACD,qBAAqB,EAAE;YACrB,IAAI,EAAE,SAAS;YACf,QAAQ,EAAE,IAAI;YACd,OAAO,EAAE,KAAK;SACf;QACD,sBAAsB,EAAE;YACtB,IAAI,EAAE,QAAQ;YACd,QAAQ,EAAE,KAAK;YACf,OAAO,EAAE,EAAE;SACZ;QACD,0BAA0B,EAAE;YAC1B,IAAI,EAAE,QAAQ;YACd,QAAQ,EAAE,KAAK;YACf,OAAO,EAAE,EAAE;SACZ;KACF;IACD,MAAM,EAAE;QACN;YACE,IAAI,EAAE,kCAAkC;YACxC,QAAQ,EAAE;gBACR,GAAG,EAAE,iBAAiB;aACvB;SACF;KACF;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"validators.d.ts","sourceRoot":"","sources":["../../src/auth/validators.ts"],"names":[],"mappings":"AAEA,wBAAgB,gBAAgB,CAAC,KAAK,EAAE,MAAM,UAE7C;AAED,wBAAgB,aAAa,CAAC,KAAK,EAAE,MAAM,UAE1C"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import { z } from 'zod';
|
|
2
|
+
export function validatePassword(value) {
|
|
3
|
+
return z.string().min(8, { message: 'Password must contain at least 8 characters' }).parse(value);
|
|
4
|
+
}
|
|
5
|
+
export function validateEmail(value) {
|
|
6
|
+
return z.string().email({ message: 'Invalid email address' }).parse(value);
|
|
7
|
+
}
|
|
8
|
+
//# sourceMappingURL=validators.js.map
|