kingkont 0.17.1 → 0.17.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/lib/projectPaths.js +62 -0
- package/package.json +1 -1
- package/renderer/board.js +16 -0
- package/server.js +29 -0
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
// lib/projectPaths.js — server-side registry абсолютных путей проектов.
|
|
2
|
+
//
|
|
3
|
+
// FSAH (File System Access API) — это browser-only. Сервер (Node.js) не
|
|
4
|
+
// может работать с его DirectoryHandle. Но в Electron мы можем извлечь
|
|
5
|
+
// абс-путь любого файла внутри папки через webUtils.getPathForFile и
|
|
6
|
+
// зарегистрировать здесь — после чего сервер напрямую читает/пишет через
|
|
7
|
+
// node:fs.
|
|
8
|
+
//
|
|
9
|
+
// Для cloud-проектов путь однозначно: userData/cloud-projects/<id>/
|
|
10
|
+
// (см. main.js cloudResolve). Их регистрировать не нужно — путь
|
|
11
|
+
// derivable.
|
|
12
|
+
//
|
|
13
|
+
// Для folder-проектов (showDirectoryPicker) путь регистрируется
|
|
14
|
+
// renderer'ом на openFilm через POST /api/project/register
|
|
15
|
+
// {projectKey, absPath}.
|
|
16
|
+
//
|
|
17
|
+
// API:
|
|
18
|
+
// register(projectKey, absPath)
|
|
19
|
+
// resolve(projectKey) → absPath или null
|
|
20
|
+
// resolveBoardPath(projectKey, kind, name) → абс-путь board-folder'a
|
|
21
|
+
|
|
22
|
+
'use strict';
|
|
23
|
+
|
|
24
|
+
const path = require('node:path');
|
|
25
|
+
|
|
26
|
+
// Map<projectKey, absPath>.
|
|
27
|
+
// projectKey формат: 'cloud:<id>' | 'folder:<name>'.
|
|
28
|
+
const paths = new Map();
|
|
29
|
+
let _userDataDir = null;
|
|
30
|
+
|
|
31
|
+
function init({ userDataDir }) {
|
|
32
|
+
_userDataDir = userDataDir;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
function register(projectKey, absPath) {
|
|
36
|
+
if (!projectKey || !absPath) return;
|
|
37
|
+
paths.set(projectKey, absPath);
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
function resolve(projectKey) {
|
|
41
|
+
if (!projectKey) return null;
|
|
42
|
+
// Cloud — derivable из userDataDir.
|
|
43
|
+
if (projectKey.startsWith('cloud:')) {
|
|
44
|
+
if (!_userDataDir) return null;
|
|
45
|
+
const id = projectKey.slice('cloud:'.length);
|
|
46
|
+
const safe = String(id).replace(/[\\/]/g, '_');
|
|
47
|
+
return path.join(_userDataDir, 'cloud-projects', safe);
|
|
48
|
+
}
|
|
49
|
+
// Folder — из registry.
|
|
50
|
+
return paths.get(projectKey) || null;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
// Полный путь к папке доски (scene) внутри проекта. kind: episode|character|location.
|
|
54
|
+
function resolveBoardPath(projectKey, kind, name) {
|
|
55
|
+
const root = resolve(projectKey);
|
|
56
|
+
if (!root) return null;
|
|
57
|
+
if (kind === 'character') return path.join(root, '_characters', name);
|
|
58
|
+
if (kind === 'location') return path.join(root, '_locations', name);
|
|
59
|
+
return path.join(root, name); // episode
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
module.exports = { init, register, resolve, resolveBoardPath };
|
package/package.json
CHANGED
package/renderer/board.js
CHANGED
|
@@ -1182,6 +1182,22 @@ async function openFilm(handle) {
|
|
|
1182
1182
|
// Запоминаем что юзер сейчас в проекте → Cmd+R откроет его снова.
|
|
1183
1183
|
try { localStorage.setItem('lastLocation', 'project'); } catch {}
|
|
1184
1184
|
window.appProject?.notifyState(true);
|
|
1185
|
+
// Register абс-путь folder-проекта на сервере — так server-side задачи
|
|
1186
|
+
// (jobsHub poller, future server-tools) могут писать файлы напрямую без
|
|
1187
|
+
// FSAH. Cloud не регистрируем (path derivable из cloudProjectId).
|
|
1188
|
+
// Не блокируем дальнейший openFilm — сервер просто не получит путь и
|
|
1189
|
+
// server-side ops для этого проекта будут недоступны.
|
|
1190
|
+
if (handle && !window.cloudFsShim?.isCloudHandle?.(handle)) {
|
|
1191
|
+
try {
|
|
1192
|
+
const absPath = await deriveFolderAbsPath(handle);
|
|
1193
|
+
if (absPath) {
|
|
1194
|
+
await fetch('/api/project/register', {
|
|
1195
|
+
method: 'POST', headers: { 'Content-Type': 'application/json' },
|
|
1196
|
+
body: JSON.stringify({ projectKey: 'folder:' + handle.name, absPath }),
|
|
1197
|
+
});
|
|
1198
|
+
}
|
|
1199
|
+
} catch (e) { vlog('warn', 'project register failed: ' + e?.message); }
|
|
1200
|
+
}
|
|
1185
1201
|
ensureClaudeMd(handle).catch(() => {});
|
|
1186
1202
|
// Подзаголовок шапки = имя открытого проекта (вместо «Видео-редактор»).
|
|
1187
1203
|
const sub = $('brandSub');
|
package/server.js
CHANGED
|
@@ -16,6 +16,7 @@ const providers = require('./lib/providers');
|
|
|
16
16
|
const chatSession = require('./lib/chatSession');
|
|
17
17
|
const jobsHub = require('./lib/jobsHub');
|
|
18
18
|
const wsHub = require('./lib/wsHub');
|
|
19
|
+
const projectPaths = require('./lib/projectPaths');
|
|
19
20
|
|
|
20
21
|
// ---------- .env loader (без зависимостей) ----------
|
|
21
22
|
function loadEnv() {
|
|
@@ -371,6 +372,30 @@ async function handleJobsList(res, url) {
|
|
|
371
372
|
} catch (e) { sendError(res, e, 500); }
|
|
372
373
|
}
|
|
373
374
|
|
|
375
|
+
// =============================================================================
|
|
376
|
+
// Projects: regission абсолютного пути folder-проекта (renderer достаёт через
|
|
377
|
+
// webUtils.getPathForFile). Cloud-проекты регистрировать не нужно — путь
|
|
378
|
+
// derivable (userData/cloud-projects/<id>/).
|
|
379
|
+
// =============================================================================
|
|
380
|
+
async function handleProjectRegister(req, res) {
|
|
381
|
+
try {
|
|
382
|
+
const body = await readJson(req);
|
|
383
|
+
const { projectKey, absPath } = body || {};
|
|
384
|
+
if (!projectKey) return send(res, 400, { error: 'projectKey обязателен' });
|
|
385
|
+
if (!absPath || typeof absPath !== 'string') return send(res, 400, { error: 'absPath обязателен' });
|
|
386
|
+
projectPaths.register(projectKey, absPath);
|
|
387
|
+
send(res, 200, { ok: true, registered: { projectKey, absPath } });
|
|
388
|
+
} catch (e) { sendError(res, e, 500); }
|
|
389
|
+
}
|
|
390
|
+
async function handleProjectResolve(res, url) {
|
|
391
|
+
try {
|
|
392
|
+
const projectKey = url.searchParams.get('projectKey');
|
|
393
|
+
if (!projectKey) return send(res, 400, { error: 'projectKey обязателен' });
|
|
394
|
+
const absPath = projectPaths.resolve(projectKey);
|
|
395
|
+
send(res, 200, { projectKey, absPath, registered: !!absPath });
|
|
396
|
+
} catch (e) { sendError(res, e, 500); }
|
|
397
|
+
}
|
|
398
|
+
|
|
374
399
|
// =============================================================================
|
|
375
400
|
// Static files (renderer assets).
|
|
376
401
|
// =============================================================================
|
|
@@ -428,6 +453,9 @@ const server = createServer(async (req, res) => {
|
|
|
428
453
|
// Jobs hub.
|
|
429
454
|
if (req.method === 'POST' && url.pathname === '/api/jobs/track') return handleJobsTrack(req, res);
|
|
430
455
|
if (req.method === 'GET' && url.pathname === '/api/jobs') return handleJobsList(res, url);
|
|
456
|
+
// Projects: register абс-путя для server-side fs-доступа.
|
|
457
|
+
if (req.method === 'POST' && url.pathname === '/api/project/register') return handleProjectRegister(req, res);
|
|
458
|
+
if (req.method === 'GET' && url.pathname === '/api/project/resolve') return handleProjectResolve(res, url);
|
|
431
459
|
// Cloud-projects routes — зеркало templates, но для редактируемых проектов.
|
|
432
460
|
if (req.method === 'GET' && url.pathname === '/api/projects') return handleProjectsList(res);
|
|
433
461
|
if (req.method === 'POST' && url.pathname === '/api/projects') return handleProjectCreate(req, res);
|
|
@@ -455,6 +483,7 @@ function start(port = PORT, opts = {}) {
|
|
|
455
483
|
// Init chatSession с userDataDir для персистентности историй.
|
|
456
484
|
// Без opts.userDataDir чат живёт только in-memory (CLI-режим).
|
|
457
485
|
chatSession.init({ userDataDir: opts.userDataDir || null });
|
|
486
|
+
projectPaths.init({ userDataDir: opts.userDataDir || null });
|
|
458
487
|
// jobsHub нужен settingsGetter чтобы поллить провайдеров (Chatium token,
|
|
459
488
|
// KIE_API_KEY и т.п.).
|
|
460
489
|
jobsHub.setSettingsGetter(getSettings);
|