@zhang_libo/resource-hub 1.0.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.
Files changed (118) hide show
  1. package/LICENSE +21 -0
  2. package/README.en.md +80 -0
  3. package/README.ja.md +80 -0
  4. package/README.md +79 -0
  5. package/README.zh-TW.md +80 -0
  6. package/bin/cli.js +10 -0
  7. package/dist/app.d.ts +2 -0
  8. package/dist/app.d.ts.map +1 -0
  9. package/dist/app.js +59 -0
  10. package/dist/app.js.map +1 -0
  11. package/dist/db/index.js +12 -0
  12. package/dist/db/index.js.map +1 -0
  13. package/dist/db/migrate.d.ts +3 -0
  14. package/dist/db/migrate.d.ts.map +1 -0
  15. package/dist/db/migrate.js +169 -0
  16. package/dist/db/migrate.js.map +1 -0
  17. package/dist/db/schema.d.ts +743 -0
  18. package/dist/db/schema.d.ts.map +1 -0
  19. package/dist/db/schema.js +88 -0
  20. package/dist/db/schema.js.map +1 -0
  21. package/dist/i18n.js +309 -0
  22. package/dist/i18n.js.map +1 -0
  23. package/dist/plugins/admin.d.ts +4 -0
  24. package/dist/plugins/admin.d.ts.map +1 -0
  25. package/dist/plugins/admin.js +19 -0
  26. package/dist/plugins/admin.js.map +1 -0
  27. package/dist/plugins/auth.d.ts +4 -0
  28. package/dist/plugins/auth.d.ts.map +1 -0
  29. package/dist/plugins/auth.js +35 -0
  30. package/dist/plugins/auth.js.map +1 -0
  31. package/dist/routes/auth.d.ts +4 -0
  32. package/dist/routes/auth.d.ts.map +1 -0
  33. package/dist/routes/auth.js +352 -0
  34. package/dist/routes/auth.js.map +1 -0
  35. package/dist/routes/categories.d.ts +4 -0
  36. package/dist/routes/categories.d.ts.map +1 -0
  37. package/dist/routes/categories.js +112 -0
  38. package/dist/routes/categories.js.map +1 -0
  39. package/dist/routes/config.d.ts +4 -0
  40. package/dist/routes/config.d.ts.map +1 -0
  41. package/dist/routes/config.js +227 -0
  42. package/dist/routes/config.js.map +1 -0
  43. package/dist/routes/resources.d.ts +4 -0
  44. package/dist/routes/resources.d.ts.map +1 -0
  45. package/dist/routes/resources.js +474 -0
  46. package/dist/routes/resources.js.map +1 -0
  47. package/dist/routes/tags.d.ts +4 -0
  48. package/dist/routes/tags.d.ts.map +1 -0
  49. package/dist/routes/tags.js +37 -0
  50. package/dist/routes/tags.js.map +1 -0
  51. package/dist/routes/users.d.ts +4 -0
  52. package/dist/routes/users.d.ts.map +1 -0
  53. package/dist/routes/users.js +181 -0
  54. package/dist/routes/users.js.map +1 -0
  55. package/dist/services/crypto.js +49 -0
  56. package/dist/services/crypto.js.map +1 -0
  57. package/dist/services/mail.d.ts +16 -0
  58. package/dist/services/mail.d.ts.map +1 -0
  59. package/dist/services/mail.js +33 -0
  60. package/dist/services/mail.js.map +1 -0
  61. package/dist/services/rsa.js +49 -0
  62. package/dist/services/rsa.js.map +1 -0
  63. package/dist/services/token.d.ts +9 -0
  64. package/dist/services/token.d.ts.map +1 -0
  65. package/dist/services/token.js +29 -0
  66. package/dist/services/token.js.map +1 -0
  67. package/dist/types.d.ts +80 -0
  68. package/dist/types.d.ts.map +1 -0
  69. package/dist/types.js +2 -0
  70. package/dist/types.js.map +1 -0
  71. package/package.json +73 -0
  72. package/public/admin/AdminCategories.jsx +310 -0
  73. package/public/admin/AdminConfig.jsx +254 -0
  74. package/public/admin/AdminEmail.jsx +279 -0
  75. package/public/admin/AdminTags.jsx +263 -0
  76. package/public/admin/AdminUsers.jsx +452 -0
  77. package/public/app.jsx +186 -0
  78. package/public/components/ConfirmDialog.jsx +78 -0
  79. package/public/components/DropdownSelect.jsx +281 -0
  80. package/public/components/EmailPreviewModal.jsx +104 -0
  81. package/public/components/EmptyState.jsx +50 -0
  82. package/public/components/Modal.jsx +127 -0
  83. package/public/components/PasswordStrength.jsx +45 -0
  84. package/public/components/Skeleton.jsx +68 -0
  85. package/public/components/Toast.jsx +80 -0
  86. package/public/components/TooltipIconButton.jsx +55 -0
  87. package/public/context/AppContext.jsx +314 -0
  88. package/public/features/BatchResourceModal.jsx +606 -0
  89. package/public/features/ChangePasswordModal.jsx +187 -0
  90. package/public/features/ProfileModal.jsx +170 -0
  91. package/public/features/ResourceCard.jsx +422 -0
  92. package/public/features/ResourceFormModal.jsx +915 -0
  93. package/public/features/ResourceRow.jsx +287 -0
  94. package/public/features/ResourceTimeline.jsx +472 -0
  95. package/public/hooks/useApi.jsx +26 -0
  96. package/public/hooks/useRouter.jsx +35 -0
  97. package/public/index.html +258 -0
  98. package/public/layout/AdminLayout.jsx +167 -0
  99. package/public/layout/AppLayout.jsx +119 -0
  100. package/public/layout/AuthLayout.jsx +503 -0
  101. package/public/layout/Header.jsx +543 -0
  102. package/public/layout/Sidebar.jsx +175 -0
  103. package/public/pages/AdminPage.jsx +30 -0
  104. package/public/pages/ForgotPasswordPage.jsx +93 -0
  105. package/public/pages/HomePage.jsx +2297 -0
  106. package/public/pages/LoginPage.jsx +191 -0
  107. package/public/pages/RegisterPage.jsx +137 -0
  108. package/public/pages/ResetPasswordPage.jsx +169 -0
  109. package/public/pages/SetupPage.jsx +157 -0
  110. package/public/utils/helpers.jsx +152 -0
  111. package/public/utils/i18n.jsx +1374 -0
  112. package/public/utils/preferences.jsx +220 -0
  113. package/public/utils/security.jsx +88 -0
  114. package/public/utils/theme.jsx +24 -0
  115. package/public/vendor/babel.min.js +2 -0
  116. package/public/vendor/lucide-react.min.js +9 -0
  117. package/public/vendor/react-dom.development.js +29869 -0
  118. package/public/vendor/react.development.js +3342 -0
