From 39f60daca70708127e1019e87fd85f54ac62f625 Mon Sep 17 00:00:00 2001 From: busy-panda Date: Thu, 18 Apr 2024 15:43:09 +0200 Subject: [PATCH] feature: Multi-part requests: user should be able to set content-type for each part in a multi-part request. #1602 --- .../MultipartFormParams/StyledWrapper.js | 2 +- .../RequestPane/MultipartFormParams/index.js | 25 ++++++++ .../ReduxStore/slices/collections/index.js | 2 + .../bruno-cli/src/runner/prepare-request.js | 51 +++++++++++---- .../src/ipc/network/prepare-request.js | 10 ++- packages/bruno-lang/v2/src/bruToJson.js | 17 ++++- packages/bruno-lang/v2/src/jsonToBru.js | 6 +- .../bruno-lang/v2/tests/fixtures/request.json | 3 + .../bruno-schema/src/collections/index.js | 1 + .../multipart/mixed-content-types.bru | 39 ++++++++++++ .../collection/multipart/small.png | Bin 0 -> 132 bytes packages/bruno-tests/src/index.js | 10 ++- .../src/multipart/form-data-parser.js | 58 ++++++++++++++++++ packages/bruno-tests/src/multipart/index.js | 10 +++ 14 files changed, 208 insertions(+), 26 deletions(-) create mode 100644 packages/bruno-tests/collection/multipart/mixed-content-types.bru create mode 100644 packages/bruno-tests/collection/multipart/small.png create mode 100644 packages/bruno-tests/src/multipart/form-data-parser.js create mode 100644 packages/bruno-tests/src/multipart/index.js diff --git a/packages/bruno-app/src/components/RequestPane/MultipartFormParams/StyledWrapper.js b/packages/bruno-app/src/components/RequestPane/MultipartFormParams/StyledWrapper.js index f04a30be0..80a146a5c 100644 --- a/packages/bruno-app/src/components/RequestPane/MultipartFormParams/StyledWrapper.js +++ b/packages/bruno-app/src/components/RequestPane/MultipartFormParams/StyledWrapper.js @@ -24,7 +24,7 @@ const Wrapper = styled.div` width: 30%; } - &:nth-child(3) { + &:nth-child(4) { width: 70px; } } diff --git a/packages/bruno-app/src/components/RequestPane/MultipartFormParams/index.js b/packages/bruno-app/src/components/RequestPane/MultipartFormParams/index.js index 1f1c9977e..794e35add 100644 --- a/packages/bruno-app/src/components/RequestPane/MultipartFormParams/index.js +++ b/packages/bruno-app/src/components/RequestPane/MultipartFormParams/index.js @@ -52,6 +52,10 @@ const MultipartFormParams = ({ item, collection }) => { param.value = e.target.value; break; } + case 'contentType': { + param.contentType = e.target.value; + break; + } case 'enabled': { param.enabled = e.target.checked; break; @@ -83,6 +87,7 @@ const MultipartFormParams = ({ item, collection }) => { Key Value + Content-Type @@ -142,6 +147,26 @@ const MultipartFormParams = ({ item, collection }) => { /> )} + + + handleParamChange( + { + target: { + value: newValue + } + }, + param, + 'contentType' + ) + } + onRun={handleRun} + collection={collection} + /> +
{ + // make axios work in node using form data + // reference: https://github.com/axios/axios/issues/1006#issuecomment-320165427 + const form = new FormData(); + datas.forEach((item) => { + const value = item.value; + const name = item.name; + let options = {}; + if (item.contentType) { + options.contentType = item.contentType; + } + if (item.type === 'file') { + const filePaths = value || []; + filePaths.forEach((filePath) => { + let trimmedFilePath = filePath.trim(); + + if (!path.isAbsolute(trimmedFilePath)) { + trimmedFilePath = path.join(collectionPath, trimmedFilePath); + } + options.filename = path.basename(trimmedFilePath); + form.append(name, fs.createReadStream(trimmedFilePath), options); + }); + } else { + form.append(name, value, options); + } + }); + return form; +}; const prepareRequest = (request, collectionRoot) => { const headers = {}; @@ -124,17 +155,11 @@ const prepareRequest = (request, collectionRoot) => { } if (request.body.mode === 'multipartForm') { - const params = {}; const enabledParams = filter(request.body.multipartForm, (p) => p.enabled); - each(enabledParams, (p) => { - if (p.type === 'file') { - params[p.name] = p.value.map((path) => fs.createReadStream(path)); - } else { - params[p.name] = p.value; - } - }); - axiosRequest.headers['content-type'] = 'multipart/form-data'; - axiosRequest.data = params; + const collectionPath = process.cwd(); + const form = parseFormData(enabledParams, collectionPath); + extend(axiosRequest.headers, form.getHeaders()); + axiosRequest.data = form; } if (request.body.mode === 'graphql') { diff --git a/packages/bruno-electron/src/ipc/network/prepare-request.js b/packages/bruno-electron/src/ipc/network/prepare-request.js index 37196589a..249b1754f 100644 --- a/packages/bruno-electron/src/ipc/network/prepare-request.js +++ b/packages/bruno-electron/src/ipc/network/prepare-request.js @@ -12,6 +12,10 @@ const parseFormData = (datas, collectionPath) => { datas.forEach((item) => { const value = item.value; const name = item.name; + let options = {}; + if (item.contentType) { + options.contentType = item.contentType; + } if (item.type === 'file') { const filePaths = value || []; filePaths.forEach((filePath) => { @@ -20,11 +24,11 @@ const parseFormData = (datas, collectionPath) => { if (!path.isAbsolute(trimmedFilePath)) { trimmedFilePath = path.join(collectionPath, trimmedFilePath); } - - form.append(name, fs.createReadStream(trimmedFilePath), path.basename(trimmedFilePath)); + options.filename = path.basename(trimmedFilePath); + form.append(name, fs.createReadStream(trimmedFilePath), options); }); } else { - form.append(name, value); + form.append(name, value, options); } }); return form; diff --git a/packages/bruno-lang/v2/src/bruToJson.js b/packages/bruno-lang/v2/src/bruToJson.js index 6f12a6ce5..32ac3a467 100644 --- a/packages/bruno-lang/v2/src/bruToJson.js +++ b/packages/bruno-lang/v2/src/bruToJson.js @@ -45,7 +45,7 @@ const grammar = ohm.grammar(`Bru { pair = st* key st* ":" st* value st* key = keychar* value = multilinetextblock | valuechar* - + // Dictionary for Assert Block assertdictionary = st* "{" assertpairlist? tagend assertpairlist = optionalnl* assertpair (~tagend stnl* assertpair)* (~tagend space)* @@ -133,16 +133,31 @@ const mapPairListToKeyValPairs = (pairList = [], parseEnabled = true) => { }); }; +const multipartExtractContentType = (pair) => { + if (_.isString(pair.value)) { + const match = pair.value.match(/^(.*?)\s*\(Content-Type=(.*?)\)\s*$/); + if (match != null && match.length > 2) { + pair.value = match[1]; + pair.contentType = match[2]; + } else { + pair.contentType = ''; + } + } +}; + const mapPairListToKeyValPairsMultipart = (pairList = [], parseEnabled = true) => { const pairs = mapPairListToKeyValPairs(pairList, parseEnabled); return pairs.map((pair) => { pair.type = 'text'; + multipartExtractContentType(pair); + if (pair.value.startsWith('@file(') && pair.value.endsWith(')')) { let filestr = pair.value.replace(/^@file\(/, '').replace(/\)$/, ''); pair.type = 'file'; pair.value = filestr.split('|'); } + return pair; }); }; diff --git a/packages/bruno-lang/v2/src/jsonToBru.js b/packages/bruno-lang/v2/src/jsonToBru.js index 3357e5d09..658d7cbd2 100644 --- a/packages/bruno-lang/v2/src/jsonToBru.js +++ b/packages/bruno-lang/v2/src/jsonToBru.js @@ -247,16 +247,18 @@ ${indentString(body.sparql)} multipartForms .map((item) => { const enabled = item.enabled ? '' : '~'; + const contentType = + item.contentType && item.contentType !== '' ? ' (Content-Type=' + item.contentType + ')' : ''; if (item.type === 'text') { - return `${enabled}${item.name}: ${item.value}`; + return `${enabled}${item.name}: ${item.value}${contentType}`; } if (item.type === 'file') { let filepaths = item.value || []; let filestr = filepaths.join('|'); const value = `@file(${filestr})`; - return `${enabled}${item.name}: ${value}`; + return `${enabled}${item.name}: ${value}${contentType}`; } }) .join('\n') diff --git a/packages/bruno-lang/v2/tests/fixtures/request.json b/packages/bruno-lang/v2/tests/fixtures/request.json index afb7ca3f9..594d02e97 100644 --- a/packages/bruno-lang/v2/tests/fixtures/request.json +++ b/packages/bruno-lang/v2/tests/fixtures/request.json @@ -103,18 +103,21 @@ ], "multipartForm": [ { + "contentType": "", "name": "apikey", "value": "secret", "enabled": true, "type": "text" }, { + "contentType": "", "name": "numbers", "value": "+91998877665", "enabled": true, "type": "text" }, { + "contentType": "", "name": "message", "value": "hello", "enabled": false, diff --git a/packages/bruno-schema/src/collections/index.js b/packages/bruno-schema/src/collections/index.js index 033e68277..c0cacc1c4 100644 --- a/packages/bruno-schema/src/collections/index.js +++ b/packages/bruno-schema/src/collections/index.js @@ -68,6 +68,7 @@ const multipartFormSchema = Yup.object({ otherwise: Yup.string().nullable() }), description: Yup.string().nullable(), + contentType: Yup.string().nullable(), enabled: Yup.boolean() }) .noUnknown(true) diff --git a/packages/bruno-tests/collection/multipart/mixed-content-types.bru b/packages/bruno-tests/collection/multipart/mixed-content-types.bru new file mode 100644 index 000000000..7fc421100 --- /dev/null +++ b/packages/bruno-tests/collection/multipart/mixed-content-types.bru @@ -0,0 +1,39 @@ +meta { + name: mixed-content-types + type: http + seq: 1 +} + +post { + url: {{host}}/api/multipart/mixed-content-types + body: multipartForm + auth: none +} + +body:multipart-form { + param1: test + param2: {"test":"i am json"} (Content-Type=application/json) + param3: @file(multipart/small.png) +} + +tests { + test("Status code is 200", function () { + expect(res.getStatus()).to.equal(200); + }); + test("param1 has no content-type", function () { + var param1 = res.body.find(p=>p.name === 'param1') + expect(param1).to.be.an('object'); + expect(param1.contentType).to.be.undefined; + }); + test("param2 has content-type application/json", function () { + var param2 = res.body.find(p=>p.name === 'param2') + expect(param2).to.be.an('object'); + expect(param2.contentType).to.equals('application/json'); + }); + test("param3 has content-type image/png", function () { + var param3 = res.body.find(p=>p.name === 'param3') + expect(param3).to.be.an('object'); + expect(param3.contentType).to.equals('image/png'); + }); + +} diff --git a/packages/bruno-tests/collection/multipart/small.png b/packages/bruno-tests/collection/multipart/small.png new file mode 100644 index 0000000000000000000000000000000000000000..2b584adf0512bb703f04eab2f012893496a03684 GIT binary patch literal 132 zcmeAS@N?(olHy`uVBq!ia0vp^0zfRp!2~4lMxL|Nn{1`ISV`@iy0Uc%|V#aIAVSkP*Bp-#WBRfKlz8hlEhU99x!f{`uCrm;pK~% Vb1~=hq=1STJYD@<);T3K0RTE@AvOR2 literal 0 HcmV?d00001 diff --git a/packages/bruno-tests/src/index.js b/packages/bruno-tests/src/index.js index 9ba6e3170..d9b921951 100644 --- a/packages/bruno-tests/src/index.js +++ b/packages/bruno-tests/src/index.js @@ -2,23 +2,25 @@ const express = require('express'); const bodyParser = require('body-parser'); const xmlparser = require('express-xml-bodyparser'); const cors = require('cors'); -const multer = require('multer'); +const formDataParser = require('./multipart/form-data-parser'); const app = new express(); const port = process.env.PORT || 8080; -const upload = multer(); app.use(cors()); app.use(xmlparser()); app.use(bodyParser.text()); app.use(bodyParser.json()); app.use(bodyParser.urlencoded({ extended: true })); +formDataParser.init(app, express); const authRouter = require('./auth'); const echoRouter = require('./echo'); +const multipartRouter = require('./multipart'); app.use('/api/auth', authRouter); app.use('/api/echo', echoRouter); +app.use('/api/multipart', multipartRouter); app.get('/ping', function (req, res) { return res.send('pong'); @@ -32,10 +34,6 @@ app.get('/query', function (req, res) { return res.json(req.query); }); -app.post('/echo/multipartForm', upload.none(), function (req, res) { - return res.json(req.body); -}); - app.get('/redirect-to-ping', function (req, res) { return res.redirect('/ping'); }); diff --git a/packages/bruno-tests/src/multipart/form-data-parser.js b/packages/bruno-tests/src/multipart/form-data-parser.js new file mode 100644 index 000000000..8b4aa500a --- /dev/null +++ b/packages/bruno-tests/src/multipart/form-data-parser.js @@ -0,0 +1,58 @@ +/** + * Instead of using multer for example to parse the multipart form data, we build our own parser + * so that we can verify the content type are set correctly by bruno (for example application/json for json content) + */ + +const extractParam = function (param, str, delimiter, quote, endDelimiter) { + let regex = new RegExp(`${param}${delimiter}\\s*${quote}(.*?)${quote}${endDelimiter}`); + const found = str.match(regex); + if (found != null && found.length > 1) { + return found[1]; + } else { + return null; + } +}; + +const init = function (app, express) { + app.use(express.raw({ type: 'multipart/form-data' })); +}; + +const parsePart = function (part) { + let result = {}; + const name = extractParam('name', part, '=', '"', ''); + if (name) { + result.name = name; + } + const filename = extractParam('filename', part, '=', '"', ''); + if (filename) { + result.filename = filename; + } + const contentType = extractParam('Content-Type', part, ':', '', ';'); + if (contentType) { + result.contentType = contentType; + } + if (!filename) { + result.value = part.substring(part.indexOf('value=') + 'value='.length); + } + if (contentType === 'application/json') { + result.value = JSON.parse(result.value); + } + return result; +}; + +const parse = function (req) { + const BOUNDARY = 'boundary='; + const contentType = req.headers['content-type']; + const boundary = '--' + contentType.substring(contentType.indexOf(BOUNDARY) + BOUNDARY.length); + const rawBody = req.body.toString(); + let parts = rawBody.split(boundary).filter((part) => part.length > 0); + parts = parts.map((part) => part.trim('\r\n')); + parts = parts.filter((part) => part != '--'); + parts = parts.map((part) => part.replace('\r\n\r\n', ';value=')); + parts = parts.map((part) => part.replace('\r\n', ';')); + parts = parts.map((part) => parsePart(part)); + return parts; +}; + +module.exports.parse = parse; +module.exports.init = init; diff --git a/packages/bruno-tests/src/multipart/index.js b/packages/bruno-tests/src/multipart/index.js new file mode 100644 index 000000000..a98837c54 --- /dev/null +++ b/packages/bruno-tests/src/multipart/index.js @@ -0,0 +1,10 @@ +const express = require('express'); +const router = express.Router(); +const formDataParser = require('./form-data-parser'); + +router.post('/mixed-content-types', (req, res) => { + const parts = formDataParser.parse(req); + return res.json(parts); +}); + +module.exports = router;