@lenne.tech/nest-server 11.6.1 → 11.6.2
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/config.env.js +141 -0
- package/dist/config.env.js.map +1 -1
- package/dist/core/common/decorators/graphql-populate.decorator.d.ts +2 -2
- package/dist/core/common/decorators/restricted.decorator.d.ts +1 -0
- package/dist/core/common/decorators/restricted.decorator.js +1 -1
- package/dist/core/common/decorators/restricted.decorator.js.map +1 -1
- package/dist/core/common/helpers/input.helper.d.ts +1 -0
- package/dist/core/common/helpers/input.helper.js +1 -1
- package/dist/core/common/helpers/input.helper.js.map +1 -1
- package/dist/core/common/interfaces/server-options.interface.d.ts +50 -0
- package/dist/core/modules/auth/auth-guard-strategy.enum.d.ts +1 -0
- package/dist/core/modules/auth/auth-guard-strategy.enum.js +1 -0
- package/dist/core/modules/auth/auth-guard-strategy.enum.js.map +1 -1
- package/dist/core/modules/auth/guards/auth.guard.js +11 -5
- package/dist/core/modules/auth/guards/auth.guard.js.map +1 -1
- package/dist/core/modules/auth/tokens.decorator.d.ts +1 -1
- package/dist/core/modules/better-auth/better-auth-auth.model.d.ts +9 -0
- package/dist/core/modules/better-auth/better-auth-auth.model.js +63 -0
- package/dist/core/modules/better-auth/better-auth-auth.model.js.map +1 -0
- package/dist/core/modules/better-auth/better-auth-models.d.ts +44 -0
- package/dist/core/modules/better-auth/better-auth-models.js +185 -0
- package/dist/core/modules/better-auth/better-auth-models.js.map +1 -0
- package/dist/core/modules/better-auth/better-auth-rate-limit.middleware.d.ts +12 -0
- package/dist/core/modules/better-auth/better-auth-rate-limit.middleware.js +70 -0
- package/dist/core/modules/better-auth/better-auth-rate-limit.middleware.js.map +1 -0
- package/dist/core/modules/better-auth/better-auth-rate-limiter.service.d.ts +32 -0
- package/dist/core/modules/better-auth/better-auth-rate-limiter.service.js +173 -0
- package/dist/core/modules/better-auth/better-auth-rate-limiter.service.js.map +1 -0
- package/dist/core/modules/better-auth/better-auth-user.mapper.d.ts +43 -0
- package/dist/core/modules/better-auth/better-auth-user.mapper.js +159 -0
- package/dist/core/modules/better-auth/better-auth-user.mapper.js.map +1 -0
- package/dist/core/modules/better-auth/better-auth.config.d.ts +9 -0
- package/dist/core/modules/better-auth/better-auth.config.js +251 -0
- package/dist/core/modules/better-auth/better-auth.config.js.map +1 -0
- package/dist/core/modules/better-auth/better-auth.middleware.d.ts +20 -0
- package/dist/core/modules/better-auth/better-auth.middleware.js +79 -0
- package/dist/core/modules/better-auth/better-auth.middleware.js.map +1 -0
- package/dist/core/modules/better-auth/better-auth.module.d.ts +30 -0
- package/dist/core/modules/better-auth/better-auth.module.js +265 -0
- package/dist/core/modules/better-auth/better-auth.module.js.map +1 -0
- package/dist/core/modules/better-auth/better-auth.resolver.d.ts +49 -0
- package/dist/core/modules/better-auth/better-auth.resolver.js +539 -0
- package/dist/core/modules/better-auth/better-auth.resolver.js.map +1 -0
- package/dist/core/modules/better-auth/better-auth.service.d.ts +38 -0
- package/dist/core/modules/better-auth/better-auth.service.js +151 -0
- package/dist/core/modules/better-auth/better-auth.service.js.map +1 -0
- package/dist/core/modules/better-auth/better-auth.types.d.ts +38 -0
- package/dist/core/modules/better-auth/better-auth.types.js +15 -0
- package/dist/core/modules/better-auth/better-auth.types.js.map +1 -0
- package/dist/core/modules/better-auth/index.d.ts +11 -0
- package/dist/core/modules/better-auth/index.js +28 -0
- package/dist/core/modules/better-auth/index.js.map +1 -0
- package/dist/core/modules/user/core-user.model.d.ts +2 -0
- package/dist/core/modules/user/core-user.model.js +21 -0
- package/dist/core/modules/user/core-user.model.js.map +1 -1
- package/dist/core.module.js +7 -0
- package/dist/core.module.js.map +1 -1
- package/dist/index.d.ts +1 -0
- package/dist/index.js +1 -0
- package/dist/index.js.map +1 -1
- package/dist/tsconfig.build.tsbuildinfo +1 -1
- package/package.json +9 -1
- package/src/config.env.ts +148 -1
- package/src/core/common/decorators/restricted.decorator.ts +2 -2
- package/src/core/common/helpers/input.helper.ts +2 -2
- package/src/core/common/interfaces/server-options.interface.ts +344 -20
- package/src/core/modules/auth/auth-guard-strategy.enum.ts +1 -0
- package/src/core/modules/auth/guards/auth.guard.ts +20 -6
- package/src/core/modules/better-auth/README.md +1096 -0
- package/src/core/modules/better-auth/better-auth-auth.model.ts +69 -0
- package/src/core/modules/better-auth/better-auth-models.ts +143 -0
- package/src/core/modules/better-auth/better-auth-rate-limit.middleware.ts +113 -0
- package/src/core/modules/better-auth/better-auth-rate-limiter.service.ts +326 -0
- package/src/core/modules/better-auth/better-auth-user.mapper.ts +269 -0
- package/src/core/modules/better-auth/better-auth.config.ts +483 -0
- package/src/core/modules/better-auth/better-auth.middleware.ts +111 -0
- package/src/core/modules/better-auth/better-auth.module.ts +433 -0
- package/src/core/modules/better-auth/better-auth.resolver.ts +678 -0
- package/src/core/modules/better-auth/better-auth.service.ts +323 -0
- package/src/core/modules/better-auth/better-auth.types.ts +75 -0
- package/src/core/modules/better-auth/index.ts +25 -0
- package/src/core/modules/user/core-user.model.ts +29 -0
- package/src/core.module.ts +12 -0
- package/src/index.ts +6 -0
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
export declare class BetterAuthUserModel {
|
|
2
|
+
id: string;
|
|
3
|
+
iamId?: string;
|
|
4
|
+
email: string;
|
|
5
|
+
name?: string;
|
|
6
|
+
emailVerified?: boolean;
|
|
7
|
+
verified?: boolean;
|
|
8
|
+
roles?: string[];
|
|
9
|
+
}
|
|
10
|
+
export declare class BetterAuthSessionModel {
|
|
11
|
+
id: string;
|
|
12
|
+
expiresAt: Date;
|
|
13
|
+
user: BetterAuthUserModel;
|
|
14
|
+
}
|
|
15
|
+
export declare class BetterAuthSessionInfoModel {
|
|
16
|
+
id?: string;
|
|
17
|
+
token?: string;
|
|
18
|
+
expiresAt?: Date;
|
|
19
|
+
}
|
|
20
|
+
export declare class BetterAuth2FASetupModel {
|
|
21
|
+
success: boolean;
|
|
22
|
+
totpUri?: string;
|
|
23
|
+
backupCodes?: string[];
|
|
24
|
+
error?: string;
|
|
25
|
+
}
|
|
26
|
+
export declare class BetterAuthPasskeyModel {
|
|
27
|
+
id: string;
|
|
28
|
+
name?: string;
|
|
29
|
+
credentialId: string;
|
|
30
|
+
createdAt: Date;
|
|
31
|
+
}
|
|
32
|
+
export declare class BetterAuthPasskeyChallengeModel {
|
|
33
|
+
success: boolean;
|
|
34
|
+
challenge?: string;
|
|
35
|
+
error?: string;
|
|
36
|
+
}
|
|
37
|
+
export declare class BetterAuthFeaturesModel {
|
|
38
|
+
enabled: boolean;
|
|
39
|
+
jwt: boolean;
|
|
40
|
+
twoFactor: boolean;
|
|
41
|
+
passkey: boolean;
|
|
42
|
+
legacyPassword: boolean;
|
|
43
|
+
socialProviders: string[];
|
|
44
|
+
}
|
|
@@ -0,0 +1,185 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
|
3
|
+
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
4
|
+
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
5
|
+
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
|
6
|
+
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
7
|
+
};
|
|
8
|
+
var __metadata = (this && this.__metadata) || function (k, v) {
|
|
9
|
+
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
|
10
|
+
};
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.BetterAuthFeaturesModel = exports.BetterAuthPasskeyChallengeModel = exports.BetterAuthPasskeyModel = exports.BetterAuth2FASetupModel = exports.BetterAuthSessionInfoModel = exports.BetterAuthSessionModel = exports.BetterAuthUserModel = void 0;
|
|
13
|
+
const graphql_1 = require("@nestjs/graphql");
|
|
14
|
+
const restricted_decorator_1 = require("../../common/decorators/restricted.decorator");
|
|
15
|
+
const role_enum_1 = require("../../common/enums/role.enum");
|
|
16
|
+
let BetterAuthUserModel = class BetterAuthUserModel {
|
|
17
|
+
};
|
|
18
|
+
exports.BetterAuthUserModel = BetterAuthUserModel;
|
|
19
|
+
__decorate([
|
|
20
|
+
(0, graphql_1.Field)(() => String, { description: 'User ID' }),
|
|
21
|
+
__metadata("design:type", String)
|
|
22
|
+
], BetterAuthUserModel.prototype, "id", void 0);
|
|
23
|
+
__decorate([
|
|
24
|
+
(0, graphql_1.Field)(() => String, { description: 'IAM provider user ID (e.g., Better-Auth)', nullable: true }),
|
|
25
|
+
__metadata("design:type", String)
|
|
26
|
+
], BetterAuthUserModel.prototype, "iamId", void 0);
|
|
27
|
+
__decorate([
|
|
28
|
+
(0, graphql_1.Field)(() => String, { description: 'Email address' }),
|
|
29
|
+
__metadata("design:type", String)
|
|
30
|
+
], BetterAuthUserModel.prototype, "email", void 0);
|
|
31
|
+
__decorate([
|
|
32
|
+
(0, graphql_1.Field)(() => String, { description: 'Display name', nullable: true }),
|
|
33
|
+
__metadata("design:type", String)
|
|
34
|
+
], BetterAuthUserModel.prototype, "name", void 0);
|
|
35
|
+
__decorate([
|
|
36
|
+
(0, graphql_1.Field)(() => Boolean, { description: 'Email verified status', nullable: true }),
|
|
37
|
+
__metadata("design:type", Boolean)
|
|
38
|
+
], BetterAuthUserModel.prototype, "emailVerified", void 0);
|
|
39
|
+
__decorate([
|
|
40
|
+
(0, graphql_1.Field)(() => Boolean, { description: 'User verified status', nullable: true }),
|
|
41
|
+
__metadata("design:type", Boolean)
|
|
42
|
+
], BetterAuthUserModel.prototype, "verified", void 0);
|
|
43
|
+
__decorate([
|
|
44
|
+
(0, graphql_1.Field)(() => [String], { description: 'User roles', nullable: true }),
|
|
45
|
+
__metadata("design:type", Array)
|
|
46
|
+
], BetterAuthUserModel.prototype, "roles", void 0);
|
|
47
|
+
exports.BetterAuthUserModel = BetterAuthUserModel = __decorate([
|
|
48
|
+
(0, graphql_1.ObjectType)({ description: 'Better-Auth User' }),
|
|
49
|
+
(0, restricted_decorator_1.Restricted)(role_enum_1.RoleEnum.S_EVERYONE)
|
|
50
|
+
], BetterAuthUserModel);
|
|
51
|
+
let BetterAuthSessionModel = class BetterAuthSessionModel {
|
|
52
|
+
};
|
|
53
|
+
exports.BetterAuthSessionModel = BetterAuthSessionModel;
|
|
54
|
+
__decorate([
|
|
55
|
+
(0, graphql_1.Field)(() => String, { description: 'Session ID' }),
|
|
56
|
+
__metadata("design:type", String)
|
|
57
|
+
], BetterAuthSessionModel.prototype, "id", void 0);
|
|
58
|
+
__decorate([
|
|
59
|
+
(0, graphql_1.Field)(() => Date, { description: 'Session expiration date' }),
|
|
60
|
+
__metadata("design:type", Date)
|
|
61
|
+
], BetterAuthSessionModel.prototype, "expiresAt", void 0);
|
|
62
|
+
__decorate([
|
|
63
|
+
(0, graphql_1.Field)(() => BetterAuthUserModel, { description: 'Session user' }),
|
|
64
|
+
__metadata("design:type", BetterAuthUserModel)
|
|
65
|
+
], BetterAuthSessionModel.prototype, "user", void 0);
|
|
66
|
+
exports.BetterAuthSessionModel = BetterAuthSessionModel = __decorate([
|
|
67
|
+
(0, graphql_1.ObjectType)({ description: 'Better-Auth Session' }),
|
|
68
|
+
(0, restricted_decorator_1.Restricted)(role_enum_1.RoleEnum.S_USER)
|
|
69
|
+
], BetterAuthSessionModel);
|
|
70
|
+
let BetterAuthSessionInfoModel = class BetterAuthSessionInfoModel {
|
|
71
|
+
};
|
|
72
|
+
exports.BetterAuthSessionInfoModel = BetterAuthSessionInfoModel;
|
|
73
|
+
__decorate([
|
|
74
|
+
(0, graphql_1.Field)(() => String, { description: 'Session ID', nullable: true }),
|
|
75
|
+
__metadata("design:type", String)
|
|
76
|
+
], BetterAuthSessionInfoModel.prototype, "id", void 0);
|
|
77
|
+
__decorate([
|
|
78
|
+
(0, graphql_1.Field)(() => String, { description: 'Session token', nullable: true }),
|
|
79
|
+
__metadata("design:type", String)
|
|
80
|
+
], BetterAuthSessionInfoModel.prototype, "token", void 0);
|
|
81
|
+
__decorate([
|
|
82
|
+
(0, graphql_1.Field)(() => Date, { description: 'Session expiration date', nullable: true }),
|
|
83
|
+
__metadata("design:type", Date)
|
|
84
|
+
], BetterAuthSessionInfoModel.prototype, "expiresAt", void 0);
|
|
85
|
+
exports.BetterAuthSessionInfoModel = BetterAuthSessionInfoModel = __decorate([
|
|
86
|
+
(0, graphql_1.ObjectType)({ description: 'Better-Auth Session Info' }),
|
|
87
|
+
(0, restricted_decorator_1.Restricted)(role_enum_1.RoleEnum.S_EVERYONE)
|
|
88
|
+
], BetterAuthSessionInfoModel);
|
|
89
|
+
let BetterAuth2FASetupModel = class BetterAuth2FASetupModel {
|
|
90
|
+
};
|
|
91
|
+
exports.BetterAuth2FASetupModel = BetterAuth2FASetupModel;
|
|
92
|
+
__decorate([
|
|
93
|
+
(0, graphql_1.Field)(() => Boolean, { description: 'Whether the operation was successful' }),
|
|
94
|
+
__metadata("design:type", Boolean)
|
|
95
|
+
], BetterAuth2FASetupModel.prototype, "success", void 0);
|
|
96
|
+
__decorate([
|
|
97
|
+
(0, graphql_1.Field)(() => String, { description: 'TOTP URI for QR code generation', nullable: true }),
|
|
98
|
+
__metadata("design:type", String)
|
|
99
|
+
], BetterAuth2FASetupModel.prototype, "totpUri", void 0);
|
|
100
|
+
__decorate([
|
|
101
|
+
(0, graphql_1.Field)(() => [String], { description: 'Backup codes for account recovery', nullable: true }),
|
|
102
|
+
__metadata("design:type", Array)
|
|
103
|
+
], BetterAuth2FASetupModel.prototype, "backupCodes", void 0);
|
|
104
|
+
__decorate([
|
|
105
|
+
(0, graphql_1.Field)(() => String, { description: 'Error message if failed', nullable: true }),
|
|
106
|
+
__metadata("design:type", String)
|
|
107
|
+
], BetterAuth2FASetupModel.prototype, "error", void 0);
|
|
108
|
+
exports.BetterAuth2FASetupModel = BetterAuth2FASetupModel = __decorate([
|
|
109
|
+
(0, graphql_1.ObjectType)({ description: 'Better-Auth 2FA setup data' }),
|
|
110
|
+
(0, restricted_decorator_1.Restricted)(role_enum_1.RoleEnum.S_USER)
|
|
111
|
+
], BetterAuth2FASetupModel);
|
|
112
|
+
let BetterAuthPasskeyModel = class BetterAuthPasskeyModel {
|
|
113
|
+
};
|
|
114
|
+
exports.BetterAuthPasskeyModel = BetterAuthPasskeyModel;
|
|
115
|
+
__decorate([
|
|
116
|
+
(0, graphql_1.Field)(() => String, { description: 'Passkey ID' }),
|
|
117
|
+
__metadata("design:type", String)
|
|
118
|
+
], BetterAuthPasskeyModel.prototype, "id", void 0);
|
|
119
|
+
__decorate([
|
|
120
|
+
(0, graphql_1.Field)(() => String, { description: 'Passkey name', nullable: true }),
|
|
121
|
+
__metadata("design:type", String)
|
|
122
|
+
], BetterAuthPasskeyModel.prototype, "name", void 0);
|
|
123
|
+
__decorate([
|
|
124
|
+
(0, graphql_1.Field)(() => String, { description: 'Credential ID' }),
|
|
125
|
+
__metadata("design:type", String)
|
|
126
|
+
], BetterAuthPasskeyModel.prototype, "credentialId", void 0);
|
|
127
|
+
__decorate([
|
|
128
|
+
(0, graphql_1.Field)(() => Date, { description: 'Creation date' }),
|
|
129
|
+
__metadata("design:type", Date)
|
|
130
|
+
], BetterAuthPasskeyModel.prototype, "createdAt", void 0);
|
|
131
|
+
exports.BetterAuthPasskeyModel = BetterAuthPasskeyModel = __decorate([
|
|
132
|
+
(0, graphql_1.ObjectType)({ description: 'Better-Auth Passkey' }),
|
|
133
|
+
(0, restricted_decorator_1.Restricted)(role_enum_1.RoleEnum.S_USER)
|
|
134
|
+
], BetterAuthPasskeyModel);
|
|
135
|
+
let BetterAuthPasskeyChallengeModel = class BetterAuthPasskeyChallengeModel {
|
|
136
|
+
};
|
|
137
|
+
exports.BetterAuthPasskeyChallengeModel = BetterAuthPasskeyChallengeModel;
|
|
138
|
+
__decorate([
|
|
139
|
+
(0, graphql_1.Field)(() => Boolean, { description: 'Whether the operation was successful' }),
|
|
140
|
+
__metadata("design:type", Boolean)
|
|
141
|
+
], BetterAuthPasskeyChallengeModel.prototype, "success", void 0);
|
|
142
|
+
__decorate([
|
|
143
|
+
(0, graphql_1.Field)(() => String, { description: 'Challenge data (JSON)', nullable: true }),
|
|
144
|
+
__metadata("design:type", String)
|
|
145
|
+
], BetterAuthPasskeyChallengeModel.prototype, "challenge", void 0);
|
|
146
|
+
__decorate([
|
|
147
|
+
(0, graphql_1.Field)(() => String, { description: 'Error message if failed', nullable: true }),
|
|
148
|
+
__metadata("design:type", String)
|
|
149
|
+
], BetterAuthPasskeyChallengeModel.prototype, "error", void 0);
|
|
150
|
+
exports.BetterAuthPasskeyChallengeModel = BetterAuthPasskeyChallengeModel = __decorate([
|
|
151
|
+
(0, graphql_1.ObjectType)({ description: 'Better-Auth Passkey registration challenge' }),
|
|
152
|
+
(0, restricted_decorator_1.Restricted)(role_enum_1.RoleEnum.S_USER)
|
|
153
|
+
], BetterAuthPasskeyChallengeModel);
|
|
154
|
+
let BetterAuthFeaturesModel = class BetterAuthFeaturesModel {
|
|
155
|
+
};
|
|
156
|
+
exports.BetterAuthFeaturesModel = BetterAuthFeaturesModel;
|
|
157
|
+
__decorate([
|
|
158
|
+
(0, graphql_1.Field)(() => Boolean, { description: 'Whether Better-Auth is enabled' }),
|
|
159
|
+
__metadata("design:type", Boolean)
|
|
160
|
+
], BetterAuthFeaturesModel.prototype, "enabled", void 0);
|
|
161
|
+
__decorate([
|
|
162
|
+
(0, graphql_1.Field)(() => Boolean, { description: 'Whether JWT plugin is enabled' }),
|
|
163
|
+
__metadata("design:type", Boolean)
|
|
164
|
+
], BetterAuthFeaturesModel.prototype, "jwt", void 0);
|
|
165
|
+
__decorate([
|
|
166
|
+
(0, graphql_1.Field)(() => Boolean, { description: 'Whether 2FA is enabled' }),
|
|
167
|
+
__metadata("design:type", Boolean)
|
|
168
|
+
], BetterAuthFeaturesModel.prototype, "twoFactor", void 0);
|
|
169
|
+
__decorate([
|
|
170
|
+
(0, graphql_1.Field)(() => Boolean, { description: 'Whether Passkey is enabled' }),
|
|
171
|
+
__metadata("design:type", Boolean)
|
|
172
|
+
], BetterAuthFeaturesModel.prototype, "passkey", void 0);
|
|
173
|
+
__decorate([
|
|
174
|
+
(0, graphql_1.Field)(() => Boolean, { description: 'Whether legacy password handling is enabled' }),
|
|
175
|
+
__metadata("design:type", Boolean)
|
|
176
|
+
], BetterAuthFeaturesModel.prototype, "legacyPassword", void 0);
|
|
177
|
+
__decorate([
|
|
178
|
+
(0, graphql_1.Field)(() => [String], { description: 'List of enabled social providers' }),
|
|
179
|
+
__metadata("design:type", Array)
|
|
180
|
+
], BetterAuthFeaturesModel.prototype, "socialProviders", void 0);
|
|
181
|
+
exports.BetterAuthFeaturesModel = BetterAuthFeaturesModel = __decorate([
|
|
182
|
+
(0, graphql_1.ObjectType)({ description: 'Better-Auth features status' }),
|
|
183
|
+
(0, restricted_decorator_1.Restricted)(role_enum_1.RoleEnum.S_EVERYONE)
|
|
184
|
+
], BetterAuthFeaturesModel);
|
|
185
|
+
//# sourceMappingURL=better-auth-models.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"better-auth-models.js","sourceRoot":"","sources":["../../../../src/core/modules/better-auth/better-auth-models.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { NestMiddleware } from '@nestjs/common';
|
|
2
|
+
import { NextFunction, Request, Response } from 'express';
|
|
3
|
+
import { BetterAuthRateLimiter } from './better-auth-rate-limiter.service';
|
|
4
|
+
import { BetterAuthService } from './better-auth.service';
|
|
5
|
+
export declare class BetterAuthRateLimitMiddleware implements NestMiddleware {
|
|
6
|
+
private readonly rateLimiter;
|
|
7
|
+
private readonly betterAuthService;
|
|
8
|
+
constructor(rateLimiter: BetterAuthRateLimiter, betterAuthService: BetterAuthService);
|
|
9
|
+
use(req: Request, res: Response, next: NextFunction): void;
|
|
10
|
+
private getClientIp;
|
|
11
|
+
private addRateLimitHeaders;
|
|
12
|
+
}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
|
3
|
+
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
4
|
+
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
5
|
+
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
|
6
|
+
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
7
|
+
};
|
|
8
|
+
var __metadata = (this && this.__metadata) || function (k, v) {
|
|
9
|
+
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
|
10
|
+
};
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.BetterAuthRateLimitMiddleware = void 0;
|
|
13
|
+
const common_1 = require("@nestjs/common");
|
|
14
|
+
const better_auth_rate_limiter_service_1 = require("./better-auth-rate-limiter.service");
|
|
15
|
+
const better_auth_service_1 = require("./better-auth.service");
|
|
16
|
+
let BetterAuthRateLimitMiddleware = class BetterAuthRateLimitMiddleware {
|
|
17
|
+
constructor(rateLimiter, betterAuthService) {
|
|
18
|
+
this.rateLimiter = rateLimiter;
|
|
19
|
+
this.betterAuthService = betterAuthService;
|
|
20
|
+
}
|
|
21
|
+
use(req, res, next) {
|
|
22
|
+
if (!this.betterAuthService.isEnabled()) {
|
|
23
|
+
return next();
|
|
24
|
+
}
|
|
25
|
+
if (!this.rateLimiter.isEnabled()) {
|
|
26
|
+
return next();
|
|
27
|
+
}
|
|
28
|
+
const ip = this.getClientIp(req);
|
|
29
|
+
const basePath = this.betterAuthService.getBasePath();
|
|
30
|
+
const path = req.path.startsWith(basePath) ? req.path.substring(basePath.length) : req.path;
|
|
31
|
+
const result = this.rateLimiter.check(ip, path);
|
|
32
|
+
this.addRateLimitHeaders(res, result);
|
|
33
|
+
if (!result.allowed) {
|
|
34
|
+
throw new common_1.HttpException({
|
|
35
|
+
error: 'Too Many Requests',
|
|
36
|
+
message: this.rateLimiter.getMessage(),
|
|
37
|
+
retryAfter: result.resetIn,
|
|
38
|
+
statusCode: common_1.HttpStatus.TOO_MANY_REQUESTS,
|
|
39
|
+
}, common_1.HttpStatus.TOO_MANY_REQUESTS);
|
|
40
|
+
}
|
|
41
|
+
next();
|
|
42
|
+
}
|
|
43
|
+
getClientIp(req) {
|
|
44
|
+
const forwardedFor = req.headers['x-forwarded-for'];
|
|
45
|
+
if (forwardedFor) {
|
|
46
|
+
const ips = Array.isArray(forwardedFor) ? forwardedFor[0] : forwardedFor.split(',')[0];
|
|
47
|
+
return ips.trim();
|
|
48
|
+
}
|
|
49
|
+
const realIp = req.headers['x-real-ip'];
|
|
50
|
+
if (realIp) {
|
|
51
|
+
return Array.isArray(realIp) ? realIp[0] : realIp;
|
|
52
|
+
}
|
|
53
|
+
return req.ip || req.socket?.remoteAddress || 'unknown';
|
|
54
|
+
}
|
|
55
|
+
addRateLimitHeaders(res, result) {
|
|
56
|
+
res.setHeader('X-RateLimit-Limit', result.limit.toString());
|
|
57
|
+
res.setHeader('X-RateLimit-Remaining', result.remaining.toString());
|
|
58
|
+
res.setHeader('X-RateLimit-Reset', result.resetIn.toString());
|
|
59
|
+
if (!result.allowed) {
|
|
60
|
+
res.setHeader('Retry-After', result.resetIn.toString());
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
};
|
|
64
|
+
exports.BetterAuthRateLimitMiddleware = BetterAuthRateLimitMiddleware;
|
|
65
|
+
exports.BetterAuthRateLimitMiddleware = BetterAuthRateLimitMiddleware = __decorate([
|
|
66
|
+
(0, common_1.Injectable)(),
|
|
67
|
+
__metadata("design:paramtypes", [better_auth_rate_limiter_service_1.BetterAuthRateLimiter,
|
|
68
|
+
better_auth_service_1.BetterAuthService])
|
|
69
|
+
], BetterAuthRateLimitMiddleware);
|
|
70
|
+
//# sourceMappingURL=better-auth-rate-limit.middleware.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"better-auth-rate-limit.middleware.js","sourceRoot":"","sources":["../../../../src/core/modules/better-auth/better-auth-rate-limit.middleware.ts"],"names":[],"mappings":";;;;;;;;;;;;AAAA,2CAAuF;AAGvF,yFAA4F;AAC5F,+DAA0D;AA4BnD,IAAM,6BAA6B,GAAnC,MAAM,6BAA6B;IACxC,YACmB,WAAkC,EAClC,iBAAoC;QADpC,gBAAW,GAAX,WAAW,CAAuB;QAClC,sBAAiB,GAAjB,iBAAiB,CAAmB;IACpD,CAAC;IAEJ,GAAG,CAAC,GAAY,EAAE,GAAa,EAAE,IAAkB;QAEjD,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,SAAS,EAAE,EAAE,CAAC;YACxC,OAAO,IAAI,EAAE,CAAC;QAChB,CAAC;QAGD,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,EAAE,CAAC;YAClC,OAAO,IAAI,EAAE,CAAC;QAChB,CAAC;QAGD,MAAM,EAAE,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;QAGjC,MAAM,QAAQ,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,EAAE,CAAC;QACtD,MAAM,IAAI,GAAG,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC;QAG5F,MAAM,MAAM,GAAG,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC;QAGhD,IAAI,CAAC,mBAAmB,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;QAGtC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpB,MAAM,IAAI,sBAAa,CACrB;gBACE,KAAK,EAAE,mBAAmB;gBAC1B,OAAO,EAAE,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE;gBACtC,UAAU,EAAE,MAAM,CAAC,OAAO;gBAC1B,UAAU,EAAE,mBAAU,CAAC,iBAAiB;aACzC,EACD,mBAAU,CAAC,iBAAiB,CAC7B,CAAC;QACJ,CAAC;QAED,IAAI,EAAE,CAAC;IACT,CAAC;IAMO,WAAW,CAAC,GAAY;QAE9B,MAAM,YAAY,GAAG,GAAG,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC;QACpD,IAAI,YAAY,EAAE,CAAC;YACjB,MAAM,GAAG,GAAG,KAAK,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;YACvF,OAAO,GAAG,CAAC,IAAI,EAAE,CAAC;QACpB,CAAC;QAGD,MAAM,MAAM,GAAG,GAAG,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;QACxC,IAAI,MAAM,EAAE,CAAC;YACX,OAAO,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;QACpD,CAAC;QAGD,OAAO,GAAG,CAAC,EAAE,IAAI,GAAG,CAAC,MAAM,EAAE,aAAa,IAAI,SAAS,CAAC;IAC1D,CAAC;IAKO,mBAAmB,CAAC,GAAa,EAAE,MAAuB;QAChE,GAAG,CAAC,SAAS,CAAC,mBAAmB,EAAE,MAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC5D,GAAG,CAAC,SAAS,CAAC,uBAAuB,EAAE,MAAM,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC,CAAC;QACpE,GAAG,CAAC,SAAS,CAAC,mBAAmB,EAAE,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC;QAE9D,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpB,GAAG,CAAC,SAAS,CAAC,aAAa,EAAE,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC1D,CAAC;IACH,CAAC;CACF,CAAA;AAhFY,sEAA6B;wCAA7B,6BAA6B;IADzC,IAAA,mBAAU,GAAE;qCAGqB,wDAAqB;QACf,uCAAiB;GAH5C,6BAA6B,CAgFzC"}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { IBetterAuthRateLimit } from '../../common/interfaces/server-options.interface';
|
|
2
|
+
export interface RateLimitResult {
|
|
3
|
+
allowed: boolean;
|
|
4
|
+
current: number;
|
|
5
|
+
limit: number;
|
|
6
|
+
remaining: number;
|
|
7
|
+
resetIn: number;
|
|
8
|
+
}
|
|
9
|
+
export declare class BetterAuthRateLimiter {
|
|
10
|
+
private readonly logger;
|
|
11
|
+
private readonly store;
|
|
12
|
+
private config;
|
|
13
|
+
private cleanupInterval;
|
|
14
|
+
constructor();
|
|
15
|
+
configure(config: IBetterAuthRateLimit | undefined): void;
|
|
16
|
+
check(ip: string, path: string): RateLimitResult;
|
|
17
|
+
getMessage(): string;
|
|
18
|
+
isEnabled(): boolean;
|
|
19
|
+
reset(ip: string): void;
|
|
20
|
+
clear(): void;
|
|
21
|
+
getStats(): {
|
|
22
|
+
activeEntries: number;
|
|
23
|
+
enabled: boolean;
|
|
24
|
+
};
|
|
25
|
+
onModuleDestroy(): void;
|
|
26
|
+
private shouldSkip;
|
|
27
|
+
private getLimit;
|
|
28
|
+
private getKey;
|
|
29
|
+
private normalizeEndpoint;
|
|
30
|
+
private maskIp;
|
|
31
|
+
private startCleanup;
|
|
32
|
+
}
|
|
@@ -0,0 +1,173 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
|
3
|
+
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
4
|
+
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
5
|
+
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
|
6
|
+
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
7
|
+
};
|
|
8
|
+
var __metadata = (this && this.__metadata) || function (k, v) {
|
|
9
|
+
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
|
10
|
+
};
|
|
11
|
+
var BetterAuthRateLimiter_1;
|
|
12
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
13
|
+
exports.BetterAuthRateLimiter = void 0;
|
|
14
|
+
const common_1 = require("@nestjs/common");
|
|
15
|
+
const DEFAULT_CONFIG = {
|
|
16
|
+
enabled: false,
|
|
17
|
+
max: 10,
|
|
18
|
+
message: 'Too many requests, please try again later.',
|
|
19
|
+
skipEndpoints: ['/session', '/callback'],
|
|
20
|
+
strictEndpoints: ['/sign-in', '/sign-up', '/forgot-password', '/reset-password'],
|
|
21
|
+
windowSeconds: 60,
|
|
22
|
+
};
|
|
23
|
+
let BetterAuthRateLimiter = BetterAuthRateLimiter_1 = class BetterAuthRateLimiter {
|
|
24
|
+
constructor() {
|
|
25
|
+
this.logger = new common_1.Logger(BetterAuthRateLimiter_1.name);
|
|
26
|
+
this.store = new Map();
|
|
27
|
+
this.config = DEFAULT_CONFIG;
|
|
28
|
+
this.cleanupInterval = null;
|
|
29
|
+
this.startCleanup();
|
|
30
|
+
}
|
|
31
|
+
configure(config) {
|
|
32
|
+
this.config = {
|
|
33
|
+
...DEFAULT_CONFIG,
|
|
34
|
+
...config,
|
|
35
|
+
skipEndpoints: config?.skipEndpoints ?? DEFAULT_CONFIG.skipEndpoints,
|
|
36
|
+
strictEndpoints: config?.strictEndpoints ?? DEFAULT_CONFIG.strictEndpoints,
|
|
37
|
+
};
|
|
38
|
+
if (this.config.enabled) {
|
|
39
|
+
this.logger.log(`Rate limiting enabled: ${this.config.max} requests per ${this.config.windowSeconds}s`);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
check(ip, path) {
|
|
43
|
+
if (!this.config.enabled) {
|
|
44
|
+
return {
|
|
45
|
+
allowed: true,
|
|
46
|
+
current: 0,
|
|
47
|
+
limit: Infinity,
|
|
48
|
+
remaining: Infinity,
|
|
49
|
+
resetIn: 0,
|
|
50
|
+
};
|
|
51
|
+
}
|
|
52
|
+
if (this.shouldSkip(path)) {
|
|
53
|
+
return {
|
|
54
|
+
allowed: true,
|
|
55
|
+
current: 0,
|
|
56
|
+
limit: Infinity,
|
|
57
|
+
remaining: Infinity,
|
|
58
|
+
resetIn: 0,
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
const limit = this.getLimit(path);
|
|
62
|
+
const key = this.getKey(ip, path);
|
|
63
|
+
const now = Date.now();
|
|
64
|
+
let entry = this.store.get(key);
|
|
65
|
+
if (!entry || now >= entry.resetTime) {
|
|
66
|
+
entry = {
|
|
67
|
+
count: 1,
|
|
68
|
+
resetTime: now + this.config.windowSeconds * 1000,
|
|
69
|
+
};
|
|
70
|
+
this.store.set(key, entry);
|
|
71
|
+
return {
|
|
72
|
+
allowed: true,
|
|
73
|
+
current: 1,
|
|
74
|
+
limit,
|
|
75
|
+
remaining: limit - 1,
|
|
76
|
+
resetIn: this.config.windowSeconds,
|
|
77
|
+
};
|
|
78
|
+
}
|
|
79
|
+
entry.count++;
|
|
80
|
+
const resetIn = Math.ceil((entry.resetTime - now) / 1000);
|
|
81
|
+
const allowed = entry.count <= limit;
|
|
82
|
+
const remaining = Math.max(0, limit - entry.count);
|
|
83
|
+
if (!allowed) {
|
|
84
|
+
this.logger.warn(`Rate limit exceeded for IP ${this.maskIp(ip)} on ${path}: ${entry.count}/${limit}`);
|
|
85
|
+
}
|
|
86
|
+
return {
|
|
87
|
+
allowed,
|
|
88
|
+
current: entry.count,
|
|
89
|
+
limit,
|
|
90
|
+
remaining,
|
|
91
|
+
resetIn,
|
|
92
|
+
};
|
|
93
|
+
}
|
|
94
|
+
getMessage() {
|
|
95
|
+
return this.config.message;
|
|
96
|
+
}
|
|
97
|
+
isEnabled() {
|
|
98
|
+
return this.config.enabled;
|
|
99
|
+
}
|
|
100
|
+
reset(ip) {
|
|
101
|
+
for (const key of this.store.keys()) {
|
|
102
|
+
if (key.startsWith(`${ip}:`)) {
|
|
103
|
+
this.store.delete(key);
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
clear() {
|
|
108
|
+
this.store.clear();
|
|
109
|
+
}
|
|
110
|
+
getStats() {
|
|
111
|
+
return {
|
|
112
|
+
activeEntries: this.store.size,
|
|
113
|
+
enabled: this.config.enabled,
|
|
114
|
+
};
|
|
115
|
+
}
|
|
116
|
+
onModuleDestroy() {
|
|
117
|
+
if (this.cleanupInterval) {
|
|
118
|
+
clearInterval(this.cleanupInterval);
|
|
119
|
+
this.cleanupInterval = null;
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
shouldSkip(path) {
|
|
123
|
+
return this.config.skipEndpoints.some((skip) => path === skip || path.endsWith(skip) || path.includes(skip));
|
|
124
|
+
}
|
|
125
|
+
getLimit(path) {
|
|
126
|
+
const isStrict = this.config.strictEndpoints.some((strict) => path === strict || path.endsWith(strict) || path.includes(strict));
|
|
127
|
+
return isStrict ? Math.ceil(this.config.max / 2) : this.config.max;
|
|
128
|
+
}
|
|
129
|
+
getKey(ip, path) {
|
|
130
|
+
const endpoint = this.normalizeEndpoint(path);
|
|
131
|
+
return `${ip}:${endpoint}`;
|
|
132
|
+
}
|
|
133
|
+
normalizeEndpoint(path) {
|
|
134
|
+
const cleanPath = path.split('?')[0];
|
|
135
|
+
if (cleanPath.includes('/callback/')) {
|
|
136
|
+
return 'callback';
|
|
137
|
+
}
|
|
138
|
+
const segments = cleanPath.split('/').filter(Boolean);
|
|
139
|
+
return segments[segments.length - 1] || 'root';
|
|
140
|
+
}
|
|
141
|
+
maskIp(ip) {
|
|
142
|
+
if (ip.includes('.')) {
|
|
143
|
+
const parts = ip.split('.');
|
|
144
|
+
return `${parts[0]}.${parts[1]}.*.*`;
|
|
145
|
+
}
|
|
146
|
+
const parts = ip.split(':');
|
|
147
|
+
return `${parts[0]}:****`;
|
|
148
|
+
}
|
|
149
|
+
startCleanup() {
|
|
150
|
+
this.cleanupInterval = setInterval(() => {
|
|
151
|
+
const now = Date.now();
|
|
152
|
+
let cleaned = 0;
|
|
153
|
+
for (const [key, entry] of this.store.entries()) {
|
|
154
|
+
if (now >= entry.resetTime) {
|
|
155
|
+
this.store.delete(key);
|
|
156
|
+
cleaned++;
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
if (cleaned > 0) {
|
|
160
|
+
this.logger.debug(`Cleaned up ${cleaned} expired rate limit entries`);
|
|
161
|
+
}
|
|
162
|
+
}, 5 * 60 * 1000);
|
|
163
|
+
if (this.cleanupInterval.unref) {
|
|
164
|
+
this.cleanupInterval.unref();
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
};
|
|
168
|
+
exports.BetterAuthRateLimiter = BetterAuthRateLimiter;
|
|
169
|
+
exports.BetterAuthRateLimiter = BetterAuthRateLimiter = BetterAuthRateLimiter_1 = __decorate([
|
|
170
|
+
(0, common_1.Injectable)(),
|
|
171
|
+
__metadata("design:paramtypes", [])
|
|
172
|
+
], BetterAuthRateLimiter);
|
|
173
|
+
//# sourceMappingURL=better-auth-rate-limiter.service.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"better-auth-rate-limiter.service.js","sourceRoot":"","sources":["../../../../src/core/modules/better-auth/better-auth-rate-limiter.service.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { Connection } from 'mongoose';
|
|
2
|
+
export interface BetterAuthSessionUser {
|
|
3
|
+
createdAt?: Date;
|
|
4
|
+
email: string;
|
|
5
|
+
emailVerified?: boolean;
|
|
6
|
+
id: string;
|
|
7
|
+
image?: string;
|
|
8
|
+
name?: string;
|
|
9
|
+
updatedAt?: Date;
|
|
10
|
+
}
|
|
11
|
+
export interface MappedUser {
|
|
12
|
+
_authenticatedViaBetterAuth: true;
|
|
13
|
+
email: string;
|
|
14
|
+
emailVerified?: boolean;
|
|
15
|
+
hasRole: (roles: string | string[]) => boolean;
|
|
16
|
+
iamId: string;
|
|
17
|
+
id: string;
|
|
18
|
+
image?: string;
|
|
19
|
+
name?: string;
|
|
20
|
+
roles: string[];
|
|
21
|
+
verified?: boolean;
|
|
22
|
+
}
|
|
23
|
+
export interface SyncedUserDocument {
|
|
24
|
+
_id: any;
|
|
25
|
+
avatar?: string;
|
|
26
|
+
createdAt: Date;
|
|
27
|
+
email: string;
|
|
28
|
+
firstName?: string;
|
|
29
|
+
iamId: string;
|
|
30
|
+
lastName?: string;
|
|
31
|
+
password?: string;
|
|
32
|
+
roles: string[];
|
|
33
|
+
updatedAt: Date;
|
|
34
|
+
verified?: boolean;
|
|
35
|
+
}
|
|
36
|
+
export declare class BetterAuthUserMapper {
|
|
37
|
+
private readonly connection?;
|
|
38
|
+
private readonly logger;
|
|
39
|
+
constructor(connection?: Connection);
|
|
40
|
+
mapSessionUser(sessionUser: BetterAuthSessionUser): Promise<MappedUser | null>;
|
|
41
|
+
private createMappedUser;
|
|
42
|
+
linkOrCreateUser(sessionUser: BetterAuthSessionUser, additionalData?: Record<string, any>): Promise<null | SyncedUserDocument>;
|
|
43
|
+
}
|