2
0
mirror of https://github.com/tenrok/axios.git synced 2026-06-17 19:21:29 +03:00
Files
axios/tests/unit/helpers/formDataToJSON.test.js
T
2026-06-11 08:36:48 +02:00

165 lines
4.1 KiB
JavaScript

import { describe, it, expect } from 'vitest';
import formDataToJSON from '../../../lib/helpers/formDataToJSON.js';
import AxiosError from '../../../lib/core/AxiosError.js';
describe('formDataToJSON', () => {
it('should convert a FormData Object to JSON Object', () => {
const formData = new FormData();
formData.append('foo[bar][baz]', '123');
expect(formDataToJSON(formData)).toEqual({
foo: {
bar: {
baz: '123',
},
},
});
});
it('should convert repeatable values as an array', () => {
const formData = new FormData();
formData.append('foo', '1');
formData.append('foo', '2');
expect(formDataToJSON(formData)).toEqual({
foo: ['1', '2'],
});
});
it('should keep repeatable values flat for 3+ entries', () => {
const formData = new FormData();
formData.append('select3', '301');
formData.append('select3', '302');
formData.append('select3', '303');
expect(formDataToJSON(formData)).toEqual({
select3: ['301', '302', '303'],
});
});
it('should keep nested repeatable values flat for 3+ entries', () => {
const formData = new FormData();
formData.append('foo[bar]', '1');
formData.append('foo[bar]', '2');
formData.append('foo[bar]', '3');
expect(formDataToJSON(formData)).toEqual({
foo: {
bar: ['1', '2', '3'],
},
});
});
it('should convert props with empty brackets to arrays', () => {
const formData = new FormData();
formData.append('foo[]', '1');
formData.append('foo[]', '2');
expect(formDataToJSON(formData)).toEqual({
foo: ['1', '2'],
});
});
it('should supported indexed arrays', () => {
const formData = new FormData();
formData.append('foo[0]', '1');
formData.append('foo[1]', '2');
expect(formDataToJSON(formData)).toEqual({
foo: ['1', '2'],
});
});
it('should resist prototype pollution CVE', () => {
const formData = new FormData();
formData.append('foo[0]', '1');
formData.append('foo[1]', '2');
formData.append('__proto__.x', 'hack');
formData.append('constructor.prototype.y', 'value');
expect(formDataToJSON(formData)).toEqual({
foo: ['1', '2'],
constructor: {
prototype: {
y: 'value',
},
},
});
expect({}.x).toEqual(undefined);
expect({}.y).toEqual(undefined);
});
it('should not write through to inherited objects on Object.prototype', () => {
Object.defineProperty(Object.prototype, 'injected', {
value: { hijack: true },
configurable: true,
writable: true,
});
try {
const formData = new FormData();
formData.append('injected.hijack', 'STOLEN');
const result = formDataToJSON(formData);
expect(result.injected).toEqual({ hijack: 'STOLEN' });
expect(Object.prototype.injected.hijack).toBe(true);
} finally {
delete Object.prototype.injected;
}
});
it('should throw AxiosError when a field path exceeds the default depth limit', () => {
const formData = new FormData();
formData.append('foo' + '[bar]'.repeat(101), '123');
try {
formDataToJSON(formData);
throw new Error('Should have thrown');
} catch (err) {
expect(err).toBeInstanceOf(AxiosError);
expect(err.code).toBe(AxiosError.ERR_FORM_DATA_DEPTH_EXCEEDED);
expect(err).not.toBeInstanceOf(RangeError);
}
});
it('should throw AxiosError while tokenizing very deep field paths', () => {
const formData = new FormData();
formData.append('foo' + '[bar]'.repeat(10000), '123');
try {
formDataToJSON(formData);
throw new Error('Should have thrown');
} catch (err) {
expect(err).toBeInstanceOf(AxiosError);
expect(err.code).toBe(AxiosError.ERR_FORM_DATA_DEPTH_EXCEEDED);
expect(err).not.toBeInstanceOf(RangeError);
}
});
it('should convert a field path at the default depth limit', () => {
const formData = new FormData();
formData.append('foo' + '[bar]'.repeat(100), '123');
let value = formDataToJSON(formData).foo;
for (let i = 0; i < 100; i++) {
value = value.bar;
}
expect(value).toBe('123');
});
});