mirror of
https://github.com/n8n-io/n8n.git
synced 2026-05-27 14:57:21 +02:00
362 lines
9.7 KiB
TypeScript
362 lines
9.7 KiB
TypeScript
import FormData from 'form-data';
|
|
import { Readable } from 'stream';
|
|
import type {
|
|
ICredentialDataDecryptedObject,
|
|
INodeExecutionData,
|
|
IRequestOptions,
|
|
} from 'n8n-workflow';
|
|
|
|
import {
|
|
REDACTED,
|
|
prepareRequestBody,
|
|
sanitizeUiMessage,
|
|
setAgentOptions,
|
|
replaceNullValues,
|
|
getSecrets,
|
|
} from '../../GenericFunctions';
|
|
import type { BodyParameter, BodyParametersReducer } from '../../GenericFunctions';
|
|
|
|
describe('HTTP Node Utils', () => {
|
|
describe('prepareRequestBody', () => {
|
|
it('should call default reducer', async () => {
|
|
const bodyParameters: BodyParameter[] = [
|
|
{
|
|
name: 'foo.bar',
|
|
value: 'baz',
|
|
},
|
|
];
|
|
const defaultReducer: BodyParametersReducer = jest.fn();
|
|
|
|
await prepareRequestBody(bodyParameters, 'json', 3, defaultReducer);
|
|
|
|
expect(defaultReducer).toBeCalledTimes(1);
|
|
expect(defaultReducer).toBeCalledWith({}, { name: 'foo.bar', value: 'baz' });
|
|
});
|
|
|
|
it('should create FormData with knownLength for stream binary data', async () => {
|
|
const streamContent = Buffer.from('test-binary-content');
|
|
const bodyParameters: BodyParameter[] = [
|
|
{
|
|
name: 'File',
|
|
value: '',
|
|
parameterType: 'formBinaryData',
|
|
},
|
|
{
|
|
name: 'Folder',
|
|
value: '/uploads',
|
|
},
|
|
];
|
|
|
|
const mockReducer: BodyParametersReducer = jest.fn().mockResolvedValue({
|
|
File: {
|
|
value: Readable.from(streamContent),
|
|
options: {
|
|
filename: 'test.pdf',
|
|
contentType: 'application/pdf',
|
|
knownLength: streamContent.length,
|
|
},
|
|
},
|
|
});
|
|
|
|
const result = await prepareRequestBody(
|
|
bodyParameters,
|
|
'multipart-form-data',
|
|
4.2,
|
|
mockReducer,
|
|
);
|
|
|
|
expect(result).toBeInstanceOf(FormData);
|
|
const formData = result as FormData;
|
|
|
|
// Verify FormData can calculate its total length (fails without knownLength)
|
|
const length = await new Promise<number>((resolve, reject) => {
|
|
formData.getLength((err, len) => (err ? reject(err) : resolve(len)));
|
|
});
|
|
expect(length).toBeGreaterThan(0);
|
|
});
|
|
|
|
it('should call process dot notations', async () => {
|
|
const bodyParameters: BodyParameter[] = [
|
|
{
|
|
name: 'foo.bar.spam',
|
|
value: 'baz',
|
|
},
|
|
];
|
|
const defaultReducer: BodyParametersReducer = jest.fn();
|
|
|
|
const result = await prepareRequestBody(bodyParameters, 'json', 4, defaultReducer);
|
|
|
|
expect(defaultReducer).toBeCalledTimes(0);
|
|
expect(result).toBeDefined();
|
|
expect(result).toEqual({ foo: { bar: { spam: 'baz' } } });
|
|
});
|
|
});
|
|
|
|
describe('setAgentOptions', () => {
|
|
it("should not have agentOptions as it's undefined", async () => {
|
|
const requestOptions: IRequestOptions = {
|
|
method: 'GET',
|
|
uri: 'https://example.com',
|
|
};
|
|
|
|
const sslCertificates = undefined;
|
|
|
|
setAgentOptions(requestOptions, sslCertificates);
|
|
|
|
expect(requestOptions).toEqual({
|
|
method: 'GET',
|
|
uri: 'https://example.com',
|
|
});
|
|
});
|
|
|
|
it('should have agentOptions set', async () => {
|
|
const requestOptions: IRequestOptions = {
|
|
method: 'GET',
|
|
uri: 'https://example.com',
|
|
};
|
|
|
|
const sslCertificates = {
|
|
ca: 'mock-ca',
|
|
};
|
|
|
|
setAgentOptions(requestOptions, sslCertificates);
|
|
|
|
expect(requestOptions).toStrictEqual({
|
|
method: 'GET',
|
|
uri: 'https://example.com',
|
|
agentOptions: {
|
|
ca: 'mock-ca',
|
|
},
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('sanitizeUiMessage', () => {
|
|
it('should remove large Buffers', async () => {
|
|
const requestOptions: IRequestOptions = {
|
|
method: 'POST',
|
|
uri: 'https://example.com',
|
|
body: Buffer.alloc(900000),
|
|
};
|
|
|
|
expect(sanitizeUiMessage(requestOptions, {}).body).toEqual(
|
|
'Binary data got replaced with this text. Original was a Buffer with a size of 900000 bytes.',
|
|
);
|
|
});
|
|
|
|
it('should remove keys that contain sensitive data and do not modify requestOptions', async () => {
|
|
const requestOptions: IRequestOptions = {
|
|
method: 'POST',
|
|
uri: 'https://example.com',
|
|
body: { sessionToken: 'secret', other: 'foo' },
|
|
headers: { authorization: 'secret', other: 'foo' },
|
|
auth: { user: 'user', password: 'secret' },
|
|
};
|
|
|
|
expect(
|
|
sanitizeUiMessage(requestOptions, {
|
|
headers: ['authorization'],
|
|
body: ['sessionToken'],
|
|
auth: ['password'],
|
|
}),
|
|
).toEqual({
|
|
body: { sessionToken: REDACTED, other: 'foo' },
|
|
headers: { other: 'foo', authorization: REDACTED },
|
|
auth: { user: 'user', password: REDACTED },
|
|
method: 'POST',
|
|
uri: 'https://example.com',
|
|
});
|
|
|
|
expect(requestOptions).toEqual({
|
|
method: 'POST',
|
|
uri: 'https://example.com',
|
|
body: { sessionToken: 'secret', other: 'foo' },
|
|
headers: { authorization: 'secret', other: 'foo' },
|
|
auth: { user: 'user', password: 'secret' },
|
|
});
|
|
});
|
|
|
|
it('should remove secrets', async () => {
|
|
const requestOptions: IRequestOptions = {
|
|
method: 'POST',
|
|
uri: 'https://example.com',
|
|
body: { nested: { secret: 'secretAccessToken' } },
|
|
headers: { authorization: 'secretAccessToken', other: 'foo' },
|
|
};
|
|
|
|
const sanitizedRequest = sanitizeUiMessage(requestOptions, {}, ['secretAccessToken']);
|
|
|
|
expect(sanitizedRequest).toEqual({
|
|
body: {
|
|
nested: {
|
|
secret: REDACTED,
|
|
},
|
|
},
|
|
headers: { authorization: REDACTED, other: 'foo' },
|
|
method: 'POST',
|
|
uri: 'https://example.com',
|
|
});
|
|
});
|
|
|
|
it('should redact secrets in auth fields not tracked by authDataKeys', async () => {
|
|
const secretValue = '{"username":"testuser","password":"fake-pass-123"}';
|
|
const requestOptions: IRequestOptions = {
|
|
method: 'GET',
|
|
uri: 'https://example.com',
|
|
auth: { user: secretValue, pass: 'regular-pass' },
|
|
};
|
|
|
|
const sanitizedRequest = sanitizeUiMessage(requestOptions, { auth: ['pass'] }, [
|
|
secretValue,
|
|
'regular-pass',
|
|
]);
|
|
|
|
expect(sanitizedRequest).toEqual({
|
|
method: 'GET',
|
|
uri: 'https://example.com',
|
|
auth: { user: REDACTED, pass: REDACTED },
|
|
});
|
|
});
|
|
|
|
it('should redact secrets appearing as object keys', async () => {
|
|
const secretJson = '{"username":"admin","password":"s3cret"}';
|
|
const requestOptions: IRequestOptions = {
|
|
method: 'GET',
|
|
uri: 'https://example.com',
|
|
qs: { [secretJson]: 'someValue' },
|
|
};
|
|
|
|
const sanitizedRequest = sanitizeUiMessage(requestOptions, {}, [secretJson]);
|
|
|
|
expect(sanitizedRequest).toEqual({
|
|
method: 'GET',
|
|
uri: 'https://example.com',
|
|
qs: { [REDACTED]: 'someValue' },
|
|
});
|
|
});
|
|
|
|
const headersToTest = [
|
|
'authorization',
|
|
'x-api-key',
|
|
'x-auth-token',
|
|
'cookie',
|
|
'proxy-authorization',
|
|
'sslclientcert',
|
|
];
|
|
|
|
headersToTest.forEach((header) => {
|
|
it(`should redact the ${header} header when the key is lowercase`, () => {
|
|
const requestOptions: IRequestOptions = {
|
|
method: 'POST',
|
|
uri: 'https://example.com',
|
|
body: { sessionToken: 'secret', other: 'foo' },
|
|
headers: { [header]: 'some-sensitive-token', other: 'foo' },
|
|
auth: { user: 'user', password: 'secret' },
|
|
};
|
|
|
|
const sanitizedRequest = sanitizeUiMessage(requestOptions, {});
|
|
|
|
expect(sanitizedRequest.headers).toEqual({ [header]: REDACTED, other: 'foo' });
|
|
});
|
|
|
|
it(`should redact the ${header} header when the key is uppercase`, () => {
|
|
const requestOptions: IRequestOptions = {
|
|
method: 'POST',
|
|
uri: 'https://example.com',
|
|
body: { sessionToken: 'secret', other: 'foo' },
|
|
headers: { [header.toUpperCase()]: 'some-sensitive-token', other: 'foo' },
|
|
auth: { user: 'user', password: 'secret' },
|
|
};
|
|
|
|
const sanitizedRequest = sanitizeUiMessage(requestOptions, {});
|
|
|
|
expect(sanitizedRequest.headers).toEqual({
|
|
[header.toUpperCase()]: REDACTED,
|
|
other: 'foo',
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should leave headers unchanged if Authorization header is not present', () => {
|
|
const requestOptions: IRequestOptions = {
|
|
method: 'POST',
|
|
uri: 'https://example.com',
|
|
body: { sessionToken: 'secret', other: 'foo' },
|
|
headers: { other: 'foo' },
|
|
auth: { user: 'user', password: 'secret' },
|
|
};
|
|
const sanitizedRequest = sanitizeUiMessage(requestOptions, {});
|
|
|
|
expect(sanitizedRequest.headers).toEqual({ other: 'foo' });
|
|
});
|
|
|
|
it('should handle case when headers are undefined', () => {
|
|
const requestOptions: IRequestOptions = {};
|
|
|
|
const sanitizedRequest = sanitizeUiMessage(requestOptions, {});
|
|
|
|
expect(sanitizedRequest.headers).toBeUndefined();
|
|
});
|
|
});
|
|
|
|
describe('replaceNullValues', () => {
|
|
it('should replace null json with an empty object', () => {
|
|
const item: INodeExecutionData = {
|
|
json: {},
|
|
};
|
|
const result = replaceNullValues(item);
|
|
expect(result.json).toEqual({});
|
|
});
|
|
|
|
it('should not modify json if it is already an object', () => {
|
|
const jsonObject = { key: 'value' };
|
|
const item: INodeExecutionData = { json: jsonObject };
|
|
const result = replaceNullValues(item);
|
|
expect(result.json).toBe(jsonObject);
|
|
});
|
|
});
|
|
|
|
describe('getSecrets', () => {
|
|
afterEach(() => {
|
|
jest.clearAllMocks();
|
|
});
|
|
|
|
it('should return all string credential values as secrets', () => {
|
|
const credentials: ICredentialDataDecryptedObject = {
|
|
apiKey: 'sensitive-api-key',
|
|
username: 'user123',
|
|
};
|
|
|
|
const secrets = getSecrets(credentials);
|
|
expect(secrets).toContain('sensitive-api-key');
|
|
expect(secrets).toContain('user123');
|
|
});
|
|
|
|
it('should not include non-string values', () => {
|
|
const credentials: ICredentialDataDecryptedObject = {
|
|
apiKey: 12345,
|
|
};
|
|
|
|
const secrets = getSecrets(credentials);
|
|
expect(secrets).toEqual([]);
|
|
});
|
|
|
|
it('should return an empty array if credentials are empty', () => {
|
|
const credentials: ICredentialDataDecryptedObject = {};
|
|
|
|
const secrets = getSecrets(credentials);
|
|
expect(secrets).toEqual([]);
|
|
});
|
|
|
|
it('should not include empty strings or non-string values', () => {
|
|
const credentials: ICredentialDataDecryptedObject = {
|
|
apiKey: {},
|
|
emptyField: '',
|
|
};
|
|
|
|
const secrets = getSecrets(credentials);
|
|
expect(secrets).toEqual([]);
|
|
});
|
|
});
|
|
});
|