mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-19 19:11:13 +00:00
fix(editor): Import form data with special characters from curl command correctly (#14898)
This commit is contained in:
@@ -1,6 +1,25 @@
|
||||
import { toHttpNodeParameters } from '@/composables/useImportCurlCommand';
|
||||
import { toHttpNodeParameters, useImportCurlCommand } from '@/composables/useImportCurlCommand';
|
||||
|
||||
const showToast = vi.fn();
|
||||
|
||||
vi.mock('@/composables/useToast', () => ({
|
||||
useToast: () => ({ showToast }),
|
||||
}));
|
||||
|
||||
describe('useImportCurlCommand', () => {
|
||||
describe('importCurlCommand', () => {
|
||||
test('Should parse cURL command with invalid protocol', () => {
|
||||
const curl = 'curl ftp://reqbin.com/echo -X POST';
|
||||
useImportCurlCommand().importCurlCommand(curl);
|
||||
expect(showToast).toHaveBeenCalledWith({
|
||||
duration: 0,
|
||||
message: 'The HTTP node doesn’t support FTP requests',
|
||||
title: 'Use the FTP node',
|
||||
type: 'error',
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
describe('toHttpNodeParameters', () => {
|
||||
test('Should parse form-urlencoded content type correctly', () => {
|
||||
const curl =
|
||||
@@ -292,5 +311,254 @@ describe('useImportCurlCommand', () => {
|
||||
expect(parameters.sendBody).toBe(false);
|
||||
expect(parameters.options.allowUnauthorizedCerts).toBe(true);
|
||||
});
|
||||
|
||||
test('Should parse form url encoded body data if parameter has base64 special characters like / or % or =', () => {
|
||||
const curl =
|
||||
'curl -X POST https://reqbin.com/echo \
|
||||
-H "Content-Type: application/x-www-form-urlencoded" \
|
||||
-d "redirect_uri=https://test.app.n8n.cloud/webhook-test/12345" \
|
||||
-d "client_secret=secret%3D%3D"';
|
||||
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('POST');
|
||||
expect(parameters.sendBody).toBe(true);
|
||||
expect(parameters.contentType).toEqual('form-urlencoded');
|
||||
expect(parameters.bodyParameters).toEqual({
|
||||
parameters: [
|
||||
{
|
||||
name: 'redirect_uri',
|
||||
value: 'https://test.app.n8n.cloud/webhook-test/12345',
|
||||
},
|
||||
{
|
||||
name: 'client_secret',
|
||||
value: 'secret==',
|
||||
},
|
||||
],
|
||||
});
|
||||
});
|
||||
|
||||
test('Should parse cURL command with no headers, body, or query parameters', () => {
|
||||
const curl = 'curl https://reqbin.com/echo';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('GET');
|
||||
expect(parameters.sendBody).toBe(false);
|
||||
expect(parameters.contentType).toBeUndefined();
|
||||
expect(parameters.sendHeaders).toBe(false);
|
||||
expect(parameters.sendQuery).toBe(false);
|
||||
});
|
||||
|
||||
test('Should parse cURL command with custom HTTP method', () => {
|
||||
const curl = 'curl -X DELETE https://reqbin.com/echo';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('DELETE');
|
||||
expect(parameters.sendBody).toBe(false);
|
||||
expect(parameters.contentType).toBeUndefined();
|
||||
expect(parameters.sendHeaders).toBe(false);
|
||||
expect(parameters.sendQuery).toBe(false);
|
||||
});
|
||||
|
||||
test('Should parse cURL command with multiple headers', () => {
|
||||
const curl =
|
||||
'curl https://reqbin.com/echo -H "Authorization: Bearer token" -H "Accept: application/json"';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.sendBody).toBe(false);
|
||||
expect(parameters.contentType).toBeUndefined();
|
||||
expect(parameters.sendHeaders).toBe(true);
|
||||
expect(parameters.headerParameters?.parameters[0].name).toBe('Authorization');
|
||||
expect(parameters.headerParameters?.parameters[0].value).toBe('Bearer token');
|
||||
expect(parameters.headerParameters?.parameters[1].name).toBe('Accept');
|
||||
expect(parameters.headerParameters?.parameters[1].value).toBe('application/json');
|
||||
});
|
||||
|
||||
test('Should parse cURL command with query parameters in URL', () => {
|
||||
const curl = 'curl https://reqbin.com/echo\\?param1\\=value1\\¶m2\\=value2';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.sendBody).toBe(false);
|
||||
expect(parameters.contentType).toBeUndefined();
|
||||
expect(parameters.sendHeaders).toBe(false);
|
||||
expect(parameters.sendQuery).toBe(true);
|
||||
expect(parameters.queryParameters).toEqual({
|
||||
parameters: [
|
||||
{
|
||||
name: 'param1',
|
||||
value: 'value1',
|
||||
},
|
||||
{
|
||||
name: 'param2',
|
||||
value: 'value2',
|
||||
},
|
||||
],
|
||||
});
|
||||
});
|
||||
|
||||
test('Should parse cURL command with both query parameters and -d flag', () => {
|
||||
const curl =
|
||||
'curl -G https://reqbin.com/echo\\?param1\\=value1 -d "param2=value2" -d "param3=value3"';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.sendBody).toBe(false);
|
||||
expect(parameters.contentType).toBeUndefined();
|
||||
expect(parameters.sendHeaders).toBe(false);
|
||||
expect(parameters.sendQuery).toBe(true);
|
||||
expect(parameters.queryParameters?.parameters[0].name).toBe('param1');
|
||||
expect(parameters.queryParameters?.parameters[0].value).toBe('value1');
|
||||
expect(parameters.queryParameters?.parameters[1].name).toBe('param2');
|
||||
expect(parameters.queryParameters?.parameters[1].value).toBe('value2');
|
||||
expect(parameters.queryParameters?.parameters[2].name).toBe('param3');
|
||||
expect(parameters.queryParameters?.parameters[2].value).toBe('value3');
|
||||
});
|
||||
|
||||
test('Should parse cURL command with data, defaulting to form urlencoded content type', () => {
|
||||
const curl = 'curl -X POST https://reqbin.com/echo -d "key=value"';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('POST');
|
||||
expect(parameters.sendBody).toBe(true);
|
||||
expect(parameters.contentType).toEqual('form-urlencoded');
|
||||
expect(parameters.bodyParameters?.parameters[0].name).toBe('key');
|
||||
expect(parameters.bodyParameters?.parameters[0].value).toBe('value');
|
||||
});
|
||||
|
||||
test('Should parse cURL command with file upload and no content type', () => {
|
||||
const curl = 'curl -X POST https://reqbin.com/echo -F "file=@/path/to/file"';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('POST');
|
||||
expect(parameters.sendBody).toBe(true);
|
||||
expect(parameters.contentType).toBe('multipart-form-data');
|
||||
expect(parameters.bodyParameters?.parameters[0].parameterType).toBe('formBinaryData');
|
||||
expect(parameters.bodyParameters?.parameters[0].name).toBe('file');
|
||||
});
|
||||
|
||||
test('Should parse cURL command with empty data flag', () => {
|
||||
const curl = 'curl -X POST https://reqbin.com/echo -d ""';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('POST');
|
||||
expect(parameters.sendBody).toBe(true);
|
||||
expect(parameters.contentType).toEqual('form-urlencoded');
|
||||
expect(parameters.bodyParameters?.parameters).toEqual([]);
|
||||
});
|
||||
|
||||
test('Should parse cURL command with custom header and case-insensitive content-type', () => {
|
||||
const curl =
|
||||
'curl -X POST https://reqbin.com/echo -H "content-TYPE: application/json" -d \'{"key":"value"}\'';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('POST');
|
||||
expect(parameters.sendBody).toBe(true);
|
||||
expect(parameters.contentType).toBe('json');
|
||||
expect(parameters.bodyParameters?.parameters[0].name).toBe('key');
|
||||
expect(parameters.bodyParameters?.parameters[0].value).toBe('value');
|
||||
});
|
||||
|
||||
test('Should parse cURL command with multiple query parameters in URL', () => {
|
||||
const curl =
|
||||
'curl https://reqbin.com/echo\\?param1\\=value1\\¶m2\\=value2\\¶m3\\=value3';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.sendBody).toBe(false);
|
||||
expect(parameters.contentType).toBeUndefined();
|
||||
expect(parameters.sendHeaders).toBe(false);
|
||||
expect(parameters.sendQuery).toBe(true);
|
||||
expect(parameters.queryParameters?.parameters).toEqual([
|
||||
{ name: 'param1', value: 'value1' },
|
||||
{ name: 'param2', value: 'value2' },
|
||||
{ name: 'param3', value: 'value3' },
|
||||
]);
|
||||
});
|
||||
|
||||
test('Should parse cURL command with custom header and no content type', () => {
|
||||
const curl = 'curl -X GET https://reqbin.com/echo -H "Authorization: Bearer token"';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('GET');
|
||||
expect(parameters.sendBody).toBe(false);
|
||||
expect(parameters.contentType).toBeUndefined();
|
||||
expect(parameters.sendHeaders).toBe(true);
|
||||
expect(parameters.headerParameters?.parameters[0].name).toBe('Authorization');
|
||||
expect(parameters.headerParameters?.parameters[0].value).toBe('Bearer token');
|
||||
});
|
||||
|
||||
test('Should parse cURL command with empty query parameters', () => {
|
||||
const curl = 'curl https://reqbin.com/echo\\?param1\\=\\¶m2=';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.sendBody).toBe(false);
|
||||
expect(parameters.contentType).toBeUndefined();
|
||||
expect(parameters.sendHeaders).toBe(false);
|
||||
expect(parameters.sendQuery).toBe(true);
|
||||
expect(parameters.queryParameters?.parameters).toEqual([
|
||||
{ name: 'param1', value: '' },
|
||||
{ name: 'param2', value: '' },
|
||||
]);
|
||||
});
|
||||
|
||||
test('Should parse cURL command with custom HTTP method and no body', () => {
|
||||
const curl = 'curl -X PUT https://reqbin.com/echo';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('PUT');
|
||||
expect(parameters.sendBody).toBe(false);
|
||||
expect(parameters.contentType).toBeUndefined();
|
||||
expect(parameters.sendHeaders).toBe(false);
|
||||
expect(parameters.sendQuery).toBe(false);
|
||||
});
|
||||
|
||||
test('Should parse cURL command with custom header and multiple data fields', () => {
|
||||
const curl =
|
||||
'curl -X POST https://reqbin.com/echo -H "Authorization: Bearer token" -d "key1=value1" -d "key2=value2"';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('POST');
|
||||
expect(parameters.sendBody).toBe(true);
|
||||
expect(parameters.contentType).toEqual('form-urlencoded');
|
||||
expect(parameters.bodyParameters?.parameters).toEqual([
|
||||
{ name: 'key1', value: 'value1' },
|
||||
{ name: 'key2', value: 'value2' },
|
||||
]);
|
||||
expect(parameters.sendHeaders).toBe(true);
|
||||
expect(parameters.headerParameters?.parameters[0].name).toBe('Authorization');
|
||||
expect(parameters.headerParameters?.parameters[0].value).toBe('Bearer token');
|
||||
});
|
||||
|
||||
test('Should parse cURL command with custom header and binary data', () => {
|
||||
const curl =
|
||||
'curl -X POST https://reqbin.com/echo -H "Content-Type: application/octet-stream" --data-binary "@/path/to/file"';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('POST');
|
||||
expect(parameters.sendBody).toBe(true);
|
||||
expect(parameters.contentType).toBe('raw');
|
||||
expect(parameters.rawContentType).toBe('application/octet-stream');
|
||||
expect(parameters.sendHeaders).toBe(false);
|
||||
expect(parameters.sendQuery).toBe(false);
|
||||
});
|
||||
|
||||
test('Should parse cURL command with multiple headers and case-insensitive keys', () => {
|
||||
const curl =
|
||||
'curl -X POST https://reqbin.com/echo -H "content-type: application/json" -H "ACCEPT: application/json" -d \'{"key":"value"}\'';
|
||||
const parameters = toHttpNodeParameters(curl);
|
||||
expect(parameters.url).toBe('https://reqbin.com/echo');
|
||||
expect(parameters.method).toBe('POST');
|
||||
expect(parameters.sendBody).toBe(true);
|
||||
expect(parameters.contentType).toBe('json');
|
||||
expect(parameters.bodyParameters?.parameters[0].name).toBe('key');
|
||||
expect(parameters.bodyParameters?.parameters[0].value).toBe('value');
|
||||
expect(parameters.sendHeaders).toBe(true);
|
||||
expect(parameters.headerParameters?.parameters[0].name).toBe('ACCEPT');
|
||||
expect(parameters.headerParameters?.parameters[0].value).toBe('application/json');
|
||||
});
|
||||
|
||||
test('Should parse cURL command with no URL', () => {
|
||||
const curl = 'curl -X POST -d "key=value"';
|
||||
expect(() => toHttpNodeParameters(curl)).toThrow('no URL specified!');
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user