@@ -0,0 +1,181 @@
1
+ import bcrypt from 'bcryptjs';
2
+ import { v4 as uuidv4 } from 'uuid';
3
+ import { db } from '../db/index.js';
4
+ import { users, resources, emailConfig } from '../db/schema.js';
5
+ import { eq, ne, and } from 'drizzle-orm';
6
+ import { deliverMail } from '../services/mail.js';
7
+ import { getAdminResetPasswordMail, getRequestLocale, localizeFields, localizeText, } from '../i18n.js';
8
+ function sendError(reply, locale, status, error, code, fields) {
9
+ const body = { success: false, error: localizeText(locale, error), code };
10
+ if (fields)
11
+ body.fields = localizeFields(locale, fields);
12
+ reply.code(status).send(body);
13
+ }
14
+ function validateUsername(v) {
15
+ return /^[a-zA-Z_][a-zA-Z0-9_]{2,19}$/.test(v);
16
+ }
17
+ function validatePassword(v) {
18
+ return v.length >= 8 && v.length <= 64 && /[a-zA-Z]/.test(v) && /[0-9]/.test(v);
19
+ }
20
+ function validateEmail(v) {
21
+ return /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(v);
22
+ }
23
+ function validateDisplayName(v) {
24
+ return typeof v === 'string' && v.trim().length >= 1 && v.trim().length <= 30;
25
+ }
26
+ function generateTempPassword() {
27
+ const upper = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ';
28
+ const digits = '0123456789';
29
+ const lower = 'abcdefghijklmnopqrstuvwxyz';
30
+ const rand = (chars, n) => Array.from({ length: n }, () => chars[Math.floor(Math.random() * chars.length)]).join('');
31
+ return rand(upper, 4) + rand(digits, 4) + rand(lower, 4);
32
+ }
33
+ function formatUser(u) {
34
+ return {
35
+ id: u.id,
36
+ username: u.username,
37
+ displayName: u.displayName,
38
+ email: u.email,
39
+ role: u.role,
40
+ status: u.status,
41
+ createdAt: u.createdAt,
42
+ };
43
+ }
44
+ function getEmailRow() {
45
+ return db.select().from(emailConfig).where(eq(emailConfig.id, 'default')).get();
46
+ }
47
+ const usersRoutes = async (fastify) => {
48
+ // GET / — admin: list all users (no passwordHash)
49
+ fastify.get('/', { preHandler: fastify.requireAdmin }, async (_req, reply) => {
50
+ const rows = db.select({
51
+ id: users.id,
52
+ username: users.username,
53
+ displayName: users.displayName,
54
+ email: users.email,
55
+ role: users.role,
56
+ status: users.status,
57
+ createdAt: users.createdAt,
58
+ }).from(users).all();
59
+ reply.send({ success: true, data: rows });
60
+ });
61
+ // POST / — admin: create user directly with provided password, no email sent
62
+ fastify.post('/', { preHandler: fastify.requireAdmin }, async (req, reply) => {
63
+ const locale = getRequestLocale(req);
64
+ const { username, displayName, email, password, role } = req.body;
65
+ const fields = {};
66
+ if (!username || !validateUsername(username))
67
+ fields.username = '用户名格式不正确(3-20字符,字母/数字/下划线,不能以数字开头)';
68
+ if (!displayName || !validateDisplayName(displayName))
69
+ fields.displayName = '显示名称须为1-30字符';
70
+ if (!email || !validateEmail(email))
71
+ fields.email = '邮箱格式不正确';
72
+ if (!password || !validatePassword(password))
73
+ fields.password = '密码须为8-64字符,且同时包含字母和数字';
74
+ if (Object.keys(fields).length > 0) {
75
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR', fields);
76
+ }
77
+ const existingUsername = db.select().from(users).where(eq(users.username, username)).get();
78
+ if (existingUsername)
79
+ return sendError(reply, locale, 422, '用户名已被占用', 'USERNAME_TAKEN');
80
+ const existingEmail = db.select().from(users).where(eq(users.email, email)).get();
81
+ if (existingEmail)
82
+ return sendError(reply, locale, 422, '邮箱已被注册', 'EMAIL_TAKEN');
83
+ const passwordHash = await bcrypt.hash(password, 10);
84
+ const now = Math.floor(Date.now() / 1000);
85
+ const id = uuidv4();
86
+ db.insert(users).values({
87
+ id,
88
+ username,
89
+ displayName,
90
+ email,
91
+ role: role === 'admin' ? 'admin' : 'user',
92
+ status: 'active',
93
+ passwordHash,
94
+ createdAt: now,
95
+ }).run();
96
+ const newUser = db.select().from(users).where(eq(users.id, id)).get();
97
+ reply.code(201).send({ success: true, data: formatUser(newUser) });
98
+ });
99
+ // PUT /:id — admin: edit user fields
100
+ fastify.put('/:id', { preHandler: fastify.requireAdmin }, async (req, reply) => {
101
+ const locale = getRequestLocale(req);
102
+ const { id } = req.params;
103
+ const user = db.select().from(users).where(eq(users.id, id)).get();
104
+ if (!user)
105
+ return sendError(reply, locale, 404, '用户不存在', 'USER_NOT_FOUND');
106
+ const { displayName, email, role, status } = req.body;
107
+ const updates = {};
108
+ if (displayName !== undefined) {
109
+ if (!validateDisplayName(displayName)) {
110
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR', { displayName: '显示名称须为1-30字符' });
111
+ }
112
+ updates.displayName = displayName;
113
+ }
114
+ if (email !== undefined) {
115
+ if (!validateEmail(email)) {
116
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR', { email: '邮箱格式不正确' });
117
+ }
118
+ const dup = db.select().from(users)
119
+ .where(and(eq(users.email, email), ne(users.id, id)))
120
+ .get();
121
+ if (dup)
122
+ return sendError(reply, locale, 422, '邮箱已被注册', 'EMAIL_TAKEN');
123
+ updates.email = email;
124
+ }
125
+ if (role !== undefined)
126
+ updates.role = role;
127
+ if (status !== undefined)
128
+ updates.status = status;
129
+ if (Object.keys(updates).length > 0) {
130
+ db.update(users).set(updates).where(eq(users.id, id)).run();
131
+ }
132
+ const updated = db.select().from(users).where(eq(users.id, id)).get();
133
+ reply.send({ success: true, data: formatUser(updated) });
134
+ });
135
+ // DELETE /:id — admin: transfer resources to admin then delete user
136
+ fastify.delete('/:id', { preHandler: fastify.requireAdmin }, async (req, reply) => {
137
+ const locale = getRequestLocale(req);
138
+ const { id } = req.params;
139
+ if (id === req.user.userId) {
140
+ return sendError(reply, locale, 422, '不能删除自身账号', 'CANNOT_DELETE_SELF');
141
+ }
142
+ const user = db.select().from(users).where(eq(users.id, id)).get();
143
+ if (!user)
144
+ return sendError(reply, locale, 404, '用户不存在', 'USER_NOT_FOUND');
145
+ // Transfer user's resources to the performing admin to avoid FK constraint
146
+ db.update(resources).set({ ownerId: req.user.userId }).where(eq(resources.ownerId, id)).run();
147
+ db.delete(users).where(eq(users.id, id)).run();
148
+ reply.send({ success: true, data: { message: localizeText(locale, '删除成功') } });
149
+ });
150
+ // POST /:id/reset-password — admin: auto-generate and send new password
151
+ fastify.post('/:id/reset-password', { preHandler: fastify.requireAdmin }, async (req, reply) => {
152
+ const locale = getRequestLocale(req);
153
+ const { id } = req.params;
154
+ const user = db.select().from(users).where(eq(users.id, id)).get();
155
+ if (!user)
156
+ return sendError(reply, locale, 404, '用户不存在', 'USER_NOT_FOUND');
157
+ const newPassword = generateTempPassword();
158
+ const passwordHash = await bcrypt.hash(newPassword, 10);
159
+ db.update(users).set({ passwordHash }).where(eq(users.id, id)).run();
160
+ const mailConfig = getEmailRow();
161
+ const { subject, body } = getAdminResetPasswordMail(locale, user.displayName, newPassword);
162
+ const preview = await deliverMail(user.email, subject, body, {
163
+ smtpHost: mailConfig?.smtpHost ?? '',
164
+ smtpPort: mailConfig?.smtpPort ?? 465,
165
+ encryption: mailConfig?.encryption ?? 'ssl',
166
+ fromEmail: mailConfig?.fromEmail ?? '',
167
+ fromName: mailConfig?.fromName ?? '资源导航系统',
168
+ smtpUser: mailConfig?.smtpUser ?? '',
169
+ smtpPassword: mailConfig?.smtpPassword ?? '',
170
+ });
171
+ const response = {
172
+ success: true,
173
+ data: { message: localizeText(locale, '密码已重置') },
174
+ };
175
+ if (preview)
176
+ response.emailPreview = preview;
177
+ reply.send(response);
178
+ });
179
+ };
180
+ export default usersRoutes;
181
+ //# sourceMappingURL=users.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"users.js","sourceRoot":"","sources":["../../src/routes/users.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,49 @@
1
+ import { constants, createPrivateKey, createPublicKey, generateKeyPairSync, privateDecrypt } from 'crypto';
2
+ let privateKey;
3
+ let publicKeyPem;
4
+ function initRsaKeyPair() {
5
+ const pem = process.env.RSA_PRIVATE_KEY_PEM;
6
+ if (pem && typeof pem === 'string' && pem.trim().length > 0) {
7
+ privateKey = createPrivateKey({ key: pem.trim(), format: 'pem' });
8
+ const pubKey = createPublicKey(privateKey);
9
+ publicKeyPem = pubKey.export({ type: 'spki', format: 'pem' });
10
+ return;
11
+ }
12
+ const { privateKey: priv, publicKey: pub } = generateKeyPairSync('rsa', {
13
+ modulusLength: 2048,
14
+ publicKeyEncoding: { type: 'spki', format: 'pem' },
15
+ privateKeyEncoding: { type: 'pkcs1', format: 'pem' },
16
+ });
17
+ privateKey = createPrivateKey(priv);
18
+ publicKeyPem = pub;
19
+ }
20
+ initRsaKeyPair();
21
+ export function getPublicKeyPem() {
22
+ return publicKeyPem;
23
+ }
24
+ /**
25
+ * Decrypt RSA-OAEP ciphertext (Base64) to UTF-8 plaintext.
26
+ * Returns null if decryption fails (invalid base64 or bad ciphertext).
27
+ */
28
+ export function decryptPassword(ciphertextBase64) {
29
+ if (typeof ciphertextBase64 !== 'string' || !ciphertextBase64.trim()) {
30
+ return null;
31
+ }
32
+ let buf;
33
+ try {
34
+ buf = Buffer.from(ciphertextBase64.trim(), 'base64');
35
+ }
36
+ catch {
37
+ return null;
38
+ }
39
+ if (buf.length === 0)
40
+ return null;
41
+ try {
42
+ const decrypted = privateDecrypt({ key: privateKey, padding: constants.RSA_PKCS1_OAEP_PADDING }, buf);
43
+ return decrypted.toString('utf8');
44
+ }
45
+ catch {
46
+ return null;
47
+ }
48
+ }
49
+ //# sourceMappingURL=crypto.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"crypto.js","sourceRoot":"","sources":["../../src/services/crypto.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,gBAAgB,EAAE,eAAe,EAAE,mBAAmB,EAAE,cAAc,EAAE,MAAM,QAAQ,CAAA;AAE1G,IAAI,UAA+C,CAAA;AACnD,IAAI,YAAoB,CAAA;AAExB,SAAS,cAAc;IACrB,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAA;IAC3C,IAAI,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ,IAAI,GAAG,CAAC,IAAI,EAAE,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC5D,UAAU,GAAG,gBAAgB,CAAC,EAAE,GAAG,EAAE,GAAG,CAAC,IAAI,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,CAAA;QACjE,MAAM,MAAM,GAAG,eAAe,CAAC,UAAU,CAAC,CAAA;QAC1C,YAAY,GAAG,MAAM,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,CAAW,CAAA;QACvE,OAAM;IACR,CAAC;IACD,MAAM,EAAE,UAAU,EAAE,IAAI,EAAE,SAAS,EAAE,GAAG,EAAE,GAAG,mBAAmB,CAAC,KAAK,EAAE;QACtE,aAAa,EAAE,IAAI;QACnB,iBAAiB,EAAE,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE;QAClD,kBAAkB,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE;KACrD,CAAC,CAAA;IACF,UAAU,GAAG,gBAAgB,CAAC,IAAI,CAAC,CAAA;IACnC,YAAY,GAAG,GAAG,CAAA;AACpB,CAAC;AAED,cAAc,EAAE,CAAA;AAEhB,MAAM,UAAU,eAAe;IAC7B,OAAO,YAAY,CAAA;AACrB,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,eAAe,CAAC,gBAAwB;IACtD,IAAI,OAAO,gBAAgB,KAAK,QAAQ,IAAI,CAAC,gBAAgB,CAAC,IAAI,EAAE,EAAE,CAAC;QACrE,OAAO,IAAI,CAAA;IACb,CAAC;IACD,IAAI,GAAW,CAAA;IACf,IAAI,CAAC;QACH,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,IAAI,EAAE,EAAE,QAAQ,CAAC,CAAA;IACtD,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAA;IACb,CAAC;IACD,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,IAAI,CAAA;IACjC,IAAI,CAAC;QACH,MAAM,SAAS,GAAG,cAAc,CAC9B,EAAE,GAAG,EAAE,UAAU,EAAE,OAAO,EAAE,SAAS,CAAC,sBAAsB,EAAE,EAC9D,GAAG,CACJ,CAAA;QACD,OAAO,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAA;IACnC,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAA;IACb,CAAC;AACH,CAAC"}
@@ -0,0 +1,16 @@
1
+ import type { EmailPreview } from '../types.js';
2
+ interface MailConfig {
3
+ smtpHost: string | null;
4
+ smtpPort: number | null;
5
+ encryption: string | null;
6
+ fromEmail: string | null;
7
+ fromName: string | null;
8
+ smtpUser: string | null;
9
+ smtpPassword: string | null;
10
+ }
11
+ /**
12
+ * Deliver an email. Returns EmailPreview if in mock mode (smtpHost empty), null otherwise.
13
+ */
14
+ export declare function deliverMail(to: string, subject: string, body: string, config: MailConfig): Promise<EmailPreview | null>;
15
+ export {};
16
+ //# sourceMappingURL=mail.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mail.d.ts","sourceRoot":"","sources":["../../src/services/mail.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,aAAa,CAAA;AAE/C,UAAU,UAAU;IAClB,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAA;IACvB,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAA;IACvB,UAAU,EAAE,MAAM,GAAG,IAAI,CAAA;IACzB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAA;IACxB,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAA;IACvB,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAA;IACvB,YAAY,EAAE,MAAM,GAAG,IAAI,CAAA;CAC5B;AAED;;GAEG;AACH,wBAAsB,WAAW,CAC/B,EAAE,EAAE,MAAM,EACV,OAAO,EAAE,MAAM,EACf,IAAI,EAAE,MAAM,EACZ,MAAM,EAAE,UAAU,GACjB,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC,CA8B9B"}
@@ -0,0 +1,33 @@
1
+ import nodemailer from 'nodemailer';
2
+ /**
3
+ * Deliver an email. Returns EmailPreview if in mock mode (smtpHost empty), null otherwise.
4
+ */
5
+ export async function deliverMail(to, subject, body, config) {
6
+ const isMock = !config.smtpHost || config.smtpHost.trim() === '';
7
+ if (isMock) {
8
+ console.log('\n[Mock Email]');
9
+ console.log(` To: ${to}`);
10
+ console.log(` Subject: ${subject}`);
11
+ console.log(` Body:\n${body}`);
12
+ return { to, subject, body };
13
+ }
14
+ const secure = config.encryption === 'ssl';
15
+ const transportOpts = {
16
+ host: config.smtpHost,
17
+ port: config.smtpPort ?? 465,
18
+ secure,
19
+ auth: {
20
+ user: config.smtpUser ?? '',
21
+ pass: config.smtpPassword ?? '',
22
+ },
23
+ };
24
+ const transporter = nodemailer.createTransport(transportOpts);
25
+ await transporter.sendMail({
26
+ from: `"${config.fromName ?? '资源导航系统'}" <${config.fromEmail ?? config.smtpUser}>`,
27
+ to,
28
+ subject,
29
+ text: body,
30
+ });
31
+ return null;
32
+ }
33
+ //# sourceMappingURL=mail.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mail.js","sourceRoot":"","sources":["../../src/services/mail.ts"],"names":[],"mappings":"AAAA,OAAO,UAAU,MAAM,YAAY,CAAA;AAanC;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,WAAW,CAC/B,EAAU,EACV,OAAe,EACf,IAAY,EACZ,MAAkB;IAElB,MAAM,MAAM,GAAG,CAAC,MAAM,CAAC,QAAQ,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,KAAK,EAAE,CAAA;IAEhE,IAAI,MAAM,EAAE,CAAC;QACX,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAA;QAC7B,OAAO,CAAC,GAAG,CAAC,SAAS,EAAE,EAAE,CAAC,CAAA;QAC1B,OAAO,CAAC,GAAG,CAAC,cAAc,OAAO,EAAE,CAAC,CAAA;QACpC,OAAO,CAAC,GAAG,CAAC,YAAY,IAAI,EAAE,CAAC,CAAA;QAC/B,OAAO,EAAE,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAA;IAC9B,CAAC;IAED,MAAM,MAAM,GAAG,MAAM,CAAC,UAAU,KAAK,KAAK,CAAA;IAC1C,MAAM,aAAa,GAAG;QACpB,IAAI,EAAE,MAAM,CAAC,QAAS;QACtB,IAAI,EAAE,MAAM,CAAC,QAAQ,IAAI,GAAG;QAC5B,MAAM;QACN,IAAI,EAAE;YACJ,IAAI,EAAE,MAAM,CAAC,QAAQ,IAAI,EAAE;YAC3B,IAAI,EAAE,MAAM,CAAC,YAAY,IAAI,EAAE;SAChC;KACF,CAAA;IACD,MAAM,WAAW,GAAG,UAAU,CAAC,eAAe,CAAC,aAAa,CAAC,CAAA;IAE7D,MAAM,WAAW,CAAC,QAAQ,CAAC;QACzB,IAAI,EAAE,IAAI,MAAM,CAAC,QAAQ,IAAI,QAAQ,MAAM,MAAM,CAAC,SAAS,IAAI,MAAM,CAAC,QAAQ,GAAG;QACjF,EAAE;QACF,OAAO;QACP,IAAI,EAAE,IAAI;KACX,CAAC,CAAA;IAEF,OAAO,IAAI,CAAA;AACb,CAAC"}
@@ -0,0 +1,49 @@
1
+ import { generateKeyPairSync, privateDecrypt, constants } from 'crypto';
2
+ import { db } from '../db/index.js';
3
+ import { rsaKeys } from '../db/schema.js';
4
+ import { eq } from 'drizzle-orm';
5
+ function nowSeconds() {
6
+ return Math.floor(Date.now() / 1000);
7
+ }
8
+ export function getOrCreateKeyPair() {
9
+ const existing = db.select().from(rsaKeys).where(eq(rsaKeys.id, 'current')).get();
10
+ if (existing)
11
+ return existing;
12
+ const { publicKey, privateKey } = generateKeyPairSync('rsa', {
13
+ modulusLength: 2048,
14
+ publicKeyEncoding: {
15
+ type: 'spki',
16
+ format: 'pem',
17
+ },
18
+ privateKeyEncoding: {
19
+ type: 'pkcs8',
20
+ format: 'pem',
21
+ },
22
+ });
23
+ const row = {
24
+ id: 'current',
25
+ publicKey,
26
+ privateKey,
27
+ createdAt: nowSeconds(),
28
+ };
29
+ db.insert(rsaKeys).values(row).run();
30
+ return row;
31
+ }
32
+ export function ensureRsaKeyPair() {
33
+ getOrCreateKeyPair();
34
+ }
35
+ export function getPublicKey() {
36
+ const row = getOrCreateKeyPair();
37
+ return row.publicKey;
38
+ }
39
+ export function decryptWithPrivateKey(ciphertextBase64) {
40
+ const row = getOrCreateKeyPair();
41
+ const buffer = Buffer.from(ciphertextBase64, 'base64');
42
+ const decrypted = privateDecrypt({
43
+ key: row.privateKey,
44
+ padding: constants.RSA_PKCS1_OAEP_PADDING,
45
+ oaepHash: 'sha256',
46
+ }, buffer);
47
+ return decrypted.toString('utf8');
48
+ }
49
+ //# sourceMappingURL=rsa.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rsa.js","sourceRoot":"","sources":["../../src/services/rsa.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,mBAAmB,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAA;AACvE,OAAO,EAAE,EAAE,EAAE,MAAM,gBAAgB,CAAA;AACnC,OAAO,EAAE,OAAO,EAAE,MAAM,iBAAiB,CAAA;AACzC,OAAO,EAAE,EAAE,EAAE,MAAM,aAAa,CAAA;AAIhC,SAAS,UAAU;IACjB,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,CAAA;AACtC,CAAC;AAED,MAAM,UAAU,kBAAkB;IAChC,MAAM,QAAQ,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,KAAK,CAAC,EAAE,CAAC,OAAO,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC,CAAC,GAAG,EAAE,CAAA;IACjF,IAAI,QAAQ;QAAE,OAAO,QAAQ,CAAA;IAE7B,MAAM,EAAE,SAAS,EAAE,UAAU,EAAE,GAAG,mBAAmB,CAAC,KAAK,EAAE;QAC3D,aAAa,EAAE,IAAI;QACnB,iBAAiB,EAAE;YACjB,IAAI,EAAE,MAAM;YACZ,MAAM,EAAE,KAAK;SACd;QACD,kBAAkB,EAAE;YAClB,IAAI,EAAE,OAAO;YACb,MAAM,EAAE,KAAK;SACd;KACF,CAAC,CAAA;IAEF,MAAM,GAAG,GAAc;QACrB,EAAE,EAAE,SAAS;QACb,SAAS;QACT,UAAU;QACV,SAAS,EAAE,UAAU,EAAE;KACxB,CAAA;IAED,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAA;IACpC,OAAO,GAAG,CAAA;AACZ,CAAC;AAED,MAAM,UAAU,gBAAgB;IAC9B,kBAAkB,EAAE,CAAA;AACtB,CAAC;AAED,MAAM,UAAU,YAAY;IAC1B,MAAM,GAAG,GAAG,kBAAkB,EAAE,CAAA;IAChC,OAAO,GAAG,CAAC,SAAS,CAAA;AACtB,CAAC;AAED,MAAM,UAAU,qBAAqB,CAAC,gBAAwB;IAC5D,MAAM,GAAG,GAAG,kBAAkB,EAAE,CAAA;IAChC,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,gBAAgB,EAAE,QAAQ,CAAC,CAAA;IAEtD,MAAM,SAAS,GAAG,cAAc,CAC9B;QACE,GAAG,EAAE,GAAG,CAAC,UAAU;QACnB,OAAO,EAAE,SAAS,CAAC,sBAAsB;QACzC,QAAQ,EAAE,QAAQ;KACnB,EACD,MAAM,CACP,CAAA;IAED,OAAO,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAA;AACnC,CAAC"}
@@ -0,0 +1,9 @@
1
+ export declare function generateResetToken(): string;
2
+ export declare function createResetToken(email: string): string;
3
+ export declare function validateResetToken(token: string): {
4
+ valid: boolean;
5
+ error?: string;
6
+ email?: string;
7
+ };
8
+ export declare function markTokenUsed(token: string): void;
9
+ //# sourceMappingURL=token.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token.d.ts","sourceRoot":"","sources":["../../src/services/token.ts"],"names":[],"mappings":"AAKA,wBAAgB,kBAAkB,IAAI,MAAM,CAE3C;AAED,wBAAgB,gBAAgB,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,CAQtD;AAED,wBAAgB,kBAAkB,CAAC,KAAK,EAAE,MAAM,GAAG;IAAE,KAAK,EAAE,OAAO,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAA;CAAE,CAMpG;AAED,wBAAgB,aAAa,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,CAEjD"}
@@ -0,0 +1,29 @@
1
+ import { randomBytes } from 'crypto';
2
+ import { db } from '../db/index.js';
3
+ import { resetTokens, systemConfig } from '../db/schema.js';
4
+ import { eq } from 'drizzle-orm';
5
+ export function generateResetToken() {
6
+ return randomBytes(32).toString('hex');
7
+ }
8
+ export function createResetToken(email) {
9
+ const config = db.select().from(systemConfig).where(eq(systemConfig.id, 'default')).get();
10
+ const expiryMinutes = config?.resetTokenExpiry ?? 60;
11
+ const expiresAt = Math.floor(Date.now() / 1000) + expiryMinutes * 60;
12
+ const token = generateResetToken();
13
+ db.insert(resetTokens).values({ token, email, expiresAt, used: false }).run();
14
+ return token;
15
+ }
16
+ export function validateResetToken(token) {
17
+ const row = db.select().from(resetTokens).where(eq(resetTokens.token, token)).get();
18
+ if (!row)
19
+ return { valid: false, error: 'RESET_TOKEN_INVALID' };
20
+ if (row.used)
21
+ return { valid: false, error: 'RESET_TOKEN_USED' };
22
+ if (row.expiresAt <= Math.floor(Date.now() / 1000))
23
+ return { valid: false, error: 'RESET_TOKEN_EXPIRED' };
24
+ return { valid: true, email: row.email };
25
+ }
26
+ export function markTokenUsed(token) {
27
+ db.update(resetTokens).set({ used: true }).where(eq(resetTokens.token, token)).run();
28
+ }
29
+ //# sourceMappingURL=token.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token.js","sourceRoot":"","sources":["../../src/services/token.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,QAAQ,CAAA;AACpC,OAAO,EAAE,EAAE,EAAE,MAAM,gBAAgB,CAAA;AACnC,OAAO,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAA;AAC3D,OAAO,EAAE,EAAE,EAAE,MAAM,aAAa,CAAA;AAEhC,MAAM,UAAU,kBAAkB;IAChC,OAAO,WAAW,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAA;AACxC,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAC,KAAa;IAC5C,MAAM,MAAM,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC,CAAC,GAAG,EAAE,CAAA;IACzF,MAAM,aAAa,GAAG,MAAM,EAAE,gBAAgB,IAAI,EAAE,CAAA;IACpD,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,GAAG,aAAa,GAAG,EAAE,CAAA;IACpE,MAAM,KAAK,GAAG,kBAAkB,EAAE,CAAA;IAElC,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,MAAM,CAAC,EAAE,KAAK,EAAE,KAAK,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC,GAAG,EAAE,CAAA;IAC7E,OAAO,KAAK,CAAA;AACd,CAAC;AAED,MAAM,UAAU,kBAAkB,CAAC,KAAa;IAC9C,MAAM,GAAG,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,KAAK,CAAC,EAAE,CAAC,WAAW,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC,GAAG,EAAE,CAAA;IACnF,IAAI,CAAC,GAAG;QAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,qBAAqB,EAAE,CAAA;IAC/D,IAAI,GAAG,CAAC,IAAI;QAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,kBAAkB,EAAE,CAAA;IAChE,IAAI,GAAG,CAAC,SAAS,IAAI,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC;QAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,qBAAqB,EAAE,CAAA;IACzG,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,CAAC,KAAK,EAAE,CAAA;AAC1C,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,KAAa;IACzC,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,KAAK,CAAC,EAAE,CAAC,WAAW,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC,GAAG,EAAE,CAAA;AACtF,CAAC"}
@@ -0,0 +1,80 @@
1
+ export interface JwtPayload {
2
+ userId: string;
3
+ role: 'admin' | 'user';
4
+ }
5
+ export interface User {
6
+ id: string;
7
+ username: string;
8
+ displayName: string;
9
+ email: string;
10
+ role: 'admin' | 'user';
11
+ isActive: boolean;
12
+ createdAt: string;
13
+ updatedAt: string;
14
+ }
15
+ export interface Resource {
16
+ id: string;
17
+ title: string;
18
+ url: string;
19
+ description: string | null;
20
+ icon: string | null;
21
+ categoryId: string | null;
22
+ authorId: string;
23
+ isPublic: boolean;
24
+ isEnabled: boolean;
25
+ sortOrder: number;
26
+ createdAt: string;
27
+ updatedAt: string;
28
+ }
29
+ export interface Category {
30
+ id: string;
31
+ name: string;
32
+ icon: string | null;
33
+ sortOrder: number;
34
+ createdAt: string;
35
+ }
36
+ export interface Tag {
37
+ id: string;
38
+ name: string;
39
+ createdAt: string;
40
+ }
41
+ export interface SystemConfig {
42
+ id: number;
43
+ siteName: string;
44
+ siteDesc: string | null;
45
+ enableRegister: boolean;
46
+ tokenExpiry: number;
47
+ updatedAt: string;
48
+ }
49
+ export interface EmailConfig {
50
+ id: number;
51
+ smtpHost: string;
52
+ smtpPort: number;
53
+ smtpUser: string;
54
+ smtpPass: string;
55
+ smtpFrom: string;
56
+ updatedAt: string;
57
+ }
58
+ export interface EmailPreview {
59
+ to: string;
60
+ subject: string;
61
+ body: string;
62
+ }
63
+ export interface ResetToken {
64
+ id: string;
65
+ userId: string;
66
+ token: string;
67
+ expiresAt: string;
68
+ used: boolean;
69
+ createdAt: string;
70
+ }
71
+ declare module 'fastify' {
72
+ interface FastifyRequest {
73
+ user: JwtPayload;
74
+ }
75
+ interface FastifyInstance {
76
+ authenticate: (request: import('fastify').FastifyRequest, reply: import('fastify').FastifyReply) => Promise<void>;
77
+ requireAdmin: (request: import('fastify').FastifyRequest, reply: import('fastify').FastifyReply) => Promise<void>;
78
+ }
79
+ }
80
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,UAAU;IACzB,MAAM,EAAE,MAAM,CAAA;IACd,IAAI,EAAE,OAAO,GAAG,MAAM,CAAA;CACvB;AAED,MAAM,WAAW,IAAI;IACnB,EAAE,EAAE,MAAM,CAAA;IACV,QAAQ,EAAE,MAAM,CAAA;IAChB,WAAW,EAAE,MAAM,CAAA;IACnB,KAAK,EAAE,MAAM,CAAA;IACb,IAAI,EAAE,OAAO,GAAG,MAAM,CAAA;IACtB,QAAQ,EAAE,OAAO,CAAA;IACjB,SAAS,EAAE,MAAM,CAAA;IACjB,SAAS,EAAE,MAAM,CAAA;CAClB;AAED,MAAM,WAAW,QAAQ;IACvB,EAAE,EAAE,MAAM,CAAA;IACV,KAAK,EAAE,MAAM,CAAA;IACb,GAAG,EAAE,MAAM,CAAA;IACX,WAAW,EAAE,MAAM,GAAG,IAAI,CAAA;IAC1B,IAAI,EAAE,MAAM,GAAG,IAAI,CAAA;IACnB,UAAU,EAAE,MAAM,GAAG,IAAI,CAAA;IACzB,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,OAAO,CAAA;IACjB,SAAS,EAAE,OAAO,CAAA;IAClB,SAAS,EAAE,MAAM,CAAA;IACjB,SAAS,EAAE,MAAM,CAAA;IACjB,SAAS,EAAE,MAAM,CAAA;CAClB;AAED,MAAM,WAAW,QAAQ;IACvB,EAAE,EAAE,MAAM,CAAA;IACV,IAAI,EAAE,MAAM,CAAA;IACZ,IAAI,EAAE,MAAM,GAAG,IAAI,CAAA;IACnB,SAAS,EAAE,MAAM,CAAA;IACjB,SAAS,EAAE,MAAM,CAAA;CAClB;AAED,MAAM,WAAW,GAAG;IAClB,EAAE,EAAE,MAAM,CAAA;IACV,IAAI,EAAE,MAAM,CAAA;IACZ,SAAS,EAAE,MAAM,CAAA;CAClB;AAED,MAAM,WAAW,YAAY;IAC3B,EAAE,EAAE,MAAM,CAAA;IACV,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAA;IACvB,cAAc,EAAE,OAAO,CAAA;IACvB,WAAW,EAAE,MAAM,CAAA;IACnB,SAAS,EAAE,MAAM,CAAA;CAClB;AAED,MAAM,WAAW,WAAW;IAC1B,EAAE,EAAE,MAAM,CAAA;IACV,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,MAAM,CAAA;IAChB,SAAS,EAAE,MAAM,CAAA;CAClB;AAED,MAAM,WAAW,YAAY;IAC3B,EAAE,EAAE,MAAM,CAAA;IACV,OAAO,EAAE,MAAM,CAAA;IACf,IAAI,EAAE,MAAM,CAAA;CACb;AAED,MAAM,WAAW,UAAU;IACzB,EAAE,EAAE,MAAM,CAAA;IACV,MAAM,EAAE,MAAM,CAAA;IACd,KAAK,EAAE,MAAM,CAAA;IACb,SAAS,EAAE,MAAM,CAAA;IACjB,IAAI,EAAE,OAAO,CAAA;IACb,SAAS,EAAE,MAAM,CAAA;CAClB;AAGD,OAAO,QAAQ,SAAS,CAAC;IACvB,UAAU,cAAc;QACtB,IAAI,EAAE,UAAU,CAAA;KACjB;IACD,UAAU,eAAe;QACvB,YAAY,EAAE,CAAC,OAAO,EAAE,OAAO,SAAS,EAAE,cAAc,EAAE,KAAK,EAAE,OAAO,SAAS,EAAE,YAAY,KAAK,OAAO,CAAC,IAAI,CAAC,CAAA;QACjH,YAAY,EAAE,CAAC,OAAO,EAAE,OAAO,SAAS,EAAE,cAAc,EAAE,KAAK,EAAE,OAAO,SAAS,EAAE,YAAY,KAAK,OAAO,CAAC,IAAI,CAAC,CAAA;KAClH;CACF"}
package/dist/types.js ADDED
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":""}
package/package.json ADDED
@@ -0,0 +1,73 @@
1
+ {
2
+ "name": "@zhang_libo/resource-hub",
3
+ "version": "1.0.2",
4
+ "description": "资源导航系统 - 集中整理与维护站点、工具、知识库与链接,支持用户登录、收藏、访问历史与后台管理",
5
+ "keywords": [
6
+ "resource",
7
+ "management",
8
+ "system",
9
+ "hub",
10
+ "resource-hub",
11
+ "资源",
12
+ "导航",
13
+ "资源管理",
14
+ "链接管理",
15
+ "收藏",
16
+ "后台管理"
17
+ ],
18
+ "type": "module",
19
+ "license": "MIT",
20
+ "author": "Libo Zhang",
21
+ "email": "zhanglibo610@gmail.com",
22
+ "repository": {
23
+ "type": "git",
24
+ "url": "https://github.com/my-bad-idea/ResourceHub.git"
25
+ },
26
+ "homepage": "https://github.com/my-bad-idea/ResourceHub#readme",
27
+ "main": "dist/app.js",
28
+ "bin": {
29
+ "resource-hub": "bin/cli.js"
30
+ },
31
+ "engines": {
32
+ "node": ">=18.0.0"
33
+ },
34
+ "files": [
35
+ "dist",
36
+ "public",
37
+ "bin"
38
+ ],
39
+ "scripts": {
40
+ "dev": "tsx watch src/app.ts",
41
+ "build": "tsc",
42
+ "start": "node dist/app.js",
43
+ "test:smoke": "node test/smoke.mjs",
44
+ "test:browser": "node test/browser-acceptance.mjs",
45
+ "check": "npm run build && npm run test:smoke",
46
+ "// release:patch": "补丁版本发布",
47
+ "release:patch": "npm run build && npm version patch && npm publish --access=public",
48
+ "// release:minor": "次版本发布",
49
+ "release:minor": "npm run build && npm version minor && npm publish --access=public",
50
+ "// release:major": "主版本发布",
51
+ "release:major": "npm run build && npm version major && npm publish --access=public"
52
+ },
53
+ "dependencies": {
54
+ "@fastify/cors": "^9",
55
+ "@fastify/jwt": "^8",
56
+ "@fastify/static": "^7",
57
+ "bcryptjs": "^3.0.3",
58
+ "better-sqlite3": "^12",
59
+ "drizzle-orm": "^0.30",
60
+ "fastify": "^4",
61
+ "fastify-plugin": "^4",
62
+ "nodemailer": "^6",
63
+ "uuid": "^9"
64
+ },
65
+ "devDependencies": {
66
+ "@types/better-sqlite3": "^7.6.13",
67
+ "@types/node": "^20",
68
+ "@types/nodemailer": "^6",
69
+ "@types/uuid": "^9",
70
+ "tsx": "^4",
71
+ "typescript": "^5"
72
+ }
73
+ }