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.
@@ -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
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "kingkont",
3
- "version": "0.17.1",
3
+ "version": "0.17.2",
4
4
  "description": "KingKont · Chatium — нод-редактор сцен с AI-генерацией (картинки/видео/голос/SFX/музыка/текст)",
5
5
  "main": "main.js",
6
6
  "bin": {
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);