diff --git a/packages/bruno-lang/package.json b/packages/bruno-lang/package.json index c243c798..632aed4c 100644 --- a/packages/bruno-lang/package.json +++ b/packages/bruno-lang/package.json @@ -10,6 +10,7 @@ "test": "jest" }, "dependencies": { - "arcsecond": "^5.0.0" + "arcsecond": "^5.0.0", + "ohm-js": "^16.6.0" } } diff --git a/packages/bruno-lang/v2/src/index.js b/packages/bruno-lang/v2/src/index.js new file mode 100644 index 00000000..b21459cc --- /dev/null +++ b/packages/bruno-lang/v2/src/index.js @@ -0,0 +1,369 @@ +const ohm = require("ohm-js"); +const _ = require('lodash'); + +/** + * A Bru file is made up of blocks. + * There are two types of blocks + * + * 1. Dictionary Blocks - These are blocks that have key value pairs + * ex: + * headers { + * content-type: application/json + * } + * + * 2. Text Blocks - These are blocks that have text + * ex: + * body:json { + * { + * "username": "John Nash", + * "password": "governingdynamics + * } + * + */ +const grammar = ohm.grammar(`Bru { + BruFile = (meta | http | querydisabled | query | headersdisabled | headers | bodies | varsandassert | script | test | docs)* + bodies = bodyjson | bodytext | bodyxml | bodygraphql | bodygraphqlvars | bodyforms + bodyforms = bodyformurlencodeddisabled | bodyformurlencoded | bodymultipartdisabled | bodymultipart + + nl = "\\r"? "\\n" + st = " " | "\\t" + tagend = nl "}" + validkey = ~(st | ":") any + validvalue = ~nl any + + // Dictionary Blocks + dictionary = st* "{" pairlist? tagend + pairlist = nl* pair (~tagend nl pair)* (~tagend space)* + pair = st* key st* ":" st* value? st* + key = ~tagend validkey* + value = ~tagend validvalue* + + // Text Blocks + textblock = textline (~tagend nl textline)* + textline = textchar* + textchar = ~nl any + + meta = "meta" dictionary + + http = get | post | put | delete | options | head | connect | trace + get = "get" dictionary + post = "post" dictionary + put = "put" dictionary + delete = "delete" dictionary + options = "options" dictionary + head = "head" dictionary + connect = "connect" dictionary + trace = "trace" dictionary + + headers = "headers" dictionary + headersdisabled = "headers:disabled" dictionary + + query = "query" dictionary + querydisabled = "query:disabled" dictionary + + varsandassert = vars | varsdisabled | varslocal | varslocaldisabled | assert | assertdisabled + vars = "vars" dictionary + varsdisabled = "vars:disabled" dictionary + varslocal = "vars:local" dictionary + varslocaldisabled = "vars:local:disabled" dictionary + assert = "assert" dictionary + assertdisabled = "assert:disabled" dictionary + + bodyjson = "body:json" st* "{" nl* textblock tagend + bodytext = "body:text" st* "{" nl* textblock tagend + bodyxml = "body:xml" st* "{" nl* textblock tagend + bodygraphql = "body:graphql" st* "{" nl* textblock tagend + bodygraphqlvars = "body:graphql:vars" st* "{" nl* textblock tagend + + bodyformurlencoded = "body:form-urlencoded" dictionary + bodyformurlencodeddisabled = "body:form-urlencoded:disabled" dictionary + bodymultipart = "body:multipart-form" dictionary + bodymultipartdisabled = "body:multipart-form:disabled" dictionary + + script = "script" st* "{" nl* textblock tagend + test = "test" st* "{" nl* textblock tagend + docs = "docs" st* "{" nl* textblock tagend +}`); + +const mapPairListToKeyValPairs = (pairList = [], enabled = true) => { + if(!pairList.length) { + return []; + } + return _.map(pairList[0], pair => { + const key = _.keys(pair)[0]; + return { + name: key, + value: pair[key], + enabled: enabled + }; + }); +}; + +const concatArrays = (objValue, srcValue) => { + if (_.isArray(objValue) && _.isArray(srcValue)) { + return objValue.concat(srcValue); + } +}; + +const mapPairListToKeyValPair = (pairList = []) => { + if(!pairList || !pairList.length) { + return {}; + } + + return _.merge({}, ...pairList[0]); +} + +const sem = grammar.createSemantics().addAttribute('ast', { + BruFile(tags) { + if(!tags || !tags.ast || !tags.ast.length) { + return {}; + } + + return _.reduce(tags.ast, (result, item) => { + return _.mergeWith(result, item, concatArrays); + }, {}); + }, + dictionary(_1, _2, pairlist, _3) { + return pairlist.ast; + }, + pairlist(_1, pair, _2, rest, _3) { + return [pair.ast, ...rest.ast]; + }, + pair(_1, key, _2, _3, _4, value, _5) { + let res = {}; + res[key.ast] = _.get(value, 'ast[0]', ''); + return res; + }, + key(chars) { + return chars.sourceString; + }, + value(chars) { + return chars.sourceString ? chars.sourceString.trim() : ''; + }, + meta(_1, dictionary) { + return { + meta: mapPairListToKeyValPair(dictionary.ast) + }; + }, + get(_1, dictionary) { + return { + http: { + method: 'GET', + ...mapPairListToKeyValPair(dictionary.ast) + } + }; + }, + post(_1, dictionary) { + return { + http: { + method: 'POST', + ...mapPairListToKeyValPair(dictionary.ast) + } + }; + }, + put(_1, dictionary) { + return { + http: { + method: 'PUT', + ...mapPairListToKeyValPair(dictionary.ast) + } + }; + }, + delete(_1, dictionary) { + return { + http: { + method: 'DELETE', + ...mapPairListToKeyValPair(dictionary.ast) + } + }; + }, + options(_1, dictionary) { + return { + http: { + method: 'OPTIONS', + ...mapPairListToKeyValPair(dictionary.ast) + } + }; + }, + head(_1, dictionary) { + return { + http: { + method: 'HEAD', + ...mapPairListToKeyValPair(dictionary.ast) + } + }; + }, + connect(_1, dictionary) { + return { + http: { + method: 'CONNECT', + ...mapPairListToKeyValPair(dictionary.ast) + } + }; + }, + query(_1, dictionary) { + return { + query: mapPairListToKeyValPairs(dictionary.ast) + }; + }, + querydisabled(_1, dictionary) { + return { + query: mapPairListToKeyValPairs(dictionary.ast, false) + }; + }, + headers(_1, dictionary) { + return { + headers: mapPairListToKeyValPairs(dictionary.ast) + }; + }, + headersdisabled(_1, dictionary) { + return { + headers: mapPairListToKeyValPairs(dictionary.ast, false) + }; + }, + bodyformurlencoded(_1, dictionary) { + return { + body: { + formUrlEncoded: mapPairListToKeyValPairs(dictionary.ast) + } + }; + }, + bodyformurlencodeddisabled(_1, dictionary) { + return { + body: { + formUrlEncoded: mapPairListToKeyValPairs(dictionary.ast, false) + } + }; + }, + bodymultipart(_1, dictionary) { + return { + body: { + multipartForm: mapPairListToKeyValPairs(dictionary.ast) + } + }; + }, + bodymultipartdisabled(_1, dictionary) { + return { + body: { + multipartForm: mapPairListToKeyValPairs(dictionary.ast, false) + } + }; + }, + bodyjson(_1, _2, _3, _4, textblock, _5) { + return { + body: { + json: textblock.sourceString + } + }; + }, + bodytext(_1, _2, _3, _4, textblock, _5) { + return { + body: { + text: textblock.sourceString + } + }; + }, + bodyxml(_1, _2, _3, _4, textblock, _5) { + return { + body: { + xml: textblock.sourceString + } + }; + }, + bodygraphql(_1, _2, _3, _4, textblock, _5) { + return { + body: { + graphql: { + query: textblock.sourceString + } + } + }; + }, + bodygraphqlvars(_1, _2, _3, _4, textblock, _5) { + return { + body: { + graphql: { + variables: textblock.sourceString + } + } + }; + }, + vars(_1, dictionary) { + return { + vars: mapPairListToKeyValPairs(dictionary.ast) + }; + }, + varsdisabled(_1, dictionary) { + return { + vars: mapPairListToKeyValPairs(dictionary.ast, false) + }; + }, + varslocal(_1, dictionary) { + return { + varsLocal: mapPairListToKeyValPairs(dictionary.ast) + }; + }, + varslocaldisabled(_1, dictionary) { + return { + varsLocal: mapPairListToKeyValPairs(dictionary.ast, false) + }; + }, + assert(_1, dictionary) { + return { + assert: mapPairListToKeyValPairs(dictionary.ast) + }; + }, + assertdisabled(_1, dictionary) { + return { + assert: mapPairListToKeyValPairs(dictionary.ast, false) + }; + }, + script(_1, _2, _3, _4, textblock, _5) { + return { + script: textblock.sourceString + }; + }, + test(_1, _2, _3, _4, textblock, _5) { + return { + test: textblock.sourceString + };; + }, + docs(_1, _2, _3, _4, textblock, _5) { + return { + docs: textblock.sourceString + }; + }, + textblock(line, _1, rest) { + return [line.ast, ...rest.ast].join('\n'); + }, + textline(chars) { + return chars.sourceString; + }, + textchar(char) { + return char.sourceString; + }, + nl(_1, _2) { + return ''; + }, + st(_) { + return ''; + }, + tagend(_1 ,_2) { + return ''; + }, + _iter(...elements) { + return elements.map(e => e.ast); + } +}); + +const parser = (input) => { + const match = grammar.match(input); + + if(match.succeeded()) { + return sem(match).ast; + } else { + throw new Error(match.message); + } +} + +module.exports = parser; diff --git a/packages/bruno-lang/v2/tests/dictionary.spec.js b/packages/bruno-lang/v2/tests/dictionary.spec.js new file mode 100644 index 00000000..46d5b31f --- /dev/null +++ b/packages/bruno-lang/v2/tests/dictionary.spec.js @@ -0,0 +1,138 @@ +/** + * This test file is used to test the text parser. + */ + +const parser = require("../src/index"); + +const assertSingleHeader = (input) => { + const output = parser(input); + + const expected = { + "headers": [{ + "name": "hello", + "value": "world", + "enabled": true + }] + }; + expect(output).toEqual(expected); +}; + +describe("headers parser", () => { + it("should parse empty header", () => { + const input = ` +headers { +}`; + + const output = parser(input); + const expected = { + "headers": [] + }; + expect(output).toEqual(expected); + }); + + it("should parse single header", () => { + const input = ` +headers { + hello: world +}`; + + assertSingleHeader(input); + }); + + it("should parse single header with spaces", () => { + const input = ` +headers { + hello: world +}`; + + assertSingleHeader(input); + }); + + it("should parse single header with spaces and newlines", () => { + const input = ` +headers { + + hello: world + + +}`; + + assertSingleHeader(input); + }); + + it("should parse single header with empty value", () => { + const input = ` +headers { + hello: +}`; + + const output = parser(input); + const expected = { + "headers": [{ + "name": "hello", + "value": "", + "enabled": true + }] + }; + expect(output).toEqual(expected); + }); + + it("should parse multi headers", () => { + const input = ` +headers { + content-type: application/json + Authorization: JWT secret +}`; + + const output = parser(input); + const expected = { + "headers": [{ + "name": "content-type", + "value": "application/json", + "enabled": true + }, { + "name": "Authorization", + "value": "JWT secret", + "enabled": true + }] + }; + expect(output).toEqual(expected); + }); + + it("should parse disabled headers", () => { + const input = ` +headers:disabled { + content-type: application/json +}`; + + const output = parser(input); + const expected = { + "headers": [{ + "name": "content-type", + "value": "application/json", + "enabled": false + }] + }; + expect(output).toEqual(expected); + }); + + it("should throw error on invalid header", () => { + const input = ` +headers { + hello: world + foo +}`; + + expect(() => parser(input)).toThrow(); + }); + + it("should throw error on invalid header", () => { + const input = ` +headers { + hello: world + foo: bar}`; + + expect(() => parser(input)).toThrow(); + }); +}); + diff --git a/packages/bruno-lang/v2/tests/fixtures/request.bru b/packages/bruno-lang/v2/tests/fixtures/request.bru new file mode 100644 index 00000000..9db94d4d --- /dev/null +++ b/packages/bruno-lang/v2/tests/fixtures/request.bru @@ -0,0 +1,114 @@ +meta { + name: Send Bulk SMS + type: http + seq: 1 +} + +get { + url: https://api.textlocal.in/send/ + body: json +} + +query { + apiKey: secret + numbers: 998877665 +} + +query:disabled { + message: hello +} + +headers { + content-type: application/json + Authorization: Bearer 123 +} + +headers:disabled { + transaction-id: {{transactionId}} +} + +body:form-urlencoded { + apikey: secret + numbers: +91998877665 +} + +body:form-urlencoded:disabled { + message: hello +} + +body:multipart-form { + apikey: secret + numbers: +91998877665 +} + +body:multipart-form:disabled { + message: hello +} + +body:json { + { + "hello": "world" + } +} + +body:text { + This is a text body +} + +body:xml { + + John + 30 + +} + +body:graphql { + { + launchesPast { + launch_site { + site_name + } + launch_success + } + } +} + +body:graphql:vars { + { + "limit": 5 + } +} + +vars { + token: $res.body.token +} + +vars:disabled { + petId: $res.body.id +} + +vars:local { + orderNumber: $res.body.orderNumber +} + +vars:local:disabled { + transactionId: $res.body.transactionId +} + +assert { + $res.status: 200 +} + +assert:disabled { + $res.body.message: success +} + +test { + function onResponse(request, response) { + expect(response.status).to.equal(200); + } +} + +docs { + This request needs auth token to be set in the headers. +} \ No newline at end of file diff --git a/packages/bruno-lang/v2/tests/fixtures/request.json b/packages/bruno-lang/v2/tests/fixtures/request.json new file mode 100644 index 00000000..167af5f6 --- /dev/null +++ b/packages/bruno-lang/v2/tests/fixtures/request.json @@ -0,0 +1,123 @@ +{ + "meta": { + "name": "Send Bulk SMS", + "type": "http", + "seq": "1" + }, + "http": { + "method": "GET", + "url": "https://api.textlocal.in/send/", + "body": "json" + }, + "query": [{ + "name": "apiKey", + "value": "secret", + "enabled": true + }, { + "name": "numbers", + "value": "998877665", + "enabled": true + }, { + "name": "message", + "value": "hello", + "enabled": false + }], + "headers": [ + { + "name": "content-type", + "value": "application/json", + "enabled": true + }, + { + "name": "Authorization", + "value": "Bearer 123", + "enabled": true + }, + { + "name": "transaction-id", + "value": "{{transactionId}}", + "enabled": false + } + ], + "body": { + "json": " {\n \"hello\": \"world\"\n }", + "text": " This is a text body", + "xml": " \n John\n 30\n ", + "graphql": { + "query": " {\n launchesPast {\n launch_site {\n site_name\n }\n launch_success\n }\n }", + "variables": " {\n \"limit\": 5\n }" + }, + "formUrlEncoded": [ + { + "name": "apikey", + "value": "secret", + "enabled": true + }, + { + "name": "numbers", + "value": "+91998877665", + "enabled": true + }, + { + "name": "message", + "value": "hello", + "enabled": false + } + ], + "multipartForm": [ + { + "name": "apikey", + "value": "secret", + "enabled": true + }, + { + "name": "numbers", + "value": "+91998877665", + "enabled": true + }, + { + "name": "message", + "value": "hello", + "enabled": false + } + ] + }, + "vars": [ + { + "name": "token", + "value": "$res.body.token", + "enabled": true + }, + { + "name": "petId", + "value": "$res.body.id", + "enabled": false + } + ], + "varsLocal": [ + { + "name": "orderNumber", + "value": "$res.body.orderNumber", + "enabled": true + }, + { + "name": "transactionId", + "value": "$res.body.transactionId", + "enabled": false + } + ], + "assert": [ + { + "name": "$res.status", + "value": "200", + "enabled": true + }, + { + "name": "$res.body.message", + "value": "success", + "enabled": false + } + ], + "test": " function onResponse(request, response) {\n expect(response.status).to.equal(200);\n }", + "docs": " This request needs auth token to be set in the headers." +} \ No newline at end of file diff --git a/packages/bruno-lang/v2/tests/index.spec.js b/packages/bruno-lang/v2/tests/index.spec.js new file mode 100644 index 00000000..c33cd763 --- /dev/null +++ b/packages/bruno-lang/v2/tests/index.spec.js @@ -0,0 +1,14 @@ +const fs = require("fs"); +const path = require("path"); +const parser = require("../src/index"); + +describe("parser", () => { + it("should parse the bru file", () => { + const input = fs.readFileSync(path.join(__dirname, 'fixtures', 'request.bru'), 'utf8'); + const expected = require("./fixtures/request.json"); + const output = parser(input); + + // console.log(JSON.stringify(output, null, 2)); + expect(output).toEqual(expected); + }); +}); diff --git a/packages/bruno-lang/v2/tests/text.spec.js b/packages/bruno-lang/v2/tests/text.spec.js new file mode 100644 index 00000000..caefa087 --- /dev/null +++ b/packages/bruno-lang/v2/tests/text.spec.js @@ -0,0 +1,20 @@ +/** + * This test file is used to test the text parser. + */ +const parser = require("../src/index"); + +describe("script parser", () => { + it("should parse script body", () => { + const input = ` +script { + function onResponse(request, response) { + expect(response.status).to.equal(200); + } +} +`; + + const output = parser(input); + const expected = " function onResponse(request, response) {\n expect(response.status).to.equal(200);\n }"; + expect(output.script).toEqual(expected); + }); +});