@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,474 @@
1
+ import { v4 as uuidv4 } from 'uuid';
2
+ import { db } from '../db/index.js';
3
+ import { resources, resourceTags, favorites, visitHistory, visitHourly, categories, users } from '../db/schema.js';
4
+ import { eq, and, or, sql, inArray, like, asc, desc } from 'drizzle-orm';
5
+ import { getRequestLocale, localizeText } from '../i18n.js';
6
+ // ── Helpers ──────────────────────────────────────────────────────────────────
7
+ function sendError(reply, locale, status, error, code) {
8
+ reply.code(status).send({ success: false, error: localizeText(locale, error), code });
9
+ }
10
+ function validateUrl(v) {
11
+ return /^https?:\/\//.test(v);
12
+ }
13
+ /** Try to extract userId/role from Bearer token without throwing on failure. */
14
+ function parseOptionalUser(fastify, req) {
15
+ try {
16
+ const auth = req.headers.authorization;
17
+ if (auth?.startsWith('Bearer ')) {
18
+ const p = fastify.jwt.verify(auth.slice(7));
19
+ const user = db.select({
20
+ id: users.id,
21
+ role: users.role,
22
+ status: users.status,
23
+ }).from(users).where(eq(users.id, p.userId)).get();
24
+ if (!user || user.status === 'disabled') {
25
+ return { userId: null, role: null };
26
+ }
27
+ return { userId: user.id, role: user.role };
28
+ }
29
+ }
30
+ catch { }
31
+ return { userId: null, role: null };
32
+ }
33
+ function getVisitHourBucket(timestamp) {
34
+ return Math.floor(timestamp / 3600) * 3600;
35
+ }
36
+ /** Build visibility WHERE condition based on auth level. Returns undefined for admin (no filter). */
37
+ function buildVisibilityCond(userId, role) {
38
+ if (role === 'admin')
39
+ return undefined;
40
+ if (userId) {
41
+ return or(and(eq(resources.visibility, 'public'), eq(resources.enabled, true)), eq(resources.ownerId, userId));
42
+ }
43
+ return and(eq(resources.visibility, 'public'), eq(resources.enabled, true));
44
+ }
45
+ /** Batch-load tags for a set of resource IDs. */
46
+ function batchFetchTags(resourceIds) {
47
+ if (resourceIds.length === 0)
48
+ return new Map();
49
+ const rows = db.select().from(resourceTags)
50
+ .where(inArray(resourceTags.resourceId, resourceIds))
51
+ .all();
52
+ const map = new Map();
53
+ for (const row of rows) {
54
+ const arr = map.get(row.resourceId) ?? [];
55
+ arr.push(row.tag);
56
+ map.set(row.resourceId, arr);
57
+ }
58
+ return map;
59
+ }
60
+ /** Batch-load categories for a set of category IDs. */
61
+ function batchFetchCategories(categoryIds) {
62
+ const validIds = [...new Set(categoryIds.filter((id) => id !== null))];
63
+ if (validIds.length === 0)
64
+ return new Map();
65
+ const rows = db.select().from(categories).where(inArray(categories.id, validIds)).all();
66
+ return new Map(rows.map((c) => [c.id, { name: c.name, color: c.color }]));
67
+ }
68
+ /** Get the set of resource IDs favorited by a user. */
69
+ function fetchFavoritedSet(userId) {
70
+ if (!userId)
71
+ return new Set();
72
+ const rows = db.select({ resourceId: favorites.resourceId })
73
+ .from(favorites)
74
+ .where(eq(favorites.userId, userId))
75
+ .all();
76
+ return new Set(rows.map((f) => f.resourceId));
77
+ }
78
+ /** Build the full API resource object from a DB row + lookup maps. */
79
+ function buildResourceResponse(resource, categoryMap, tagMap, favoritedSet) {
80
+ const cat = resource.categoryId ? categoryMap.get(resource.categoryId) : null;
81
+ return {
82
+ id: resource.id,
83
+ name: resource.name,
84
+ url: resource.url,
85
+ categoryId: resource.categoryId,
86
+ categoryName: cat?.name ?? null,
87
+ categoryColor: cat?.color ?? null,
88
+ visibility: resource.visibility,
89
+ logoUrl: resource.logoUrl,
90
+ description: resource.description,
91
+ enabled: resource.enabled,
92
+ ownerId: resource.ownerId,
93
+ visitCount: resource.visitCount || 0,
94
+ tags: tagMap.get(resource.id) ?? [],
95
+ isFavorited: favoritedSet.has(resource.id),
96
+ createdAt: resource.createdAt,
97
+ updatedAt: resource.updatedAt,
98
+ };
99
+ }
100
+ /** Check whether a resource is visible to a given user (for single-resource access). */
101
+ function isVisible(resource, userId, role) {
102
+ if (role === 'admin')
103
+ return true;
104
+ if (!userId)
105
+ return resource.visibility === 'public' && resource.enabled;
106
+ return (resource.visibility === 'public' && resource.enabled) || resource.ownerId === userId;
107
+ }
108
+ // ── Route plugin ──────────────────────────────────────────────────────────────
109
+ const resourcesRoutes = async (fastify) => {
110
+ // ── GET / — resource list (optional auth, visibility-aware) ─────────────────
111
+ fastify.get('/', async (req, reply) => {
112
+ const { userId, role } = parseOptionalUser(fastify, req);
113
+ const query = req.query;
114
+ const q = query.q?.trim() || '';
115
+ const category = query.category?.trim() || '';
116
+ const tagsParam = query.tags?.trim() || '';
117
+ const sort = query.sort || 'hot';
118
+ const tagList = tagsParam ? tagsParam.split(',').map((t) => t.trim()).filter(Boolean) : [];
119
+ const sortMap = {
120
+ createdAt: desc(resources.createdAt),
121
+ updatedAt: desc(resources.updatedAt),
122
+ };
123
+ const orderCol = sortMap[sort] ?? sortMap.updatedAt;
124
+ const conditions = [];
125
+ const visCond = buildVisibilityCond(userId, role);
126
+ if (visCond)
127
+ conditions.push(visCond);
128
+ if (category)
129
+ conditions.push(eq(resources.categoryId, category));
130
+ if (q) {
131
+ conditions.push(or(like(resources.name, `%${q}%`), like(resources.description, `%${q}%`), like(resources.url, `%${q}%`), sql `EXISTS (SELECT 1 FROM resource_tags rt WHERE rt.resource_id = ${resources.id} AND rt.tag LIKE ${'%' + q + '%'})`));
132
+ }
133
+ for (const tag of tagList) {
134
+ conditions.push(sql `EXISTS (SELECT 1 FROM resource_tags rt WHERE rt.resource_id = ${resources.id} AND rt.tag = ${tag})`);
135
+ }
136
+ const rows = conditions.length > 0
137
+ ? db.select().from(resources).where(and(...conditions)).orderBy(orderCol).all()
138
+ : db.select().from(resources).orderBy(orderCol).all();
139
+ const ids = rows.map((r) => r.id);
140
+ const tagMap = batchFetchTags(ids);
141
+ const categoryMap = batchFetchCategories(rows.map((r) => r.categoryId));
142
+ const favoritedSet = fetchFavoritedSet(userId);
143
+ const data = rows
144
+ .map((r) => buildResourceResponse(r, categoryMap, tagMap, favoritedSet))
145
+ .sort((a, b) => {
146
+ if (sort === 'createdAt')
147
+ return (b.createdAt || 0) - (a.createdAt || 0);
148
+ if (sort === 'updatedAt')
149
+ return (b.updatedAt || 0) - (a.updatedAt || 0);
150
+ return (b.visitCount || 0) - (a.visitCount || 0) || (b.updatedAt || 0) - (a.updatedAt || 0);
151
+ });
152
+ reply.send({ success: true, data });
153
+ });
154
+ // ── GET /analytics — global hourly visit summary ───────────────────────────
155
+ fastify.get('/analytics', async (_req, reply) => {
156
+ const now = Math.floor(Date.now() / 1000);
157
+ const recent30Threshold = now - 30 * 24 * 3600;
158
+ const recent24Threshold = now - 24 * 3600;
159
+ const hourlyRows = db.select({
160
+ visitHour: visitHourly.visitHour,
161
+ visitCount: visitHourly.visitCount,
162
+ }).from(visitHourly).all();
163
+ const totalVisits = hourlyRows.reduce((sum, row) => sum + (row.visitCount || 0), 0);
164
+ const monthlyVisits = hourlyRows.reduce((sum, row) => sum + (row.visitHour >= recent30Threshold ? row.visitCount : 0), 0);
165
+ const dailyVisits = hourlyRows.reduce((sum, row) => sum + (row.visitHour >= recent24Threshold ? row.visitCount : 0), 0);
166
+ reply.send({
167
+ success: true,
168
+ data: {
169
+ totalVisits,
170
+ monthlyVisits,
171
+ dailyVisits,
172
+ },
173
+ });
174
+ });
175
+ // ── POST /analytics/visit — record homepage visit ─────────────────────────
176
+ fastify.post('/analytics/visit', async (_req, reply) => {
177
+ const now = Math.floor(Date.now() / 1000);
178
+ const visitHour = getVisitHourBucket(now);
179
+ db.insert(visitHourly)
180
+ .values({ visitHour, visitCount: 1 })
181
+ .onConflictDoUpdate({
182
+ target: [visitHourly.visitHour],
183
+ set: { visitCount: sql `${visitHourly.visitCount} + 1` },
184
+ })
185
+ .run();
186
+ reply.send({ success: true, data: { ok: true } });
187
+ });
188
+ // ── POST / — create resource ────────────────────────────────────────────────
189
+ fastify.post('/', { preHandler: fastify.authenticate }, async (req, reply) => {
190
+ const locale = getRequestLocale(req);
191
+ const body = req.body;
192
+ const { name, url, categoryId, visibility, logoUrl, description, tags, enabled } = body;
193
+ // Validate required fields
194
+ if (!name || typeof name !== 'string' || name.trim().length < 1 || name.trim().length > 50) {
195
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR');
196
+ }
197
+ if (!url || !validateUrl(url)) {
198
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR');
199
+ }
200
+ if (logoUrl && logoUrl !== '' && !validateUrl(logoUrl)) {
201
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR');
202
+ }
203
+ if (description !== undefined && description.length > 200) {
204
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR');
205
+ }
206
+ if (tags !== undefined) {
207
+ if (!Array.isArray(tags) || tags.length > 10) {
208
+ return sendError(reply, locale, 422, '标签最多 10 个', 'VALIDATION_ERROR');
209
+ }
210
+ for (const tag of tags) {
211
+ if (typeof tag !== 'string' || tag.length < 1 || tag.length > 20 || /\s/.test(tag)) {
212
+ return sendError(reply, locale, 422, '标签格式不正确(1-20字符,不可含空格)', 'VALIDATION_ERROR');
213
+ }
214
+ }
215
+ }
216
+ // Validate categoryId if provided
217
+ if (categoryId) {
218
+ const cat = db.select().from(categories).where(eq(categories.id, categoryId)).get();
219
+ if (!cat)
220
+ return sendError(reply, locale, 422, '类别不存在', 'CATEGORY_NOT_FOUND');
221
+ }
222
+ const now = Math.floor(Date.now() / 1000);
223
+ const id = uuidv4();
224
+ db.insert(resources).values({
225
+ id,
226
+ name: name.trim(),
227
+ url,
228
+ categoryId: categoryId ?? null,
229
+ visibility: visibility ?? 'public',
230
+ logoUrl: logoUrl ?? '',
231
+ description: description ?? '',
232
+ enabled: enabled ?? true,
233
+ ownerId: req.user.userId,
234
+ visitCount: 0,
235
+ createdAt: now,
236
+ updatedAt: now,
237
+ }).run();
238
+ // Insert tags (deduplicated)
239
+ const uniqueTags = [...new Set(tags ?? [])];
240
+ for (const tag of uniqueTags) {
241
+ db.insert(resourceTags).values({ resourceId: id, tag }).onConflictDoNothing().run();
242
+ }
243
+ const resource = db.select().from(resources).where(eq(resources.id, id)).get();
244
+ const tagMap = batchFetchTags([id]);
245
+ const categoryMap = batchFetchCategories([resource.categoryId]);
246
+ const favoritedSet = fetchFavoritedSet(req.user.userId);
247
+ reply.code(201).send({
248
+ success: true,
249
+ data: buildResourceResponse(resource, categoryMap, tagMap, favoritedSet),
250
+ });
251
+ });
252
+ // ── GET /favorites — user's favorite resources ──────────────────────────────
253
+ fastify.get('/favorites', { preHandler: fastify.authenticate }, async (req, reply) => {
254
+ const userId = req.user.userId;
255
+ const favRows = db.select().from(favorites)
256
+ .where(eq(favorites.userId, userId))
257
+ .orderBy(desc(favorites.createdAt))
258
+ .all();
259
+ if (favRows.length === 0) {
260
+ return reply.send({ success: true, data: [] });
261
+ }
262
+ const resourceIds = favRows.map((f) => f.resourceId);
263
+ const resourceRows = db.select().from(resources)
264
+ .where(inArray(resources.id, resourceIds))
265
+ .all();
266
+ const resourceMap = new Map(resourceRows.map((r) => [r.id, r]));
267
+ const tagMap = batchFetchTags(resourceIds);
268
+ const categoryMap = batchFetchCategories(resourceRows.map((r) => r.categoryId));
269
+ const favoritedSet = new Set(resourceIds);
270
+ // Return in favorites order (createdAt desc)
271
+ const data = resourceIds
272
+ .map((id) => resourceMap.get(id))
273
+ .filter((r) => r !== undefined)
274
+ .map((r) => buildResourceResponse(r, categoryMap, tagMap, favoritedSet));
275
+ reply.send({ success: true, data });
276
+ });
277
+ // ── GET /history — user's visit history ────────────────────────────────────
278
+ fastify.get('/history', { preHandler: fastify.authenticate }, async (req, reply) => {
279
+ const userId = req.user.userId;
280
+ const histRows = db.select().from(visitHistory)
281
+ .where(eq(visitHistory.userId, userId))
282
+ .orderBy(desc(visitHistory.visitedAt))
283
+ .all();
284
+ if (histRows.length === 0) {
285
+ return reply.send({ success: true, data: [] });
286
+ }
287
+ const resourceIds = histRows.map((h) => h.resourceId);
288
+ const uniqueIds = [...new Set(resourceIds)];
289
+ const resourceRows = db.select().from(resources)
290
+ .where(inArray(resources.id, uniqueIds))
291
+ .all();
292
+ const resourceMap = new Map(resourceRows.map((r) => [r.id, r]));
293
+ const tagMap = batchFetchTags(uniqueIds);
294
+ const categoryMap = batchFetchCategories(resourceRows.map((r) => r.categoryId));
295
+ const favoritedSet = fetchFavoritedSet(userId);
296
+ // Return in history order (visitedAt desc), same resource can appear multiple times
297
+ const data = resourceIds
298
+ .map((id) => resourceMap.get(id))
299
+ .filter((r) => r !== undefined)
300
+ .map((r) => buildResourceResponse(r, categoryMap, tagMap, favoritedSet));
301
+ reply.send({ success: true, data });
302
+ });
303
+ // ── GET /mine — resources created by current user ──────────────────────────
304
+ fastify.get('/mine', { preHandler: fastify.authenticate }, async (req, reply) => {
305
+ const userId = req.user.userId;
306
+ const rows = db.select().from(resources)
307
+ .where(eq(resources.ownerId, userId))
308
+ .orderBy(desc(resources.createdAt))
309
+ .all();
310
+ const ids = rows.map((r) => r.id);
311
+ const tagMap = batchFetchTags(ids);
312
+ const categoryMap = batchFetchCategories(rows.map((r) => r.categoryId));
313
+ const favoritedSet = fetchFavoritedSet(userId);
314
+ const data = rows.map((r) => buildResourceResponse(r, categoryMap, tagMap, favoritedSet));
315
+ reply.send({ success: true, data });
316
+ });
317
+ // ── PUT /:id — edit resource (owner or admin) ───────────────────────────────
318
+ fastify.put('/:id', { preHandler: fastify.authenticate }, async (req, reply) => {
319
+ const locale = getRequestLocale(req);
320
+ const { id } = req.params;
321
+ const { userId, role } = req.user;
322
+ const resource = db.select().from(resources).where(eq(resources.id, id)).get();
323
+ if (!resource)
324
+ return sendError(reply, locale, 404, '资源不存在', 'RESOURCE_NOT_FOUND');
325
+ if (role !== 'admin' && resource.ownerId !== userId) {
326
+ return sendError(reply, locale, 403, '权限不足', 'PERMISSION_DENIED');
327
+ }
328
+ const body = req.body;
329
+ const updates = {};
330
+ if (body.name !== undefined) {
331
+ const name = body.name;
332
+ if (!name || name.trim().length < 1 || name.trim().length > 50) {
333
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR');
334
+ }
335
+ updates.name = name.trim();
336
+ }
337
+ if (body.url !== undefined) {
338
+ if (!validateUrl(body.url)) {
339
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR');
340
+ }
341
+ updates.url = body.url;
342
+ }
343
+ if (body.logoUrl !== undefined) {
344
+ const rawLogoUrl = body.logoUrl;
345
+ if (rawLogoUrl !== null && typeof rawLogoUrl !== 'string') {
346
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR');
347
+ }
348
+ const logoUrl = typeof rawLogoUrl === 'string' ? rawLogoUrl : '';
349
+ if (logoUrl !== '' && !validateUrl(logoUrl)) {
350
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR');
351
+ }
352
+ updates.logoUrl = logoUrl;
353
+ }
354
+ if (body.description !== undefined) {
355
+ if (body.description.length > 200) {
356
+ return sendError(reply, locale, 422, '请求参数校验失败', 'VALIDATION_ERROR');
357
+ }
358
+ updates.description = body.description;
359
+ }
360
+ if (body.categoryId !== undefined) {
361
+ const catId = body.categoryId;
362
+ if (catId) {
363
+ const cat = db.select().from(categories).where(eq(categories.id, catId)).get();
364
+ if (!cat)
365
+ return sendError(reply, locale, 422, '类别不存在', 'CATEGORY_NOT_FOUND');
366
+ }
367
+ updates.categoryId = catId;
368
+ }
369
+ if (body.visibility !== undefined)
370
+ updates.visibility = body.visibility;
371
+ if (body.enabled !== undefined)
372
+ updates.enabled = body.enabled;
373
+ // Validate and replace tags if provided
374
+ let newTags;
375
+ if (body.tags !== undefined) {
376
+ const tags = body.tags;
377
+ if (!Array.isArray(tags) || tags.length > 10) {
378
+ return sendError(reply, locale, 422, '标签最多 10 个', 'VALIDATION_ERROR');
379
+ }
380
+ for (const tag of tags) {
381
+ if (typeof tag !== 'string' || tag.length < 1 || tag.length > 20 || /\s/.test(tag)) {
382
+ return sendError(reply, locale, 422, '标签格式不正确(1-20字符,不可含空格)', 'VALIDATION_ERROR');
383
+ }
384
+ }
385
+ newTags = [...new Set(tags)];
386
+ }
387
+ updates.updatedAt = Math.floor(Date.now() / 1000);
388
+ if (Object.keys(updates).length > 0) {
389
+ db.update(resources).set(updates).where(eq(resources.id, id)).run();
390
+ }
391
+ // Full replace tags if provided
392
+ if (newTags !== undefined) {
393
+ db.delete(resourceTags).where(eq(resourceTags.resourceId, id)).run();
394
+ for (const tag of newTags) {
395
+ db.insert(resourceTags).values({ resourceId: id, tag }).onConflictDoNothing().run();
396
+ }
397
+ }
398
+ const updated = db.select().from(resources).where(eq(resources.id, id)).get();
399
+ const tagMap = batchFetchTags([id]);
400
+ const categoryMap = batchFetchCategories([updated.categoryId]);
401
+ const favoritedSet = fetchFavoritedSet(userId);
402
+ reply.send({ success: true, data: buildResourceResponse(updated, categoryMap, tagMap, favoritedSet) });
403
+ });
404
+ // ── DELETE /:id — delete resource (owner or admin) ─────────────────────────
405
+ fastify.delete('/:id', { preHandler: fastify.authenticate }, async (req, reply) => {
406
+ const locale = getRequestLocale(req);
407
+ const { id } = req.params;
408
+ const { userId, role } = req.user;
409
+ const resource = db.select().from(resources).where(eq(resources.id, id)).get();
410
+ if (!resource)
411
+ return sendError(reply, locale, 404, '资源不存在', 'RESOURCE_NOT_FOUND');
412
+ if (role !== 'admin' && resource.ownerId !== userId) {
413
+ return sendError(reply, locale, 403, '权限不足', 'PERMISSION_DENIED');
414
+ }
415
+ db.delete(resources).where(eq(resources.id, id)).run();
416
+ reply.send({ success: true, data: { message: localizeText(locale, '删除成功') } });
417
+ });
418
+ // ── POST /:id/visit — record a visit ────────────────────────────────────────
419
+ fastify.post('/:id/visit', async (req, reply) => {
420
+ const locale = getRequestLocale(req);
421
+ const { id } = req.params;
422
+ const { userId, role } = parseOptionalUser(fastify, req);
423
+ const resource = db.select().from(resources).where(eq(resources.id, id)).get();
424
+ if (!resource || !isVisible(resource, userId, role)) {
425
+ return sendError(reply, locale, 404, '资源不存在', 'RESOURCE_NOT_FOUND');
426
+ }
427
+ const now = Math.floor(Date.now() / 1000);
428
+ db.update(resources)
429
+ .set({ visitCount: sql `${resources.visitCount} + 1` })
430
+ .where(eq(resources.id, id))
431
+ .run();
432
+ if (userId) {
433
+ db.insert(visitHistory).values({ userId, resourceId: id, visitedAt: now }).run();
434
+ // Cap visit history at 200 per user
435
+ const histRows = db.select({ id: visitHistory.id })
436
+ .from(visitHistory)
437
+ .where(eq(visitHistory.userId, userId))
438
+ .orderBy(asc(visitHistory.id))
439
+ .all();
440
+ if (histRows.length > 200) {
441
+ const toDelete = histRows.slice(0, histRows.length - 200).map((r) => r.id);
442
+ db.delete(visitHistory).where(inArray(visitHistory.id, toDelete)).run();
443
+ }
444
+ }
445
+ const updatedResource = db.select({ visitCount: resources.visitCount }).from(resources).where(eq(resources.id, id)).get();
446
+ reply.send({ success: true, data: { visitCount: updatedResource?.visitCount || 0 } });
447
+ });
448
+ // ── POST /:id/favorite — toggle favorite ────────────────────────────────────
449
+ fastify.post('/:id/favorite', { preHandler: fastify.authenticate }, async (req, reply) => {
450
+ const locale = getRequestLocale(req);
451
+ const { id } = req.params;
452
+ const { userId, role } = req.user;
453
+ const resource = db.select().from(resources).where(eq(resources.id, id)).get();
454
+ if (!resource || !isVisible(resource, userId, role)) {
455
+ return sendError(reply, locale, 404, '资源不存在', 'RESOURCE_NOT_FOUND');
456
+ }
457
+ const existing = db.select().from(favorites)
458
+ .where(and(eq(favorites.userId, userId), eq(favorites.resourceId, id)))
459
+ .get();
460
+ if (existing) {
461
+ db.delete(favorites)
462
+ .where(and(eq(favorites.userId, userId), eq(favorites.resourceId, id)))
463
+ .run();
464
+ return reply.send({ success: true, data: { isFavorited: false } });
465
+ }
466
+ else {
467
+ const now = Math.floor(Date.now() / 1000);
468
+ db.insert(favorites).values({ userId, resourceId: id, createdAt: now }).run();
469
+ return reply.send({ success: true, data: { isFavorited: true } });
470
+ }
471
+ });
472
+ };
473
+ export default resourcesRoutes;
474
+ //# sourceMappingURL=resources.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resources.js","sourceRoot":"","sources":["../../src/routes/resources.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { FastifyPluginAsync } from 'fastify';
2
+ declare const tagsRoutes: FastifyPluginAsync;
3
+ export default tagsRoutes;
4
+ //# sourceMappingURL=tags.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tags.d.ts","sourceRoot":"","sources":["../../src/routes/tags.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,kBAAkB,EAAgB,MAAM,SAAS,CAAA;AAS/D,QAAA,MAAM,UAAU,EAAE,kBA2BjB,CAAA;AAED,eAAe,UAAU,CAAA"}
@@ -0,0 +1,37 @@
1
+ import { db } from '../db/index.js';
2
+ import { resourceTags } from '../db/schema.js';
3
+ import { eq, sql, desc } from 'drizzle-orm';
4
+ import { getRequestLocale, localizeText } from '../i18n.js';
5
+ function sendError(reply, locale, status, error, code) {
6
+ reply.code(status).send({ success: false, error: localizeText(locale, error), code });
7
+ }
8
+ const tagsRoutes = async (fastify) => {
9
+ // GET / — public, count across all resources (no visibility filter)
10
+ fastify.get('/', async (_req, reply) => {
11
+ const rows = db
12
+ .select({ tag: resourceTags.tag, count: sql `count(*)`.as('count') })
13
+ .from(resourceTags)
14
+ .groupBy(resourceTags.tag)
15
+ .orderBy(desc(sql `count(*)`))
16
+ .all();
17
+ reply.send({ success: true, data: rows });
18
+ });
19
+ // DELETE /:tag — admin
20
+ fastify.delete('/:tag', { preHandler: fastify.requireAdmin }, async (req, reply) => {
21
+ const locale = getRequestLocale(req);
22
+ const { tag } = req.params;
23
+ const countRow = db
24
+ .select({ count: sql `count(distinct resource_id)` })
25
+ .from(resourceTags)
26
+ .where(eq(resourceTags.tag, tag))
27
+ .get();
28
+ const affectedResources = countRow?.count ?? 0;
29
+ db.delete(resourceTags).where(eq(resourceTags.tag, tag)).run();
30
+ reply.send({
31
+ success: true,
32
+ data: { message: localizeText(locale, '删除成功'), affectedResources },
33
+ });
34
+ });
35
+ };
36
+ export default tagsRoutes;
37
+ //# sourceMappingURL=tags.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tags.js","sourceRoot":"","sources":["../../src/routes/tags.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,EAAE,EAAE,MAAM,gBAAgB,CAAA;AACnC,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAA;AAC9C,OAAO,EAAE,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,aAAa,CAAA;AAC3C,OAAO,EAAE,gBAAgB,EAAE,YAAY,EAAE,MAAM,YAAY,CAAA;AAE3D,SAAS,SAAS,CAChB,KAAmB,EACnB,MAA2C,EAC3C,MAAc,EACd,KAAa,EACb,IAAY;IAEZ,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,YAAY,CAAC,MAAM,EAAE,KAAK,CAAC,EAAE,IAAI,EAAE,CAAC,CAAA;AACvF,CAAC;AAED,MAAM,UAAU,GAAuB,KAAK,EAAE,OAAO,EAAE,EAAE;IAEvD,oEAAoE;IACpE,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QACrC,MAAM,IAAI,GAAG,EAAE;aACZ,MAAM,CAAC,EAAE,GAAG,EAAE,YAAY,CAAC,GAAG,EAAE,KAAK,EAAE,GAAG,CAAQ,UAAU,CAAC,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC;aAC3E,IAAI,CAAC,YAAY,CAAC;aAClB,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC;aACzB,OAAO,CAAC,IAAI,CAAC,GAAG,CAAA,UAAU,CAAC,CAAC;aAC5B,GAAG,EAAE,CAAA;QACR,KAAK,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAA;IAC3C,CAAC,CAAC,CAAA;IAEF,uBAAuB;IACvB,OAAO,CAAC,MAAM,CAAC,OAAO,EAAE,EAAE,UAAU,EAAE,OAAO,CAAC,YAAY,EAAE,EAAE,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACjF,MAAM,MAAM,GAAG,gBAAgB,CAAC,GAAG,CAAC,CAAA;QACpC,MAAM,EAAE,GAAG,EAAE,GAAG,GAAG,CAAC,MAAyB,CAAA;QAE7C,MAAM,QAAQ,GAAG,EAAE;aAChB,MAAM,CAAC,EAAE,KAAK,EAAE,GAAG,CAAQ,6BAA6B,EAAE,CAAC;aAC3D,IAAI,CAAC,YAAY,CAAC;aAClB,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;aAChC,GAAG,EAAE,CAAA;QACR,MAAM,iBAAiB,GAAG,QAAQ,EAAE,KAAK,IAAI,CAAC,CAAA;QAE9C,EAAE,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,CAAA;QAC9D,KAAK,CAAC,IAAI,CAAC;YACT,OAAO,EAAE,IAAI;YACb,IAAI,EAAE,EAAE,OAAO,EAAE,YAAY,CAAC,MAAM,EAAE,MAAM,CAAC,EAAE,iBAAiB,EAAE;SACnE,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;AACJ,CAAC,CAAA;AAED,eAAe,UAAU,CAAA"}
@@ -0,0 +1,4 @@
1
+ import type { FastifyPluginAsync } from 'fastify';
2
+ declare const usersRoutes: FastifyPluginAsync;
3
+ export default usersRoutes;
4
+ //# sourceMappingURL=users.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"users.d.ts","sourceRoot":"","sources":["../../src/routes/users.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,kBAAkB,EAAgB,MAAM,SAAS,CAAA;AAuD/D,QAAA,MAAM,WAAW,EAAE,kBAoJlB,CAAA;AAED,eAAe,WAAW,CAAA"}