@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.
- package/LICENSE +21 -0
- package/README.en.md +80 -0
- package/README.ja.md +80 -0
- package/README.md +79 -0
- package/README.zh-TW.md +80 -0
- package/bin/cli.js +10 -0
- package/dist/app.d.ts +2 -0
- package/dist/app.d.ts.map +1 -0
- package/dist/app.js +59 -0
- package/dist/app.js.map +1 -0
- package/dist/db/index.js +12 -0
- package/dist/db/index.js.map +1 -0
- package/dist/db/migrate.d.ts +3 -0
- package/dist/db/migrate.d.ts.map +1 -0
- package/dist/db/migrate.js +169 -0
- package/dist/db/migrate.js.map +1 -0
- package/dist/db/schema.d.ts +743 -0
- package/dist/db/schema.d.ts.map +1 -0
- package/dist/db/schema.js +88 -0
- package/dist/db/schema.js.map +1 -0
- package/dist/i18n.js +309 -0
- package/dist/i18n.js.map +1 -0
- package/dist/plugins/admin.d.ts +4 -0
- package/dist/plugins/admin.d.ts.map +1 -0
- package/dist/plugins/admin.js +19 -0
- package/dist/plugins/admin.js.map +1 -0
- package/dist/plugins/auth.d.ts +4 -0
- package/dist/plugins/auth.d.ts.map +1 -0
- package/dist/plugins/auth.js +35 -0
- package/dist/plugins/auth.js.map +1 -0
- package/dist/routes/auth.d.ts +4 -0
- package/dist/routes/auth.d.ts.map +1 -0
- package/dist/routes/auth.js +352 -0
- package/dist/routes/auth.js.map +1 -0
- package/dist/routes/categories.d.ts +4 -0
- package/dist/routes/categories.d.ts.map +1 -0
- package/dist/routes/categories.js +112 -0
- package/dist/routes/categories.js.map +1 -0
- package/dist/routes/config.d.ts +4 -0
- package/dist/routes/config.d.ts.map +1 -0
- package/dist/routes/config.js +227 -0
- package/dist/routes/config.js.map +1 -0
- package/dist/routes/resources.d.ts +4 -0
- package/dist/routes/resources.d.ts.map +1 -0
- package/dist/routes/resources.js +474 -0
- package/dist/routes/resources.js.map +1 -0
- package/dist/routes/tags.d.ts +4 -0
- package/dist/routes/tags.d.ts.map +1 -0
- package/dist/routes/tags.js +37 -0
- package/dist/routes/tags.js.map +1 -0
- package/dist/routes/users.d.ts +4 -0
- package/dist/routes/users.d.ts.map +1 -0
- package/dist/routes/users.js +181 -0
- package/dist/routes/users.js.map +1 -0
- package/dist/services/crypto.js +49 -0
- package/dist/services/crypto.js.map +1 -0
- package/dist/services/mail.d.ts +16 -0
- package/dist/services/mail.d.ts.map +1 -0
- package/dist/services/mail.js +33 -0
- package/dist/services/mail.js.map +1 -0
- package/dist/services/rsa.js +49 -0
- package/dist/services/rsa.js.map +1 -0
- package/dist/services/token.d.ts +9 -0
- package/dist/services/token.d.ts.map +1 -0
- package/dist/services/token.js +29 -0
- package/dist/services/token.js.map +1 -0
- package/dist/types.d.ts +80 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/package.json +73 -0
- package/public/admin/AdminCategories.jsx +310 -0
- package/public/admin/AdminConfig.jsx +254 -0
- package/public/admin/AdminEmail.jsx +279 -0
- package/public/admin/AdminTags.jsx +263 -0
- package/public/admin/AdminUsers.jsx +452 -0
- package/public/app.jsx +186 -0
- package/public/components/ConfirmDialog.jsx +78 -0
- package/public/components/DropdownSelect.jsx +281 -0
- package/public/components/EmailPreviewModal.jsx +104 -0
- package/public/components/EmptyState.jsx +50 -0
- package/public/components/Modal.jsx +127 -0
- package/public/components/PasswordStrength.jsx +45 -0
- package/public/components/Skeleton.jsx +68 -0
- package/public/components/Toast.jsx +80 -0
- package/public/components/TooltipIconButton.jsx +55 -0
- package/public/context/AppContext.jsx +314 -0
- package/public/features/BatchResourceModal.jsx +606 -0
- package/public/features/ChangePasswordModal.jsx +187 -0
- package/public/features/ProfileModal.jsx +170 -0
- package/public/features/ResourceCard.jsx +422 -0
- package/public/features/ResourceFormModal.jsx +915 -0
- package/public/features/ResourceRow.jsx +287 -0
- package/public/features/ResourceTimeline.jsx +472 -0
- package/public/hooks/useApi.jsx +26 -0
- package/public/hooks/useRouter.jsx +35 -0
- package/public/index.html +258 -0
- package/public/layout/AdminLayout.jsx +167 -0
- package/public/layout/AppLayout.jsx +119 -0
- package/public/layout/AuthLayout.jsx +503 -0
- package/public/layout/Header.jsx +543 -0
- package/public/layout/Sidebar.jsx +175 -0
- package/public/pages/AdminPage.jsx +30 -0
- package/public/pages/ForgotPasswordPage.jsx +93 -0
- package/public/pages/HomePage.jsx +2297 -0
- package/public/pages/LoginPage.jsx +191 -0
- package/public/pages/RegisterPage.jsx +137 -0
- package/public/pages/ResetPasswordPage.jsx +169 -0
- package/public/pages/SetupPage.jsx +157 -0
- package/public/utils/helpers.jsx +152 -0
- package/public/utils/i18n.jsx +1374 -0
- package/public/utils/preferences.jsx +220 -0
- package/public/utils/security.jsx +88 -0
- package/public/utils/theme.jsx +24 -0
- package/public/vendor/babel.min.js +2 -0
- package/public/vendor/lucide-react.min.js +9 -0
- package/public/vendor/react-dom.development.js +29869 -0
- 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"}
|
package/dist/types.d.ts
ADDED
|
@@ -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 @@
|
|
|
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
|
+
}
|