Skip to content

Commit

Permalink
test(fetch): make prepareData async to enforce fakeTime order
Browse files Browse the repository at this point in the history
  • Loading branch information
Ugzuzg committed May 14, 2022
1 parent 22085fc commit b1d9b86
Showing 1 changed file with 71 additions and 88 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -155,8 +155,7 @@ describe('fetch', () => {
lastResponse = undefined;
};

const prepareData = (
done: any,
const prepareData = async (
fileUrl: string,
config: FetchInstrumentationConfig,
method?: string,
Expand Down Expand Up @@ -239,36 +238,25 @@ describe('fetch', () => {
);

rootSpan = webTracerWithZone.startSpan('root');
api.context.with(api.trace.setSpan(api.context.active(), rootSpan), () => {
await api.context.with(api.trace.setSpan(api.context.active(), rootSpan), async () => {
fakeNow = 0;
void getData(fileUrl, method)
.then(
response => {
// this is a bit tricky as the only way to get all request headers from
// fetch is to use json()
return response.json().then(
json => {
lastResponse = json;
const headers: { [key: string]: string } = {};
Object.keys(lastResponse.headers).forEach(key => {
headers[key.toLowerCase()] = lastResponse.headers[key];
});
lastResponse.headers = headers;
},
() => {
lastResponse = undefined;
}
);
},
() => {
lastResponse = undefined;
}
)
.then(sinon.clock.runAllAsync)
.then(() => {
done();
try {
const responsePromise = getData(fileUrl, method);
fakeNow = 300;
const response = await responsePromise;
// this is a bit tricky as the only way to get all request headers from
// fetch is to use json()
const json = await response.json();
lastResponse = json;
const headers: { [key: string]: string } = {};
Object.keys(lastResponse.headers).forEach(key => {
headers[key.toLowerCase()] = lastResponse.headers[key];
});
fakeNow = 300;
lastResponse.headers = headers;
} catch (e) {
lastResponse = undefined;
}
await sinon.clock.runAllAsync();
});
};

Expand All @@ -290,9 +278,9 @@ describe('fetch', () => {
});

describe('when request is successful', () => {
beforeEach(done => {
beforeEach(async () => {
const propagateTraceHeaderCorsUrls = [url];
prepareData(done, url, { propagateTraceHeaderCorsUrls });
await prepareData(url, { propagateTraceHeaderCorsUrls });
});

afterEach(() => {
Expand Down Expand Up @@ -580,13 +568,13 @@ describe('fetch', () => {

describe('when propagateTraceHeaderCorsUrls does NOT MATCH', () => {
let spyDebug: sinon.SinonSpy;
beforeEach(done => {
beforeEach(async () => {
const diagLogger = new api.DiagConsoleLogger();
spyDebug = sinon.spy();
diagLogger.debug = spyDebug;
api.diag.setLogger(diagLogger, api.DiagLogLevel.ALL);
clearData();
prepareData(done, url, {});
await prepareData(url, {});
});
afterEach(() => {
sinon.restore();
Expand Down Expand Up @@ -619,15 +607,13 @@ describe('fetch', () => {
});

describe('applyCustomAttributesOnSpan option', () => {
const noop = () => {};
const prepare = (
const prepare = async (
url: string,
applyCustomAttributesOnSpan: FetchCustomAttributeFunction,
cb: VoidFunction = noop
) => {
const propagateTraceHeaderCorsUrls = [url];

prepareData(cb, url, {
await prepareData(url, {
propagateTraceHeaderCorsUrls,
applyCustomAttributesOnSpan,
});
Expand All @@ -637,74 +623,71 @@ describe('fetch', () => {
clearData();
});

it('applies attributes when the request is succesful', done => {
prepare(
it('applies attributes when the request is succesful', async () => {
await prepare(
url,
span => {
span.setAttribute(CUSTOM_ATTRIBUTE_KEY, 'custom value');
},
() => {
const span: tracing.ReadableSpan = exportSpy.args[1][0][0];
const attributes = span.attributes;

assert.ok(attributes[CUSTOM_ATTRIBUTE_KEY] === 'custom value');
done();
}
);
const span: tracing.ReadableSpan = exportSpy.args[1][0][0];
const attributes = span.attributes;

assert.ok(attributes[CUSTOM_ATTRIBUTE_KEY] === 'custom value');
});

it('applies custom attributes when the request fails', done => {
prepare(
it('applies custom attributes when the request fails', async () => {
await prepare(
badUrl,
span => {
span.setAttribute(CUSTOM_ATTRIBUTE_KEY, 'custom value');
},
() => {
const span: tracing.ReadableSpan = exportSpy.args[1][0][0];
const attributes = span.attributes;

assert.ok(attributes[CUSTOM_ATTRIBUTE_KEY] === 'custom value');
done();
}
);
const span: tracing.ReadableSpan = exportSpy.args[1][0][0];
const attributes = span.attributes;

assert.ok(attributes[CUSTOM_ATTRIBUTE_KEY] === 'custom value');
});

it('has request and response objects in callback arguments', done => {
it('has request and response objects in callback arguments', async () => {
let request: any;
let response: any;
const applyCustomAttributes: FetchCustomAttributeFunction = (
span,
request,
response
req,
res
) => {
assert.ok(request.method === 'GET');
assert.ok(response.status === 200);

done();
request = req;
response = res;
};

prepare(url, applyCustomAttributes);
await prepare(url, applyCustomAttributes);
assert.ok(request.method === 'GET');
assert.ok(response.status === 200);
});

it('get response body from callback arguments response', done => {
it('get response body from callback arguments response', async () => {
let response: any;
const applyCustomAttributes: FetchCustomAttributeFunction = async (
span,
request,
response
req,
res
) => {
if(response instanceof Response ){
const rsp = await response.json();
assert.deepStrictEqual(rsp.args, {});
done();
if (res instanceof Response) {
response = res;
}
};

prepare(url, applyCustomAttributes);
await prepare(url, applyCustomAttributes);
const rsp = await response.json();
assert.deepStrictEqual(rsp.args, {});
});
});

describe('when url is ignored', () => {
beforeEach(done => {
beforeEach(async () => {
const propagateTraceHeaderCorsUrls = url;
prepareData(done, url, {
await prepareData(url, {
propagateTraceHeaderCorsUrls,
ignoreUrls: [propagateTraceHeaderCorsUrls],
});
Expand All @@ -726,9 +709,9 @@ describe('fetch', () => {
});

describe('when clearTimingResources is TRUE', () => {
beforeEach(done => {
beforeEach(async () => {
const propagateTraceHeaderCorsUrls = url;
prepareData(done, url, {
await prepareData(url, {
propagateTraceHeaderCorsUrls,
clearTimingResources: true,
});
Expand All @@ -746,9 +729,9 @@ describe('fetch', () => {
});

describe('when request is NOT successful (wrong url)', () => {
beforeEach(done => {
beforeEach(async () => {
const propagateTraceHeaderCorsUrls = badUrl;
prepareData(done, badUrl, { propagateTraceHeaderCorsUrls });
await prepareData(badUrl, { propagateTraceHeaderCorsUrls });
});
afterEach(() => {
clearData();
Expand All @@ -764,9 +747,9 @@ describe('fetch', () => {
});

describe('when request is NOT successful (405)', () => {
beforeEach(done => {
beforeEach(async () => {
const propagateTraceHeaderCorsUrls = url;
prepareData(done, url, { propagateTraceHeaderCorsUrls }, 'DELETE');
await prepareData(url, { propagateTraceHeaderCorsUrls }, 'DELETE');
});
afterEach(() => {
clearData();
Expand All @@ -783,11 +766,11 @@ describe('fetch', () => {
});

describe('when PerformanceObserver is used by default', () => {
beforeEach(done => {
beforeEach(async () => {
// All above tests test it already but just in case
// lets explicitly turn getEntriesByType off so we can be sure
// that the perf entries come from the observer.
prepareData(done, url, {}, undefined, false, true);
await prepareData(url, {}, undefined, false, true);
});
afterEach(() => {
clearData();
Expand All @@ -813,8 +796,8 @@ describe('fetch', () => {
});

describe('when fetching with relative url', () => {
beforeEach(done => {
prepareData(done, '/get', {}, undefined, false, true);
beforeEach(async () => {
await prepareData('/get', {}, undefined, false, true);
});
afterEach(() => {
clearData();
Expand All @@ -841,8 +824,8 @@ describe('fetch', () => {
});

describe('when PerformanceObserver is undefined', () => {
beforeEach(done => {
prepareData(done, url, {}, undefined, true, false);
beforeEach(async () => {
await prepareData(url, {}, undefined, true, false);
});

afterEach(() => {
Expand All @@ -868,8 +851,8 @@ describe('fetch', () => {
});

describe('when PerformanceObserver and performance.getEntriesByType are undefined', () => {
beforeEach(done => {
prepareData(done, url, {}, undefined, true, true);
beforeEach(async () => {
await prepareData(url, {}, undefined, true, true);
});
afterEach(() => {
clearData();
Expand Down

0 comments on commit b1d9b86

Please sign in to comment.