@react-grab/codex 0.0.82 → 0.0.84

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/dist/client.cjs CHANGED
@@ -1,29 +1,25 @@
1
1
  'use strict';
2
2
 
3
- // src/constants.ts
4
- var DEFAULT_PORT = 7567;
3
+ // ../utils/dist/client.js
5
4
  var CONNECTION_CHECK_TTL_MS = 5e3;
6
-
7
- // src/client.ts
8
- var DEFAULT_SERVER_URL = `http://localhost:${DEFAULT_PORT}`;
9
5
  var STORAGE_KEY = "react-grab:agent-sessions";
10
- var parseServerSentEvent = (eventStringBlock) => {
6
+ var parseSSEEvent = (eventBlock) => {
11
7
  let eventType = "";
12
8
  let data = "";
13
- for (const line of eventStringBlock.split("\n")) {
9
+ for (const line of eventBlock.split("\n")) {
14
10
  if (line.startsWith("event:")) eventType = line.slice(6).trim();
15
11
  else if (line.startsWith("data:")) data = line.slice(5).trim();
16
12
  }
17
13
  return { eventType, data };
18
14
  };
19
- var streamSSE = async function* (stream, signal) {
20
- const streamReader = stream.getReader();
21
- const textDecoder = new TextDecoder();
22
- let textBuffer = "";
15
+ async function* streamSSE(stream, signal) {
16
+ const reader = stream.getReader();
17
+ const decoder = new TextDecoder();
18
+ let buffer = "";
23
19
  let aborted = false;
24
20
  const onAbort = () => {
25
21
  aborted = true;
26
- streamReader.cancel().catch(() => {
22
+ reader.cancel().catch(() => {
27
23
  });
28
24
  };
29
25
  signal.addEventListener("abort", onAbort);
@@ -32,18 +28,16 @@ var streamSSE = async function* (stream, signal) {
32
28
  throw new DOMException("Aborted", "AbortError");
33
29
  }
34
30
  while (true) {
35
- const result = await streamReader.read();
31
+ const result = await reader.read();
36
32
  if (aborted || signal.aborted) {
37
33
  throw new DOMException("Aborted", "AbortError");
38
34
  }
39
35
  const { done, value } = result;
40
- if (value) textBuffer += textDecoder.decode(value, { stream: true });
41
- let boundaryIndex;
42
- while ((boundaryIndex = textBuffer.indexOf("\n\n")) !== -1) {
43
- const { eventType, data } = parseServerSentEvent(
44
- textBuffer.slice(0, boundaryIndex)
45
- );
46
- textBuffer = textBuffer.slice(boundaryIndex + 2);
36
+ if (value) buffer += decoder.decode(value, { stream: true });
37
+ let boundary;
38
+ while ((boundary = buffer.indexOf("\n\n")) !== -1) {
39
+ const { eventType, data } = parseSSEEvent(buffer.slice(0, boundary));
40
+ buffer = buffer.slice(boundary + 2);
47
41
  if (eventType === "done") return;
48
42
  if (eventType === "error") throw new Error(data || "Agent error");
49
43
  if (data) yield data;
@@ -53,12 +47,19 @@ var streamSSE = async function* (stream, signal) {
53
47
  } finally {
54
48
  signal.removeEventListener("abort", onAbort);
55
49
  try {
56
- streamReader.releaseLock();
50
+ reader.releaseLock();
57
51
  } catch {
58
52
  }
59
53
  }
60
- };
54
+ }
55
+
56
+ // src/constants.ts
57
+ var DEFAULT_PORT = 7567;
58
+
59
+ // src/client.ts
60
+ var DEFAULT_SERVER_URL = `http://localhost:${DEFAULT_PORT}`;
61
61
  var streamFromServer = async function* (serverUrl, context, signal) {
62
+ const startTime = Date.now();
62
63
  const sessionId = context.sessionId;
63
64
  const handleAbort = () => {
64
65
  if (sessionId) {
@@ -82,7 +83,37 @@ var streamFromServer = async function* (serverUrl, context, signal) {
82
83
  if (!response.body) {
83
84
  throw new Error("No response body");
84
85
  }
85
- yield* streamSSE(response.body, signal);
86
+ const iterator = streamSSE(response.body, signal)[Symbol.asyncIterator]();
87
+ let done = false;
88
+ let pendingNext = iterator.next();
89
+ while (!done) {
90
+ const result = await Promise.race([
91
+ pendingNext.then((iteratorResult) => ({
92
+ type: "status",
93
+ iteratorResult
94
+ })),
95
+ new Promise(
96
+ (resolve) => setTimeout(() => resolve({ type: "timeout" }), 100)
97
+ )
98
+ ]);
99
+ if (result.type === "timeout") {
100
+ const elapsedSeconds = (Date.now() - startTime) / 1e3;
101
+ yield `Working\u2026 ${elapsedSeconds.toFixed(1)}s`;
102
+ } else {
103
+ const iteratorResult = result.iteratorResult;
104
+ done = iteratorResult.done ?? false;
105
+ if (!done && iteratorResult.value) {
106
+ const status = iteratorResult.value;
107
+ if (status === "Completed successfully") {
108
+ const totalSeconds = ((Date.now() - startTime) / 1e3).toFixed(1);
109
+ yield `Completed in ${totalSeconds}s`;
110
+ } else {
111
+ yield status;
112
+ }
113
+ pendingNext = iterator.next();
114
+ }
115
+ }
116
+ }
86
117
  } finally {
87
118
  signal.removeEventListener("abort", handleAbort);
88
119
  }
@@ -1,4 +1,5 @@
1
- var ReactGrabCodex=(function(exports){'use strict';var m=`http://localhost:${7567}`,E="react-grab:agent-sessions",y=s=>{let t="",o="";for(let n of s.split(`
2
- `))n.startsWith("event:")?t=n.slice(6).trim():n.startsWith("data:")&&(o=n.slice(5).trim());return {eventType:t,data:o}},C=async function*(s,t){let o=s.getReader(),n=new TextDecoder,r="",e=false,a=()=>{e=true,o.cancel().catch(()=>{});};t.addEventListener("abort",a);try{if(t.aborted)throw new DOMException("Aborted","AbortError");for(;;){let i=await o.read();if(e||t.aborted)throw new DOMException("Aborted","AbortError");let{done:l,value:f}=i;f&&(r+=n.decode(f,{stream:!0}));let c;for(;(c=r.indexOf(`
1
+ var ReactGrabCodex=(function(exports){'use strict';var f=5e3,m="react-grab:agent-sessions",y=c=>{let e="",n="";for(let t of c.split(`
2
+ `))t.startsWith("event:")?e=t.slice(6).trim():t.startsWith("data:")&&(n=t.slice(5).trim());return {eventType:e,data:n}};async function*g(c,e){let n=c.getReader(),t=new TextDecoder,r="",o=false,i=()=>{o=true,n.cancel().catch(()=>{});};e.addEventListener("abort",i);try{if(e.aborted)throw new DOMException("Aborted","AbortError");for(;;){let a=await n.read();if(o||e.aborted)throw new DOMException("Aborted","AbortError");let{done:l,value:u}=a;u&&(r+=t.decode(u,{stream:!0}));let d;for(;(d=r.indexOf(`
3
3
 
4
- `))!==-1;){let{eventType:d,data:p}=y(r.slice(0,c));if(r=r.slice(c+2),d==="done")return;if(d==="error")throw new Error(p||"Agent error");p&&(yield p);}if(l)break}}finally{t.removeEventListener("abort",a);try{o.releaseLock();}catch{}}},g=async function*(s,t,o){let n=t.sessionId,r=()=>{n&&fetch(`${s}/abort/${n}`,{method:"POST"}).catch(()=>{});};o.addEventListener("abort",r);try{let e=await fetch(`${s}/agent`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(t),signal:o});if(!e.ok)throw new Error(`Server error: ${e.status}`);if(!e.body)throw new Error("No response body");yield*C(e.body,o);}finally{o.removeEventListener("abort",r);}},b=(s={})=>{let{serverUrl:t=m,getOptions:o}=s,n=null,r=e=>({...o?.()??{},...e??{}});return {send:async function*(e,a){let i={...e,options:r(e.options)};yield*g(t,i,a);},resume:async function*(e,a,i){let l=i.getItem(E);if(!l)throw new Error("No sessions to resume");let c=JSON.parse(l)[e];if(!c)throw new Error(`Session ${e} not found`);let d=c.context,p={...d,options:r(d.options)};yield "Resuming...",yield*g(t,p,a);},supportsResume:true,supportsFollowUp:true,checkConnection:async()=>{let e=Date.now();if(n&&e-n.timestamp<5e3)return n.result;try{let i=(await fetch(`${t}/health`,{method:"GET"})).ok;return n={result:i,timestamp:e},i}catch{return n={result:false,timestamp:e},false}},undo:async()=>{try{await fetch(`${t}/undo`,{method:"POST"});}catch{}}}},S=async()=>{if(typeof window>"u")return;let s=b(),t=r=>{r.setAgent({provider:s,storage:sessionStorage});},o=window.__REACT_GRAB__;if(o){t(o);return}window.addEventListener("react-grab:init",r=>{t(r.detail);},{once:true});let n=window.__REACT_GRAB__;n&&t(n);};S();exports.attachAgent=S;exports.createCodexAgentProvider=b;return exports;})({});
4
+ `))!==-1;){let{eventType:s,data:p}=y(r.slice(0,d));if(r=r.slice(d+2),s==="done")return;if(s==="error")throw new Error(p||"Agent error");p&&(yield p);}if(l)break}}finally{e.removeEventListener("abort",i);try{n.releaseLock();}catch{}}}var C=`http://localhost:${7567}`,A=async function*(c,e,n){let t=Date.now(),r=e.sessionId,o=()=>{r&&fetch(`${c}/abort/${r}`,{method:"POST"}).catch(()=>{});};n.addEventListener("abort",o);try{let i=await fetch(`${c}/agent`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(e),signal:n});if(!i.ok)throw new Error(`Server error: ${i.status}`);if(!i.body)throw new Error("No response body");let a=g(i.body,n)[Symbol.asyncIterator](),l=!1,u=a.next();for(;!l;){let d=await Promise.race([u.then(s=>({type:"status",iteratorResult:s})),new Promise(s=>setTimeout(()=>s({type:"timeout"}),100))]);if(d.type==="timeout")yield `Working\u2026 ${((Date.now()-t)/1e3).toFixed(1)}s`;else {let s=d.iteratorResult;if(l=s.done??!1,!l&&s.value){let p=s.value;p==="Completed successfully"?yield `Completed in ${((Date.now()-t)/1e3).toFixed(1)}s`:yield p,u=a.next();}}}}finally{n.removeEventListener("abort",o);}},E=(c={})=>{let{serverUrl:e=C,getOptions:n}=c,t=null,r=o=>({...n?.()??{},...o??{}});return {send:async function*(o,i){let a={...o,options:r(o.options)};yield*A(e,a,i);},resume:async function*(o,i,a){let l=a.getItem(m);if(!l)throw new Error("No sessions to resume");let d=JSON.parse(l)[o];if(!d)throw new Error(`Session ${o} not found`);let s=d.context,p={...s,options:r(s.options)};yield "Resuming...",yield*A(e,p,i);},supportsResume:true,supportsFollowUp:true,checkConnection:async()=>{let o=Date.now();if(t&&o-t.timestamp<f)return t.result;try{let a=(await fetch(`${e}/health`,{method:"GET"})).ok;return t={result:a,timestamp:o},a}catch{return t={result:false,timestamp:o},false}},undo:async()=>{try{await fetch(`${e}/undo`,{method:"POST"});}catch{}}}},S=async()=>{if(typeof window>"u")return;let c=E(),e=r=>{r.setAgent({provider:c,storage:sessionStorage});},n=window.__REACT_GRAB__;if(n){e(n);return}window.addEventListener("react-grab:init",r=>{e(r.detail);},{once:true});let t=window.__REACT_GRAB__;t&&e(t);};S();
5
+ exports.attachAgent=S;exports.createCodexAgentProvider=E;return exports;})({});
package/dist/client.js CHANGED
@@ -1,27 +1,23 @@
1
- // src/constants.ts
2
- var DEFAULT_PORT = 7567;
1
+ // ../utils/dist/client.js
3
2
  var CONNECTION_CHECK_TTL_MS = 5e3;
4
-
5
- // src/client.ts
6
- var DEFAULT_SERVER_URL = `http://localhost:${DEFAULT_PORT}`;
7
3
  var STORAGE_KEY = "react-grab:agent-sessions";
8
- var parseServerSentEvent = (eventStringBlock) => {
4
+ var parseSSEEvent = (eventBlock) => {
9
5
  let eventType = "";
10
6
  let data = "";
11
- for (const line of eventStringBlock.split("\n")) {
7
+ for (const line of eventBlock.split("\n")) {
12
8
  if (line.startsWith("event:")) eventType = line.slice(6).trim();
13
9
  else if (line.startsWith("data:")) data = line.slice(5).trim();
14
10
  }
15
11
  return { eventType, data };
16
12
  };
17
- var streamSSE = async function* (stream, signal) {
18
- const streamReader = stream.getReader();
19
- const textDecoder = new TextDecoder();
20
- let textBuffer = "";
13
+ async function* streamSSE(stream, signal) {
14
+ const reader = stream.getReader();
15
+ const decoder = new TextDecoder();
16
+ let buffer = "";
21
17
  let aborted = false;
22
18
  const onAbort = () => {
23
19
  aborted = true;
24
- streamReader.cancel().catch(() => {
20
+ reader.cancel().catch(() => {
25
21
  });
26
22
  };
27
23
  signal.addEventListener("abort", onAbort);
@@ -30,18 +26,16 @@ var streamSSE = async function* (stream, signal) {
30
26
  throw new DOMException("Aborted", "AbortError");
31
27
  }
32
28
  while (true) {
33
- const result = await streamReader.read();
29
+ const result = await reader.read();
34
30
  if (aborted || signal.aborted) {
35
31
  throw new DOMException("Aborted", "AbortError");
36
32
  }
37
33
  const { done, value } = result;
38
- if (value) textBuffer += textDecoder.decode(value, { stream: true });
39
- let boundaryIndex;
40
- while ((boundaryIndex = textBuffer.indexOf("\n\n")) !== -1) {
41
- const { eventType, data } = parseServerSentEvent(
42
- textBuffer.slice(0, boundaryIndex)
43
- );
44
- textBuffer = textBuffer.slice(boundaryIndex + 2);
34
+ if (value) buffer += decoder.decode(value, { stream: true });
35
+ let boundary;
36
+ while ((boundary = buffer.indexOf("\n\n")) !== -1) {
37
+ const { eventType, data } = parseSSEEvent(buffer.slice(0, boundary));
38
+ buffer = buffer.slice(boundary + 2);
45
39
  if (eventType === "done") return;
46
40
  if (eventType === "error") throw new Error(data || "Agent error");
47
41
  if (data) yield data;
@@ -51,12 +45,19 @@ var streamSSE = async function* (stream, signal) {
51
45
  } finally {
52
46
  signal.removeEventListener("abort", onAbort);
53
47
  try {
54
- streamReader.releaseLock();
48
+ reader.releaseLock();
55
49
  } catch {
56
50
  }
57
51
  }
58
- };
52
+ }
53
+
54
+ // src/constants.ts
55
+ var DEFAULT_PORT = 7567;
56
+
57
+ // src/client.ts
58
+ var DEFAULT_SERVER_URL = `http://localhost:${DEFAULT_PORT}`;
59
59
  var streamFromServer = async function* (serverUrl, context, signal) {
60
+ const startTime = Date.now();
60
61
  const sessionId = context.sessionId;
61
62
  const handleAbort = () => {
62
63
  if (sessionId) {
@@ -80,7 +81,37 @@ var streamFromServer = async function* (serverUrl, context, signal) {
80
81
  if (!response.body) {
81
82
  throw new Error("No response body");
82
83
  }
83
- yield* streamSSE(response.body, signal);
84
+ const iterator = streamSSE(response.body, signal)[Symbol.asyncIterator]();
85
+ let done = false;
86
+ let pendingNext = iterator.next();
87
+ while (!done) {
88
+ const result = await Promise.race([
89
+ pendingNext.then((iteratorResult) => ({
90
+ type: "status",
91
+ iteratorResult
92
+ })),
93
+ new Promise(
94
+ (resolve) => setTimeout(() => resolve({ type: "timeout" }), 100)
95
+ )
96
+ ]);
97
+ if (result.type === "timeout") {
98
+ const elapsedSeconds = (Date.now() - startTime) / 1e3;
99
+ yield `Working\u2026 ${elapsedSeconds.toFixed(1)}s`;
100
+ } else {
101
+ const iteratorResult = result.iteratorResult;
102
+ done = iteratorResult.done ?? false;
103
+ if (!done && iteratorResult.value) {
104
+ const status = iteratorResult.value;
105
+ if (status === "Completed successfully") {
106
+ const totalSeconds = ((Date.now() - startTime) / 1e3).toFixed(1);
107
+ yield `Completed in ${totalSeconds}s`;
108
+ } else {
109
+ yield status;
110
+ }
111
+ pendingNext = iterator.next();
112
+ }
113
+ }
114
+ }
84
115
  } finally {
85
116
  signal.removeEventListener("abort", handleAbort);
86
117
  }
package/dist/server.cjs CHANGED
@@ -19,8 +19,10 @@ var pc__default = /*#__PURE__*/_interopDefault(pc);
19
19
 
20
20
  // src/constants.ts
21
21
  var DEFAULT_PORT = 7567;
22
- var VERSION = "0.0.82";
22
+
23
+ // ../utils/dist/server.js
23
24
  var sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
25
+ var VERSION = "0.0.84";
24
26
  var codexInstance = null;
25
27
  var threadMap = /* @__PURE__ */ new Map();
26
28
  var abortControllers = /* @__PURE__ */ new Map();
@@ -148,7 +150,9 @@ var startServer = async (port = DEFAULT_PORT) => {
148
150
  await sleep(100);
149
151
  const honoApplication = createServer();
150
152
  nodeServer.serve({ fetch: honoApplication.fetch, port });
151
- console.log(`${pc__default.default.magenta("\u269B")} ${pc__default.default.bold("React Grab")} ${pc__default.default.gray(VERSION)} ${pc__default.default.dim("(Codex)")}`);
153
+ console.log(
154
+ `${pc__default.default.magenta("\u269B")} ${pc__default.default.bold("React Grab")} ${pc__default.default.gray(VERSION)} ${pc__default.default.dim("(Codex)")}`
155
+ );
152
156
  console.log(`- Local: ${pc__default.default.cyan(`http://localhost:${port}`)}`);
153
157
  };
154
158
  if ((typeof document === 'undefined' ? require('u' + 'rl').pathToFileURL(__filename).href : (_documentCurrentScript && _documentCurrentScript.tagName.toUpperCase() === 'SCRIPT' && _documentCurrentScript.src || new URL('server.cjs', document.baseURI).href)) === url.pathToFileURL(process.argv[1]).href) {
package/dist/server.js CHANGED
@@ -11,8 +11,10 @@ import { pathToFileURL } from 'url';
11
11
 
12
12
  // src/constants.ts
13
13
  var DEFAULT_PORT = 7567;
14
- var VERSION = "0.0.82";
14
+
15
+ // ../utils/dist/server.js
15
16
  var sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
17
+ var VERSION = "0.0.84";
16
18
  var codexInstance = null;
17
19
  var threadMap = /* @__PURE__ */ new Map();
18
20
  var abortControllers = /* @__PURE__ */ new Map();
@@ -140,7 +142,9 @@ var startServer = async (port = DEFAULT_PORT) => {
140
142
  await sleep(100);
141
143
  const honoApplication = createServer();
142
144
  serve({ fetch: honoApplication.fetch, port });
143
- console.log(`${pc.magenta("\u269B")} ${pc.bold("React Grab")} ${pc.gray(VERSION)} ${pc.dim("(Codex)")}`);
145
+ console.log(
146
+ `${pc.magenta("\u269B")} ${pc.bold("React Grab")} ${pc.gray(VERSION)} ${pc.dim("(Codex)")}`
147
+ );
144
148
  console.log(`- Local: ${pc.cyan(`http://localhost:${port}`)}`);
145
149
  };
146
150
  if (import.meta.url === pathToFileURL(process.argv[1]).href) {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@react-grab/codex",
3
- "version": "0.0.82",
3
+ "version": "0.0.84",
4
4
  "type": "module",
5
5
  "bin": {
6
6
  "react-grab-codex": "./dist/cli.cjs"
@@ -24,17 +24,17 @@
24
24
  "dist"
25
25
  ],
26
26
  "devDependencies": {
27
- "@types/cross-spawn": "^6.0.6",
28
- "tsup": "^8.4.0"
27
+ "tsup": "^8.4.0",
28
+ "@react-grab/utils": "0.0.84"
29
29
  },
30
30
  "dependencies": {
31
31
  "@hono/node-server": "^1.19.6",
32
- "cross-spawn": "^7.0.6",
32
+ "execa": "^9.6.0",
33
33
  "@openai/codex-sdk": "^0.66.0",
34
34
  "fkill": "^9.0.0",
35
35
  "hono": "^4.0.0",
36
36
  "picocolors": "^1.1.1",
37
- "react-grab": "0.0.82"
37
+ "react-grab": "0.0.84"
38
38
  },
39
39
  "scripts": {
40
40
  "dev": "tsup --watch",