@hkonda/loco-translate 1.0.0

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 (92) hide show
  1. package/README.md +284 -0
  2. package/bin/loco.js +5 -0
  3. package/dist/assets/index-CGo6e-bA.js +59 -0
  4. package/dist/assets/index-DBcQDZ75.css +1 -0
  5. package/dist/index.html +13 -0
  6. package/dist-server/app.d.ts +3 -0
  7. package/dist-server/app.d.ts.map +1 -0
  8. package/dist-server/app.js +154 -0
  9. package/dist-server/app.js.map +1 -0
  10. package/dist-server/config.d.ts +16 -0
  11. package/dist-server/config.d.ts.map +1 -0
  12. package/dist-server/config.js +41 -0
  13. package/dist-server/config.js.map +1 -0
  14. package/dist-server/db/index.d.ts +8 -0
  15. package/dist-server/db/index.d.ts.map +1 -0
  16. package/dist-server/db/index.js +28 -0
  17. package/dist-server/db/index.js.map +1 -0
  18. package/dist-server/db/schema.d.ts +851 -0
  19. package/dist-server/db/schema.d.ts.map +1 -0
  20. package/dist-server/db/schema.js +65 -0
  21. package/dist-server/db/schema.js.map +1 -0
  22. package/dist-server/db/seed.d.ts +14 -0
  23. package/dist-server/db/seed.d.ts.map +1 -0
  24. package/dist-server/db/seed.js +229 -0
  25. package/dist-server/db/seed.js.map +1 -0
  26. package/dist-server/index.d.ts +2 -0
  27. package/dist-server/index.d.ts.map +1 -0
  28. package/dist-server/index.js +31 -0
  29. package/dist-server/index.js.map +1 -0
  30. package/dist-server/routes/ai-jobs.d.ts +5 -0
  31. package/dist-server/routes/ai-jobs.d.ts.map +1 -0
  32. package/dist-server/routes/ai-jobs.js +141 -0
  33. package/dist-server/routes/ai-jobs.js.map +1 -0
  34. package/dist-server/routes/backup.d.ts +5 -0
  35. package/dist-server/routes/backup.d.ts.map +1 -0
  36. package/dist-server/routes/backup.js +125 -0
  37. package/dist-server/routes/backup.js.map +1 -0
  38. package/dist-server/routes/chrome-extension.d.ts +5 -0
  39. package/dist-server/routes/chrome-extension.d.ts.map +1 -0
  40. package/dist-server/routes/chrome-extension.js +140 -0
  41. package/dist-server/routes/chrome-extension.js.map +1 -0
  42. package/dist-server/routes/export.d.ts +5 -0
  43. package/dist-server/routes/export.d.ts.map +1 -0
  44. package/dist-server/routes/export.js +95 -0
  45. package/dist-server/routes/export.js.map +1 -0
  46. package/dist-server/routes/languages.d.ts +5 -0
  47. package/dist-server/routes/languages.d.ts.map +1 -0
  48. package/dist-server/routes/languages.js +36 -0
  49. package/dist-server/routes/languages.js.map +1 -0
  50. package/dist-server/routes/project.d.ts +5 -0
  51. package/dist-server/routes/project.d.ts.map +1 -0
  52. package/dist-server/routes/project.js +151 -0
  53. package/dist-server/routes/project.js.map +1 -0
  54. package/dist-server/routes/prompts.d.ts +5 -0
  55. package/dist-server/routes/prompts.d.ts.map +1 -0
  56. package/dist-server/routes/prompts.js +90 -0
  57. package/dist-server/routes/prompts.js.map +1 -0
  58. package/dist-server/routes/screenshots.d.ts +5 -0
  59. package/dist-server/routes/screenshots.d.ts.map +1 -0
  60. package/dist-server/routes/screenshots.js +71 -0
  61. package/dist-server/routes/screenshots.js.map +1 -0
  62. package/dist-server/routes/textnodes.d.ts +5 -0
  63. package/dist-server/routes/textnodes.d.ts.map +1 -0
  64. package/dist-server/routes/textnodes.js +318 -0
  65. package/dist-server/routes/textnodes.js.map +1 -0
  66. package/dist-server/routes/translations.d.ts +5 -0
  67. package/dist-server/routes/translations.d.ts.map +1 -0
  68. package/dist-server/routes/translations.js +224 -0
  69. package/dist-server/routes/translations.js.map +1 -0
  70. package/dist-server/scripts/backup.d.ts +2 -0
  71. package/dist-server/scripts/backup.d.ts.map +1 -0
  72. package/dist-server/scripts/backup.js +26 -0
  73. package/dist-server/scripts/backup.js.map +1 -0
  74. package/dist-server/services/ai-translation.d.ts +21 -0
  75. package/dist-server/services/ai-translation.d.ts.map +1 -0
  76. package/dist-server/services/ai-translation.js +137 -0
  77. package/dist-server/services/ai-translation.js.map +1 -0
  78. package/dist-server/services/job-manager.d.ts +67 -0
  79. package/dist-server/services/job-manager.d.ts.map +1 -0
  80. package/dist-server/services/job-manager.js +159 -0
  81. package/dist-server/services/job-manager.js.map +1 -0
  82. package/dist-server/services/slot-pattern.d.ts +16 -0
  83. package/dist-server/services/slot-pattern.d.ts.map +1 -0
  84. package/dist-server/services/slot-pattern.js +68 -0
  85. package/dist-server/services/slot-pattern.js.map +1 -0
  86. package/dist-server/utils/network.d.ts +2 -0
  87. package/dist-server/utils/network.d.ts.map +1 -0
  88. package/dist-server/utils/network.js +16 -0
  89. package/dist-server/utils/network.js.map +1 -0
  90. package/package.json +47 -0
  91. package/public/loco.js +2386 -0
  92. package/public/loco.min.js +4 -0
