2020-01-15 19:41:47 +00:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright 2020 Balena Ltd.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import * as nock from 'nock';
|
2021-05-19 22:12:49 +00:00
|
|
|
import * as fs from 'fs';
|
2020-01-15 19:41:47 +00:00
|
|
|
|
|
|
|
export interface ScopeOpts {
|
|
|
|
optional?: boolean;
|
|
|
|
persist?: boolean;
|
2023-12-11 18:55:49 +02:00
|
|
|
times?: number;
|
2020-01-15 19:41:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Base class for tests using nock to intercept HTTP requests.
|
|
|
|
* Subclasses include BalenaAPIMock, DockerMock and BuilderMock.
|
|
|
|
*/
|
|
|
|
export class NockMock {
|
|
|
|
public readonly scope: nock.Scope;
|
|
|
|
// Expose `scope` as `expect` to allow for better semantics in tests
|
2020-08-27 11:50:57 +01:00
|
|
|
public readonly expect;
|
2020-01-15 19:41:47 +00:00
|
|
|
protected static instanceCount = 0;
|
|
|
|
|
2021-07-28 13:19:57 +00:00
|
|
|
constructor(
|
|
|
|
public basePathPattern: string | RegExp,
|
|
|
|
public allowUnmocked: boolean = false,
|
|
|
|
) {
|
2020-01-15 19:41:47 +00:00
|
|
|
if (NockMock.instanceCount === 0) {
|
|
|
|
if (!nock.isActive()) {
|
|
|
|
nock.activate();
|
|
|
|
}
|
|
|
|
nock.emitter.on('no match', this.handleUnexpectedRequest);
|
|
|
|
} else if (process.env.DEBUG) {
|
|
|
|
console.error(
|
2020-01-20 21:21:05 +00:00
|
|
|
`[debug] NockMock.constructor() instance count is ${NockMock.instanceCount}`,
|
2020-01-15 19:41:47 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
NockMock.instanceCount += 1;
|
2021-07-28 13:19:57 +00:00
|
|
|
this.scope = nock(this.basePathPattern, { allowUnmocked });
|
2020-08-27 11:50:57 +01:00
|
|
|
this.expect = this.scope;
|
2020-01-15 19:41:47 +00:00
|
|
|
}
|
|
|
|
|
2023-12-11 18:55:49 +02:00
|
|
|
public optMethod(
|
|
|
|
method: 'get' | 'delete' | 'patch' | 'post',
|
|
|
|
uri: string | RegExp | ((uri: string) => boolean),
|
|
|
|
{ optional = false, persist = false, times = undefined }: ScopeOpts,
|
|
|
|
) {
|
|
|
|
let scope = this.scope;
|
|
|
|
if (persist) {
|
|
|
|
scope = scope.persist();
|
|
|
|
}
|
|
|
|
let reqInterceptor = scope[method](uri);
|
|
|
|
if (times != null) {
|
|
|
|
reqInterceptor = reqInterceptor.times(times);
|
|
|
|
} else if (optional) {
|
|
|
|
reqInterceptor = reqInterceptor.optionally();
|
|
|
|
}
|
|
|
|
return reqInterceptor;
|
|
|
|
}
|
|
|
|
|
2020-01-15 19:41:47 +00:00
|
|
|
public optGet(
|
|
|
|
uri: string | RegExp | ((uri: string) => boolean),
|
2023-12-11 18:55:49 +02:00
|
|
|
opts: ScopeOpts,
|
2020-01-15 19:41:47 +00:00
|
|
|
): nock.Interceptor {
|
2023-12-11 18:55:49 +02:00
|
|
|
return this.optMethod('get', uri, opts);
|
2020-01-15 19:41:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public optDelete(
|
|
|
|
uri: string | RegExp | ((uri: string) => boolean),
|
2023-12-11 18:55:49 +02:00
|
|
|
opts: ScopeOpts,
|
2020-01-15 19:41:47 +00:00
|
|
|
) {
|
2023-12-11 18:55:49 +02:00
|
|
|
return this.optMethod('delete', uri, opts);
|
2020-01-15 19:41:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public optPatch(
|
|
|
|
uri: string | RegExp | ((uri: string) => boolean),
|
2023-12-11 18:55:49 +02:00
|
|
|
opts: ScopeOpts,
|
2020-01-15 19:41:47 +00:00
|
|
|
) {
|
2023-12-11 18:55:49 +02:00
|
|
|
return this.optMethod('patch', uri, opts);
|
2020-01-15 19:41:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public optPost(
|
|
|
|
uri: string | RegExp | ((uri: string) => boolean),
|
2023-12-11 18:55:49 +02:00
|
|
|
opts: ScopeOpts,
|
2020-01-15 19:41:47 +00:00
|
|
|
) {
|
2023-12-11 18:55:49 +02:00
|
|
|
return this.optMethod('post', uri, opts);
|
2020-01-15 19:41:47 +00:00
|
|
|
}
|
|
|
|
|
2020-05-21 14:51:10 +01:00
|
|
|
protected inspectNoOp(_uri: string, _requestBody: nock.Body): void {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected getInspectedReplyBodyFunction(
|
|
|
|
inspectRequest: (uri: string, requestBody: nock.Body) => void,
|
|
|
|
replyBody: nock.ReplyBody,
|
|
|
|
) {
|
2020-06-15 23:53:07 +01:00
|
|
|
return function (
|
2020-05-21 14:51:10 +01:00
|
|
|
this: nock.ReplyFnContext,
|
|
|
|
uri: string,
|
|
|
|
requestBody: nock.Body,
|
|
|
|
cb: (err: NodeJS.ErrnoException | null, result: nock.ReplyBody) => void,
|
|
|
|
) {
|
|
|
|
try {
|
|
|
|
inspectRequest(uri, requestBody);
|
|
|
|
} catch (err) {
|
|
|
|
cb(err, '');
|
|
|
|
}
|
|
|
|
cb(null, replyBody);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-05-19 22:12:49 +00:00
|
|
|
protected getInspectedReplyFileFunction(
|
|
|
|
inspectRequest: (uri: string, requestBody: nock.Body) => void,
|
|
|
|
replyBodyFile: string,
|
|
|
|
) {
|
|
|
|
return function (
|
|
|
|
this: nock.ReplyFnContext,
|
|
|
|
uri: string,
|
|
|
|
requestBody: nock.Body,
|
|
|
|
cb: (err: NodeJS.ErrnoException | null, result: nock.ReplyBody) => void,
|
|
|
|
) {
|
|
|
|
try {
|
|
|
|
inspectRequest(uri, requestBody);
|
|
|
|
} catch (err) {
|
|
|
|
cb(err, '');
|
|
|
|
}
|
|
|
|
|
|
|
|
const replyBody = fs.readFileSync(replyBodyFile);
|
|
|
|
cb(null, replyBody);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-01-15 19:41:47 +00:00
|
|
|
public done() {
|
|
|
|
try {
|
|
|
|
// scope.done() will throw an error if there are expected api calls that have not happened.
|
|
|
|
// So ensure that all expected calls have been made.
|
|
|
|
this.scope.done();
|
|
|
|
} finally {
|
|
|
|
const count = NockMock.instanceCount - 1;
|
|
|
|
if (count < 0 && process.env.DEBUG) {
|
|
|
|
console.error(
|
|
|
|
`[debug] Warning: NockMock.instanceCount is negative (${count})`,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
NockMock.instanceCount = Math.max(0, count);
|
|
|
|
if (NockMock.instanceCount === 0) {
|
|
|
|
// Remove 'no match' handler, for tests using nock without this module
|
|
|
|
nock.emitter.removeAllListeners('no match');
|
|
|
|
nock.cleanAll();
|
|
|
|
nock.restore();
|
|
|
|
} else if (process.env.DEBUG) {
|
|
|
|
console.error(
|
|
|
|
`[debug] NockMock.done() instance count is ${NockMock.instanceCount}`,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected handleUnexpectedRequest(req: any) {
|
2021-07-20 14:57:00 +01:00
|
|
|
const { interceptorServerPort } =
|
|
|
|
require('./proxy-server') as typeof import('./proxy-server');
|
2020-01-15 19:41:47 +00:00
|
|
|
const o = req.options || {};
|
|
|
|
const u = o.uri || {};
|
2020-07-09 15:39:33 +01:00
|
|
|
const method = req.method;
|
|
|
|
const proto = req.protocol || req.proto || o.proto || u.protocol;
|
2020-06-15 23:53:04 +01:00
|
|
|
const host = req.host || req.headers?.host || o.host || u.host;
|
2020-07-09 15:39:33 +01:00
|
|
|
const path = req.path || o.path || u.path;
|
2020-06-15 23:53:04 +01:00
|
|
|
|
|
|
|
// Requests made by the local proxy/interceptor server are OK
|
|
|
|
if (host === `127.0.0.1:${interceptorServerPort}`) {
|
|
|
|
return;
|
|
|
|
}
|
2020-07-09 15:39:33 +01:00
|
|
|
console.error(
|
|
|
|
`NockMock: Unexpected HTTP request: ${method} ${proto}//${host}${path}`,
|
|
|
|
);
|
2020-01-15 19:41:47 +00:00
|
|
|
// Errors thrown here are not causing the tests to fail for some reason.
|
|
|
|
// Possibly due to CLI global error handlers? (error.js)
|
|
|
|
// (Also, nock should automatically throw an error, but also not happening)
|
|
|
|
// For now, the console.error is sufficient (will fail the test)
|
|
|
|
}
|
|
|
|
|
|
|
|
// For debugging tests
|
|
|
|
get unfulfilledCallCount(): number {
|
|
|
|
return this.scope.pendingMocks().length;
|
|
|
|
}
|
|
|
|
|
|
|
|
public debug() {
|
|
|
|
const scope = this.scope;
|
|
|
|
let mocks = scope.pendingMocks();
|
|
|
|
console.error(`pending mocks ${mocks.length}: ${mocks}`);
|
|
|
|
|
2020-06-15 23:53:07 +01:00
|
|
|
this.scope.on('request', function (_req, _interceptor, _body) {
|
2020-01-15 19:41:47 +00:00
|
|
|
console.log(`>> REQUEST:` + _req.path);
|
|
|
|
mocks = scope.pendingMocks();
|
|
|
|
console.error(`pending mocks ${mocks.length}: ${mocks}`);
|
|
|
|
});
|
|
|
|
|
2020-06-15 23:53:07 +01:00
|
|
|
this.scope.on('replied', function (_req) {
|
2020-01-15 19:41:47 +00:00
|
|
|
console.log(`<< REPLIED:` + _req.path);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|