wormclaude 1.0.126 → 1.0.128

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.
@@ -10,6 +10,97 @@ const ARG_KEYS = ['arguments', 'parameters', 'input', 'args', 'params'];
10
10
  // Model araç adını JSON DIŞINA koyabiliyor: `Bash {"command":...}` / `Write {"file_path":...}`.
11
11
  const _TOOLS = 'Bash|PowerShell|Read|Write|Edit|Glob|Grep|WebFetch|WebSearch|Sleep|TaskOutput';
12
12
  const TOOL_PREFIX_RE = new RegExp('(?:^|[\\n>*`\\s])(' + _TOOLS + ')\\s*\\{');
13
+ // Fonksiyon-çağrı pseudo-syntax'i: `Write(file_path="...", content="""...""")`.
14
+ const TOOL_PAREN_RE = new RegExp('(' + _TOOLS + ')\\s*\\(');
15
+ // `key="val"` / `key='val'` / `key="""val"""` argümanlarını çözer (content çok-satırlı olabilir).
16
+ function _parseKwCall(s) {
17
+ const unesc = (str) => str.replace(/\\(["'\\/nrt])/g, (_m, c) => (c === 'n' ? '\n' : c === 'r' ? '\r' : c === 't' ? '\t' : c));
18
+ const args = {};
19
+ let i = 0;
20
+ const n = s.length;
21
+ let guard = 0;
22
+ while (i < n && guard++ < 20) {
23
+ while (i < n && /[\s,]/.test(s[i]))
24
+ i++;
25
+ if (i >= n || s[i] === ')')
26
+ break;
27
+ const km = /^([a-zA-Z_]\w*)\s*[:=]\s*/.exec(s.slice(i));
28
+ if (!km)
29
+ break;
30
+ i += km[0].length;
31
+ const key = km[1];
32
+ if (s.startsWith('"""', i)) {
33
+ const e = s.indexOf('"""', i + 3);
34
+ if (e < 0)
35
+ break;
36
+ args[key] = s.slice(i + 3, e);
37
+ i = e + 3;
38
+ }
39
+ else if (s.startsWith("'''", i)) {
40
+ const e = s.indexOf("'''", i + 3);
41
+ if (e < 0)
42
+ break;
43
+ args[key] = s.slice(i + 3, e);
44
+ i = e + 3;
45
+ }
46
+ else if (s[i] === '"' || s[i] === "'") {
47
+ const q = s[i];
48
+ i++;
49
+ let v = '';
50
+ while (i < n && s[i] !== q) {
51
+ if (s[i] === '\\' && i + 1 < n) {
52
+ v += s.slice(i, i + 2);
53
+ i += 2;
54
+ }
55
+ else {
56
+ v += s[i];
57
+ i++;
58
+ }
59
+ }
60
+ i++;
61
+ args[key] = unesc(v);
62
+ }
63
+ else
64
+ break;
65
+ }
66
+ return Object.keys(args).length ? args : null;
67
+ }
68
+ // `ToolName(` açılışından eşleşen `)`'ye kadar (string/triple-quote duyarlı) bitiş indexi.
69
+ function _findCallEnd(s, open) {
70
+ let depth = 0, i = open;
71
+ while (i < s.length) {
72
+ if (s.startsWith('"""', i)) {
73
+ const e = s.indexOf('"""', i + 3);
74
+ i = e < 0 ? s.length : e + 3;
75
+ continue;
76
+ }
77
+ if (s.startsWith("'''", i)) {
78
+ const e = s.indexOf("'''", i + 3);
79
+ i = e < 0 ? s.length : e + 3;
80
+ continue;
81
+ }
82
+ const c = s[i];
83
+ if (c === '"' || c === "'") {
84
+ i++;
85
+ while (i < s.length && s[i] !== c) {
86
+ if (s[i] === '\\')
87
+ i++;
88
+ i++;
89
+ }
90
+ i++;
91
+ continue;
92
+ }
93
+ if (c === '(')
94
+ depth++;
95
+ else if (c === ')') {
96
+ depth--;
97
+ if (depth === 0)
98
+ return i + 1;
99
+ }
100
+ i++;
101
+ }
102
+ return -1;
103
+ }
13
104
  function toToolCall(obj, i) {
14
105
  if (!obj || typeof obj !== 'object')
15
106
  return null;
@@ -76,8 +167,8 @@ function extractTopLevelJsonObjects(text) {
76
167
  * (<tool_call>, ```json) veya mesajın TAMAMI tek JSON çağrısıysa onu alır. */
77
168
  export function recoverInlineToolCalls(text) {
78
169
  const t = (text || '').trim();
79
- // JSON/prose/ToolName{…} çağrısı yoksa erken çık.
80
- if (!t || (!t.includes('"name"') && !t.includes('"tool"') && !t.includes('"function"') && !/AskUserQuestion/i.test(t) && !TOOL_PREFIX_RE.test(t)))
170
+ // JSON/prose/ToolName{…}/ToolName(…) çağrısı yoksa erken çık.
171
+ if (!t || (!t.includes('"name"') && !t.includes('"tool"') && !t.includes('"function"') && !/AskUserQuestion/i.test(t) && !TOOL_PREFIX_RE.test(t) && !TOOL_PAREN_RE.test(t)))
81
172
  return [];
82
173
  const out = [];
83
174
  const seen = new Set();
@@ -143,6 +234,22 @@ export function recoverInlineToolCalls(text) {
143
234
  }
144
235
  }
145
236
  }
237
+ // 6) "ToolName(key=...)" fonksiyon-çağrı pseudo-syntax'i (Write(file_path="...", content="""...""")).
238
+ if (!out.length) {
239
+ const re = new RegExp('(' + _TOOLS + ')\\s*\\(', 'g');
240
+ let mm;
241
+ while ((mm = re.exec(t)) !== null) {
242
+ const open = mm.index + mm[0].length - 1; // '(' konumu
243
+ const end = _findCallEnd(t, open);
244
+ if (end < 0)
245
+ continue;
246
+ const args = _parseKwCall(t.slice(open + 1, end - 1));
247
+ if (args) {
248
+ push({ name: mm[1], arguments: args });
249
+ re.lastIndex = end;
250
+ }
251
+ }
252
+ }
146
253
  return out;
147
254
  }
148
255
  /** Modelin komut/çıktıyı TEKRAR yazdığı kod-bloklarını siler (```plaintext/bash/http/console…).
@@ -207,5 +314,21 @@ export function stripInlineToolCalls(text) {
207
314
  for (let i = spans.length - 1; i >= 0; i--)
208
315
  s = s.slice(0, spans[i][0]) + s.slice(spans[i][1]);
209
316
  }
317
+ // ToolName(…) biçimi (Write(file_path="...", content="""...""")) — adı + () bloğunu sil.
318
+ {
319
+ const re = new RegExp('(' + _TOOLS + ')\\s*\\(', 'g');
320
+ let mm;
321
+ const spans = [];
322
+ while ((mm = re.exec(s)) !== null) {
323
+ const open = mm.index + mm[0].length - 1;
324
+ const end = _findCallEnd(s, open);
325
+ if (end > 0) {
326
+ spans.push([mm.index, end]);
327
+ re.lastIndex = end;
328
+ }
329
+ }
330
+ for (let i = spans.length - 1; i >= 0; i--)
331
+ s = s.slice(0, spans[i][0]) + s.slice(spans[i][1]);
332
+ }
210
333
  return s.trim();
211
334
  }
package/dist/theme.js CHANGED
@@ -16,4 +16,4 @@ export const theme = {
16
16
  synType: '#a78bfa', // tip/sınıf adları, sabitler
17
17
  synProp: '#e0e0e0', // özellik/anahtar adları
18
18
  };
19
- export const VERSION = '1.0.126';
19
+ export const VERSION = '1.0.128';
package/dist/tui.js CHANGED
@@ -83,7 +83,6 @@ export async function runTui() {
83
83
  const displayItems = [{ kind: 'banner' }];
84
84
  let inputBuf = '', inputCur = 0, busy = false, streamChars = 0, spin = 0;
85
85
  let taskStart = 0; // görev başlangıç zamanı (spinner'da geçen süre için)
86
- let awaitingWorkspace = true; // açılışta çalışma klasörü sorulur (ilk submit = klasör)
87
86
  // Canlı akış artık FOOTER'da DEĞİL — içerik akışına (mesajın altından aşağı) satır-satır basılır.
88
87
  const SPIN = ['·', '✢', '✳', '✶', '✻', '✽', '✶', '✳', '✢'];
89
88
  // Canlı bağlam ölçer: son isteğin prompt_tokens'ı = o anki kullanılan bağlam (footer'da gösterilir).
@@ -449,10 +448,25 @@ export async function runTui() {
449
448
  catch { }
450
449
  process.stdout.write('\x1b[?25l'); // gerçek imleci gizle (kendi ▌ bloğumuzu çiziyoruz)
451
450
  redrawAll();
452
- // Açılış: Claude tarzı GÜVEN onayı (yol yazdırma yok cwd zaten kullanıcının açtığı klasör).
453
- printItem({ kind: 'note', text: getLang() === 'en'
454
- ? `🔒 Accessing workspace:\n ${process.cwd()}\n\n Trust this folder? It's where files will be created. Press Enter to continue · type "no" to exit · /cd <path> to change later.`
455
- : `🔒 Çalışma alanına erişiliyor:\n ${process.cwd()}\n\n Bu klasöre güveniyor musun? Dosyalar burada oluşturulur. Devam için Enter · çıkmak için "hayır" yaz · sonra değiştirmek için /cd <yol>.` });
451
+ // Açılış: Claude tarzı GÜVEN ekranı trust metni (içerik) + ok-tuşlu Evet/Hayır seçici (ask modal).
452
+ {
453
+ const _en = getLang() === 'en';
454
+ printItem({ kind: 'note', text: (_en ? 'Accessing workspace:' : 'Çalışma alanına erişiliyor:') + '\n\n' +
455
+ process.cwd() + '\n\n' +
456
+ t('trust.check') + '\n\n' +
457
+ t('trust.canDo') });
458
+ const _noLabel = _en ? 'No, exit' : 'Hayır, çık';
459
+ ask = {
460
+ question: _en ? 'Trust this folder?' : 'Bu klasöre güveniyor musun?',
461
+ options: [{ label: _en ? 'Yes, I trust this folder' : 'Evet, bu klasöre güveniyorum' }, { label: _noLabel }],
462
+ sel: 0,
463
+ resolve: (v) => { if (v === _noLabel)
464
+ quit();
465
+ else
466
+ refresh(); },
467
+ };
468
+ refresh();
469
+ }
456
470
  fetchAccount(config).then((a) => { if (a) {
457
471
  account = a;
458
472
  redrawAll();
@@ -621,21 +635,6 @@ export async function runTui() {
621
635
  inputBuf = '';
622
636
  inputCur = 0;
623
637
  histIdx = -1;
624
- // Açılış GÜVEN onayı (Claude tarzı): Enter = onayla, "hayır"/"no" = çık. Yol yazdırma YOK.
625
- if (awaitingWorkspace) {
626
- awaitingWorkspace = false;
627
- const _low = v.toLowerCase();
628
- if (['hayır', 'hayir', 'no', 'n', 'q', 'exit', 'quit', 'çık', 'cik'].includes(_low)) {
629
- quit();
630
- return;
631
- }
632
- printItem({ kind: 'note', text: '✓ ' + (getLang() === 'en' ? 'Workspace trusted: ' : 'Klasöre güvenildi: ') + process.cwd() });
633
- if (!v) {
634
- refresh();
635
- return;
636
- } // sadece Enter → onayla, görev bekle
637
- // görev yazıldıysa → onayla + aşağı düşüp çalıştır
638
- }
639
638
  if (!v) {
640
639
  refresh();
641
640
  return;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "wormclaude",
3
- "version": "1.0.126",
3
+ "version": "1.0.128",
4
4
  "description": "WormClaude CLI - uncensored security+code assistant (ink TUI, Claude-style)",
5
5
  "type": "module",
6
6
  "bin": {