@@ -0,0 +1,125 @@
1
+ import { join, dirname, basename } from 'path';
2
+ import { fileURLToPath } from 'url';
3
+ import { existsSync, mkdirSync, createReadStream, readdirSync, statSync, unlinkSync } from 'fs';
4
+ import { rawDb } from '../db/index.js';
5
+ const __filename = fileURLToPath(import.meta.url);
6
+ const __dirname = dirname(__filename);
7
+ const dataDir = join(__dirname, '..', 'data');
8
+ const backupsDir = join(dataDir, 'backups');
9
+ function sanitizeFilename(name) {
10
+ const base = basename(name);
11
+ if (base !== name || !base.startsWith('loco-backup-') || !base.endsWith('.db'))
12
+ return null;
13
+ if (/[^a-zA-Z0-9._-]/.test(base))
14
+ return null;
15
+ return base;
16
+ }
17
+ export default async function backupRoutes(app, opts) {
18
+ // List all backups
19
+ app.get('/api/backups', {
20
+ schema: {
21
+ tags: ['Backup'],
22
+ summary: 'List all available backups',
23
+ response: {
24
+ 200: {
25
+ type: 'array',
26
+ items: {
27
+ type: 'object',
28
+ properties: {
29
+ filename: { type: 'string' },
30
+ size: { type: 'number' },
31
+ created: { type: 'string' },
32
+ },
33
+ },
34
+ },
35
+ },
36
+ },
37
+ }, async (_req, reply) => {
38
+ mkdirSync(backupsDir, { recursive: true });
39
+ const files = readdirSync(backupsDir)
40
+ .filter(f => f.startsWith('loco-backup-') && f.endsWith('.db'))
41
+ .map(f => {
42
+ const st = statSync(join(backupsDir, f));
43
+ return { filename: f, size: st.size, created: st.mtime.toISOString() };
44
+ })
45
+ .sort((a, b) => b.created.localeCompare(a.created));
46
+ return reply.send(files);
47
+ });
48
+ // Create a new backup (without downloading)
49
+ app.post('/api/backups', {
50
+ schema: {
51
+ tags: ['Backup'],
52
+ summary: 'Create a new database backup',
53
+ response: {
54
+ 200: {
55
+ type: 'object',
56
+ properties: {
57
+ filename: { type: 'string' },
58
+ size: { type: 'number' },
59
+ created: { type: 'string' },
60
+ },
61
+ },
62
+ },
63
+ },
64
+ }, async (_req, reply) => {
65
+ mkdirSync(backupsDir, { recursive: true });
66
+ const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
67
+ const filename = `loco-backup-${timestamp}.db`;
68
+ const backupPath = join(backupsDir, filename);
69
+ rawDb.exec(`VACUUM INTO '${backupPath.replace(/'/g, "''")}'`);
70
+ if (!existsSync(backupPath)) {
71
+ return reply.code(500).send({ error: 'Backup failed' });
72
+ }
73
+ const st = statSync(backupPath);
74
+ return reply.send({ filename, size: st.size, created: st.mtime.toISOString() });
75
+ });
76
+ // Delete a backup
77
+ app.delete('/api/backups/:filename', {
78
+ schema: {
79
+ tags: ['Backup'],
80
+ summary: 'Delete a backup file',
81
+ params: {
82
+ type: 'object',
83
+ required: ['filename'],
84
+ properties: { filename: { type: 'string' } },
85
+ },
86
+ },
87
+ }, async (req, reply) => {
88
+ const safe = sanitizeFilename(req.params.filename);
89
+ if (!safe)
90
+ return reply.code(400).send({ error: 'Invalid filename' });
91
+ const backupPath = join(backupsDir, safe);
92
+ if (!existsSync(backupPath)) {
93
+ return reply.code(404).send({ error: 'Backup not found' });
94
+ }
95
+ unlinkSync(backupPath);
96
+ return reply.send({ ok: true });
97
+ });
98
+ // Download a backup (legacy + direct download)
99
+ app.get('/api/backup', {
100
+ schema: {
101
+ tags: ['Backup'],
102
+ summary: 'Download a timestamped SQLite database backup',
103
+ response: {
104
+ 200: {
105
+ type: 'string',
106
+ description: 'SQLite database file (binary)',
107
+ },
108
+ },
109
+ },
110
+ }, async (_req, reply) => {
111
+ mkdirSync(backupsDir, { recursive: true });
112
+ const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
113
+ const backupPath = join(backupsDir, `loco-backup-${timestamp}.db`);
114
+ rawDb.exec(`VACUUM INTO '${backupPath.replace(/'/g, "''")}'`);
115
+ if (!existsSync(backupPath)) {
116
+ return reply.code(500).send({ error: 'Backup failed' });
117
+ }
118
+ const stream = createReadStream(backupPath);
119
+ return reply
120
+ .header('Content-Type', 'application/octet-stream')
121
+ .header('Content-Disposition', `attachment; filename="loco-backup-${timestamp}.db"`)
122
+ .send(stream);
123
+ });
124
+ }
125
+ //# sourceMappingURL=backup.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"backup.js","sourceRoot":"","sources":["../../server/routes/backup.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import { FastifyInstance } from 'fastify';
2
+ export default function chromeExtensionRoutes(app: FastifyInstance, opts: {
3
+ project: any;
4
+ }): Promise<void>;
5
+ //# sourceMappingURL=chrome-extension.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"chrome-extension.d.ts","sourceRoot":"","sources":["../../server/routes/chrome-extension.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAY1C,wBAA8B,qBAAqB,CAAC,GAAG,EAAE,eAAe,EAAE,IAAI,EAAE;IAAE,OAAO,EAAE,GAAG,CAAA;CAAE,iBA4I/F"}
@@ -0,0 +1,140 @@
1
+ import { rawDb } from '../db/index.js';
2
+ function cleanUrl(raw) {
3
+ try {
4
+ const u = new URL(raw);
5
+ return u.origin + u.pathname;
6
+ }
7
+ catch {
8
+ return raw;
9
+ }
10
+ }
11
+ export default async function chromeExtensionRoutes(app, opts) {
12
+ // ── GET /api/ext/phrases
13
+ app.get('/api/ext/phrases', {
14
+ schema: {
15
+ tags: ['Chrome Extension'],
16
+ summary: 'Get phrases + translations filtered by URL',
17
+ querystring: {
18
+ type: 'object',
19
+ properties: {
20
+ url: { type: 'string' },
21
+ lang: { type: 'string' },
22
+ status: { type: 'string', enum: ['pending', 'approved', 'trash'] },
23
+ search: { type: 'string' },
24
+ translated: { type: 'string', enum: ['yes', 'no'] },
25
+ },
26
+ required: ['url'],
27
+ },
28
+ response: {
29
+ 200: {
30
+ type: 'array',
31
+ items: { type: 'object', additionalProperties: true },
32
+ },
33
+ },
34
+ },
35
+ }, (req) => {
36
+ const { project } = opts;
37
+ const { url, lang, status, search, translated } = req.query;
38
+ if (!url)
39
+ return { error: 'url query param is required' };
40
+ const targetLang = lang || 'zh-Hans';
41
+ const targetStatus = ['pending', 'approved', 'trash'].includes(status) ? status : 'pending';
42
+ const normalizedUrl = cleanUrl(url);
43
+ let sql = `
44
+ SELECT k.id AS key_id, k.key, k.context, k.var_slots, k.url, k.status, k.created_at,
45
+ t.id AS translation_id, t.value
46
+ FROM textnodes k
47
+ LEFT JOIN translations t
48
+ ON t.project_id = k.project_id AND t.key = k.key AND t.lang = ?
49
+ WHERE k.project_id = ? AND k.status = ?`;
50
+ const params = [targetLang, project.id, targetStatus];
51
+ if (translated === 'yes') {
52
+ sql += ` AND t.value IS NOT NULL AND t.value != ''`;
53
+ }
54
+ else if (translated === 'no') {
55
+ sql += ` AND (t.value IS NULL OR t.value = '')`;
56
+ }
57
+ if (search && typeof search === 'string' && search.trim()) {
58
+ sql += ` AND (k.key LIKE ? OR k.context LIKE ?) ESCAPE '\\\\'`;
59
+ const escaped = search.trim().replace(/[%_]/g, c => '\\' + c);
60
+ const term = `%${escaped}%`;
61
+ params.push(term, term);
62
+ }
63
+ sql += ` ORDER BY k.created_at DESC`;
64
+ const rows = rawDb.prepare(sql).all(...params);
65
+ const filtered = rows.filter((r) => r.url && cleanUrl(r.url) === normalizedUrl);
66
+ return filtered;
67
+ });
68
+ // ── GET /api/ext/phrases/counts
69
+ app.get('/api/ext/phrases/counts', {
70
+ schema: {
71
+ tags: ['Chrome Extension'],
72
+ summary: 'Get phrase counts by status filtered by URL',
73
+ querystring: {
74
+ type: 'object',
75
+ properties: { url: { type: 'string' } },
76
+ required: ['url'],
77
+ },
78
+ response: {
79
+ 200: {
80
+ type: 'object',
81
+ properties: {
82
+ pending: { type: 'number' },
83
+ approved: { type: 'number' },
84
+ trash: { type: 'number' },
85
+ },
86
+ },
87
+ },
88
+ },
89
+ }, (req) => {
90
+ const { project } = opts;
91
+ const { url } = req.query;
92
+ if (!url)
93
+ return { error: 'url query param is required' };
94
+ const normalizedUrl = cleanUrl(url);
95
+ const allRows = rawDb.prepare(`SELECT status, url FROM textnodes WHERE project_id = ?`).all(project.id);
96
+ const result = { pending: 0, approved: 0, trash: 0 };
97
+ for (const r of allRows) {
98
+ if (r.url && cleanUrl(r.url) === normalizedUrl && result.hasOwnProperty(r.status)) {
99
+ result[r.status]++;
100
+ }
101
+ }
102
+ return result;
103
+ });
104
+ // ── GET /api/ext/phrases/keys
105
+ app.get('/api/ext/phrases/keys', {
106
+ schema: {
107
+ tags: ['Chrome Extension'],
108
+ summary: 'Get lightweight keys + var_slots for URL pattern matching',
109
+ querystring: {
110
+ type: 'object',
111
+ properties: { url: { type: 'string' } },
112
+ required: ['url'],
113
+ },
114
+ response: {
115
+ 200: {
116
+ type: 'array',
117
+ items: {
118
+ type: 'object',
119
+ properties: {
120
+ key: { type: 'string' },
121
+ var_slots: { type: ['string', 'null'] },
122
+ },
123
+ },
124
+ },
125
+ },
126
+ },
127
+ }, (req) => {
128
+ const { project } = opts;
129
+ const { url } = req.query;
130
+ if (!url)
131
+ return { error: 'url query param is required' };
132
+ const normalizedUrl = cleanUrl(url);
133
+ const allWithUrl = rawDb.prepare(`SELECT key, var_slots, url FROM textnodes WHERE project_id = ? AND status != 'trash'`).all(project.id);
134
+ const filtered = allWithUrl
135
+ .filter(r => r.url && cleanUrl(r.url) === normalizedUrl)
136
+ .map(r => ({ key: r.key, var_slots: r.var_slots || null }));
137
+ return filtered;
138
+ });
139
+ }
140
+ //# sourceMappingURL=chrome-extension.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"chrome-extension.js","sourceRoot":"","sources":["../../server/routes/chrome-extension.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import { FastifyInstance } from 'fastify';
2
+ export default function exportRoutes(app: FastifyInstance, opts: {
3
+ project: any;
4
+ }): Promise<void>;
5
+ //# sourceMappingURL=export.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"export.d.ts","sourceRoot":"","sources":["../../server/routes/export.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAG1C,wBAA8B,YAAY,CAAC,GAAG,EAAE,eAAe,EAAE,IAAI,EAAE;IAAE,OAAO,EAAE,GAAG,CAAA;CAAE,iBA8CtF"}
@@ -0,0 +1,95 @@
1
+ import { rawDb } from '../db/index.js';
2
+ export default async function exportRoutes(app, opts) {
3
+ app.get('/api/export', {
4
+ schema: {
5
+ tags: ['Export'],
6
+ summary: 'Export all approved translations + config as JSON',
7
+ response: {
8
+ 200: {
9
+ type: 'object',
10
+ properties: {
11
+ languages: { type: 'array', items: { type: 'string' } },
12
+ languageNames: { type: 'object', additionalProperties: { type: 'string' } },
13
+ translations: { type: 'object', additionalProperties: true },
14
+ timestamp: { type: 'number' },
15
+ },
16
+ },
17
+ },
18
+ },
19
+ }, () => {
20
+ return buildExport(opts.project);
21
+ });
22
+ app.get('/api/export/:lang', {
23
+ schema: {
24
+ tags: ['Export'],
25
+ summary: 'Export translations for a single language',
26
+ params: {
27
+ type: 'object',
28
+ properties: { lang: { type: 'string' } },
29
+ required: ['lang'],
30
+ },
31
+ response: {
32
+ 200: {
33
+ type: 'object',
34
+ properties: {
35
+ languages: { type: 'array', items: { type: 'string' } },
36
+ languageNames: { type: 'object', additionalProperties: { type: 'string' } },
37
+ translations: { type: 'object', additionalProperties: true },
38
+ timestamp: { type: 'number' },
39
+ },
40
+ },
41
+ },
42
+ },
43
+ }, (req) => {
44
+ return buildExportSingleLang(opts.project, req.params.lang);
45
+ });
46
+ }
47
+ function getLanguageNames(project) {
48
+ let settingsLangs = [];
49
+ try {
50
+ settingsLangs = JSON.parse(project.settings || '{}').languages || [];
51
+ }
52
+ catch { }
53
+ const languageNames = {};
54
+ settingsLangs.forEach((l) => { if (l.code && l.name)
55
+ languageNames[l.code] = l.name; });
56
+ return languageNames;
57
+ }
58
+ function buildExport(project) {
59
+ const langRows = rawDb.prepare(`
60
+ SELECT DISTINCT t.lang FROM translations t
61
+ INNER JOIN textnodes k ON k.project_id = t.project_id AND k.key = t.key AND k.context = t.context
62
+ WHERE t.project_id = ? AND k.status = 'approved'
63
+ ORDER BY t.lang
64
+ `).all(project.id);
65
+ const languages = langRows.map(r => r.lang);
66
+ const languageNames = getLanguageNames(project);
67
+ const translations = {};
68
+ const stmt = rawDb.prepare(`
69
+ SELECT t.key, t.context, t.value FROM translations t
70
+ INNER JOIN textnodes k ON k.project_id = t.project_id AND k.key = t.key AND k.context = t.context
71
+ WHERE t.project_id = ? AND t.lang = ? AND k.status = 'approved'
72
+ `);
73
+ for (const lang of languages) {
74
+ const rows = stmt.all(project.id, lang);
75
+ translations[lang] = rows.map(r => ({ key: r.key, context: r.context || '', value: r.value }));
76
+ }
77
+ return { languages, languageNames, translations, timestamp: Date.now() };
78
+ }
79
+ function buildExportSingleLang(project, lang) {
80
+ const languageNames = getLanguageNames(project);
81
+ const rows = rawDb.prepare(`
82
+ SELECT t.key, t.context, t.value FROM translations t
83
+ INNER JOIN textnodes k ON k.project_id = t.project_id AND k.key = t.key AND k.context = t.context
84
+ WHERE t.project_id = ? AND t.lang = ? AND k.status = 'approved'
85
+ `).all(project.id, lang);
86
+ const translations = {};
87
+ translations[lang] = rows.map(r => ({ key: r.key, context: r.context || '', value: r.value }));
88
+ return {
89
+ languages: [lang],
90
+ languageNames,
91
+ translations,
92
+ timestamp: Date.now(),
93
+ };
94
+ }
95
+ //# sourceMappingURL=export.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"export.js","sourceRoot":"","sources":["../../server/routes/export.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import { FastifyInstance } from 'fastify';
2
+ export default function languagesRoutes(app: FastifyInstance, opts: {
3
+ project: any;
4
+ }): Promise<void>;
5
+ //# sourceMappingURL=languages.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"languages.d.ts","sourceRoot":"","sources":["../../server/routes/languages.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAG1C,wBAA8B,eAAe,CAAC,GAAG,EAAE,eAAe,EAAE,IAAI,EAAE;IAAE,OAAO,EAAE,GAAG,CAAA;CAAE,iBA8BzF"}
@@ -0,0 +1,36 @@
1
+ import { rawDb } from '../db/index.js';
2
+ export default async function languagesRoutes(app, opts) {
3
+ app.get('/api/languages', {
4
+ schema: {
5
+ tags: ['Languages'],
6
+ summary: 'Get available languages',
7
+ response: {
8
+ 200: {
9
+ type: 'array',
10
+ items: {
11
+ type: 'object',
12
+ properties: {
13
+ code: { type: 'string' },
14
+ name: { type: 'string' },
15
+ },
16
+ },
17
+ },
18
+ },
19
+ },
20
+ }, () => {
21
+ const { project } = opts;
22
+ const rows = rawDb
23
+ .prepare('SELECT DISTINCT lang FROM translations WHERE project_id = ? ORDER BY lang')
24
+ .all(project.id);
25
+ let settingsLangs = [];
26
+ try {
27
+ settingsLangs = JSON.parse(project.settings || '{}').languages || [];
28
+ }
29
+ catch { }
30
+ const nameMap = {};
31
+ settingsLangs.forEach((l) => { if (l.code && l.name)
32
+ nameMap[l.code] = l.name; });
33
+ return rows.map(r => ({ code: r.lang, name: nameMap[r.lang] || r.lang }));
34
+ });
35
+ }
36
+ //# sourceMappingURL=languages.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"languages.js","sourceRoot":"","sources":["../../server/routes/languages.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,KAAK,EAAE,MAAM,gBAAgB,CAAC;AAEvC,MAAM,CAAC,OAAO,CAAC,KAAK,UAAU,eAAe,CAAC,GAAoB,EAAE,IAAsB;IAExF,GAAG,CAAC,GAAG,CAAC,gBAAgB,EAAE;QACxB,MAAM,EAAE;YACN,IAAI,EAAE,CAAC,WAAW,CAAC;YACnB,OAAO,EAAE,yBAAyB;YAClC,QAAQ,EAAE;gBACR,GAAG,EAAE;oBACH,IAAI,EAAE,OAAO;oBACb,KAAK,EAAE;wBACL,IAAI,EAAE,QAAQ;wBACd,UAAU,EAAE;4BACV,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;4BACxB,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;yBACzB;qBACF;iBACF;aACF;SACF;KACF,EAAE,GAAG,EAAE;QACN,MAAM,EAAE,OAAO,EAAE,GAAG,IAAI,CAAC;QACzB,MAAM,IAAI,GAAG,KAAK;aACf,OAAO,CAAC,2EAA2E,CAAC;aACpF,GAAG,CAAC,OAAO,CAAC,EAAE,CAAuB,CAAC;QACzC,IAAI,aAAa,GAAU,EAAE,CAAC;QAC9B,IAAI,CAAC;YAAC,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,SAAS,IAAI,EAAE,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;QACtF,MAAM,OAAO,GAA2B,EAAE,CAAC;QAC3C,aAAa,CAAC,OAAO,CAAC,CAAC,CAAM,EAAE,EAAE,GAAG,IAAI,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,IAAI;YAAE,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QACvF,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;IAC5E,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,5 @@
1
+ import { FastifyInstance } from 'fastify';
2
+ export default function projectRoutes(app: FastifyInstance, opts: {
3
+ project: any;
4
+ }): Promise<void>;
5
+ //# sourceMappingURL=project.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"project.d.ts","sourceRoot":"","sources":["../../server/routes/project.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAM1C,wBAA8B,aAAa,CAAC,GAAG,EAAE,eAAe,EAAE,IAAI,EAAE;IAAE,OAAO,EAAE,GAAG,CAAA;CAAE,iBAmJvF"}
@@ -0,0 +1,151 @@
1
+ import { randomUUID } from 'crypto';
2
+ import { rawDb } from '../db/index.js';
3
+ import { reloadProject } from '../db/seed.js';
4
+ import { LIMITS } from '../config.js';
5
+ export default async function projectRoutes(app, opts) {
6
+ let { project } = opts;
7
+ // ── Health check
8
+ app.get('/api', {
9
+ schema: {
10
+ tags: ['Project'],
11
+ summary: 'Health check',
12
+ response: {
13
+ 200: {
14
+ type: 'object',
15
+ properties: {
16
+ name: { type: 'string' },
17
+ status: { type: 'string' },
18
+ version: { type: 'string' },
19
+ },
20
+ },
21
+ },
22
+ },
23
+ }, () => {
24
+ return { name: 'loco', status: 'running', version: '1.0.0' };
25
+ });
26
+ // ── Project info
27
+ app.get('/api/project', {
28
+ schema: {
29
+ tags: ['Project'],
30
+ summary: 'Get project info',
31
+ response: {
32
+ 200: {
33
+ type: 'object',
34
+ properties: {
35
+ id: { type: 'number' },
36
+ name: { type: 'string' },
37
+ api_key: { type: 'string' },
38
+ settings: { type: 'string' },
39
+ created_at: { type: 'string' },
40
+ },
41
+ },
42
+ },
43
+ },
44
+ }, () => {
45
+ return project;
46
+ });
47
+ app.patch('/api/project', {
48
+ schema: {
49
+ tags: ['Project'],
50
+ summary: 'Update project name',
51
+ body: {
52
+ type: 'object',
53
+ properties: { name: { type: 'string' } },
54
+ },
55
+ response: { 200: { type: 'object', additionalProperties: true } },
56
+ },
57
+ }, (req) => {
58
+ const { name } = req.body;
59
+ if (name && typeof name === 'string') {
60
+ rawDb.prepare('UPDATE projects SET name = ? WHERE id = ?').run(name.trim(), project.id);
61
+ project = reloadProject(project.id);
62
+ opts.project = project;
63
+ }
64
+ return project;
65
+ });
66
+ app.post('/api/project/regenerate-key', {
67
+ schema: {
68
+ tags: ['Project'],
69
+ summary: 'Regenerate API key',
70
+ response: { 200: { type: 'object', additionalProperties: true } },
71
+ },
72
+ }, () => {
73
+ const newKey = randomUUID().replace(/-/g, '').slice(0, LIMITS.API_KEY_LENGTH);
74
+ rawDb.prepare('UPDATE projects SET api_key = ? WHERE id = ?').run(newKey, project.id);
75
+ project = reloadProject(project.id);
76
+ opts.project = project;
77
+ return project;
78
+ });
79
+ app.patch('/api/project/settings', {
80
+ schema: {
81
+ tags: ['Project'],
82
+ summary: 'Update project settings',
83
+ body: {
84
+ type: 'object',
85
+ properties: { settings: { type: 'object', additionalProperties: true } },
86
+ },
87
+ response: { 200: { type: 'object', additionalProperties: true } },
88
+ },
89
+ }, (req) => {
90
+ const { settings } = req.body;
91
+ if (settings && typeof settings === 'object') {
92
+ rawDb.prepare('UPDATE projects SET settings = ? WHERE id = ?').run(JSON.stringify(settings), project.id);
93
+ project = reloadProject(project.id);
94
+ opts.project = project;
95
+ }
96
+ return project;
97
+ });
98
+ app.get('/api/project/crawler-config', {
99
+ schema: {
100
+ tags: ['Project'],
101
+ summary: 'Get crawler configuration',
102
+ response: {
103
+ 200: {
104
+ type: 'object',
105
+ properties: {
106
+ blockedTags: { type: 'object', additionalProperties: true },
107
+ inlineTags: { type: 'object', additionalProperties: true },
108
+ domContextSelectors: { type: 'object', additionalProperties: true },
109
+ contextDepth: { type: 'number' },
110
+ domContextEnabled: { type: 'boolean' },
111
+ screenshotsEnabled: { type: 'boolean' },
112
+ },
113
+ },
114
+ },
115
+ },
116
+ }, () => {
117
+ let s;
118
+ try {
119
+ s = JSON.parse(project.settings || '{}');
120
+ }
121
+ catch {
122
+ s = {};
123
+ }
124
+ return {
125
+ blockedTags: s.blockedTags || {},
126
+ inlineTags: s.inlineTags || {},
127
+ domContextSelectors: s.domContextSelectors || {},
128
+ contextDepth: typeof s.contextDepth === 'number' ? s.contextDepth : 0,
129
+ domContextEnabled: s.domContextEnabled !== false,
130
+ screenshotsEnabled: s.screenshotsEnabled !== false,
131
+ };
132
+ });
133
+ app.post('/api/project/reset-database', {
134
+ schema: {
135
+ tags: ['Project'],
136
+ summary: 'Reset database (delete all data for this project)',
137
+ response: { 200: { type: 'object', properties: { ok: { type: 'boolean' } } } },
138
+ },
139
+ }, () => {
140
+ const run = rawDb.transaction(() => {
141
+ rawDb.prepare('DELETE FROM translations WHERE project_id = ?').run(project.id);
142
+ rawDb.prepare('DELETE FROM textnodes WHERE project_id = ?').run(project.id);
143
+ rawDb.prepare('DELETE FROM page_screenshots WHERE project_id = ?').run(project.id);
144
+ // Reset auto-increment counters so IDs restart from 1
145
+ rawDb.prepare("DELETE FROM sqlite_sequence WHERE name IN ('textnodes','translations','page_screenshots')").run();
146
+ });
147
+ run();
148
+ return { ok: true };
149
+ });
150
+ }
151
+ //# sourceMappingURL=project.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"project.js","sourceRoot":"","sources":["../../server/routes/project.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import { FastifyInstance } from 'fastify';
2
+ export default function promptsRoutes(app: FastifyInstance, opts: {
3
+ project: any;
4
+ }): Promise<void>;
5
+ //# sourceMappingURL=prompts.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompts.d.ts","sourceRoot":"","sources":["../../server/routes/prompts.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAG1C,wBAA8B,aAAa,CAAC,GAAG,EAAE,eAAe,EAAE,IAAI,EAAE;IAAE,OAAO,EAAE,GAAG,CAAA;CAAE,iBA2FvF"}