Skip to content

Commit 0e08fc5

Browse files
authored
feat: Group Concurrent Access Token Requests for Base External Clients (#1840)
1 parent 76666f8 commit 0e08fc5

File tree

2 files changed

+82
-0
lines changed

2 files changed

+82
-0
lines changed

src/auth/baseexternalclient.ts

+18
Original file line numberDiff line numberDiff line change
@@ -252,6 +252,11 @@ export abstract class BaseExternalAccountClient extends AuthClient {
252252
*/
253253
protected cloudResourceManagerURL: URL | string;
254254
protected supplierContext: ExternalAccountSupplierContext;
255+
/**
256+
* A pending access token request. Used for concurrent calls.
257+
*/
258+
#pendingAccessToken: Promise<CredentialsWithResponse> | null = null;
259+
255260
/**
256261
* Instantiate a BaseExternalAccountClient instance using the provided JSON
257262
* object loaded from an external account credentials file.
@@ -545,6 +550,19 @@ export abstract class BaseExternalAccountClient extends AuthClient {
545550
* @return A promise that resolves with the fresh GCP access tokens.
546551
*/
547552
protected async refreshAccessTokenAsync(): Promise<CredentialsWithResponse> {
553+
// Use an existing access token request, or cache a new one
554+
this.#pendingAccessToken =
555+
this.#pendingAccessToken || this.#internalRefreshAccessTokenAsync();
556+
557+
try {
558+
return await this.#pendingAccessToken;
559+
} finally {
560+
// clear pending access token for future requests
561+
this.#pendingAccessToken = null;
562+
}
563+
}
564+
565+
async #internalRefreshAccessTokenAsync(): Promise<CredentialsWithResponse> {
548566
// Retrieve the external credential.
549567
const subjectToken = await this.retrieveSubjectToken();
550568
// Construct the STS credentials options.

test/test.baseexternalclient.ts

+64
Original file line numberDiff line numberDiff line change
@@ -326,6 +326,70 @@ describe('BaseExternalAccountClient', () => {
326326

327327
scope.done();
328328
});
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+
});
329393
});
330394

331395
describe('projectNumber', () => {

0 commit comments

Comments
 (0)