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(/"+$/, \'"]\');',
+ '}'
+ ])
+ }
}