mirror of
https://github.com/Unleash/unleash.git
synced 2025-04-19 01:17:18 +02:00
https://linear.app/unleash/issue/2-2462/register-integration-events-new-relic Registers integration events in the **New Relic** integration. Similar to: - #7635 - #7634 - #7631 - #7626 - #7621
213 lines
6.5 KiB
TypeScript
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,
|
|
},
|
|
},
|
|
});
|
|
});
|
|
});
|