1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-04-19 01:17:18 +02:00
unleash.unleash/src/lib/addons/new-relic.test.ts
2024-07-23 10:07:31 +01:00

213 lines
6.5 KiB
TypeScript

import {
FEATURE_ARCHIVED,
FEATURE_CREATED,
FEATURE_ENVIRONMENT_DISABLED,
type IFlagResolver,
type IAddonConfig,
type IEvent,
serializeDates,
} from '../types';
import type { Logger } from '../logger';
import NewRelicAddon, { type INewRelicParameters } from './new-relic';
import noLogger from '../../test/fixtures/no-logger';
import { gunzip } from 'node:zlib';
import { promisify } from 'util';
import type { IntegrationEventsService } from '../services';
const asyncGunzip = promisify(gunzip);
let fetchRetryCalls: any[] = [];
const registerEventMock = jest.fn();
const INTEGRATION_ID = 1337;
const ARGS: IAddonConfig = {
getLogger: noLogger,
unleashUrl: 'http://some-url.com',
integrationEventsService: {} as IntegrationEventsService,
flagResolver: {} as IFlagResolver,
};
jest.mock(
'./addon',
() =>
class Addon {
logger: Logger;
constructor(definition, { getLogger }) {
this.logger = getLogger('addon/test');
fetchRetryCalls = [];
}
async fetchRetry(url, options, retries, backoff) {
fetchRetryCalls.push({
url,
options,
retries,
backoff,
});
return Promise.resolve({ ok: true, status: 200 });
}
async registerEvent(event) {
return registerEventMock(event);
}
},
);
const defaultParameters = {
url: 'fakeUrl',
licenseKey: 'fakeLicenseKey',
} as INewRelicParameters;
const defaultEvent = {
id: 1,
createdAt: new Date(),
type: FEATURE_CREATED,
createdBy: 'some@user.com',
createdByUserId: -1337,
featureName: 'some-toggle',
data: {
name: 'some-toggle',
enabled: false,
strategies: [{ name: 'default' }],
},
} as IEvent;
const makeAddHandleEvent = (event: IEvent, parameters: INewRelicParameters) => {
const addon = new NewRelicAddon(ARGS);
return () => addon.handleEvent(event, parameters, INTEGRATION_ID);
};
describe('New Relic integration', () => {
beforeEach(() => {
registerEventMock.mockClear();
});
test.each([
{
partialEvent: { type: FEATURE_CREATED },
test: '$type toggle',
},
{
partialEvent: {
type: FEATURE_ARCHIVED,
data: {
name: 'some-toggle',
},
},
test: 'FEATURE_ARCHIVED toggle with project info',
},
{
partialEvent: {
type: FEATURE_ARCHIVED,
project: 'some-project',
data: {
name: 'some-toggle',
},
},
test: 'FEATURE_ARCHIVED with project info',
},
{
partialEvent: {
type: FEATURE_ENVIRONMENT_DISABLED,
environment: 'development',
},
test: 'toggled environment',
},
{
partialEvent: {
type: FEATURE_ENVIRONMENT_DISABLED,
environment: 'development',
},
partialParameters: {
customHeaders: `{ "MY_CUSTOM_HEADER": "MY_CUSTOM_VALUE" }`,
},
test: 'customHeaders in headers when calling service',
},
{
partialEvent: {
type: FEATURE_ENVIRONMENT_DISABLED,
environment: 'development',
},
partialParameters: {
bodyTemplate:
'{\n "eventType": "{{event.type}}",\n "createdBy": "{{event.createdBy}}"\n}',
},
test: 'custom body template',
},
] as Array<{
partialEvent: Partial<IEvent>;
partialParameters?: Partial<INewRelicParameters>;
test: String;
}>)(
'Should call New Relic Event API for $test',
async ({ partialEvent, partialParameters }) => {
const event = {
...defaultEvent,
...partialEvent,
};
const parameters = {
...defaultParameters,
...partialParameters,
};
const handleEvent = makeAddHandleEvent(event, parameters);
await handleEvent();
expect(fetchRetryCalls.length).toBe(1);
const { url, options } = fetchRetryCalls[0];
const jsonBody = JSON.parse(
(await asyncGunzip(options.body)).toString(),
);
expect(url).toBe(parameters.url);
expect(options.method).toBe('POST');
expect(options.headers['Api-Key']).toBe(parameters.licenseKey);
expect(options.headers['Content-Type']).toBe('application/json');
expect(options.headers['Content-Encoding']).toBe('gzip');
expect(options.headers).toMatchSnapshot();
expect(jsonBody.eventType).toBe('UnleashServiceEvent');
expect(jsonBody.unleashEventType).toBe(event.type);
expect(jsonBody.featureName).toBe(event.data.name);
expect(jsonBody.environment).toBe(event.environment);
expect(jsonBody.createdBy).toBe(event.createdBy);
expect(jsonBody.createdByUserId).toBe(event.createdByUserId);
expect(jsonBody.createdAt).toBe(event.createdAt.getTime());
},
);
test('Should call registerEvent', async () => {
const handleEvent = makeAddHandleEvent(defaultEvent, defaultParameters);
await handleEvent();
expect(registerEventMock).toHaveBeenCalledTimes(1);
expect(registerEventMock).toHaveBeenCalledWith({
integrationId: INTEGRATION_ID,
state: 'success',
stateDetails:
'New Relic Events API request was successful with status code: 200.',
event: serializeDates(defaultEvent),
details: {
url: defaultParameters.url,
body: {
eventType: 'UnleashServiceEvent',
unleashEventType: defaultEvent.type,
featureName: defaultEvent.featureName,
environment: defaultEvent.environment,
createdBy: defaultEvent.createdBy,
createdByUserId: defaultEvent.createdByUserId,
createdAt: defaultEvent.createdAt.getTime(),
...defaultEvent.data,
},
},
});
});
});