@@ -143,6 +143,50 @@ export abstract class Runnable<
143
143
return output ;
144
144
}
145
145
146
+ protected async * _streamWithConfig < T extends RunOutput > (
147
+ generator : AsyncGenerator < T > ,
148
+ options ?: RunnableConfig & { runType ?: string }
149
+ ) {
150
+ const callbackManager_ = await CallbackManager . configure (
151
+ options ?. callbacks ,
152
+ undefined ,
153
+ options ?. tags ,
154
+ undefined ,
155
+ options ?. metadata
156
+ ) ;
157
+ // TODO: Find a way to pass the entire streamed value into the callback.
158
+ const runManager = await callbackManager_ ?. handleChainStart (
159
+ this . toJSON ( ) ,
160
+ _coerceToDict ( "<streamed value>" , "input" ) ,
161
+ undefined ,
162
+ options ?. runType
163
+ ) ;
164
+ let output ;
165
+ let concatSupported = true ;
166
+ try {
167
+ for await ( const chunk of generator ) {
168
+ yield chunk ;
169
+ if ( concatSupported ) {
170
+ if ( output === undefined ) {
171
+ output = chunk ;
172
+ } else {
173
+ try {
174
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
175
+ output = ( output as any ) . concat ( chunk ) ;
176
+ } catch ( e ) {
177
+ output = undefined ;
178
+ concatSupported = false ;
179
+ }
180
+ }
181
+ }
182
+ }
183
+ } catch ( e ) {
184
+ await runManager ?. handleChainError ( e ) ;
185
+ throw e ;
186
+ }
187
+ await runManager ?. handleChainEnd ( _coerceToDict ( output , "output" ) ) ;
188
+ }
189
+
146
190
_patchConfig (
147
191
config : Partial < CallOptions > = { } ,
148
192
callbackManager : CallbackManager | undefined = undefined
@@ -160,6 +204,11 @@ export abstract class Runnable<
160
204
} ) ;
161
205
}
162
206
207
+ transform ?(
208
+ generator : AsyncGenerator < RunInput > ,
209
+ options : Partial < CallOptions >
210
+ ) : AsyncGenerator < RunOutput > ;
211
+
163
212
// eslint-disable-next-line @typescript-eslint/no-explicit-any
164
213
static isRunnable ( thing : any ) : thing is Runnable {
165
214
return thing . lc_runnable ;
@@ -314,8 +363,17 @@ export class RunnableSequence<
314
363
_coerceToDict ( input , "input" )
315
364
) ;
316
365
let nextStepInput = input ;
366
+ const steps = [ this . first , ...this . middle , this . last ] ;
367
+ // Find the index of the last runnable in the sequence that doesn't have a .transform() method
368
+ // and start streaming from there
369
+ const streamingStartStepIndex =
370
+ steps . length -
371
+ [ ...steps ]
372
+ . reverse ( )
373
+ . findIndex ( ( step ) => typeof step . transform !== "function" ) -
374
+ 1 ;
317
375
try {
318
- for ( const step of [ this . first , ... this . middle ] ) {
376
+ for ( const step of steps . slice ( 0 , streamingStartStepIndex ) ) {
319
377
nextStepInput = await step . invoke (
320
378
nextStepInput ,
321
379
this . _patchConfig ( options , runManager ?. getChild ( ) )
@@ -328,11 +386,18 @@ export class RunnableSequence<
328
386
let concatSupported = true ;
329
387
let finalOutput ;
330
388
try {
331
- const iterator = await this . last . _streamIterator (
389
+ let finalGenerator = await steps [ streamingStartStepIndex ] . _streamIterator (
332
390
nextStepInput ,
333
391
this . _patchConfig ( options , runManager ?. getChild ( ) )
334
392
) ;
335
- for await ( const chunk of iterator ) {
393
+ for ( const step of steps . slice ( streamingStartStepIndex + 1 ) ) {
394
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
395
+ finalGenerator = await step . transform ! (
396
+ finalGenerator ,
397
+ this . _patchConfig ( options , runManager ?. getChild ( ) )
398
+ ) ;
399
+ }
400
+ for await ( const chunk of finalGenerator ) {
336
401
yield chunk ;
337
402
if ( concatSupported ) {
338
403
if ( finalOutput === undefined ) {
0 commit comments