From b0d9ad75324355ce10d0af4802cb64e557ddec0a Mon Sep 17 00:00:00 2001 From: Mark Herwege Date: Mon, 19 Aug 2024 11:16:32 +0200 Subject: [PATCH] [blockly] HTTP block enhancements (#2607) Signed-off-by: Mark Herwege --- .../assets/definitions/blockly/blocks-http.js | 238 +++++++++++++----- 1 file changed, 170 insertions(+), 68 deletions(-) diff --git a/bundles/org.openhab.ui/web/src/assets/definitions/blockly/blocks-http.js b/bundles/org.openhab.ui/web/src/assets/definitions/blockly/blocks-http.js index 791a570169..7b84c4b824 100644 --- a/bundles/org.openhab.ui/web/src/assets/definitions/blockly/blocks-http.js +++ b/bundles/org.openhab.ui/web/src/assets/definitions/blockly/blocks-http.js @@ -8,6 +8,7 @@ import { javascriptGenerator } from 'blockly/javascript.js' export default function (f7, isGraalJs) { const timeoutImage = '' const headerImage = '' + const queryImage = '' const unavailMsg = 'HTTP blocks aren\'t supported in "application/javascript;version=ECMAScript-5.1"' /* @@ -21,10 +22,14 @@ export default function (f7, isGraalJs) { const imageHeaderField = new Blockly.FieldImage(headerImage, 15, 15, undefined, this.onClickHeader) imageHeaderField.setTooltip('Add headers to the request.') + const imageQueryField = new Blockly.FieldImage(queryImage, 15, 15, undefined, this.onClickQuery) + imageQueryField.setTooltip('Add query to the request url.') + this.appendValueInput('url') .setCheck('String') .appendField(imageTimeoutField, 'imgTimeout') .appendField(imageHeaderField, 'imgHeader') + .appendField(imageQueryField, 'imgQuery') .appendField('send', 'methodField') .appendField(new Blockly.FieldDropdown([ ['HttpGetRequest', 'HttpGetRequest'], @@ -34,115 +39,177 @@ export default function (f7, isGraalJs) { ], this.handleRequestTypeSelection.bind(this)), 'requestType') .appendField('to') - this.updateShape(this.hasTimeout, true, this.hasHeader) - this.handleRequestTypeSelection(this.getFieldValue('requestType')) - this.setInputsInline(false) this.setOutput(true, null) this.setColour(230) this.setTooltip('Send HTTP requests') this.setHelpUrl('https://www.openhab.org/docs/configuration/blockly/rules-blockly-http.html#requests') + + this.updateShape(this.hasTimeout, this.hasHeader, this.hasQuery) }, handleRequestTypeSelection: function (requestType) { if (this.requestType !== requestType) { this.requestType = requestType - const getOrDelete = (requestType === 'HttpGetRequest' || requestType === 'HttpDeleteRequest') - if (!getOrDelete) { - if (!this.getInput('payload')) { - this.appendValueInput('payload') - .setCheck(null) - .appendField('with Payload') - .appendField(new Blockly.FieldDropdown([ - ['application/json', 'application/json'], - ['none', 'none'], - ['application/javascript', 'application/javascript'], - ['application/xhtml+xml ', 'application/xhtml+xml '], - ['application/xml ', 'application/xml '], - ['application/x-www-form-urlencoded ', 'application/x-www-form-urlencoded '], - ['text/html', 'text/html'], - ['text/javascript', 'text/javascript'], - ['text/plain', 'text/plain'], - ['text/xml', 'text/xml']]), 'contentType') - } - } else { - if (this.getInput('payload')) { - this.removeInput('payload') - } - } - this.updateShape(this.hasTimeout, true, this.hasHeader) + this.updateShape(this.hasTimeout, this.hasHeader, this.hasQuery) } }, - updateShape: function (hasTimeout, addNumBlock, hasHeader) { + handleContentTypeSelection: function (contentType) { + if (this.contentType !== contentType) { + this.contentType = contentType + this.updateShape(this.hasTimeout, this.hasHeader, this.hasQuery) + } + }, + updateShape: function (hasTimeout, hasHeader, hasQuery) { this.hasTimeout = hasTimeout this.hasHeader = hasHeader + this.hasQuery = hasQuery + + let timeoutInput = this.getInput('timeoutInput') if (hasTimeout) { - if (!this.getInput('timeoutInput')) { - const timeoutInput = this.appendValueInput('timeoutInput') + if (!timeoutInput) { + timeoutInput = this.appendValueInput('timeoutInput') .setCheck('Number') .appendField('with Timeout (ms)') - - const blockAfter = (this.getInput('requestHeader')) ? 'requestHeader' : (this.getInput('payload')) ? 'payload' : undefined - + const blockAfter = this.getInput('requestHeader') ? 'requestHeader' : (this.getInput('query') ? 'query' : (this.getInput('payload') ? 'payload' : undefined)) if (blockAfter) { this.moveInputBefore('timeoutInput', blockAfter) } - if (addNumBlock) { - const parentConnection = timeoutInput.connection - const mathNumberBlock = this.workspace.newBlock('math_number') - mathNumberBlock.setFieldValue('3000', 'NUM') - mathNumberBlock.initSvg() - mathNumberBlock.render() - parentConnection.connect(mathNumberBlock.outputConnection) - } - } - } else { - if (this.getInput('timeoutInput')) { - const parentConnection = this.getInput('timeoutInput').connection - const targetBlock = parentConnection.targetBlock() - if (targetBlock) { - targetBlock.unplug(true) - targetBlock.dispose(true) - } - this.removeInput('timeoutInput') + timeoutInput.setShadowDom(Blockly.utils.xml.textToDom('3000')) } + } else if (timeoutInput) { + timeoutInput.setShadowDom(null) + this.removeInput('timeoutInput') } + + let headerInput = this.getInput('requestHeader') if (hasHeader) { - if (!this.getInput('requestHeader')) { - const headerInput = this.appendValueInput('requestHeader') + if (!headerInput) { + headerInput = this.appendValueInput('requestHeader') .setCheck('Dictionary') .appendField('with Headers') - const blockAfter = (this.getInput('payload')) ? 'payload' : undefined - + const blockAfter = this.getInput('query') ? 'query' : (this.getInput('payload') ? 'payload' : undefined) if (blockAfter) { this.moveInputBefore('requestHeader', blockAfter) } + this.addDictShadowBlock(headerInput, 'header') + } + } else if (headerInput) { + headerInput.setShadowDom(null) + this.removeInput('requestHeader') + } + + let queryInput = this.getInput('query') + if (hasQuery) { + if (!queryInput) { + queryInput = this.appendValueInput('query') + .setCheck('Dictionary') + .appendField('with Query') + const blockAfter = this.getInput('payload') ? 'payload' : undefined + if (blockAfter) { + this.moveInputBefore('query', blockAfter) + } + this.addDictShadowBlock(queryInput, 'param') + } + } else if (queryInput) { + queryInput.setShadowDom(null) + this.removeInput('query') + } + + if (['HttpPostRequest', 'HttpPutRequest'].includes(this.requestType)) { + let payloadInput = this.getInput('payload') + let hasPayload = (this.contentType !== 'none') + if (!hasPayload && payloadInput && (payloadInput.type === Blockly.inputs.inputTypes.VALUE)) { + this.removePayloadInput() + payloadInput = null + } else if (hasPayload && payloadInput && (payloadInput.type === Blockly.inputs.inputTypes.DUMMY)) { + this.removePayloadInput() + payloadInput = null + } + if (!payloadInput) { + if (!hasPayload) { + payloadInput = this.appendDummyInput('payload') + } else { + payloadInput = this.appendValueInput('payload') + } + payloadInput.appendField('with Payload') + .appendField(new Blockly.FieldDropdown([ + ['application/json', 'application/json'], + ['none', 'none'], + ['application/javascript', 'application/javascript'], + ['application/xhtml+xml', 'application/xhtml+xml'], + ['application/xml', 'application/xml'], + ['application/x-www-form-urlencoded', 'application/x-www-form-urlencoded'], + ['text/html', 'text/html'], + ['text/javascript', 'text/javascript'], + ['text/plain', 'text/plain'], + ['text/xml', 'text/xml']], this.handleContentTypeSelection.bind(this)), 'contentType') + if (this.contentType) { + this.setFieldValue(this.contentType, 'contentType') + } + } + if (hasPayload) { + payloadInput.setShadowDom(null) + if (this.contentType === 'application/x-www-form-urlencoded') { + payloadInput.setCheck(['Dictionary', 'String']) + this.addDictShadowBlock(payloadInput, 'param') + } else { + if (this.contentType && (this.contentType !== 'application/json')) { + payloadInput.setCheck('String') + } else { + payloadInput.setCheck(null) + } + payloadInput.setShadowDom(Blockly.utils.xml.textToDom('payload')) + } } } else { - if (this.getInput('requestHeader')) { - this.removeInput('requestHeader') + this.removePayloadInput() + } + }, + removePayloadInput () { + let payloadInput = this.getInput('payload') + if (payloadInput) { + if (payloadInput.type === Blockly.inputs.inputTypes.VALUE) { + payloadInput.setShadowDom(null) } + this.removeInput('payload') } }, + addDictShadowBlock (input, param) { + input.setShadowDom(Blockly.utils.xml.textToDom(` + + ${param}1 + ${param}1 + ${param}2 + ${param}2 + `)) + }, onClickTimeout () { let block = this.getSourceBlock() block.hasTimeout = !block.hasTimeout - block.updateShape(block.hasTimeout, true, block.hasHeader) + block.updateShape(block.hasTimeout, block.hasHeader, block.hasQuery) }, onClickHeader () { let block = this.getSourceBlock() block.hasHeader = !block.hasHeader - block.updateShape(block.hasTimeout, true, block.hasHeader) + block.updateShape(block.hasTimeout, block.hasHeader, block.hasQuery) + }, + onClickQuery () { + let block = this.getSourceBlock() + block.hasQuery = !block.hasQuery + block.updateShape(block.hasTimeout, block.hasHeader, block.hasQuery) }, mutationToDom: function () { let container = Blockly.utils.xml.createElement('mutation') container.setAttribute('hasTimeout', this.hasTimeout) container.setAttribute('hasHeader', this.hasHeader) + container.setAttribute('hasQuery', this.hasQuery) return container }, domToMutation: function (xmlElement) { let hasTimeout = xmlElement.getAttribute('hasTimeout') === 'true' let hasHeader = xmlElement.getAttribute('hasHeader') === 'true' - this.updateShape(hasTimeout, false, hasHeader) + let hasQuery = xmlElement.getAttribute('hasQuery') === 'true' + this.updateShape(hasTimeout, hasHeader, hasQuery) } } @@ -152,23 +219,39 @@ export default function (f7, isGraalJs) { * @param block * @returns {[string,number]} */ - javascriptGenerator.forBlock['oh_httprequest'] = function (block) { + javascriptGenerator.forBlock['oh_httprequest'] = function (block, generator) { const requestType = block.getFieldValue('requestType') + + let url = javascriptGenerator.valueToCode(block, 'url', javascriptGenerator.ORDER_ATOMIC) + + const query = javascriptGenerator.valueToCode(block, 'query', javascriptGenerator.ORDER_ATOMIC) + if (query) { + const queryEncodeFunction = encodeParams(query) + url = `${url} + '?' + ${queryEncodeFunction}(${query})` + } + const contentType = block.getFieldValue('contentType') - const headers = javascriptGenerator.valueToCode(block, 'requestHeader', javascriptGenerator.ORDER_ATOMIC) - const timeout = javascriptGenerator.valueToCode(block, 'timeoutInput', javascriptGenerator.ORDER_ATOMIC) || 3000 + let payload = javascriptGenerator.valueToCode(block, 'payload', javascriptGenerator.ORDER_ATOMIC) + if (payload) { + if (contentType === 'application/x-www-form-urlencoded') { + const payloadEncodeFunction = encodeParams(payload) + payload = `${payloadEncodeFunction}(${payload})` + } else if (contentType === 'application/json') { + const payloadToJSONStringFunction = toJSONString(payload) + payload = `${payloadToJSONStringFunction}(${payload})` + } + } - const url = javascriptGenerator.valueToCode(block, 'url', javascriptGenerator.ORDER_ATOMIC) - const payload = javascriptGenerator.valueToCode(block, 'payload', javascriptGenerator.ORDER_ATOMIC) + const headers = javascriptGenerator.valueToCode(block, 'requestHeader', javascriptGenerator.ORDER_ATOMIC) + const timeout = javascriptGenerator.valueToCode(block, 'timeoutInput', javascriptGenerator.ORDER_ATOMIC) || 3000 - const hasContent = (requestType !== 'HttpGetRequest' && requestType !== 'HttpDeleteRequest') let code = '' - if (hasContent) { + if (payload) { if (!headers) { code = `actions.HTTP.send${requestType}(${url}, '${contentType}', ${payload}, ${timeout})` } else { - code = `actions.HTTP.send${requestType}(${url},' ${contentType}', ${payload}, ${headers}, ${timeout})` + code = `actions.HTTP.send${requestType}(${url}, '${contentType}', ${payload}, ${headers}, ${timeout})` } } else { if (!headers) { @@ -183,4 +266,23 @@ export default function (f7, isGraalJs) { throw new Error(unavailMsg) } } + + function encodeParams (params) { + return javascriptGenerator.provideFunction_('encodeParams', [ + 'function encodeParams(params) {', + ' if ((typeof params === \'string\') || (params instanceof String)) return params;', + ' const encodedParams = Object.entries(params).map(([key, value]) => [key, encodeURIComponent(value)]);', + ' return encodedParams.map(p => p.join(\'=\')).join(\'&\');', + '}' + ]) + } + + function toJSONString (params) { + return javascriptGenerator.provideFunction_('toJSONString', [ + 'function toJSONString(params) {', + ' if ((typeof params === \'string\') || (params instanceof String)) return params;', + ' return JSON.stringify(params).replace(/^"+/, \'["\').replace(/"+$/, \'"]\');', + '}' + ]) + } }