Skip to content

Commit

Permalink
fix: add user-agent header to transfer manager and resumable upload (#…
Browse files Browse the repository at this point in the history
  • Loading branch information
ddelgrosso1 authored Oct 19, 2023
1 parent f20ef3c commit 0520867
Show file tree
Hide file tree
Showing 9 changed files with 87 additions and 64 deletions.
4 changes: 2 additions & 2 deletions src/nodejs-common/service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ import {
PackageJson,
util,
} from './util';
import {getRuntimeTrackingString} from '../util';
import {getRuntimeTrackingString, getUserAgentString} from '../util';

export const DEFAULT_PROJECT_ID_TOKEN = '{{projectId}}';

Expand Down Expand Up @@ -242,7 +242,7 @@ export class Service {
delete reqOpts.interceptors_;

const pkg = this.packageJson;
let userAgent = util.getUserAgentFromPackageJson(pkg);
let userAgent = getUserAgentString();
if (this.providedUserAgent) {
userAgent = `${this.providedUserAgent} ${userAgent}`;
}
Expand Down
18 changes: 2 additions & 16 deletions src/nodejs-common/util.ts
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ import {teenyRequest} from 'teeny-request';
import {Interceptor} from './service-object';
import * as uuid from 'uuid';
import {DEFAULT_PROJECT_ID_TOKEN} from './service';
import {getRuntimeTrackingString} from '../util';
import {getRuntimeTrackingString, getUserAgentString} from '../util';

const packageJson = require('../../../package.json');

Expand Down Expand Up @@ -1001,20 +1001,6 @@ export class Util {
}
}

/**
* Create a properly-formatted User-Agent string from a package.json file.
*
* @param {object} packageJson - A module's package.json file.
* @return {string} userAgent - The formatted User-Agent string.
*/
getUserAgentFromPackageJson(packageJson: PackageJson) {
const hyphenatedPackageName = packageJson.name
.replace('@google-cloud', 'gcloud-node') // For legacy purposes.
.replace('/', '-'); // For UA spec-compliance purposes.

return hyphenatedPackageName + '/' + packageJson.version;
}

/**
* Given two parameters, figure out if this is either:
* - Just a callback function
Expand All @@ -1033,7 +1019,7 @@ export class Util {

_getDefaultHeaders(gcclGcsCmd?: string) {
const headers = {
'User-Agent': util.getUserAgentFromPackageJson(packageJson),
'User-Agent': getUserAgentString(),
'x-goog-api-client': `${getRuntimeTrackingString()} gccl/${
packageJson.version
} gccl-invocation-id/${uuid.v4()}`,
Expand Down
5 changes: 4 additions & 1 deletion src/resumable-upload.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ import {Readable, Writable, WritableOptions} from 'stream';
import retry = require('async-retry');
import {RetryOptions, PreconditionOptions} from './storage';
import * as uuid from 'uuid';
import {getRuntimeTrackingString} from './util';
import {getRuntimeTrackingString, getUserAgentString} from './util';
import {GCCL_GCS_CMD_KEY} from './nodejs-common/util';

const NOT_FOUND_STATUS_CODE = 404;
Expand Down Expand Up @@ -612,6 +612,7 @@ export class Upload extends Writable {
),
data: metadata,
headers: {
'User-Agent': getUserAgentString(),
'x-goog-api-client': googAPIClient,
...headers,
},
Expand Down Expand Up @@ -787,6 +788,7 @@ export class Upload extends Writable {
}

const headers: GaxiosOptions['headers'] = {
'User-Agent': getUserAgentString(),
'x-goog-api-client': googAPIClient,
};

Expand Down Expand Up @@ -936,6 +938,7 @@ export class Upload extends Writable {
headers: {
'Content-Length': 0,
'Content-Range': 'bytes */*',
'User-Agent': getUserAgentString(),
'x-goog-api-client': googAPIClient,
},
};
Expand Down
11 changes: 9 additions & 2 deletions src/transfer-manager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ import {ApiError} from './nodejs-common';
import {GaxiosResponse, Headers} from 'gaxios';
import {createHash} from 'crypto';
import {GCCL_GCS_CMD_KEY} from './nodejs-common/util';
import {getRuntimeTrackingString} from './util';
import {getRuntimeTrackingString, getUserAgentString} from './util';

const packageJson = require('../../package.json');

Expand Down Expand Up @@ -205,6 +205,7 @@ class XMLMultiPartUploadHelper implements MultiPartUploadHelper {

#setGoogApiClientHeaders(headers: Headers = {}): Headers {
let headerFound = false;
let userAgentFound = false;

for (const [key, value] of Object.entries(headers)) {
if (key.toLocaleLowerCase().trim() === 'x-goog-api-client') {
Expand All @@ -216,7 +217,8 @@ class XMLMultiPartUploadHelper implements MultiPartUploadHelper {
key
] = `${value} gccl-gcs-cmd/${GCCL_GCS_CMD_FEATURE.UPLOAD_SHARDED}`;
}
break;
} else if (key.toLocaleLowerCase().trim() === 'user-agent') {
userAgentFound = true;
}
}

