@@ -2,6 +2,7 @@ import http from "http"
2
2
import path from "path"
3
3
import child_process from "child_process"
4
4
import { fileURLToPath } from "url"
5
+ import net from "net"
5
6
6
7
export interface RunOpts {
7
8
input ?: string
@@ -31,35 +32,45 @@ export enum RunEventType {
31
32
Prompt = "prompt"
32
33
}
33
34
34
- let serverProcess : child_process . ChildProcess
35
- let clientCount : number = 0
35
+ export class GPTScript {
36
+ private static serverURL : string = ""
37
+ private static serverProcess : child_process . ChildProcess
38
+ private static instanceCount : number = 0
36
39
37
- export class Client {
38
- private readonly gptscriptURL : string
39
- private clientReady : boolean
40
40
41
- constructor ( ) {
42
- this . clientReady = false
43
- this . gptscriptURL = "http://" + ( process . env . GPTSCRIPT_URL || "127.0.0.1:9090" )
44
- clientCount ++
45
- if ( clientCount === 1 && process . env . GPTSCRIPT_DISABLE_SERVER !== "true" ) {
46
- serverProcess = child_process . spawn ( getCmdPath ( ) , [ "--listen-address" , this . gptscriptURL . replace ( "http://" , "" ) . replace ( "https://" , "" ) , "sdkserver" ] , {
47
- env : process . env ,
48
- stdio : [ "pipe" ]
49
- } )
41
+ private ready : boolean
50
42
51
- process . on ( "exit" , ( code ) => {
52
- serverProcess . stdin ?. end ( )
53
- serverProcess . kill ( code )
54
- } )
43
+ constructor ( ) {
44
+ this . ready = false
45
+ GPTScript . instanceCount ++
46
+ if ( GPTScript . instanceCount === 1 && process . env . GPTSCRIPT_DISABLE_SERVER !== "true" ) {
47
+ GPTScript . serverURL = process . env . GPTSCRIPT_URL || "http://127.0.0.1:0"
48
+ const u = new URL ( GPTScript . serverURL )
49
+ if ( u . port === "0" ) {
50
+ const srv = net . createServer ( )
51
+ const s = srv . listen ( 0 , ( ) => {
52
+ GPTScript . serverURL = "http://" + u . hostname + ":" + String ( ( s . address ( ) as net . AddressInfo ) . port )
53
+ srv . close ( )
54
+
55
+ GPTScript . serverProcess = child_process . spawn ( getCmdPath ( ) , [ "--listen-address" , GPTScript . serverURL . replace ( "http://" , "" ) , "sdkserver" ] , {
56
+ env : process . env ,
57
+ stdio : [ "pipe" ]
58
+ } )
59
+
60
+ process . on ( "exit" , ( code ) => {
61
+ GPTScript . serverProcess . stdin ?. end ( )
62
+ GPTScript . serverProcess . kill ( code )
63
+ } )
64
+ } )
65
+ }
55
66
}
56
67
}
57
68
58
69
close ( ) : void {
59
- clientCount --
60
- if ( clientCount === 0 && serverProcess ) {
61
- serverProcess . kill ( "SIGTERM" )
62
- serverProcess . stdin ?. end ( )
70
+ GPTScript . instanceCount --
71
+ if ( GPTScript . instanceCount === 0 && GPTScript . serverProcess ) {
72
+ GPTScript . serverProcess . kill ( "SIGTERM" )
73
+ GPTScript . serverProcess . stdin ?. end ( )
63
74
}
64
75
}
65
76
@@ -76,10 +87,10 @@ export class Client {
76
87
}
77
88
78
89
async runBasicCommand ( cmd : string ) : Promise < string > {
79
- if ( ! this . clientReady ) {
80
- this . clientReady = await this . testGPTScriptURL ( 20 )
90
+ if ( ! this . ready ) {
91
+ this . ready = await this . testGPTScriptURL ( 20 )
81
92
}
82
- const r = new RunSubcommand ( cmd , "" , "" , { } , this . gptscriptURL )
93
+ const r = new RunSubcommand ( cmd , "" , "" , { } , GPTScript . serverURL )
83
94
r . requestNoStream ( null )
84
95
return r . text ( )
85
96
}
@@ -92,10 +103,10 @@ export class Client {
92
103
* @return {Run } The Run object representing the running tool.
93
104
*/
94
105
async run ( toolName : string , opts : RunOpts = { } ) : Promise < Run > {
95
- if ( ! this . clientReady ) {
96
- this . clientReady = await this . testGPTScriptURL ( 20 )
106
+ if ( ! this . ready ) {
107
+ this . ready = await this . testGPTScriptURL ( 20 )
97
108
}
98
- return ( new Run ( "run" , toolName , "" , opts , this . gptscriptURL ) ) . nextChat ( opts . input )
109
+ return ( new Run ( "run" , toolName , "" , opts , GPTScript . serverURL ) ) . nextChat ( opts . input )
99
110
}
100
111
101
112
/**
@@ -106,8 +117,8 @@ export class Client {
106
117
* @return {Run } The Run object representing the evaluation.
107
118
*/
108
119
async evaluate ( tool : ToolDef | ToolDef [ ] | string , opts : RunOpts = { } ) : Promise < Run > {
109
- if ( ! this . clientReady ) {
110
- this . clientReady = await this . testGPTScriptURL ( 20 )
120
+ if ( ! this . ready ) {
121
+ this . ready = await this . testGPTScriptURL ( 20 )
111
122
}
112
123
let toolString : string = ""
113
124
@@ -119,30 +130,30 @@ export class Client {
119
130
toolString = toolDefToString ( tool )
120
131
}
121
132
122
- return ( new Run ( "evaluate" , "" , toolString , opts , this . gptscriptURL ) ) . nextChat ( opts . input )
133
+ return ( new Run ( "evaluate" , "" , toolString , opts , GPTScript . serverURL ) ) . nextChat ( opts . input )
123
134
}
124
135
125
136
async parse ( fileName : string ) : Promise < Block [ ] > {
126
- if ( ! this . clientReady ) {
127
- this . clientReady = await this . testGPTScriptURL ( 20 )
137
+ if ( ! this . ready ) {
138
+ this . ready = await this . testGPTScriptURL ( 20 )
128
139
}
129
- const r : Run = new RunSubcommand ( "parse" , fileName , "" , { } , this . gptscriptURL )
140
+ const r : Run = new RunSubcommand ( "parse" , fileName , "" , { } , GPTScript . serverURL )
130
141
r . request ( { file : fileName } )
131
142
return parseBlocksFromNodes ( ( await r . json ( ) ) . nodes )
132
143
}
133
144
134
145
async parseTool ( toolContent : string ) : Promise < Block [ ] > {
135
- if ( ! this . clientReady ) {
136
- this . clientReady = await this . testGPTScriptURL ( 20 )
146
+ if ( ! this . ready ) {
147
+ this . ready = await this . testGPTScriptURL ( 20 )
137
148
}
138
- const r : Run = new RunSubcommand ( "parse" , "" , toolContent , { } , this . gptscriptURL )
149
+ const r : Run = new RunSubcommand ( "parse" , "" , toolContent , { } , GPTScript . serverURL )
139
150
r . request ( { content : toolContent } )
140
151
return parseBlocksFromNodes ( ( await r . json ( ) ) . nodes )
141
152
}
142
153
143
154
async stringify ( blocks : Block [ ] ) : Promise < string > {
144
- if ( ! this . clientReady ) {
145
- this . clientReady = await this . testGPTScriptURL ( 20 )
155
+ if ( ! this . ready ) {
156
+ this . ready = await this . testGPTScriptURL ( 20 )
146
157
}
147
158
const nodes : any [ ] = [ ]
148
159
@@ -162,16 +173,16 @@ export class Client {
162
173
}
163
174
}
164
175
165
- const r : Run = new RunSubcommand ( "fmt" , "" , JSON . stringify ( { nodes : nodes } ) , { } , this . gptscriptURL )
176
+ const r : Run = new RunSubcommand ( "fmt" , "" , JSON . stringify ( { nodes : nodes } ) , { } , GPTScript . serverURL )
166
177
r . request ( { nodes : nodes } )
167
178
return r . text ( )
168
179
}
169
180
170
181
async confirm ( response : AuthResponse ) : Promise < void > {
171
- if ( ! this . clientReady ) {
172
- this . clientReady = await this . testGPTScriptURL ( 20 )
182
+ if ( ! this . ready ) {
183
+ this . ready = await this . testGPTScriptURL ( 20 )
173
184
}
174
- const resp = await fetch ( `${ this . gptscriptURL } /confirm/${ response . id } ` , {
185
+ const resp = await fetch ( `${ GPTScript . serverURL } /confirm/${ response . id } ` , {
175
186
method : "POST" ,
176
187
body : JSON . stringify ( response )
177
188
} )
@@ -182,10 +193,10 @@ export class Client {
182
193
}
183
194
184
195
async promptResponse ( response : PromptResponse ) : Promise < void > {
185
- if ( ! this . clientReady ) {
186
- this . clientReady = await this . testGPTScriptURL ( 20 )
196
+ if ( ! this . ready ) {
197
+ this . ready = await this . testGPTScriptURL ( 20 )
187
198
}
188
- const resp = await fetch ( `${ this . gptscriptURL } /prompt-response/${ response . id } ` , {
199
+ const resp = await fetch ( `${ GPTScript . serverURL } /prompt-response/${ response . id } ` , {
189
200
method : "POST" ,
190
201
body : JSON . stringify ( response . responses )
191
202
} )
@@ -197,7 +208,7 @@ export class Client {
197
208
198
209
private async testGPTScriptURL ( count : number ) : Promise < boolean > {
199
210
try {
200
- await fetch ( `${ this . gptscriptURL } /healthz` )
211
+ await fetch ( `${ GPTScript . serverURL } /healthz` )
201
212
return true
202
213
} catch {
203
214
if ( count === 0 ) {
0 commit comments