2023-09-20 05:40:06 -07:00
import { mkdtempSync , readFileSync } from 'fs' ;
2024-11-28 05:31:54 -08:00
import { IncomingMessage } from 'http' ;
2024-09-30 06:38:56 -07:00
import type { Agent } from 'https' ;
2023-04-12 05:58:05 -07:00
import { mock } from 'jest-mock-extended' ;
import type {
IBinaryData ,
2024-02-22 08:56:48 -08:00
IHttpRequestMethods ,
2024-03-14 04:17:20 -07:00
IHttpRequestOptions ,
2023-04-12 05:58:05 -07:00
INode ,
2024-03-14 04:17:20 -07:00
IRequestOptions ,
2023-04-12 05:58:05 -07:00
ITaskDataConnections ,
IWorkflowExecuteAdditionalData ,
Workflow ,
WorkflowHooks ,
} from 'n8n-workflow' ;
2023-09-20 05:40:06 -07:00
import nock from 'nock' ;
import { tmpdir } from 'os' ;
import { join } from 'path' ;
2024-11-28 05:31:54 -08:00
import { Readable } from 'stream' ;
2024-09-30 06:38:56 -07:00
import type { SecureContextOptions } from 'tls' ;
2023-09-22 08:22:12 -07:00
import Container from 'typedi' ;
2024-09-30 06:38:56 -07:00
import { BinaryDataService } from '@/BinaryData/BinaryData.service' ;
2024-08-09 07:40:50 -07:00
import { InstanceSettings } from '@/InstanceSettings' ;
2024-09-30 06:38:56 -07:00
import {
2024-11-28 05:31:54 -08:00
binaryToString ,
2024-09-30 06:38:56 -07:00
copyInputItems ,
getBinaryDataBuffer ,
isFilePathBlocked ,
2024-11-28 06:53:18 -08:00
parseContentDisposition ,
parseContentType ,
2024-09-30 06:38:56 -07:00
parseIncomingMessage ,
parseRequestObject ,
proxyRequestToAxios ,
removeEmptyBody ,
setBinaryDataBuffer ,
} from '@/NodeExecuteFunctions' ;
2022-09-11 07:42:09 -07:00
const temporaryDir = mkdtempSync ( join ( tmpdir ( ) , 'n8n' ) ) ;
describe ( 'NodeExecuteFunctions' , ( ) = > {
2023-04-12 05:58:05 -07:00
describe ( 'test binary data helper methods' , ( ) = > {
2023-05-02 01:37:19 -07:00
test ( "test getBinaryDataBuffer(...) & setBinaryDataBuffer(...) methods in 'default' mode" , async ( ) = > {
2022-09-11 07:42:09 -07:00
// Setup a 'default' binary data manager instance
2023-09-22 08:22:12 -07:00
Container . set ( BinaryDataService , new BinaryDataService ( ) ) ;
await Container . get ( BinaryDataService ) . init ( {
2022-09-11 07:42:09 -07:00
mode : 'default' ,
2023-09-22 08:22:12 -07:00
availableModes : [ 'default' ] ,
2022-09-11 07:42:09 -07:00
localStoragePath : temporaryDir ,
} ) ;
// Set our binary data buffer
2023-05-02 01:37:19 -07:00
const inputData : Buffer = Buffer . from ( 'This is some binary data' , 'utf8' ) ;
const setBinaryDataBufferResponse : IBinaryData = await setBinaryDataBuffer (
2022-09-11 07:42:09 -07:00
{
mimeType : 'txt' ,
data : 'This should be overwritten by the actual payload in the response' ,
} ,
inputData ,
2023-09-27 00:42:35 -07:00
'workflowId' ,
2022-09-11 07:42:09 -07:00
'executionId' ,
) ;
// Expect our return object to contain the base64 encoding of the input data, as it should be stored in memory.
expect ( setBinaryDataBufferResponse . data ) . toEqual ( inputData . toString ( 'base64' ) ) ;
// Now, re-fetch our data.
// An ITaskDataConnections object is used to share data between nodes. The top level property, 'main', represents the successful output object from a previous node.
2023-05-02 01:37:19 -07:00
const taskDataConnectionsInput : ITaskDataConnections = {
2022-09-11 07:42:09 -07:00
main : [ ] ,
} ;
// We add an input set, with one item at index 0, to this input. It contains an empty json payload and our binary data.
taskDataConnectionsInput . main . push ( [
{
json : { } ,
binary : {
data : setBinaryDataBufferResponse ,
} ,
} ,
] ) ;
// Now, lets fetch our data! The item will be item index 0.
2023-05-02 01:37:19 -07:00
const getBinaryDataBufferResponse : Buffer = await getBinaryDataBuffer (
2022-09-11 07:42:09 -07:00
taskDataConnectionsInput ,
0 ,
'data' ,
0 ,
) ;
expect ( getBinaryDataBufferResponse ) . toEqual ( inputData ) ;
} ) ;
2023-05-02 01:37:19 -07:00
test ( "test getBinaryDataBuffer(...) & setBinaryDataBuffer(...) methods in 'filesystem' mode" , async ( ) = > {
2023-09-22 08:22:12 -07:00
Container . set ( BinaryDataService , new BinaryDataService ( ) ) ;
2022-09-11 07:42:09 -07:00
// Setup a 'filesystem' binary data manager instance
2023-09-22 08:22:12 -07:00
await Container . get ( BinaryDataService ) . init ( {
2022-09-11 07:42:09 -07:00
mode : 'filesystem' ,
2023-09-22 08:22:12 -07:00
availableModes : [ 'filesystem' ] ,
2022-09-11 07:42:09 -07:00
localStoragePath : temporaryDir ,
} ) ;
// Set our binary data buffer
2023-05-02 01:37:19 -07:00
const inputData : Buffer = Buffer . from ( 'This is some binary data' , 'utf8' ) ;
const setBinaryDataBufferResponse : IBinaryData = await setBinaryDataBuffer (
2022-09-11 07:42:09 -07:00
{
mimeType : 'txt' ,
data : 'This should be overwritten with the name of the configured data manager' ,
} ,
inputData ,
2023-09-27 00:42:35 -07:00
'workflowId' ,
2022-09-11 07:42:09 -07:00
'executionId' ,
) ;
// Expect our return object to contain the name of the configured data manager.
2023-10-10 01:06:06 -07:00
expect ( setBinaryDataBufferResponse . data ) . toEqual ( 'filesystem-v2' ) ;
2022-09-11 07:42:09 -07:00
// Ensure that the input data was successfully persisted to disk.
expect (
readFileSync (
2023-10-10 01:06:06 -07:00
` ${ temporaryDir } / ${ setBinaryDataBufferResponse . id ? . replace ( 'filesystem-v2:' , '' ) } ` ,
2022-09-11 07:42:09 -07:00
) ,
) . toEqual ( inputData ) ;
// Now, re-fetch our data.
// An ITaskDataConnections object is used to share data between nodes. The top level property, 'main', represents the successful output object from a previous node.
2023-05-02 01:37:19 -07:00
const taskDataConnectionsInput : ITaskDataConnections = {
2022-09-11 07:42:09 -07:00
main : [ ] ,
} ;
// We add an input set, with one item at index 0, to this input. It contains an empty json payload and our binary data.
taskDataConnectionsInput . main . push ( [
{
json : { } ,
binary : {
data : setBinaryDataBufferResponse ,
} ,
} ,
] ) ;
// Now, lets fetch our data! The item will be item index 0.
2023-05-02 01:37:19 -07:00
const getBinaryDataBufferResponse : Buffer = await getBinaryDataBuffer (
2022-09-11 07:42:09 -07:00
taskDataConnectionsInput ,
0 ,
'data' ,
0 ,
) ;
expect ( getBinaryDataBufferResponse ) . toEqual ( inputData ) ;
} ) ;
} ) ;
2023-04-12 05:58:05 -07:00
2024-11-28 06:53:18 -08:00
describe ( 'parseContentType' , ( ) = > {
const testCases = [
{
input : 'text/plain' ,
expected : {
type : 'text/plain' ,
parameters : {
charset : 'utf-8' ,
} ,
} ,
description : 'should parse basic content type' ,
} ,
{
input : 'TEXT/PLAIN' ,
expected : {
type : 'text/plain' ,
parameters : {
charset : 'utf-8' ,
} ,
} ,
description : 'should convert type to lowercase' ,
} ,
{
input : 'text/html; charset=iso-8859-1' ,
expected : {
type : 'text/html' ,
parameters : {
charset : 'iso-8859-1' ,
} ,
} ,
description : 'should parse content type with charset' ,
} ,
{
input : 'application/json; charset=utf-8; boundary=---123' ,
expected : {
type : 'application/json' ,
parameters : {
charset : 'utf-8' ,
boundary : '---123' ,
} ,
} ,
description : 'should parse content type with multiple parameters' ,
} ,
{
input : 'text/plain; charset="utf-8"; filename="test.txt"' ,
expected : {
type : 'text/plain' ,
parameters : {
charset : 'utf-8' ,
filename : 'test.txt' ,
} ,
} ,
description : 'should handle quoted parameter values' ,
} ,
{
input : 'text/plain; filename=%22test%20file.txt%22' ,
expected : {
type : 'text/plain' ,
parameters : {
charset : 'utf-8' ,
filename : 'test file.txt' ,
} ,
} ,
description : 'should handle encoded parameter values' ,
} ,
{
input : undefined ,
expected : null ,
description : 'should return null for undefined input' ,
} ,
{
input : '' ,
expected : null ,
description : 'should return null for empty string' ,
} ,
] ;
test . each ( testCases ) ( '$description' , ( { input , expected } ) = > {
expect ( parseContentType ( input ) ) . toEqual ( expected ) ;
} ) ;
} ) ;
describe ( 'parseContentDisposition' , ( ) = > {
const testCases = [
{
input : 'attachment; filename="file.txt"' ,
expected : { type : 'attachment' , filename : 'file.txt' } ,
description : 'should parse basic content disposition' ,
} ,
{
input : 'attachment; filename=file.txt' ,
expected : { type : 'attachment' , filename : 'file.txt' } ,
description : 'should parse filename without quotes' ,
} ,
{
input : 'inline; filename="image.jpg"' ,
expected : { type : 'inline' , filename : 'image.jpg' } ,
description : 'should parse inline disposition' ,
} ,
{
input : 'attachment; filename="my file.pdf"' ,
expected : { type : 'attachment' , filename : 'my file.pdf' } ,
description : 'should parse filename with spaces' ,
} ,
{
input : "attachment; filename*=UTF-8''my%20file.txt" ,
expected : { type : 'attachment' , filename : 'my file.txt' } ,
description : 'should parse filename* parameter (RFC 5987)' ,
} ,
{
input : 'filename="test.txt"' ,
expected : { type : 'attachment' , filename : 'test.txt' } ,
description : 'should handle invalid syntax but with filename' ,
} ,
{
input : 'filename=test.txt' ,
expected : { type : 'attachment' , filename : 'test.txt' } ,
description : 'should handle invalid syntax with only filename parameter' ,
} ,
{
input : undefined ,
expected : null ,
description : 'should return null for undefined input' ,
} ,
{
input : '' ,
expected : null ,
description : 'should return null for empty string' ,
} ,
{
input : 'attachment; filename="%F0%9F%98%80.txt"' ,
expected : { type : 'attachment' , filename : '😀.txt' } ,
description : 'should handle encoded filenames' ,
} ,
{
input : 'attachment; size=123; filename="test.txt"; creation-date="Thu, 1 Jan 2020"' ,
expected : { type : 'attachment' , filename : 'test.txt' } ,
description : 'should handle multiple parameters' ,
} ,
] ;
test . each ( testCases ) ( '$description' , ( { input , expected } ) = > {
expect ( parseContentDisposition ( input ) ) . toEqual ( expected ) ;
} ) ;
} ) ;
2023-09-20 05:40:06 -07:00
describe ( 'parseIncomingMessage' , ( ) = > {
it ( 'parses valid content-type header' , ( ) = > {
const message = mock < IncomingMessage > ( {
headers : { 'content-type' : 'application/json' , 'content-disposition' : undefined } ,
} ) ;
parseIncomingMessage ( message ) ;
expect ( message . contentType ) . toEqual ( 'application/json' ) ;
} ) ;
it ( 'parses valid content-type header with parameters' , ( ) = > {
const message = mock < IncomingMessage > ( {
headers : {
'content-type' : 'application/json; charset=utf-8' ,
'content-disposition' : undefined ,
} ,
} ) ;
parseIncomingMessage ( message ) ;
expect ( message . contentType ) . toEqual ( 'application/json' ) ;
2024-11-28 06:53:18 -08:00
expect ( message . encoding ) . toEqual ( 'utf-8' ) ;
} ) ;
it ( 'parses valid content-type header with encoding wrapped in quotes' , ( ) = > {
const message = mock < IncomingMessage > ( {
headers : {
'content-type' : 'application/json; charset="utf-8"' ,
'content-disposition' : undefined ,
} ,
} ) ;
parseIncomingMessage ( message ) ;
expect ( message . contentType ) . toEqual ( 'application/json' ) ;
expect ( message . encoding ) . toEqual ( 'utf-8' ) ;
2023-09-20 05:40:06 -07:00
} ) ;
it ( 'parses valid content-disposition header with filename*' , ( ) = > {
const message = mock < IncomingMessage > ( {
headers : {
'content-type' : undefined ,
'content-disposition' :
'attachment; filename="screenshot%20(1).png"; filename*=UTF-8\'\'screenshot%20(1).png' ,
} ,
} ) ;
parseIncomingMessage ( message ) ;
expect ( message . contentDisposition ) . toEqual ( {
filename : 'screenshot (1).png' ,
type : 'attachment' ,
} ) ;
} ) ;
2023-09-21 05:54:10 -07:00
it ( 'parses valid content-disposition header with filename* (quoted)' , ( ) = > {
const message = mock < IncomingMessage > ( {
headers : {
'content-type' : undefined ,
'content-disposition' : ' attachment;filename*="utf-8\' \'test-unsplash.jpg"' ,
} ,
} ) ;
parseIncomingMessage ( message ) ;
expect ( message . contentDisposition ) . toEqual ( {
filename : 'test-unsplash.jpg' ,
type : 'attachment' ,
} ) ;
} ) ;
2023-09-20 05:40:06 -07:00
it ( 'parses valid content-disposition header with filename and trailing ";"' , ( ) = > {
const message = mock < IncomingMessage > ( {
headers : {
'content-type' : undefined ,
'content-disposition' : 'inline; filename="screenshot%20(1).png";' ,
} ,
} ) ;
parseIncomingMessage ( message ) ;
expect ( message . contentDisposition ) . toEqual ( {
filename : 'screenshot (1).png' ,
type : 'inline' ,
} ) ;
} ) ;
it ( 'parses non standard content-disposition with missing type' , ( ) = > {
const message = mock < IncomingMessage > ( {
headers : {
'content-type' : undefined ,
'content-disposition' : 'filename="screenshot%20(1).png";' ,
} ,
} ) ;
parseIncomingMessage ( message ) ;
expect ( message . contentDisposition ) . toEqual ( {
filename : 'screenshot (1).png' ,
type : 'attachment' ,
} ) ;
} ) ;
} ) ;
2023-04-12 05:58:05 -07:00
describe ( 'proxyRequestToAxios' , ( ) = > {
const baseUrl = 'http://example.de' ;
const workflow = mock < Workflow > ( ) ;
const hooks = mock < WorkflowHooks > ( ) ;
const additionalData = mock < IWorkflowExecuteAdditionalData > ( { hooks } ) ;
const node = mock < INode > ( ) ;
beforeEach ( ( ) = > {
hooks . executeHookFunctions . mockClear ( ) ;
} ) ;
2024-04-03 09:00:27 -07:00
test ( 'should rethrow an error with `status` property' , async ( ) = > {
nock ( baseUrl ) . get ( '/test' ) . reply ( 400 ) ;
try {
await proxyRequestToAxios ( workflow , additionalData , node , ` ${ baseUrl } /test ` ) ;
} catch ( error ) {
expect ( error . status ) . toEqual ( 400 ) ;
}
} ) ;
2023-04-12 05:58:05 -07:00
test ( 'should not throw if the response status is 200' , async ( ) = > {
nock ( baseUrl ) . get ( '/test' ) . reply ( 200 ) ;
await proxyRequestToAxios ( workflow , additionalData , node , ` ${ baseUrl } /test ` ) ;
expect ( hooks . executeHookFunctions ) . toHaveBeenCalledWith ( 'nodeFetchedData' , [
workflow . id ,
node ,
] ) ;
} ) ;
test ( 'should throw if the response status is 403' , async ( ) = > {
const headers = { 'content-type' : 'text/plain' } ;
nock ( baseUrl ) . get ( '/test' ) . reply ( 403 , 'Forbidden' , headers ) ;
try {
await proxyRequestToAxios ( workflow , additionalData , node , ` ${ baseUrl } /test ` ) ;
} catch ( error ) {
expect ( error . statusCode ) . toEqual ( 403 ) ;
expect ( error . request ) . toBeUndefined ( ) ;
expect ( error . response ) . toMatchObject ( { headers , status : 403 } ) ;
expect ( error . options ) . toMatchObject ( {
headers : { Accept : '*/*' } ,
method : 'get' ,
url : 'http://example.de/test' ,
} ) ;
expect ( error . config ) . toBeUndefined ( ) ;
expect ( error . message ) . toEqual ( '403 - "Forbidden"' ) ;
}
expect ( hooks . executeHookFunctions ) . not . toHaveBeenCalled ( ) ;
} ) ;
test ( 'should not throw if the response status is 404, but `simple` option is set to `false`' , async ( ) = > {
nock ( baseUrl ) . get ( '/test' ) . reply ( 404 , 'Not Found' ) ;
const response = await proxyRequestToAxios ( workflow , additionalData , node , {
url : ` ${ baseUrl } /test ` ,
simple : false ,
} ) ;
expect ( response ) . toEqual ( 'Not Found' ) ;
expect ( hooks . executeHookFunctions ) . toHaveBeenCalledWith ( 'nodeFetchedData' , [
workflow . id ,
node ,
] ) ;
} ) ;
test ( 'should return full response when `resolveWithFullResponse` is set to true' , async ( ) = > {
nock ( baseUrl ) . get ( '/test' ) . reply ( 404 , 'Not Found' ) ;
const response = await proxyRequestToAxios ( workflow , additionalData , node , {
url : ` ${ baseUrl } /test ` ,
resolveWithFullResponse : true ,
simple : false ,
} ) ;
expect ( response ) . toMatchObject ( {
body : 'Not Found' ,
headers : { } ,
statusCode : 404 ,
statusMessage : null ,
} ) ;
expect ( hooks . executeHookFunctions ) . toHaveBeenCalledWith ( 'nodeFetchedData' , [
workflow . id ,
node ,
] ) ;
} ) ;
2024-01-31 06:11:29 -08:00
describe ( 'redirects' , ( ) = > {
test ( 'should forward authorization header' , async ( ) = > {
nock ( baseUrl ) . get ( '/redirect' ) . reply ( 301 , '' , { Location : 'https://otherdomain.com/test' } ) ;
nock ( 'https://otherdomain.com' )
. get ( '/test' )
. reply ( 200 , function ( ) {
return this . req . headers ;
} ) ;
const response = await proxyRequestToAxios ( workflow , additionalData , node , {
url : ` ${ baseUrl } /redirect ` ,
auth : {
username : 'testuser' ,
password : 'testpassword' ,
} ,
headers : {
'X-Other-Header' : 'otherHeaderContent' ,
} ,
resolveWithFullResponse : true ,
} ) ;
expect ( response . statusCode ) . toBe ( 200 ) ;
const forwardedHeaders = JSON . parse ( response . body ) ;
expect ( forwardedHeaders . authorization ) . toBe ( 'Basic dGVzdHVzZXI6dGVzdHBhc3N3b3Jk' ) ;
expect ( forwardedHeaders [ 'x-other-header' ] ) . toBe ( 'otherHeaderContent' ) ;
} ) ;
test ( 'should follow redirects by default' , async ( ) = > {
nock ( baseUrl )
. get ( '/redirect' )
. reply ( 301 , '' , { Location : ` ${ baseUrl } /test ` } ) ;
nock ( baseUrl ) . get ( '/test' ) . reply ( 200 , 'Redirected' ) ;
const response = await proxyRequestToAxios ( workflow , additionalData , node , {
url : ` ${ baseUrl } /redirect ` ,
resolveWithFullResponse : true ,
} ) ;
expect ( response ) . toMatchObject ( {
body : 'Redirected' ,
headers : { } ,
statusCode : 200 ,
} ) ;
} ) ;
test ( 'should not follow redirects when configured' , async ( ) = > {
nock ( baseUrl )
. get ( '/redirect' )
. reply ( 301 , '' , { Location : ` ${ baseUrl } /test ` } ) ;
nock ( baseUrl ) . get ( '/test' ) . reply ( 200 , 'Redirected' ) ;
await expect (
proxyRequestToAxios ( workflow , additionalData , node , {
url : ` ${ baseUrl } /redirect ` ,
resolveWithFullResponse : true ,
followRedirect : false ,
} ) ,
) . rejects . toThrowError ( expect . objectContaining ( { statusCode : 301 } ) ) ;
} ) ;
} ) ;
2023-04-12 05:58:05 -07:00
} ) ;
2023-10-06 07:25:58 -07:00
2024-02-06 10:38:36 -08:00
describe ( 'parseRequestObject' , ( ) = > {
test ( 'should not use Host header for SNI' , async ( ) = > {
const axiosOptions = await parseRequestObject ( {
url : 'https://example.de/foo/bar' ,
headers : { Host : 'other.host.com' } ,
} ) ;
expect ( ( axiosOptions . httpsAgent as Agent ) . options . servername ) . toEqual ( 'example.de' ) ;
} ) ;
2024-02-22 08:56:48 -08:00
2024-04-24 07:28:02 -07:00
describe ( 'should set SSL certificates' , ( ) = > {
const agentOptions : SecureContextOptions = {
ca : '-----BEGIN CERTIFICATE-----\nTEST\n-----END CERTIFICATE-----' ,
} ;
const requestObject : IRequestOptions = {
method : 'GET' ,
uri : 'https://example.de' ,
agentOptions ,
} ;
test ( 'on regular requests' , async ( ) = > {
const axiosOptions = await parseRequestObject ( requestObject ) ;
expect ( ( axiosOptions . httpsAgent as Agent ) . options ) . toEqual ( {
servername : 'example.de' ,
. . . agentOptions ,
noDelay : true ,
path : null ,
} ) ;
} ) ;
test ( 'on redirected requests' , async ( ) = > {
const axiosOptions = await parseRequestObject ( requestObject ) ;
expect ( axiosOptions . beforeRedirect ) . toBeDefined ;
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const redirectOptions : Record < string , any > = { agents : { } , hostname : 'example.de' } ;
axiosOptions . beforeRedirect ! ( redirectOptions , mock ( ) ) ;
expect ( redirectOptions . agent ) . toEqual ( redirectOptions . agents . https ) ;
expect ( ( redirectOptions . agent as Agent ) . options ) . toEqual ( {
servername : 'example.de' ,
. . . agentOptions ,
noDelay : true ,
path : null ,
} ) ;
} ) ;
} ) ;
2024-02-22 08:56:48 -08:00
describe ( 'when followRedirect is true' , ( ) = > {
test . each ( [ 'GET' , 'HEAD' ] as IHttpRequestMethods [ ] ) (
'should set maxRedirects on %s ' ,
async ( method ) = > {
const axiosOptions = await parseRequestObject ( {
method ,
followRedirect : true ,
maxRedirects : 1234 ,
} ) ;
expect ( axiosOptions . maxRedirects ) . toEqual ( 1234 ) ;
} ,
) ;
test . each ( [ 'POST' , 'PUT' , 'PATCH' , 'DELETE' ] as IHttpRequestMethods [ ] ) (
'should not set maxRedirects on %s ' ,
async ( method ) = > {
const axiosOptions = await parseRequestObject ( {
method ,
followRedirect : true ,
maxRedirects : 1234 ,
} ) ;
expect ( axiosOptions . maxRedirects ) . toEqual ( 0 ) ;
} ,
) ;
} ) ;
describe ( 'when followAllRedirects is true' , ( ) = > {
test . each ( [ 'GET' , 'HEAD' , 'POST' , 'PUT' , 'PATCH' , 'DELETE' ] as IHttpRequestMethods [ ] ) (
'should set maxRedirects on %s ' ,
async ( method ) = > {
const axiosOptions = await parseRequestObject ( {
method ,
followAllRedirects : true ,
maxRedirects : 1234 ,
} ) ;
expect ( axiosOptions . maxRedirects ) . toEqual ( 1234 ) ;
} ,
) ;
} ) ;
2024-02-06 10:38:36 -08:00
} ) ;
2023-10-06 07:25:58 -07:00
describe ( 'copyInputItems' , ( ) = > {
it ( 'should pick only selected properties' , ( ) = > {
const output = copyInputItems (
[
{
json : {
a : 1 ,
b : true ,
c : { } ,
} ,
} ,
] ,
[ 'a' ] ,
) ;
expect ( output ) . toEqual ( [ { a : 1 } ] ) ;
} ) ;
it ( 'should convert undefined to null' , ( ) = > {
const output = copyInputItems (
[
{
json : {
a : undefined ,
} ,
} ,
] ,
[ 'a' ] ,
) ;
expect ( output ) . toEqual ( [ { a : null } ] ) ;
} ) ;
it ( 'should clone objects' , ( ) = > {
const input = {
a : { b : 5 } ,
} ;
const output = copyInputItems (
[
{
json : input ,
} ,
] ,
[ 'a' ] ,
) ;
expect ( output [ 0 ] . a ) . toEqual ( input . a ) ;
expect ( output [ 0 ] . a === input . a ) . toEqual ( false ) ;
} ) ;
} ) ;
2024-03-14 04:17:20 -07:00
describe ( 'removeEmptyBody' , ( ) = > {
test . each ( [ 'GET' , 'HEAD' , 'OPTIONS' ] as IHttpRequestMethods [ ] ) (
'Should remove empty body for %s' ,
async ( method ) = > {
const requestOptions = {
method ,
body : { } ,
} as IHttpRequestOptions | IRequestOptions ;
removeEmptyBody ( requestOptions ) ;
expect ( requestOptions . body ) . toEqual ( undefined ) ;
} ,
) ;
test . each ( [ 'GET' , 'HEAD' , 'OPTIONS' ] as IHttpRequestMethods [ ] ) (
'Should not remove non-empty body for %s' ,
async ( method ) = > {
const requestOptions = {
method ,
body : { test : true } ,
} as IHttpRequestOptions | IRequestOptions ;
removeEmptyBody ( requestOptions ) ;
expect ( requestOptions . body ) . toEqual ( { test : true } ) ;
} ,
) ;
test . each ( [ 'POST' , 'PUT' , 'PATCH' , 'DELETE' ] as IHttpRequestMethods [ ] ) (
'Should not remove empty body for %s' ,
async ( method ) = > {
const requestOptions = {
method ,
body : { } ,
} as IHttpRequestOptions | IRequestOptions ;
removeEmptyBody ( requestOptions ) ;
expect ( requestOptions . body ) . toEqual ( { } ) ;
} ,
) ;
} ) ;
2024-11-28 05:31:54 -08:00
describe ( 'binaryToString' , ( ) = > {
const ENCODING_SAMPLES = {
utf8 : {
text : 'Hello, 世界! τεστ мир ⚡️ é à ü ñ' ,
buffer : Buffer.from ( [
0x48 , 0x65 , 0x6c , 0x6c , 0x6f , 0x2c , 0x20 , 0xe4 , 0xb8 , 0x96 , 0xe7 , 0x95 , 0x8c , 0x21 , 0x20 ,
0xcf , 0x84 , 0xce , 0xb5 , 0xcf , 0x83 , 0xcf , 0x84 , 0x20 , 0xd0 , 0xbc , 0xd0 , 0xb8 , 0xd1 , 0x80 ,
0x20 , 0xe2 , 0x9a , 0xa1 , 0xef , 0xb8 , 0x8f , 0x20 , 0xc3 , 0xa9 , 0x20 , 0xc3 , 0xa0 , 0x20 , 0xc3 ,
0xbc , 0x20 , 0xc3 , 0xb1 ,
] ) ,
} ,
'iso-8859-15' : {
text : 'Café € personnalité' ,
buffer : Buffer.from ( [
0x43 , 0x61 , 0x66 , 0xe9 , 0x20 , 0xa4 , 0x20 , 0x70 , 0x65 , 0x72 , 0x73 , 0x6f , 0x6e , 0x6e , 0x61 ,
0x6c , 0x69 , 0x74 , 0xe9 ,
] ) ,
} ,
latin1 : {
text : 'señor année déjà' ,
buffer : Buffer.from ( [
0x73 , 0x65 , 0xf1 , 0x6f , 0x72 , 0x20 , 0x61 , 0x6e , 0x6e , 0xe9 , 0x65 , 0x20 , 0x64 , 0xe9 , 0x6a ,
0xe0 ,
] ) ,
} ,
ascii : {
text : 'Hello, World! 123' ,
buffer : Buffer.from ( [
0x48 , 0x65 , 0x6c , 0x6c , 0x6f , 0x2c , 0x20 , 0x57 , 0x6f , 0x72 , 0x6c , 0x64 , 0x21 , 0x20 , 0x31 ,
0x32 , 0x33 ,
] ) ,
} ,
'windows-1252' : {
text : '€ Smart "quotes" • bullet' ,
buffer : Buffer.from ( [
0x80 , 0x20 , 0x53 , 0x6d , 0x61 , 0x72 , 0x74 , 0x20 , 0x22 , 0x71 , 0x75 , 0x6f , 0x74 , 0x65 , 0x73 ,
0x22 , 0x20 , 0x95 , 0x20 , 0x62 , 0x75 , 0x6c , 0x6c , 0x65 , 0x74 ,
] ) ,
} ,
'shift-jis' : {
text : 'こんにちは世界' ,
buffer : Buffer.from ( [
0x82 , 0xb1 , 0x82 , 0xf1 , 0x82 , 0xc9 , 0x82 , 0xbf , 0x82 , 0xcd , 0x90 , 0xa2 , 0x8a , 0x45 ,
] ) ,
} ,
big5 : {
text : '哈囉世界' ,
buffer : Buffer.from ( [ 0xab , 0xa2 , 0xc5 , 0x6f , 0xa5 , 0x40 , 0xac , 0xc9 ] ) ,
} ,
'koi8-r' : {
text : 'Привет мир' ,
buffer : Buffer.from ( [ 0xf0 , 0xd2 , 0xc9 , 0xd7 , 0xc5 , 0xd4 , 0x20 , 0xcd , 0xc9 , 0xd2 ] ) ,
} ,
} ;
describe ( 'should handle Buffer' , ( ) = > {
for ( const [ encoding , { text , buffer } ] of Object . entries ( ENCODING_SAMPLES ) ) {
test ( ` with ${ encoding } ` , async ( ) = > {
const data = await binaryToString ( buffer , encoding ) ;
expect ( data ) . toBe ( text ) ;
} ) ;
}
} ) ;
describe ( 'should handle streams' , ( ) = > {
for ( const [ encoding , { text , buffer } ] of Object . entries ( ENCODING_SAMPLES ) ) {
test ( ` with ${ encoding } ` , async ( ) = > {
const stream = Readable . from ( buffer ) ;
const data = await binaryToString ( stream , encoding ) ;
expect ( data ) . toBe ( text ) ;
} ) ;
}
} ) ;
describe ( 'should handle IncomingMessage' , ( ) = > {
for ( const [ encoding , { text , buffer } ] of Object . entries ( ENCODING_SAMPLES ) ) {
test ( ` with ${ encoding } ` , async ( ) = > {
const response = Readable . from ( buffer ) as IncomingMessage ;
response . headers = { 'content-type' : ` application/json;charset= ${ encoding } ` } ;
// @ts-expect-error need this hack to fake `instanceof IncomingMessage` checks
response . __proto__ = IncomingMessage . prototype ;
const data = await binaryToString ( response ) ;
expect ( data ) . toBe ( text ) ;
} ) ;
}
} ) ;
} ) ;
2022-09-11 07:42:09 -07:00
} ) ;
2024-08-09 07:40:50 -07:00
describe ( 'isFilePathBlocked' , ( ) = > {
test ( 'should return true for static cache dir' , ( ) = > {
const filePath = Container . get ( InstanceSettings ) . staticCacheDir ;
expect ( isFilePathBlocked ( filePath ) ) . toBe ( true ) ;
} ) ;
} ) ;