patchrelay 0.53.0 → 0.53.1

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.
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "service": "patchrelay",
3
- "version": "0.53.0",
4
- "commit": "93105eaf009d",
5
- "builtAt": "2026-04-23T17:09:54.595Z"
3
+ "version": "0.53.1",
4
+ "commit": "6cfb95653641",
5
+ "builtAt": "2026-04-24T01:17:31.633Z"
6
6
  }
@@ -45,6 +45,7 @@ export class CodexAppServerClient extends EventEmitter {
45
45
  stdoutBuffer = "";
46
46
  started = false;
47
47
  stopping = false;
48
+ startPromise;
48
49
  constructor(config, logger, spawnProcess = spawn) {
49
50
  super();
50
51
  this.config = config;
@@ -68,76 +69,16 @@ export class CodexAppServerClient extends EventEmitter {
68
69
  if (this.started) {
69
70
  return;
70
71
  }
71
- this.stopping = false;
72
- const launch = resolveCodexAppServerLaunch(this.config);
73
- this.logger.info({ command: launch.command, args: launch.args }, "Starting Codex app-server");
74
- this.child = this.spawnProcess(launch.command, launch.args, {
75
- stdio: ["pipe", "pipe", "pipe"],
76
- });
77
- this.child.stdin.on("error", (error) => {
78
- this.logger.error({ error: sanitizeDiagnosticText(error.message) }, "Codex app-server stdin error");
79
- });
80
- this.child.stdout.on("error", (error) => {
81
- this.logger.error({ error: sanitizeDiagnosticText(error.message) }, "Codex app-server stdout error");
82
- });
83
- this.child.stderr.on("error", (error) => {
84
- this.logger.error({ error: sanitizeDiagnosticText(error.message) }, "Codex app-server stderr error");
85
- });
86
- this.child.stderr.on("data", (chunk) => {
87
- const line = chunk.toString().trim();
88
- if (line) {
89
- this.logger.warn({ output: sanitizeDiagnosticText(line) }, "Codex app-server stderr");
90
- }
91
- });
92
- this.child.on("error", (error) => {
93
- const err = error instanceof Error ? error : new Error(String(error));
94
- this.logger.error({
95
- error: sanitizeDiagnosticText(err.message),
96
- pendingRequestCount: this.pending.size,
97
- }, "Codex app-server process errored");
98
- this.rejectAllPending(err);
99
- });
100
- this.child.on("close", (code, signal) => {
101
- this.started = false;
102
- const log = this.stopping ? this.logger.info.bind(this.logger) : this.logger.warn.bind(this.logger);
103
- log({
104
- code: code ?? 1,
105
- signal: signal ?? null,
106
- pendingRequestCount: this.pending.size,
107
- }, this.stopping ? "Codex app-server stopped" : "Codex app-server exited");
108
- this.stopping = false;
109
- this.rejectAllPending(new Error(`Codex app-server exited with code ${code ?? 1}`));
110
- });
111
- this.child.stdout.on("data", (chunk) => {
112
- this.stdoutBuffer += chunk.toString("utf8");
113
- if (this.stdoutBuffer.length > 50 * 1024 * 1024) {
114
- this.logger.error({ bufferSize: this.stdoutBuffer.length }, "Codex app-server stdout buffer exceeded 50 MB — killing process");
115
- this.stdoutBuffer = "";
116
- this.rejectAllPending(new Error("Codex app-server stdout buffer overflow"));
117
- this.child?.kill("SIGTERM");
118
- return;
119
- }
120
- this.drainMessages();
121
- });
122
- const initializeResponse = await this.sendRequest("initialize", {
123
- clientInfo: {
124
- name: "patchrelay",
125
- title: "PatchRelay",
126
- version: "0.1.0",
127
- },
128
- capabilities: {
129
- experimentalApi: true,
130
- },
131
- });
132
- const serverInfo = initializeResponse && typeof initializeResponse === "object" && "serverInfo" in initializeResponse
133
- ? initializeResponse.serverInfo
134
- : undefined;
135
- this.logger.info({
136
- serverName: typeof serverInfo?.name === "string" ? serverInfo.name : undefined,
137
- serverVersion: typeof serverInfo?.version === "string" ? serverInfo.version : undefined,
138
- }, "Connected to Codex app-server");
139
- this.sendNotification("initialized");
140
- this.started = true;
72
+ if (this.startPromise) {
73
+ return await this.startPromise;
74
+ }
75
+ this.startPromise = this.startInternal();
76
+ try {
77
+ await this.startPromise;
78
+ }
79
+ finally {
80
+ this.startPromise = undefined;
81
+ }
141
82
  }
142
83
  async stop() {
143
84
  const child = this.child;
@@ -276,9 +217,7 @@ export class CodexAppServerClient extends EventEmitter {
276
217
  });
277
218
  }
278
219
  async sendRequest(method, params) {
279
- if (!this.child?.stdin) {
280
- throw new Error("Codex app-server is not running");
281
- }
220
+ await this.ensureRunningForRequest(method);
282
221
  const id = this.nextRequestId++;
283
222
  const requestTimeoutMs = this.config.requestTimeoutMs ?? CodexAppServerClient.DEFAULT_REQUEST_TIMEOUT_MS;
284
223
  const promise = new Promise((resolve, reject) => {
@@ -316,6 +255,93 @@ export class CodexAppServerClient extends EventEmitter {
316
255
  throw err;
317
256
  });
318
257
  }
258
+ async startInternal() {
259
+ this.stopping = false;
260
+ this.stdoutBuffer = "";
261
+ const launch = resolveCodexAppServerLaunch(this.config);
262
+ this.logger.info({ command: launch.command, args: launch.args }, "Starting Codex app-server");
263
+ this.child = this.spawnProcess(launch.command, launch.args, {
264
+ stdio: ["pipe", "pipe", "pipe"],
265
+ });
266
+ this.child.stdin.on("error", (error) => {
267
+ this.logger.error({ error: sanitizeDiagnosticText(error.message) }, "Codex app-server stdin error");
268
+ });
269
+ this.child.stdout.on("error", (error) => {
270
+ this.logger.error({ error: sanitizeDiagnosticText(error.message) }, "Codex app-server stdout error");
271
+ });
272
+ this.child.stderr.on("error", (error) => {
273
+ this.logger.error({ error: sanitizeDiagnosticText(error.message) }, "Codex app-server stderr error");
274
+ });
275
+ this.child.stderr.on("data", (chunk) => {
276
+ const line = chunk.toString().trim();
277
+ if (line) {
278
+ this.logger.warn({ output: sanitizeDiagnosticText(line) }, "Codex app-server stderr");
279
+ }
280
+ });
281
+ this.child.on("error", (error) => {
282
+ const err = error instanceof Error ? error : new Error(String(error));
283
+ this.logger.error({
284
+ error: sanitizeDiagnosticText(err.message),
285
+ pendingRequestCount: this.pending.size,
286
+ }, "Codex app-server process errored");
287
+ this.rejectAllPending(err);
288
+ });
289
+ this.child.on("close", (code, signal) => {
290
+ this.started = false;
291
+ this.child = undefined;
292
+ this.stdoutBuffer = "";
293
+ const log = this.stopping ? this.logger.info.bind(this.logger) : this.logger.warn.bind(this.logger);
294
+ log({
295
+ code: code ?? 1,
296
+ signal: signal ?? null,
297
+ pendingRequestCount: this.pending.size,
298
+ }, this.stopping ? "Codex app-server stopped" : "Codex app-server exited");
299
+ this.stopping = false;
300
+ this.rejectAllPending(new Error(`Codex app-server exited with code ${code ?? 1}`));
301
+ });
302
+ this.child.stdout.on("data", (chunk) => {
303
+ this.stdoutBuffer += chunk.toString("utf8");
304
+ if (this.stdoutBuffer.length > 50 * 1024 * 1024) {
305
+ this.logger.error({ bufferSize: this.stdoutBuffer.length }, "Codex app-server stdout buffer exceeded 50 MB — killing process");
306
+ this.stdoutBuffer = "";
307
+ this.rejectAllPending(new Error("Codex app-server stdout buffer overflow"));
308
+ this.child?.kill("SIGTERM");
309
+ return;
310
+ }
311
+ this.drainMessages();
312
+ });
313
+ const initializeResponse = await this.sendRequest("initialize", {
314
+ clientInfo: {
315
+ name: "patchrelay",
316
+ title: "PatchRelay",
317
+ version: "0.1.0",
318
+ },
319
+ capabilities: {
320
+ experimentalApi: true,
321
+ },
322
+ });
323
+ const serverInfo = initializeResponse && typeof initializeResponse === "object" && "serverInfo" in initializeResponse
324
+ ? initializeResponse.serverInfo
325
+ : undefined;
326
+ this.logger.info({
327
+ serverName: typeof serverInfo?.name === "string" ? serverInfo.name : undefined,
328
+ serverVersion: typeof serverInfo?.version === "string" ? serverInfo.version : undefined,
329
+ }, "Connected to Codex app-server");
330
+ this.sendNotification("initialized");
331
+ this.started = true;
332
+ }
333
+ async ensureRunningForRequest(method) {
334
+ if (this.child?.stdin) {
335
+ return;
336
+ }
337
+ if (method !== "initialize") {
338
+ this.logger.warn({ method }, "Codex app-server is unavailable before request; restarting");
339
+ }
340
+ await this.start();
341
+ if (!this.child?.stdin) {
342
+ throw new Error("Codex app-server is not running");
343
+ }
344
+ }
319
345
  writeMessage(message) {
320
346
  if (!this.child?.stdin) {
321
347
  throw new Error("Codex app-server stdin is unavailable");
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "patchrelay",
3
- "version": "0.53.0",
3
+ "version": "0.53.1",
4
4
  "license": "MIT",
5
5
  "type": "module",
6
6
  "repository": {