@comasoft/nestjs 0.0.97 → 0.1.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/README.md +31 -0
- package/dist/aws/aws-s3.service.d.ts +17 -0
- package/dist/aws/aws-s3.service.js +158 -0
- package/dist/aws/aws.module.d.ts +2 -0
- package/dist/aws/aws.module.js +25 -0
- package/dist/aws/aws.service.d.ts +3 -0
- package/dist/aws/aws.service.js +21 -0
- package/dist/aws/constants/storage.constant.d.ts +4 -0
- package/dist/aws/constants/storage.constant.js +25 -0
- package/dist/aws/dto/create-s3-presigned.dto.d.ts +12 -0
- package/dist/aws/dto/create-s3-presigned.dto.js +78 -0
- package/dist/aws/dto/index.d.ts +2 -0
- package/dist/aws/dto/index.js +18 -0
- package/dist/aws/dto/response-s3-presigned.dto.d.ts +30 -0
- package/dist/aws/dto/response-s3-presigned.dto.js +110 -0
- package/dist/aws/index.d.ts +4 -0
- package/dist/aws/index.js +20 -0
- package/dist/aws/types/s3.interface.d.ts +14 -0
- package/dist/aws/types/s3.interface.js +2 -0
- package/dist/aws/types/storage.interface.d.ts +19 -0
- package/dist/aws/types/storage.interface.js +2 -0
- package/dist/category/category.module.d.ts +2 -0
- package/dist/category/category.module.js +23 -0
- package/dist/category/category.service.d.ts +13 -0
- package/dist/category/category.service.js +124 -0
- package/dist/category/dto/create-category.dto.d.ts +7 -0
- package/dist/category/dto/create-category.dto.js +34 -0
- package/dist/category/dto/delete-category.dto.d.ts +3 -0
- package/dist/category/dto/delete-category.dto.js +6 -0
- package/dist/category/dto/index.d.ts +5 -0
- package/dist/category/dto/index.js +21 -0
- package/dist/category/dto/response-categories.dto.d.ts +12 -0
- package/dist/category/dto/response-categories.dto.js +52 -0
- package/dist/category/dto/update-category-sort.dto.d.ts +9 -0
- package/dist/category/dto/update-category-sort.dto.js +33 -0
- package/dist/category/dto/update-category.dto.d.ts +6 -0
- package/dist/category/dto/update-category.dto.js +9 -0
- package/dist/category/index.d.ts +3 -0
- package/dist/category/index.js +19 -0
- package/dist/comment/comment.module.d.ts +2 -0
- package/dist/comment/comment.module.js +23 -0
- package/dist/comment/comment.service.d.ts +33 -0
- package/dist/comment/comment.service.js +240 -0
- package/dist/comment/dto/create-comment.dto.d.ts +7 -0
- package/dist/comment/dto/create-comment.dto.js +41 -0
- package/dist/comment/dto/index.d.ts +3 -0
- package/dist/comment/dto/index.js +19 -0
- package/dist/comment/dto/response-comment.dto.d.ts +26 -0
- package/dist/comment/dto/response-comment.dto.js +97 -0
- package/dist/comment/dto/update-comment.dto.d.ts +4 -0
- package/dist/comment/dto/update-comment.dto.js +22 -0
- package/dist/comment/index.d.ts +3 -0
- package/dist/comment/index.js +19 -0
- package/dist/database/entities/category.entity.d.ts +14 -0
- package/dist/database/entities/category.entity.js +70 -0
- package/dist/database/entities/comment.entity.d.ts +16 -0
- package/dist/database/entities/comment.entity.js +77 -0
- package/dist/database/entities/config.entity.d.ts +8 -0
- package/dist/database/entities/config.entity.js +44 -0
- package/dist/database/entities/file.entity.d.ts +20 -0
- package/dist/database/entities/file.entity.js +91 -0
- package/dist/database/entities/index.d.ts +10 -0
- package/dist/database/entities/index.js +26 -0
- package/dist/database/entities/like.entity.d.ts +10 -0
- package/dist/database/entities/like.entity.js +49 -0
- package/dist/database/entities/logs-audit.entity.d.ts +16 -0
- package/dist/database/entities/logs-audit.entity.js +72 -0
- package/dist/database/entities/logs-event.entity.d.ts +16 -0
- package/dist/database/entities/logs-event.entity.js +72 -0
- package/dist/database/entities/notification-recipient.entity.d.ts +19 -0
- package/dist/database/entities/notification-recipient.entity.js +74 -0
- package/dist/database/entities/notification.entity.d.ts +18 -0
- package/dist/database/entities/notification.entity.js +69 -0
- package/dist/database/entities/role.entity.d.ts +13 -0
- package/dist/database/entities/role.entity.js +54 -0
- package/dist/database/entities/roles.entity.d.ts +12 -0
- package/dist/database/entities/roles.entity.js +1 -0
- package/dist/decorators/auth.decorator.factory.d.ts +1 -1
- package/dist/decorators/auth.decorator.factory.js +23 -2
- package/dist/decorators/delete-columns.decorator.js +1 -33
- package/dist/decorators/patch-body.decorator.js +20 -12
- package/dist/decorators/soft-delete.decorator.js +1 -15
- package/dist/decorators/timestamps.decorator.js +5 -13
- package/dist/dto/file.dto.d.ts +18 -0
- package/dist/dto/file.dto.js +70 -0
- package/dist/dto/index.d.ts +2 -0
- package/dist/dto/index.js +2 -0
- package/dist/dto/response-uid.dto.d.ts +2 -0
- package/dist/dto/response-uid.dto.js +8 -0
- package/dist/dto/update-ids.dto.d.ts +6 -1
- package/dist/dto/update-ids.dto.js +24 -10
- package/dist/dto/update-sort.dto.d.ts +9 -0
- package/dist/dto/update-sort.dto.js +50 -0
- package/dist/enums.common.d.ts +23 -0
- package/dist/enums.common.js +28 -1
- package/dist/events/event-bus.module.d.ts +2 -0
- package/dist/events/event-bus.module.js +44 -0
- package/dist/events/index.d.ts +4 -0
- package/dist/events/index.js +20 -0
- package/dist/events/listeners/audit-event.listener.d.ts +9 -0
- package/dist/events/listeners/audit-event.listener.js +58 -0
- package/dist/events/listeners/index.d.ts +1 -0
- package/dist/events/listeners/index.js +17 -0
- package/dist/events/listeners/system-event.listener.d.ts +9 -0
- package/dist/events/listeners/system-event.listener.js +58 -0
- package/dist/events/services/event-logger.service.d.ts +5 -0
- package/dist/events/services/event-logger.service.js +34 -0
- package/dist/events/services/event-notification.service.d.ts +7 -0
- package/dist/events/services/event-notification.service.js +104 -0
- package/dist/events/services/event-storage.service.d.ts +11 -0
- package/dist/events/services/event-storage.service.js +79 -0
- package/dist/events/services/index.d.ts +4 -0
- package/dist/events/services/index.js +20 -0
- package/dist/events/services/token.service.d.ts +23 -0
- package/dist/events/services/token.service.js +88 -0
- package/dist/events/types/audit-events.type.d.ts +12 -0
- package/dist/events/types/audit-events.type.js +2 -0
- package/dist/events/types/index.d.ts +18 -0
- package/dist/events/types/index.js +17 -0
- package/dist/filters/http-exception.filter.d.ts +4 -0
- package/dist/filters/http-exception.filter.js +26 -4
- package/dist/guards/auth.guard.d.ts +3 -1
- package/dist/guards/auth.guard.js +33 -11
- package/dist/index.d.ts +20 -8
- package/dist/index.js +12 -0
- package/dist/library.module.d.ts +2 -0
- package/dist/library.module.js +20 -0
- package/dist/like/dto/create-like.dto.d.ts +7 -0
- package/dist/like/dto/create-like.dto.js +47 -0
- package/dist/like/dto/index.d.ts +2 -0
- package/dist/like/dto/index.js +18 -0
- package/dist/like/dto/response-like.dto.d.ts +19 -0
- package/dist/like/dto/response-like.dto.js +79 -0
- package/dist/like/index.d.ts +3 -0
- package/dist/like/index.js +19 -0
- package/dist/like/like.module.d.ts +2 -0
- package/dist/like/like.module.js +23 -0
- package/dist/like/like.service.d.ts +13 -0
- package/dist/like/like.service.js +155 -0
- package/dist/logger/index.d.ts +2 -0
- package/dist/logger/index.js +18 -0
- package/dist/logger/pino-options.d.ts +31 -0
- package/dist/logger/pino-options.js +77 -0
- package/dist/logger/pino-pretty-transport.d.ts +1 -0
- package/dist/logger/pino-pretty-transport.js +164 -0
- package/dist/modules/index.d.ts +1 -0
- package/dist/modules/index.js +1 -0
- package/dist/modules/redis.module.d.ts +9 -0
- package/dist/modules/redis.module.js +1 -0
- package/dist/notifications/index.d.ts +70 -0
- package/dist/notifications/index.js +18 -0
- package/dist/notifications/notification.module.d.ts +2 -0
- package/dist/notifications/notification.module.js +32 -0
- package/dist/notifications/services/email.service.d.ts +31 -0
- package/dist/notifications/services/email.service.js +69 -0
- package/dist/notifications/services/index.d.ts +7 -0
- package/dist/notifications/services/index.js +23 -0
- package/dist/notifications/services/kakao.service.d.ts +41 -0
- package/dist/notifications/services/kakao.service.js +94 -0
- package/dist/notifications/services/message-template.service.d.ts +15 -0
- package/dist/notifications/services/message-template.service.js +74 -0
- package/dist/notifications/services/notification-send.service.d.ts +30 -0
- package/dist/notifications/services/notification-send.service.js +146 -0
- package/dist/notifications/services/notification.service.d.ts +37 -0
- package/dist/notifications/services/notification.service.js +125 -0
- package/dist/notifications/services/slack.service.d.ts +11 -0
- package/dist/notifications/services/slack.service.js +53 -0
- package/dist/notifications/services/sms.service.d.ts +19 -0
- package/dist/notifications/services/sms.service.js +62 -0
- package/dist/redis/index.d.ts +2 -0
- package/dist/redis/index.js +18 -0
- package/dist/redis/redis-constraint.d.ts +1 -0
- package/dist/redis/redis-constraint.js +4 -0
- package/dist/redis/redis.module.d.ts +8 -0
- package/dist/redis/redis.module.js +62 -0
- package/dist/redis/redis.service.d.ts +12 -0
- package/dist/redis/redis.service.js +76 -0
- package/dist/role/dto/create-role.dto.d.ts +5 -0
- package/dist/role/dto/create-role.dto.js +41 -0
- package/dist/role/dto/index.d.ts +3 -0
- package/dist/role/dto/index.js +19 -0
- package/dist/role/dto/response-roles.dto.d.ts +15 -0
- package/dist/role/dto/response-roles.dto.js +62 -0
- package/dist/role/dto/update-role.dto.d.ts +6 -0
- package/dist/role/dto/update-role.dto.js +9 -0
- package/dist/role/index.d.ts +3 -0
- package/dist/role/index.js +19 -0
- package/dist/role/role.module.d.ts +2 -0
- package/dist/role/role.module.js +23 -0
- package/dist/role/role.service.d.ts +17 -0
- package/dist/role/role.service.js +144 -0
- package/dist/services/file.service.d.ts +55 -0
- package/dist/services/file.service.js +82 -0
- package/dist/services/index.d.ts +2 -0
- package/dist/services/index.js +18 -0
- package/dist/services/message-template.service.d.ts +15 -0
- package/dist/services/message-template.service.js +1 -0
- package/dist/services/permission-checker.service.d.ts +38 -0
- package/dist/services/permission-checker.service.js +65 -0
- package/dist/shared/error-codes.d.ts +5 -0
- package/dist/shared/error-codes.js +9 -0
- package/dist/utils/common.utils.d.ts +2 -0
- package/dist/utils/common.utils.js +11 -1
- package/dist/utils/index.d.ts +1 -0
- package/dist/utils/index.js +1 -0
- package/dist/utils/object-diff.d.ts +1 -0
- package/dist/utils/object-diff.js +33 -0
- package/package.json +61 -16
- package/eslint.config.js +0 -31
- package/tsconfig.build.json +0 -19
- package/tsconfig.json +0 -18
|
@@ -0,0 +1,155 @@
|
|
|
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 __param = (this && this.__param) || function (paramIndex, decorator) {
|
|
12
|
+
return function (target, key) { decorator(target, key, paramIndex); }
|
|
13
|
+
};
|
|
14
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
15
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
16
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
17
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
18
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
19
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
20
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
21
|
+
});
|
|
22
|
+
};
|
|
23
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
24
|
+
exports.LikeService = void 0;
|
|
25
|
+
const common_1 = require("@nestjs/common");
|
|
26
|
+
const typeorm_1 = require("@nestjs/typeorm");
|
|
27
|
+
const class_transformer_1 = require("class-transformer");
|
|
28
|
+
const typeorm_2 = require("typeorm");
|
|
29
|
+
const entities_1 = require("../database/entities");
|
|
30
|
+
const enums_common_1 = require("../enums.common");
|
|
31
|
+
const dto_1 = require("./dto");
|
|
32
|
+
let LikeService = class LikeService {
|
|
33
|
+
constructor(repository, dataSource) {
|
|
34
|
+
this.repository = repository;
|
|
35
|
+
this.dataSource = dataSource;
|
|
36
|
+
}
|
|
37
|
+
resolveTargetId(target, target_id, target_uid) {
|
|
38
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
39
|
+
if (target_id)
|
|
40
|
+
return target_id;
|
|
41
|
+
if (!target_uid) {
|
|
42
|
+
throw new common_1.BadRequestException('Either target_id or target_uid is required');
|
|
43
|
+
}
|
|
44
|
+
throw new common_1.BadRequestException('resolveTargetId must be implemented in app service');
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
validateTarget(target, target_id) {
|
|
48
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
49
|
+
throw new common_1.BadRequestException('validateTarget must be implemented in app service');
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
updateTargetLikeCount(manager, target, target_id, increment) {
|
|
53
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
54
|
+
throw new common_1.BadRequestException('updateTargetLikeCount must be implemented in app service');
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
toggleLike(user_id, _body) {
|
|
58
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
59
|
+
const { target, target_id, target_uid, type } = _body;
|
|
60
|
+
const resolved_target_id = yield this.resolveTargetId(target, target_id, target_uid);
|
|
61
|
+
yield this.validateTarget(target, resolved_target_id);
|
|
62
|
+
const finalLike = yield this.dataSource.transaction((manager) => __awaiter(this, void 0, void 0, function* () {
|
|
63
|
+
const existingLike = yield manager.findOne(entities_1.Like, {
|
|
64
|
+
where: {
|
|
65
|
+
target: target,
|
|
66
|
+
target_id: resolved_target_id,
|
|
67
|
+
user_id,
|
|
68
|
+
},
|
|
69
|
+
});
|
|
70
|
+
if (existingLike) {
|
|
71
|
+
if (existingLike.type === type) {
|
|
72
|
+
yield manager.update(entities_1.Like, existingLike.id, { type: enums_common_1.LIKE_TYPE.NONE });
|
|
73
|
+
const countChange = type === enums_common_1.LIKE_TYPE.LIKE ? -1 : 0;
|
|
74
|
+
if (countChange !== 0) {
|
|
75
|
+
yield this.updateTargetLikeCount(manager, target, resolved_target_id, countChange);
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
else {
|
|
79
|
+
yield manager.update(entities_1.Like, existingLike.id, { type });
|
|
80
|
+
const previousIncrement = existingLike.type === enums_common_1.LIKE_TYPE.LIKE ? -1 : 0;
|
|
81
|
+
const newIncrement = type === enums_common_1.LIKE_TYPE.LIKE ? 1 : 0;
|
|
82
|
+
const totalChange = previousIncrement + newIncrement;
|
|
83
|
+
if (totalChange !== 0) {
|
|
84
|
+
yield this.updateTargetLikeCount(manager, target, resolved_target_id, totalChange);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
else {
|
|
89
|
+
yield manager.save(entities_1.Like, {
|
|
90
|
+
target: target,
|
|
91
|
+
target_id: resolved_target_id,
|
|
92
|
+
user_id,
|
|
93
|
+
type,
|
|
94
|
+
});
|
|
95
|
+
const countChange = type === enums_common_1.LIKE_TYPE.LIKE ? 1 : 0;
|
|
96
|
+
if (countChange !== 0) {
|
|
97
|
+
yield this.updateTargetLikeCount(manager, target, resolved_target_id, countChange);
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
return yield manager.findOne(entities_1.Like, {
|
|
101
|
+
where: {
|
|
102
|
+
target: target,
|
|
103
|
+
target_id: resolved_target_id,
|
|
104
|
+
user_id,
|
|
105
|
+
},
|
|
106
|
+
});
|
|
107
|
+
}));
|
|
108
|
+
return finalLike ? (0, class_transformer_1.plainToClass)(dto_1.ResponseLikeBaseDto, finalLike) : null;
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
getStats(target, target_id, target_uid, user_id) {
|
|
112
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
113
|
+
const resolved_target_id = yield this.resolveTargetId(target, target_id, target_uid);
|
|
114
|
+
yield this.validateTarget(target, resolved_target_id);
|
|
115
|
+
const likeCount = yield this.repository.count({
|
|
116
|
+
where: {
|
|
117
|
+
target: target,
|
|
118
|
+
target_id: resolved_target_id,
|
|
119
|
+
type: enums_common_1.LIKE_TYPE.LIKE,
|
|
120
|
+
},
|
|
121
|
+
});
|
|
122
|
+
const dislikeCount = yield this.repository.count({
|
|
123
|
+
where: {
|
|
124
|
+
target: target,
|
|
125
|
+
target_id: resolved_target_id,
|
|
126
|
+
type: enums_common_1.LIKE_TYPE.DISLIKE,
|
|
127
|
+
},
|
|
128
|
+
});
|
|
129
|
+
const stats = {
|
|
130
|
+
cnt_likes: likeCount,
|
|
131
|
+
cnt_dislikes: dislikeCount,
|
|
132
|
+
};
|
|
133
|
+
if (user_id) {
|
|
134
|
+
const userLike = yield this.repository.findOne({
|
|
135
|
+
where: {
|
|
136
|
+
target: target,
|
|
137
|
+
target_id: resolved_target_id,
|
|
138
|
+
user_id,
|
|
139
|
+
},
|
|
140
|
+
});
|
|
141
|
+
if (userLike && userLike.type !== enums_common_1.LIKE_TYPE.NONE) {
|
|
142
|
+
stats.user_like_type = userLike.type;
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
return (0, class_transformer_1.plainToClass)(dto_1.ResponseLikeStatsBaseDto, stats);
|
|
146
|
+
});
|
|
147
|
+
}
|
|
148
|
+
};
|
|
149
|
+
exports.LikeService = LikeService;
|
|
150
|
+
exports.LikeService = LikeService = __decorate([
|
|
151
|
+
(0, common_1.Injectable)(),
|
|
152
|
+
__param(0, (0, typeorm_1.InjectRepository)(entities_1.Like)),
|
|
153
|
+
__metadata("design:paramtypes", [typeorm_2.Repository,
|
|
154
|
+
typeorm_2.DataSource])
|
|
155
|
+
], LikeService);
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./pino-options"), exports);
|
|
18
|
+
__exportStar(require("./pino-pretty-transport"), exports);
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
export declare const COMMON_LOGGER_OPTIONS: {
|
|
2
|
+
customLogLevel: (req: any, res: any, err: any) => "warn" | "error" | "info";
|
|
3
|
+
customProps: (req: any, res: any) => {
|
|
4
|
+
method: any;
|
|
5
|
+
url: any;
|
|
6
|
+
ip: any;
|
|
7
|
+
statusCode: any;
|
|
8
|
+
browser: string;
|
|
9
|
+
os: string;
|
|
10
|
+
deviceType: string;
|
|
11
|
+
};
|
|
12
|
+
serializers: {
|
|
13
|
+
req: () => any;
|
|
14
|
+
res: () => any;
|
|
15
|
+
err(err: any): {
|
|
16
|
+
type: any;
|
|
17
|
+
message: any;
|
|
18
|
+
code: any;
|
|
19
|
+
stack: any;
|
|
20
|
+
} | {
|
|
21
|
+
type: any;
|
|
22
|
+
message: any;
|
|
23
|
+
code: any;
|
|
24
|
+
stack?: undefined;
|
|
25
|
+
};
|
|
26
|
+
};
|
|
27
|
+
redact: {
|
|
28
|
+
remove: boolean;
|
|
29
|
+
paths: string[];
|
|
30
|
+
};
|
|
31
|
+
};
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.COMMON_LOGGER_OPTIONS = void 0;
|
|
4
|
+
const ua_parser_js_1 = require("ua-parser-js");
|
|
5
|
+
exports.COMMON_LOGGER_OPTIONS = {
|
|
6
|
+
customLogLevel: (req, res, err) => {
|
|
7
|
+
if (res.statusCode >= 400 && res.statusCode < 500)
|
|
8
|
+
return 'warn';
|
|
9
|
+
if (res.statusCode >= 500 || err)
|
|
10
|
+
return 'error';
|
|
11
|
+
return 'info';
|
|
12
|
+
},
|
|
13
|
+
customProps: (req, res) => {
|
|
14
|
+
const uaParser = new ua_parser_js_1.UAParser(req.headers['user-agent']);
|
|
15
|
+
const browser = uaParser.getBrowser();
|
|
16
|
+
const os = uaParser.getOS();
|
|
17
|
+
const deviceType = uaParser.getDevice().type || 'desktop';
|
|
18
|
+
return {
|
|
19
|
+
method: req.method,
|
|
20
|
+
url: req.url,
|
|
21
|
+
ip: req.headers['x-forwarded-for'] || req.connection.remoteAddress,
|
|
22
|
+
statusCode: res.statusCode,
|
|
23
|
+
browser: browser.name || 'unknown',
|
|
24
|
+
os: os.name || 'unknown',
|
|
25
|
+
deviceType: deviceType || 'unknown',
|
|
26
|
+
};
|
|
27
|
+
},
|
|
28
|
+
serializers: {
|
|
29
|
+
req: () => undefined,
|
|
30
|
+
res: () => undefined,
|
|
31
|
+
err(err) {
|
|
32
|
+
var _a, _b;
|
|
33
|
+
if (process.env.NODE_ENV !== 'production') {
|
|
34
|
+
return {
|
|
35
|
+
type: err === null || err === void 0 ? void 0 : err.name,
|
|
36
|
+
message: err === null || err === void 0 ? void 0 : err.message,
|
|
37
|
+
code: (_a = err === null || err === void 0 ? void 0 : err.response) === null || _a === void 0 ? void 0 : _a.code,
|
|
38
|
+
stack: err === null || err === void 0 ? void 0 : err.stack,
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
else {
|
|
42
|
+
return {
|
|
43
|
+
type: err === null || err === void 0 ? void 0 : err.name,
|
|
44
|
+
message: err === null || err === void 0 ? void 0 : err.message,
|
|
45
|
+
code: (_b = err === null || err === void 0 ? void 0 : err.response) === null || _b === void 0 ? void 0 : _b.code,
|
|
46
|
+
};
|
|
47
|
+
}
|
|
48
|
+
},
|
|
49
|
+
},
|
|
50
|
+
redact: {
|
|
51
|
+
remove: true,
|
|
52
|
+
paths: [
|
|
53
|
+
'email',
|
|
54
|
+
'password',
|
|
55
|
+
'req.query',
|
|
56
|
+
'req.params',
|
|
57
|
+
'req.query',
|
|
58
|
+
'res.headers',
|
|
59
|
+
'req.headers.host',
|
|
60
|
+
'req.headers.connection',
|
|
61
|
+
'req.headers.accept',
|
|
62
|
+
'req.headers.origin',
|
|
63
|
+
'req.headers.referer',
|
|
64
|
+
'req.headers["content-type"]',
|
|
65
|
+
'req.headers["sec-ch-ua"]',
|
|
66
|
+
'req.headers["sec-ch-ua-mobile"]',
|
|
67
|
+
'req.headers["user-agent"]',
|
|
68
|
+
'req.headers["sec-ch-ua-platform"]',
|
|
69
|
+
'req.headers["sec-fetch-site"]',
|
|
70
|
+
'req.headers["sec-fetch-mode"]',
|
|
71
|
+
'req.headers["sec-fetch-dest"]',
|
|
72
|
+
'req.headers["accept-encoding"]',
|
|
73
|
+
'req.headers["accept-language"]',
|
|
74
|
+
'req.headers["if-none-match"]',
|
|
75
|
+
],
|
|
76
|
+
},
|
|
77
|
+
};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const pinoPretty = require("pino-pretty");
|
|
4
|
+
const date_fns_tz_1 = require("date-fns-tz");
|
|
5
|
+
function getLevel(level, extras) {
|
|
6
|
+
let levelStr, levelColor;
|
|
7
|
+
switch (level) {
|
|
8
|
+
case 10:
|
|
9
|
+
levelStr = 'TRACE';
|
|
10
|
+
levelColor = extras.colors.gray;
|
|
11
|
+
break;
|
|
12
|
+
case 20:
|
|
13
|
+
levelStr = 'DEBUG';
|
|
14
|
+
levelColor = extras.colors.blue;
|
|
15
|
+
break;
|
|
16
|
+
case 30:
|
|
17
|
+
levelStr = 'INFO';
|
|
18
|
+
levelColor = extras.colors.green;
|
|
19
|
+
break;
|
|
20
|
+
case 40:
|
|
21
|
+
levelStr = 'WARN';
|
|
22
|
+
levelColor = extras.colors.yellow;
|
|
23
|
+
break;
|
|
24
|
+
case 50:
|
|
25
|
+
levelStr = 'ERROR';
|
|
26
|
+
levelColor = extras.colors.red;
|
|
27
|
+
break;
|
|
28
|
+
case 60:
|
|
29
|
+
levelStr = 'FATAL';
|
|
30
|
+
levelColor = extras.colors.magenta;
|
|
31
|
+
break;
|
|
32
|
+
default:
|
|
33
|
+
levelStr = `LEVEL:${level}`;
|
|
34
|
+
levelColor = extras.colors.white;
|
|
35
|
+
}
|
|
36
|
+
return { levelStr, levelColor };
|
|
37
|
+
}
|
|
38
|
+
function getMethodInfo(method, colors) {
|
|
39
|
+
let methodColor;
|
|
40
|
+
switch (method.toUpperCase()) {
|
|
41
|
+
case 'GET':
|
|
42
|
+
methodColor = colors.blue;
|
|
43
|
+
break;
|
|
44
|
+
case 'POST':
|
|
45
|
+
methodColor = colors.green;
|
|
46
|
+
break;
|
|
47
|
+
case 'PUT':
|
|
48
|
+
methodColor = colors.yellow;
|
|
49
|
+
break;
|
|
50
|
+
case 'PATCH':
|
|
51
|
+
methodColor = colors.blueBright;
|
|
52
|
+
break;
|
|
53
|
+
case 'DELETE':
|
|
54
|
+
methodColor = colors.redBright;
|
|
55
|
+
break;
|
|
56
|
+
default:
|
|
57
|
+
methodColor = colors.white;
|
|
58
|
+
}
|
|
59
|
+
return { methodColor, methodStr: method.toUpperCase() };
|
|
60
|
+
}
|
|
61
|
+
function getResponseTimeInfo(_responseTime, colors) {
|
|
62
|
+
let responseTimeColor = colors.white;
|
|
63
|
+
if (_responseTime) {
|
|
64
|
+
if (_responseTime < 50)
|
|
65
|
+
responseTimeColor = colors.gray;
|
|
66
|
+
else if (_responseTime < 100)
|
|
67
|
+
responseTimeColor = colors.yellow;
|
|
68
|
+
else
|
|
69
|
+
responseTimeColor = colors.red;
|
|
70
|
+
}
|
|
71
|
+
return { responseTimeColor, responseTimeStr: _responseTime + 'ms' };
|
|
72
|
+
}
|
|
73
|
+
function getStatusCodeInfo(statusCode, colors) {
|
|
74
|
+
let statusCodeColor = colors.white;
|
|
75
|
+
if (statusCode) {
|
|
76
|
+
if (statusCode < 300)
|
|
77
|
+
statusCodeColor = colors.green;
|
|
78
|
+
else if (statusCode < 400)
|
|
79
|
+
statusCodeColor = colors.cyan;
|
|
80
|
+
else if (statusCode < 500)
|
|
81
|
+
statusCodeColor = colors.red;
|
|
82
|
+
else
|
|
83
|
+
statusCodeColor = colors.redBright;
|
|
84
|
+
}
|
|
85
|
+
return { statusCodeColor, statusCodeStr: statusCode.toString() };
|
|
86
|
+
}
|
|
87
|
+
function pinoPrettyTransport(opts) {
|
|
88
|
+
const timezone = opts.timezone || 'Asia/Seoul';
|
|
89
|
+
const ignoreContext = opts.ignoreContext || [
|
|
90
|
+
'InstanceLoader',
|
|
91
|
+
'RouterExplorer',
|
|
92
|
+
];
|
|
93
|
+
return pinoPretty(Object.assign(Object.assign({ singleLine: true, colorize: true, colorizeObjects: false, hideObject: true, ignore: 'time,pid,hostname,context,level' }, opts), { levelFirst: true, messageFormat(log, messageKey, levelLabel, extras) {
|
|
94
|
+
if (ignoreContext.includes(log.context)) {
|
|
95
|
+
return;
|
|
96
|
+
}
|
|
97
|
+
const timestamp = log.time
|
|
98
|
+
? (0, date_fns_tz_1.formatInTimeZone)(new Date(log.time), timezone, 'HH:mm:ss')
|
|
99
|
+
: '';
|
|
100
|
+
const tz = log.time
|
|
101
|
+
? (0, date_fns_tz_1.formatInTimeZone)(new Date(log.time), timezone, 'zzz')
|
|
102
|
+
: '';
|
|
103
|
+
if (log.method) {
|
|
104
|
+
const { methodColor, methodStr } = getMethodInfo(log.method, extras.colors);
|
|
105
|
+
const { responseTimeColor, responseTimeStr } = getResponseTimeInfo(log.responseTime, extras.colors);
|
|
106
|
+
const { statusCodeColor, statusCodeStr } = getStatusCodeInfo(log.statusCode, extras.colors);
|
|
107
|
+
if (log.err) {
|
|
108
|
+
const arr = [
|
|
109
|
+
extras.colors.dim(extras.colors.white(timestamp)),
|
|
110
|
+
extras.colors.gray(tz),
|
|
111
|
+
'',
|
|
112
|
+
extras.colors.underline(statusCodeColor(statusCodeStr)),
|
|
113
|
+
extras.colors.bold(methodColor(methodStr.padEnd(6))),
|
|
114
|
+
methodColor(log.url),
|
|
115
|
+
responseTimeColor(responseTimeStr),
|
|
116
|
+
];
|
|
117
|
+
const err_arr = [];
|
|
118
|
+
if (log.err.code)
|
|
119
|
+
err_arr.push(extras.colors.italic(extras.colors.red(log.err.code)));
|
|
120
|
+
if (log.err.stack)
|
|
121
|
+
err_arr.push(extras.colors.red(log.err.stack));
|
|
122
|
+
return arr.join(' ') + '\n' + err_arr.join('\n') + '\n';
|
|
123
|
+
}
|
|
124
|
+
else {
|
|
125
|
+
const arr = [
|
|
126
|
+
extras.colors.dim(extras.colors.white(timestamp)),
|
|
127
|
+
extras.colors.gray(tz),
|
|
128
|
+
'',
|
|
129
|
+
statusCodeColor(statusCodeStr),
|
|
130
|
+
extras.colors.bold(methodColor(methodStr.padEnd(6))),
|
|
131
|
+
methodColor(log.url),
|
|
132
|
+
responseTimeColor(responseTimeStr),
|
|
133
|
+
];
|
|
134
|
+
return arr.join(' ') + '\n';
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
const { levelStr, levelColor } = getLevel(log.level, extras);
|
|
138
|
+
const context = log.context
|
|
139
|
+
? extras.colors.yellow(`[${log.context}]`)
|
|
140
|
+
: '';
|
|
141
|
+
const arr = [
|
|
142
|
+
extras.colors.dim(extras.colors.white(timestamp)),
|
|
143
|
+
extras.colors.gray(tz),
|
|
144
|
+
'',
|
|
145
|
+
levelColor(levelStr),
|
|
146
|
+
context,
|
|
147
|
+
levelColor(log[messageKey]),
|
|
148
|
+
];
|
|
149
|
+
return arr.join(' ') + '\n';
|
|
150
|
+
}, customPrettifiers: {
|
|
151
|
+
time: (timestamp, key, log, extras) => {
|
|
152
|
+
return `⏰ ${timestamp}`;
|
|
153
|
+
},
|
|
154
|
+
level: (logLevel, key, log, extras) => {
|
|
155
|
+
return `LEVEL: ${logLevel} LABEL: ${extras.label} COLORIZED LABEL: ${extras.labelColorized}x`;
|
|
156
|
+
},
|
|
157
|
+
hostname: (hostname) => `MY HOST: ${hostname}`,
|
|
158
|
+
pid: (pid) => pid,
|
|
159
|
+
name: (name, key, log, { colors }) => `${colors.blue(name)}`,
|
|
160
|
+
caller: (caller, key, log, { colors }) => `${colors.greenBright(caller)}`,
|
|
161
|
+
myCustomLogProp: (value, key, log, { colors }) => `My Prop -> ${colors.bold(value)} <--`,
|
|
162
|
+
} }));
|
|
163
|
+
}
|
|
164
|
+
module.exports = pinoPrettyTransport;
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './redis.module';
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
var e=this&&this.__createBinding||(Object.create?function(e,t,r,i){void 0===i&&(i=r);var o=Object.getOwnPropertyDescriptor(t,r);o&&!("get"in o?!t.__esModule:o.writable||o.configurable)||(o={enumerable:!0,get:function(){return t[r]}}),Object.defineProperty(e,i,o)}:function(e,t,r,i){void 0===i&&(i=r),e[i]=t[r]}),t=this&&this.__exportStar||function(t,r){for(var i in t)"default"===i||Object.prototype.hasOwnProperty.call(r,i)||e(r,t,i)};Object.defineProperty(exports,"__esModule",{value:!0}),t(require("./redis.module"),exports);
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { DynamicModule } from '@nestjs/common';
|
|
2
|
+
import { RedisOptions } from 'ioredis';
|
|
3
|
+
export declare const REDIS_CLIENT = "REDIS_CLIENT";
|
|
4
|
+
export declare class RedisModule {
|
|
5
|
+
static forRootAsync(options: {
|
|
6
|
+
useFactory: (...args: any[]) => RedisOptions;
|
|
7
|
+
inject?: any[];
|
|
8
|
+
}): DynamicModule;
|
|
9
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
var e,t=this&&this.__decorate||function(e,t,o,r){var n,i=arguments.length,s=i<3?t:null===r?r=Object.getOwnPropertyDescriptor(t,o):r;if("object"==typeof Reflect&&"function"==typeof Reflect.decorate)s=Reflect.decorate(e,t,o,r);else for(var c=e.length-1;c>=0;c--)(n=e[c])&&(s=(i<3?n(s):i>3?n(t,o,s):n(t,o))||s);return i>3&&s&&Object.defineProperty(t,o,s),s},o=this&&this.__awaiter||function(e,t,o,r){return new(o||(o=Promise))(function(n,i){function fulfilled(e){try{step(r.next(e))}catch(e){i(e)}}function rejected(e){try{step(r.throw(e))}catch(e){i(e)}}function step(e){e.done?n(e.value):function adopt(e){return e instanceof o?e:new o(function(t){t(e)})}(e.value).then(fulfilled,rejected)}step((r=r.apply(e,t||[])).next())})};Object.defineProperty(exports,"__esModule",{value:!0}),exports.RedisModule=exports.REDIS_CLIENT=void 0;const r=require("@nestjs/common"),n=require("@nestjs/config"),i=require("ioredis");exports.REDIS_CLIENT="REDIS_CLIENT";let s=e=class RedisModule{static forRootAsync(t){return{module:e,imports:[n.ConfigModule],providers:[{provide:exports.REDIS_CLIENT,useFactory:(...e)=>o(this,void 0,void 0,function*(){const o=t.useFactory(...e);o.retryStrategy=e=>{const t=new r.Logger("Redis");return e>5?(t.error("Redis Reconnect Max Times, Stop Reconnect"),null):2e3};const n=new i.default(o),s=new r.Logger("Redis");return n.on("ready",()=>s.log("Redis Ready")),n.on("error",e=>s.error("Redis Error: "+e)),n.on("reconnecting",()=>s.warn("Redis Reconnecting")),n}),inject:t.inject||[]}],exports:[exports.REDIS_CLIENT]}}};exports.RedisModule=s,exports.RedisModule=s=e=t([(0,r.Global)(),(0,r.Module)({})],s);
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
export * from './notification.module';
|
|
2
|
+
export * from './services';
|
|
3
|
+
export interface NotificationRecipient {
|
|
4
|
+
user_id: number;
|
|
5
|
+
recipient: string;
|
|
6
|
+
variables?: Record<string, any>;
|
|
7
|
+
}
|
|
8
|
+
export interface NotificationOptions {
|
|
9
|
+
email?: {
|
|
10
|
+
recipients: NotificationRecipient[];
|
|
11
|
+
subject?: string;
|
|
12
|
+
body?: string;
|
|
13
|
+
template_key?: string;
|
|
14
|
+
variables?: Record<string, any>;
|
|
15
|
+
};
|
|
16
|
+
slack?: {
|
|
17
|
+
channel: string;
|
|
18
|
+
message?: string;
|
|
19
|
+
template_key?: string;
|
|
20
|
+
variables?: Record<string, any>;
|
|
21
|
+
};
|
|
22
|
+
kakao?: {
|
|
23
|
+
recipients: NotificationRecipient[];
|
|
24
|
+
template_key: string;
|
|
25
|
+
variables?: Record<string, any>;
|
|
26
|
+
};
|
|
27
|
+
sms?: {
|
|
28
|
+
recipients: NotificationRecipient[];
|
|
29
|
+
message?: string;
|
|
30
|
+
template_key?: string;
|
|
31
|
+
variables?: Record<string, any>;
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
export type NotificationPayload = {
|
|
35
|
+
type: 'email';
|
|
36
|
+
recipients: Array<{
|
|
37
|
+
recipient: string;
|
|
38
|
+
user_id: number;
|
|
39
|
+
variables?: Record<string, any>;
|
|
40
|
+
}>;
|
|
41
|
+
subject?: string;
|
|
42
|
+
body?: string;
|
|
43
|
+
template_key?: string;
|
|
44
|
+
variables?: Record<string, any>;
|
|
45
|
+
} | {
|
|
46
|
+
type: 'sms';
|
|
47
|
+
recipients: Array<{
|
|
48
|
+
recipient: string;
|
|
49
|
+
user_id: number;
|
|
50
|
+
variables?: Record<string, any>;
|
|
51
|
+
}>;
|
|
52
|
+
message?: string;
|
|
53
|
+
template_key?: string;
|
|
54
|
+
variables?: Record<string, any>;
|
|
55
|
+
} | {
|
|
56
|
+
type: 'kakao';
|
|
57
|
+
recipients: Array<{
|
|
58
|
+
recipient: string;
|
|
59
|
+
user_id: number;
|
|
60
|
+
variables?: Record<string, any>;
|
|
61
|
+
}>;
|
|
62
|
+
template_key: string;
|
|
63
|
+
variables?: Record<string, any>;
|
|
64
|
+
} | {
|
|
65
|
+
type: 'slack';
|
|
66
|
+
channel: string;
|
|
67
|
+
message?: string;
|
|
68
|
+
template_key?: string;
|
|
69
|
+
variables?: Record<string, any>;
|
|
70
|
+
};
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./notification.module"), exports);
|
|
18
|
+
__exportStar(require("./services"), exports);
|
|
@@ -0,0 +1,32 @@
|
|
|
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
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
9
|
+
exports.NotificationModule = void 0;
|
|
10
|
+
const common_1 = require("@nestjs/common");
|
|
11
|
+
const typeorm_1 = require("@nestjs/typeorm");
|
|
12
|
+
const entities_1 = require("../database/entities");
|
|
13
|
+
const services_1 = require("./services");
|
|
14
|
+
let NotificationModule = class NotificationModule {
|
|
15
|
+
};
|
|
16
|
+
exports.NotificationModule = NotificationModule;
|
|
17
|
+
exports.NotificationModule = NotificationModule = __decorate([
|
|
18
|
+
(0, common_1.Global)(),
|
|
19
|
+
(0, common_1.Module)({
|
|
20
|
+
imports: [typeorm_1.TypeOrmModule.forFeature([entities_1.Config, entities_1.Notification, entities_1.NotificationRecipient])],
|
|
21
|
+
providers: [
|
|
22
|
+
services_1.NotificationService,
|
|
23
|
+
services_1.NotificationSendService,
|
|
24
|
+
services_1.MessageTemplateService,
|
|
25
|
+
services_1.EmailService,
|
|
26
|
+
services_1.KakaoService,
|
|
27
|
+
services_1.SmsService,
|
|
28
|
+
services_1.SlackService,
|
|
29
|
+
],
|
|
30
|
+
exports: [services_1.NotificationSendService, services_1.NotificationService],
|
|
31
|
+
})
|
|
32
|
+
], NotificationModule);
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { ConfigService } from '@nestjs/config';
|
|
2
|
+
export declare class EmailService {
|
|
3
|
+
private readonly configService;
|
|
4
|
+
constructor(configService: ConfigService);
|
|
5
|
+
send(recipients: Array<{
|
|
6
|
+
recipient: string;
|
|
7
|
+
user_id: number;
|
|
8
|
+
subject: string;
|
|
9
|
+
text: string;
|
|
10
|
+
variables?: Record<string, any>;
|
|
11
|
+
}>): Promise<{
|
|
12
|
+
user_id: number;
|
|
13
|
+
recipient: string;
|
|
14
|
+
success: boolean;
|
|
15
|
+
messageId: string;
|
|
16
|
+
status: string;
|
|
17
|
+
}[]>;
|
|
18
|
+
sendAws(recipients: Array<{
|
|
19
|
+
recipient: string;
|
|
20
|
+
user_id: number;
|
|
21
|
+
subject: string;
|
|
22
|
+
text: string;
|
|
23
|
+
variables?: Record<string, any>;
|
|
24
|
+
}>): Promise<{
|
|
25
|
+
user_id: number;
|
|
26
|
+
recipient: string;
|
|
27
|
+
success: boolean;
|
|
28
|
+
messageId: string;
|
|
29
|
+
status: string;
|
|
30
|
+
}[]>;
|
|
31
|
+
}
|