Expand All @@ -227,6 +229,11 @@ class XMLMultiPartUploadHelper implements MultiPartUploadHelper {
} gccl-gcs-cmd/${GCCL_GCS_CMD_FEATURE.UPLOAD_SHARDED}`;
}

// If the User-Agent isn't present, add it
if (!userAgentFound) {
headers['User-Agent'] = getUserAgentString();
}

return headers;
}

Expand Down
13 changes: 13 additions & 0 deletions src/util.ts
Original file line number Diff line number Diff line change
Expand Up @@ -192,6 +192,19 @@ export function getRuntimeTrackingString(): string {
}
}

/**
* Looks at package.json and creates the user-agent string to be applied to request headers.
* @returns {string} user agent string.
*/
export function getUserAgentString(): string {
const pkg = require('../../package.json');
const hyphenatedPackageName = pkg.name
.replace('@google-cloud', 'gcloud-node') // For legacy purposes.
.replace('/', '-'); // For UA spec-compliance purposes.

return hyphenatedPackageName + '/' + pkg.version;
}

export class PassThroughShim extends PassThrough {
private shouldEmitReading = true;
private shouldEmitWriting = true;
Expand Down
33 changes: 2 additions & 31 deletions test/nodejs-common/service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ import {
util,
Util,
} from '../../src/nodejs-common/util';
import {getUserAgentString} from '../../src/util';

proxyquire.noPreserveCache();

Expand Down Expand Up @@ -473,40 +474,10 @@ describe('Service', () => {
});

it('should add the User Agent', done => {
const userAgent = 'user-agent/0.0.0';

const getUserAgentFn = util.getUserAgentFromPackageJson;
util.getUserAgentFromPackageJson = packageJson => {
util.getUserAgentFromPackageJson = getUserAgentFn;
assert.strictEqual(packageJson, service.packageJson);
return userAgent;
};

service.makeAuthenticatedRequest = (reqOpts: DecorateRequestOptions) => {
assert.strictEqual(reqOpts.headers!['User-Agent'], userAgent);
done();
};

service.request_(reqOpts, assert.ifError);
});

it('should add the provided User Agent', done => {
const userAgent = 'user-agent/0.0.0';
const providedUserAgent = 'test';

service.providedUserAgent = providedUserAgent;

const getUserAgentFn = util.getUserAgentFromPackageJson;
util.getUserAgentFromPackageJson = packageJson => {
util.getUserAgentFromPackageJson = getUserAgentFn;
assert.strictEqual(packageJson, service.packageJson);
return userAgent;
};

service.makeAuthenticatedRequest = (reqOpts: DecorateRequestOptions) => {
assert.strictEqual(
reqOpts.headers!['User-Agent'],
`${providedUserAgent} ${userAgent}`
getUserAgentString()
);
done();
};
Expand Down
11 changes: 0 additions & 11 deletions test/nodejs-common/util.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1869,17 +1869,6 @@ describe('common/util', () => {
});
});

describe('getUserAgentFromPackageJson', () => {
it('should format a User Agent string from a package.json', () => {
const userAgent = util.getUserAgentFromPackageJson({
name: '@google-cloud/storage',
version: '0.1.0',
});

assert.strictEqual(userAgent, 'gcloud-node-storage/0.1.0');
});
});

describe('maybeOptionsOrCallback', () => {
it('should allow passing just a callback', () => {
const optionsOrCallback = () => {};
Expand Down
15 changes: 15 additions & 0 deletions test/resumable-upload.ts
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,7 @@ const CHUNK_SIZE_MULTIPLE = 2 ** 18;
const queryPath = '/?userProject=user-project-id';
const X_GOOG_API_HEADER_REGEX =
/^gl-node\/(?<nodeVersion>[^W]+) gccl\/(?<gccl>[^W]+) gccl-invocation-id\/(?<gcclInvocationId>[^W]+) gccl-gcs-cmd\/(?<gcclGcsCmd>[^W]+)$/;
const USER_AGENT_REGEX = /^gcloud-node-storage\/(?<libVersion>[^W]+)$/;

function mockAuthorizeRequest(
code = 200,
Expand Down Expand Up @@ -1162,6 +1163,7 @@ describe('resumable-upload', () => {
assert.ok(
X_GOOG_API_HEADER_REGEX.test(reqOpts.headers['x-goog-api-client'])
);
assert.ok(USER_AGENT_REGEX.test(reqOpts.headers['User-Agent']));

const data = await getAllDataFromRequest();

Expand All @@ -1178,6 +1180,7 @@ describe('resumable-upload', () => {
assert.ok(
X_GOOG_API_HEADER_REGEX.test(reqOpts.headers['x-goog-api-client'])
);
assert.ok(USER_AGENT_REGEX.test(reqOpts.headers['User-Agent']));

const data = await getAllDataFromRequest();

Expand Down Expand Up @@ -1211,6 +1214,7 @@ describe('resumable-upload', () => {
assert.ok(
X_GOOG_API_HEADER_REGEX.test(reqOpts.headers['x-goog-api-client'])
);
assert.ok(USER_AGENT_REGEX.test(reqOpts.headers['User-Agent']));

const data = await getAllDataFromRequest();

Expand Down Expand Up @@ -1242,6 +1246,7 @@ describe('resumable-upload', () => {
assert.ok(
X_GOOG_API_HEADER_REGEX.test(reqOpts.headers['x-goog-api-client'])
);
assert.ok(USER_AGENT_REGEX.test(reqOpts.headers['User-Agent']));

const data = await getAllDataFromRequest();

Expand Down Expand Up @@ -1272,6 +1277,7 @@ describe('resumable-upload', () => {
assert.ok(
X_GOOG_API_HEADER_REGEX.test(reqOpts.headers['x-goog-api-client'])
);
assert.ok(USER_AGENT_REGEX.test(reqOpts.headers['User-Agent']));
const data = await getAllDataFromRequest();

assert.equal(data.byteLength, CONTENT_LENGTH - NUM_BYTES_WRITTEN);
Expand Down Expand Up @@ -1456,6 +1462,7 @@ describe('resumable-upload', () => {
assert.ok(
X_GOOG_API_HEADER_REGEX.test(reqOpts.headers['x-goog-api-client'])
);
assert.ok(USER_AGENT_REGEX.test(reqOpts.headers['User-Agent']));
done();
return {};
};
Expand Down Expand Up @@ -2242,6 +2249,7 @@ describe('resumable-upload', () => {
request.opts.headers['x-goog-api-client']
)
);
assert.ok(USER_AGENT_REGEX.test(request.opts.headers['User-Agent']));

done();
});
Expand Down Expand Up @@ -2413,6 +2421,9 @@ describe('resumable-upload', () => {
request.opts.headers['x-goog-api-client']
)
);
assert.ok(
USER_AGENT_REGEX.test(request.opts.headers['User-Agent'])
);
} else {
// The preceding chunks
const endByte = offset + CHUNK_SIZE - 1;
Expand All @@ -2429,6 +2440,9 @@ describe('resumable-upload', () => {
request.opts.headers['x-goog-api-client']
)
);
assert.ok(
USER_AGENT_REGEX.test(request.opts.headers['User-Agent'])
);
}
}

Expand Down Expand Up @@ -2527,6 +2541,7 @@ describe('resumable-upload', () => {
request.opts.headers['x-goog-api-client']
)
);
assert.ok(USER_AGENT_REGEX.test(request.opts.headers['User-Agent']));

done();
});
Expand Down
41 changes: 40 additions & 1 deletion test/transfer-manager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -443,6 +443,7 @@ describe('Transfer Manager', () => {
const headersToAdd = {
'Content-Type': 'foo/bar',
'x-goog-meta-foo': 'foobar',
'User-Agent': 'barfoo',
};

mockGeneratorFunction = (bucket, fileName, uploadId, partsMap) => {
Expand Down Expand Up @@ -497,7 +498,7 @@ describe('Transfer Manager', () => {
});

it('should set the appropriate `GCCL_GCS_CMD_KEY`', async () => {
let called = true;
let called = false;
class TestAuthClient extends AuthClient {
async getAccessToken() {
return {token: '', res: undefined};
Expand Down Expand Up @@ -536,5 +537,43 @@ describe('Transfer Manager', () => {

assert(called);
});

it('should set User-Agent correctly based on package.json', async () => {
let called = false;
class TestAuthClient extends AuthClient {
async getAccessToken() {
return {token: '', res: undefined};
}

async getRequestHeaders() {
return {};
}

async request(opts: GaxiosOptions) {
called = true;

assert(opts.headers);
assert('User-Agent' in opts.headers);
assert.match(opts.headers['User-Agent'], /gcloud-node/);

return {
data: Buffer.from(
`<InitiateMultipartUploadResult>
<UploadId>1</UploadId>
</InitiateMultipartUploadResult>`
),
headers: {},
} as GaxiosResponse;
}
}

transferManager.bucket.storage.authClient = new GoogleAuth({
authClient: new TestAuthClient(),
});

await transferManager.uploadFileInChunks(filePath);

assert(called);
});
});
});

0 comments on commit 0520867

Please sign in to comment.