@@ -326,6 +326,70 @@ describe('BaseExternalAccountClient', () => {
326
326
327
327
scope . done ( ) ;
328
328
} ) ;
329
+
330
+ it ( 'should not duplicate access token requests for concurrent requests' , async ( ) => {
331
+ const client = new TestExternalAccountClient ( externalAccountOptionsNoUrl ) ;
332
+ const RESPONSE_A = {
333
+ access_token : 'ACCESS_TOKEN' ,
334
+ issued_token_type : 'urn:ietf:params:oauth:token-type:access_token' ,
335
+ token_type : 'Bearer' ,
336
+ expires_in : ONE_HOUR_IN_SECS ,
337
+ scope : 'scope1 scope2' ,
338
+ } ;
339
+
340
+ const RESPONSE_B = {
341
+ ...RESPONSE_A ,
342
+ access_token : 'ACCESS_TOKEN_2' ,
343
+ } ;
344
+
345
+ const scope = mockStsTokenExchange ( [
346
+ {
347
+ statusCode : 200 ,
348
+ response : RESPONSE_A ,
349
+ request : {
350
+ grant_type : 'urn:ietf:params:oauth:grant-type:token-exchange' ,
351
+ audience,
352
+ scope : 'https://www.googleapis.com/auth/cloud-platform' ,
353
+ requested_token_type :
354
+ 'urn:ietf:params:oauth:token-type:access_token' ,
355
+ subject_token : 'subject_token_0' ,
356
+ subject_token_type : 'urn:ietf:params:oauth:token-type:jwt' ,
357
+ } ,
358
+ } ,
359
+ {
360
+ statusCode : 200 ,
361
+ response : RESPONSE_B ,
362
+ request : {
363
+ grant_type : 'urn:ietf:params:oauth:grant-type:token-exchange' ,
364
+ audience,
365
+ scope : 'https://www.googleapis.com/auth/cloud-platform' ,
366
+ requested_token_type :
367
+ 'urn:ietf:params:oauth:token-type:access_token' ,
368
+ subject_token : 'subject_token_1' ,
369
+ subject_token_type : 'urn:ietf:params:oauth:token-type:jwt' ,
370
+ } ,
371
+ } ,
372
+ ] ) ;
373
+
374
+ // simulate 5 concurrent requests
375
+ const calls = [
376
+ client . getAccessToken ( ) ,
377
+ client . getAccessToken ( ) ,
378
+ client . getAccessToken ( ) ,
379
+ client . getAccessToken ( ) ,
380
+ client . getAccessToken ( ) ,
381
+ ] ;
382
+
383
+ for ( const { token} of await Promise . all ( calls ) ) {
384
+ assert . strictEqual ( token , RESPONSE_A . access_token ) ;
385
+ }
386
+
387
+ // this should be handled in a second request as the above were all awaited and we're forcing an expiration
388
+ client . eagerRefreshThresholdMillis = RESPONSE_A . expires_in * 1000 ;
389
+ assert ( ( await client . getAccessToken ( ) ) . token , RESPONSE_B . access_token ) ;
390
+
391
+ scope . done ( ) ;
392
+ } ) ;
329
393
} ) ;
330
394
331
395
describe ( 'projectNumber' , ( ) => {
0 commit comments