Commit 9c4d094075bac3dbd6cba30397fa829d5249d28c

Authored by vzikratyi
2 parents 060a54cd 8c0ae686

Merge remote-tracking branch 'upstream/master' into bug/string-to-number-auto-convertion

1 -before_install:  
2 - - sudo rm -f /etc/mavenrc  
3 - - export M2_HOME=/usr/local/maven  
4 - - export MAVEN_OPTS="-Dmaven.repo.local=$HOME/.m2/repository -Xms1024m -Xmx3072m"  
5 - - export HTTP_LOG_CONTROLLER_ERROR_STACK_TRACE=false  
6 -jdk:  
7 - - openjdk8  
8 -language: java  
9 -sudo: required  
10 -services:  
11 - - docker  
12 -script: mvn clean verify -Ddockerfile.skip=false  
@@ -6,19 +6,19 @@ @@ -6,19 +6,19 @@
6 }, 6 },
7 "widgetTypes": [ 7 "widgetTypes": [
8 { 8 {
9 - "alias": "openstreetmap",  
10 - "name": "OpenStreetMap", 9 + "alias": "route_map_tencent_maps",
  10 + "name": "Route Map - Tencent Maps",
11 "descriptor": { 11 "descriptor": {
12 - "type": "latest", 12 + "type": "timeseries",
13 "sizeX": 8.5, 13 "sizeX": 8.5,
14 "sizeY": 6, 14 "sizeY": 6,
15 "resources": [], 15 "resources": [],
16 "templateHtml": "", 16 "templateHtml": "",
17 - "templateCss": ".leaflet-zoom-box {\n\tz-index: 9;\n}\n\n.leaflet-pane { z-index: 4; }\n\n.leaflet-tile-pane { z-index: 2; }\n.leaflet-overlay-pane { z-index: 4; }\n.leaflet-shadow-pane { z-index: 5; }\n.leaflet-marker-pane { z-index: 6; }\n.leaflet-tooltip-pane { z-index: 7; }\n.leaflet-popup-pane { z-index: 8; }\n\n.leaflet-map-pane canvas { z-index: 1; }\n.leaflet-map-pane svg { z-index: 2; }\n\n.leaflet-control {\n\tz-index: 9;\n}\n.leaflet-top,\n.leaflet-bottom {\n\tz-index: 11;\n}\n\n.tb-marker-label {\n border: none;\n background: none;\n box-shadow: none;\n}\n\n.tb-marker-label:before {\n border: none;\n background: none;\n}\n",  
18 - "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('openstreet-map', false, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('openstreet-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('openstreet-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}", 17 + "templateCss": ".error {\n color: red;\n}\n.tb-labels {\n color: #222;\n font: 12px/1.5 \"Helvetica Neue\", Arial, Helvetica, sans-serif;\n text-align: center;\n width: 200px;\n white-space: nowrap;\n}",
  18 + "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('tencent-map', true, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('tencent-map', true);\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('tencent-map', true);\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}",
19 "settingsSchema": "{}", 19 "settingsSchema": "{}",
20 "dataKeySettingsSchema": "{}\n", 20 "dataKeySettingsSchema": "{}\n",
21 - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#9c27b0\",\"settings\":{},\"_hash\":0.9430343126300238,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#8bc34a\",\"settings\":{},\"_hash\":0.1784452363910778,\"funcBody\":\"return \\\"colorpin\\\";\"}]},{\"type\":\"function\",\"name\":\"Second point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#8bc34a\",\"settings\":{},\"_hash\":0.773875863339494,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#3f51b5\",\"settings\":{},\"_hash\":0.405822538899673,\"funcBody\":\"return \\\"thermomether\\\";\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"markerImageSize\":34,\"useColorFunction\":true,\"markerImages\":[\"\",\"\",\"\",\"\"],\"useMarkerImageFunction\":true,\"colorFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'colorpin') {\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120 * 100;\\n\\t return tinycolor.mix('blue', 'red', amount = percent).toHexString();\\n\\t}\\n\\treturn 'blue';\\n}\\n\",\"markerImageFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'thermomether') {\\n\\tvar res = {\\n\\t url: images[0],\\n\\t size: 40\\n\\t}\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120;\\n\\t var index = Math.min(3, Math.floor(4 * percent));\\n\\t res.url = images[index];\\n\\t}\\n\\treturn res;\\n}\",\"color\":\"#fe7569\",\"mapProvider\":\"OpenStreetMap.Mapnik\",\"showTooltip\":true,\"autocloseTooltip\":true,\"tooltipFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Energy:</b> ${energy:2} kWt<br/>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Temperature:</b> ${temperature:2} °C<br/>';\\r\\n }\\r\\n}\",\"labelFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<span style=\\\"color:orange;\\\">${entityName}, ${energy:2} kWt</span>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<span style=\\\"color:blue;\\\">${entityName}, ${temperature:2} °C</span>';\\r\\n }\\r\\n}\",\"provider\":\"openstreet-map\",\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\"},\"title\":\"OpenStreetMap\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}" 21 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First route\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.5851719234007373,\"funcBody\":\"var lats = [37.7696499,\\n37.7699074,\\n37.7699536,\\n37.7697242,\\n37.7695189,\\n37.7696889,\\n37.7697153,\\n37.7701244,\\n37.7700604,\\n37.7705491,\\n37.7715705,\\n37.771752,\\n37.7707533,\\n37.769866];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lats[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.9015113051937396,\"funcBody\":\"var lons = [-122.4261215,\\n-122.4219157,\\n-122.4199623,\\n-122.4179074,\\n-122.4155876,\\n-122.4155521,\\n-122.4163203,\\n-122.4193876,\\n-122.4210496,\\n-122.422284,\\n-122.4232717,\\n-122.4235138,\\n-122.4247605,\\n-122.4258812];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lons[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.7253460349565717,\"funcBody\":\"var value = prevValue;\\nif (time % 500 < 100) {\\n value = value + Math.random() * 40 - 20;\\n if (value < 45) {\\n \\tvalue = 45;\\n } else if (value > 130) {\\n \\tvalue = 130;\\n }\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<div style='font-size: 13px;'><b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Speed:</b> ${Speed} MPH<br/><small>See advanced settings for details</small></div>\",\"markerImageSize\":34,\"useColorFunction\":true,\"markerImages\":[\"\",\"\",\"\"],\"useMarkerImageFunction\":true,\"colorFunction\":\"var speed = dsData[dsIndex]['Speed'];\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n if (percent < 0.5) {\\n percent *=2*100; \\n return tinycolor.mix('green', 'yellow', amount = percent).toHexString();\\n } else {\\n percent = (percent - 0.5)*2*100;\\n return tinycolor.mix('yellow', 'red', amount = percent).toHexString();\\n }\\n}\",\"markerImageFunction\":\"var speed = dsData[dsIndex]['Speed'];\\nvar res = {\\n url: images[0],\\n size: 55\\n};\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n var index = Math.min(2, Math.floor(3 * percent));\\n res.url = images[index];\\n}\\nreturn res;\",\"strokeWeight\":4,\"strokeOpacity\":0.65,\"color\":\"#1976d3\",\"tmDefaultMapType\":\"roadmap\",\"showTooltip\":true,\"autocloseTooltip\":true,\"tmApiKey\":\"84d6d83e0e51e481e50454ccbe8986b\",\"labelFunction\":\"var vehicleType = dsData[dsIndex]['vehicleType'];\\r\\nif (typeof vehicleType !== undefined) {\\r\\n if (vehicleType == \\\"bus\\\") {\\r\\n return '<span style=\\\"color:orange;\\\">Bus: ${entityName}</span>';\\r\\n } else if (vehicleType == \\\"car\\\") {\\r\\n return '<span style=\\\"color:blue;\\\">Car: ${entityName}</span>';\\r\\n }\\r\\n}\",\"tooltipFunction\":\"var vehicleType = dsData[dsIndex]['vehicleType'];\\r\\nif (typeof vehicleType !== undefined) {\\r\\n if (vehicleType == \\\"bus\\\") {\\r\\n return '<b>Bus: ${entityName}</b><br/><b>Bus route:</b> ${busRoute}<br/>';\\r\\n } else if (vehicleType == \\\"car\\\") {\\r\\n return '<b>Car: ${entityName}</b><br/><b>Current destination:</b> ${destination}<br/>';\\r\\n }\\r\\n}\",\"provider\":\"tencent-map\",\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\"},\"title\":\"Route Map - Tencent Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
22 } 22 }
23 }, 23 },
24 { 24 {
@@ -38,8 +38,8 @@ @@ -38,8 +38,8 @@
38 } 38 }
39 }, 39 },
40 { 40 {
41 - "alias": "route_map_tencent_maps",  
42 - "name": "Route Map - Tencent Maps", 41 + "alias": "route_map",
  42 + "name": "Route Map",
43 "descriptor": { 43 "descriptor": {
44 "type": "timeseries", 44 "type": "timeseries",
45 "sizeX": 8.5, 45 "sizeX": 8.5,
@@ -47,10 +47,10 @@ @@ -47,10 +47,10 @@
47 "resources": [], 47 "resources": [],
48 "templateHtml": "", 48 "templateHtml": "",
49 "templateCss": ".error {\n color: red;\n}\n.tb-labels {\n color: #222;\n font: 12px/1.5 \"Helvetica Neue\", Arial, Helvetica, sans-serif;\n text-align: center;\n width: 200px;\n white-space: nowrap;\n}", 49 "templateCss": ".error {\n color: red;\n}\n.tb-labels {\n color: #222;\n font: 12px/1.5 \"Helvetica Neue\", Arial, Helvetica, sans-serif;\n text-align: center;\n width: 200px;\n white-space: nowrap;\n}",
50 - "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('tencent-map', true, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('tencent-map', true);\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('tencent-map', true);\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}", 50 + "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('google-map', true, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('google-map', true);\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('google-map', true);\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}",
51 "settingsSchema": "{}", 51 "settingsSchema": "{}",
52 "dataKeySettingsSchema": "{}\n", 52 "dataKeySettingsSchema": "{}\n",
53 - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First route\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.5851719234007373,\"funcBody\":\"var lats = [37.7696499,\\n37.7699074,\\n37.7699536,\\n37.7697242,\\n37.7695189,\\n37.7696889,\\n37.7697153,\\n37.7701244,\\n37.7700604,\\n37.7705491,\\n37.7715705,\\n37.771752,\\n37.7707533,\\n37.769866];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lats[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.9015113051937396,\"funcBody\":\"var lons = [-122.4261215,\\n-122.4219157,\\n-122.4199623,\\n-122.4179074,\\n-122.4155876,\\n-122.4155521,\\n-122.4163203,\\n-122.4193876,\\n-122.4210496,\\n-122.422284,\\n-122.4232717,\\n-122.4235138,\\n-122.4247605,\\n-122.4258812];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lons[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.7253460349565717,\"funcBody\":\"var value = prevValue;\\nif (time % 500 < 100) {\\n value = value + Math.random() * 40 - 20;\\n if (value < 45) {\\n \\tvalue = 45;\\n } else if (value > 130) {\\n \\tvalue = 130;\\n }\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<div style='font-size: 13px;'><b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Speed:</b> ${Speed} MPH<br/><small>See advanced settings for details</small></div>\",\"markerImageSize\":34,\"useColorFunction\":true,\"markerImages\":[\"\",\"\",\"\"],\"useMarkerImageFunction\":true,\"colorFunction\":\"var speed = dsData[dsIndex]['Speed'];\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n if (percent < 0.5) {\\n percent *=2*100; \\n return tinycolor.mix('green', 'yellow', amount = percent).toHexString();\\n } else {\\n percent = (percent - 0.5)*2*100;\\n return tinycolor.mix('yellow', 'red', amount = percent).toHexString();\\n }\\n}\",\"markerImageFunction\":\"var speed = dsData[dsIndex]['Speed'];\\nvar res = {\\n url: images[0],\\n size: 55\\n};\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n var index = Math.min(2, Math.floor(3 * percent));\\n res.url = images[index];\\n}\\nreturn res;\",\"strokeWeight\":4,\"strokeOpacity\":0.65,\"color\":\"#1976d3\",\"tmDefaultMapType\":\"roadmap\",\"showTooltip\":true,\"autocloseTooltip\":true,\"tmApiKey\":\"84d6d83e0e51e481e50454ccbe8986b\",\"labelFunction\":\"var vehicleType = dsData[dsIndex]['vehicleType'];\\r\\nif (typeof vehicleType !== undefined) {\\r\\n if (vehicleType == \\\"bus\\\") {\\r\\n return '<span style=\\\"color:orange;\\\">Bus: ${entityName}</span>';\\r\\n } else if (vehicleType == \\\"car\\\") {\\r\\n return '<span style=\\\"color:blue;\\\">Car: ${entityName}</span>';\\r\\n }\\r\\n}\",\"tooltipFunction\":\"var vehicleType = dsData[dsIndex]['vehicleType'];\\r\\nif (typeof vehicleType !== undefined) {\\r\\n if (vehicleType == \\\"bus\\\") {\\r\\n return '<b>Bus: ${entityName}</b><br/><b>Bus route:</b> ${busRoute}<br/>';\\r\\n } else if (vehicleType == \\\"car\\\") {\\r\\n return '<b>Car: ${entityName}</b><br/><b>Current destination:</b> ${destination}<br/>';\\r\\n }\\r\\n}\",\"provider\":\"tencent-map\",\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\"},\"title\":\"Route Map - Tencent Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}" 53 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First route\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.5851719234007373,\"funcBody\":\"var lats = [37.7696499,\\n37.7699074,\\n37.7699536,\\n37.7697242,\\n37.7695189,\\n37.7696889,\\n37.7697153,\\n37.7701244,\\n37.7700604,\\n37.7705491,\\n37.7715705,\\n37.771752,\\n37.7707533,\\n37.769866];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lats[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.9015113051937396,\"funcBody\":\"var lons = [-122.4261215,\\n-122.4219157,\\n-122.4199623,\\n-122.4179074,\\n-122.4155876,\\n-122.4155521,\\n-122.4163203,\\n-122.4193876,\\n-122.4210496,\\n-122.422284,\\n-122.4232717,\\n-122.4235138,\\n-122.4247605,\\n-122.4258812];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lons[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.7253460349565717,\"funcBody\":\"var value = prevValue;\\nif (time % 500 < 100) {\\n value = value + Math.random() * 40 - 20;\\n if (value < 45) {\\n \\tvalue = 45;\\n } else if (value > 130) {\\n \\tvalue = 130;\\n }\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Speed:</b> ${Speed} MPH<br/><small>See advanced settings for details</small>\",\"markerImageSize\":34,\"gmDefaultMapType\":\"roadmap\",\"gmApiKey\":\"AIzaSyDoEx2kaGz3PxwbI9T7ccTSg5xjdw8Nw8Q\",\"useColorFunction\":true,\"markerImages\":[\"\",\"\",\"\"],\"useMarkerImageFunction\":true,\"colorFunction\":\"var speed = dsData[dsIndex]['Speed'];\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n if (percent < 0.5) {\\n percent *=2*100; \\n return tinycolor.mix('green', 'yellow', amount = percent).toHexString();\\n } else {\\n percent = (percent - 0.5)*2*100;\\n return tinycolor.mix('yellow', 'red', amount = percent).toHexString();\\n }\\n}\",\"markerImageFunction\":\"var speed = dsData[dsIndex]['Speed'];\\nvar res = {\\n url: images[0],\\n size: 55\\n};\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n var index = Math.min(2, Math.floor(3 * percent));\\n res.url = images[index];\\n}\\nreturn res;\",\"strokeWeight\":4,\"strokeOpacity\":0.65,\"color\":\"#1976d2\",\"showTooltip\":true,\"autocloseTooltip\":true,\"labelFunction\":\"var vehicleType = dsData[dsIndex]['vehicleType'];\\r\\nif (typeof vehicleType !== undefined) {\\r\\n if (vehicleType == \\\"bus\\\") {\\r\\n return '<span style=\\\"color:orange;\\\">Bus: ${entityName}</span>';\\r\\n } else if (vehicleType == \\\"car\\\") {\\r\\n return '<span style=\\\"color:blue;\\\">Car: ${entityName}</span>';\\r\\n }\\r\\n}\",\"tooltipFunction\":\"var vehicleType = dsData[dsIndex]['vehicleType'];\\r\\nif (typeof vehicleType !== undefined) {\\r\\n if (vehicleType == \\\"bus\\\") {\\r\\n return '<b>Bus: ${entityName}</b><br/><b>Bus route:</b> ${busRoute}<br/>';\\r\\n } else if (vehicleType == \\\"car\\\") {\\r\\n return '<b>Car: ${entityName}</b><br/><b>Current destination:</b> ${destination}<br/>';\\r\\n }\\r\\n}\",\"provider\":\"google-map\",\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\"},\"title\":\"Route Map\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
54 } 54 }
55 }, 55 },
56 { 56 {
@@ -70,19 +70,19 @@ @@ -70,19 +70,19 @@
70 } 70 }
71 }, 71 },
72 { 72 {
73 - "alias": "route_map",  
74 - "name": "Route Map", 73 + "alias": "tencent_maps",
  74 + "name": "Tencent Maps",
75 "descriptor": { 75 "descriptor": {
76 - "type": "timeseries",  
77 - "sizeX": 8.5, 76 + "type": "latest",
  77 + "sizeX": 9,
78 "sizeY": 6, 78 "sizeY": 6,
79 "resources": [], 79 "resources": [],
80 "templateHtml": "", 80 "templateHtml": "",
81 "templateCss": ".error {\n color: red;\n}\n.tb-labels {\n color: #222;\n font: 12px/1.5 \"Helvetica Neue\", Arial, Helvetica, sans-serif;\n text-align: center;\n width: 200px;\n white-space: nowrap;\n}", 81 "templateCss": ".error {\n color: red;\n}\n.tb-labels {\n color: #222;\n font: 12px/1.5 \"Helvetica Neue\", Arial, Helvetica, sans-serif;\n text-align: center;\n width: 200px;\n white-space: nowrap;\n}",
82 - "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('google-map', true, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('google-map', true);\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('google-map', true);\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}", 82 + "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('tencent-map', false, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('tencent-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('tencent-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}",
83 "settingsSchema": "{}", 83 "settingsSchema": "{}",
84 "dataKeySettingsSchema": "{}\n", 84 "dataKeySettingsSchema": "{}\n",
85 - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First route\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.5851719234007373,\"funcBody\":\"var lats = [37.7696499,\\n37.7699074,\\n37.7699536,\\n37.7697242,\\n37.7695189,\\n37.7696889,\\n37.7697153,\\n37.7701244,\\n37.7700604,\\n37.7705491,\\n37.7715705,\\n37.771752,\\n37.7707533,\\n37.769866];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lats[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.9015113051937396,\"funcBody\":\"var lons = [-122.4261215,\\n-122.4219157,\\n-122.4199623,\\n-122.4179074,\\n-122.4155876,\\n-122.4155521,\\n-122.4163203,\\n-122.4193876,\\n-122.4210496,\\n-122.422284,\\n-122.4232717,\\n-122.4235138,\\n-122.4247605,\\n-122.4258812];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lons[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.7253460349565717,\"funcBody\":\"var value = prevValue;\\nif (time % 500 < 100) {\\n value = value + Math.random() * 40 - 20;\\n if (value < 45) {\\n \\tvalue = 45;\\n } else if (value > 130) {\\n \\tvalue = 130;\\n }\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Speed:</b> ${Speed} MPH<br/><small>See advanced settings for details</small>\",\"markerImageSize\":34,\"gmDefaultMapType\":\"roadmap\",\"gmApiKey\":\"AIzaSyDoEx2kaGz3PxwbI9T7ccTSg5xjdw8Nw8Q\",\"useColorFunction\":true,\"markerImages\":[\"\",\"\",\"\"],\"useMarkerImageFunction\":true,\"colorFunction\":\"var speed = dsData[dsIndex]['Speed'];\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n if (percent < 0.5) {\\n percent *=2*100; \\n return tinycolor.mix('green', 'yellow', amount = percent).toHexString();\\n } else {\\n percent = (percent - 0.5)*2*100;\\n return tinycolor.mix('yellow', 'red', amount = percent).toHexString();\\n }\\n}\",\"markerImageFunction\":\"var speed = dsData[dsIndex]['Speed'];\\nvar res = {\\n url: images[0],\\n size: 55\\n};\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n var index = Math.min(2, Math.floor(3 * percent));\\n res.url = images[index];\\n}\\nreturn res;\",\"strokeWeight\":4,\"strokeOpacity\":0.65,\"color\":\"#1976d2\",\"showTooltip\":true,\"autocloseTooltip\":true,\"labelFunction\":\"var vehicleType = dsData[dsIndex]['vehicleType'];\\r\\nif (typeof vehicleType !== undefined) {\\r\\n if (vehicleType == \\\"bus\\\") {\\r\\n return '<span style=\\\"color:orange;\\\">Bus: ${entityName}</span>';\\r\\n } else if (vehicleType == \\\"car\\\") {\\r\\n return '<span style=\\\"color:blue;\\\">Car: ${entityName}</span>';\\r\\n }\\r\\n}\",\"tooltipFunction\":\"var vehicleType = dsData[dsIndex]['vehicleType'];\\r\\nif (typeof vehicleType !== undefined) {\\r\\n if (vehicleType == \\\"bus\\\") {\\r\\n return '<b>Bus: ${entityName}</b><br/><b>Bus route:</b> ${busRoute}<br/>';\\r\\n } else if (vehicleType == \\\"car\\\") {\\r\\n return '<b>Car: ${entityName}</b><br/><b>Current destination:</b> ${destination}<br/>';\\r\\n }\\r\\n}\",\"provider\":\"google-map\",\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\"},\"title\":\"Route Map\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}" 85 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.24727730589425012,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.8437014651129422,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.7558240907832925,\"funcBody\":\"return \\\"colorpin\\\";\"}]},{\"type\":\"function\",\"name\":\"Second Point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.19266205227372524,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#9c27b0\",\"settings\":{},\"_hash\":0.7995830793603149,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#8bc34a\",\"settings\":{},\"_hash\":0.04902495467943502,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#3f51b5\",\"settings\":{},\"_hash\":0.44120841439482095,\"funcBody\":\"return \\\"thermomether\\\";\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"tmDefaultMapType\":\"roadmap\",\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"showTooltip\":true,\"autocloseTooltip\":true,\"tooltipPattern\":\"<div style='font-size: 13px;'><b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small></div>\",\"markerImageSize\":34,\"tmApiKey\":\"84d6d83e0e51e481e50454ccbe8986b\",\"color\":\"#fe7569\",\"useColorFunction\":true,\"colorFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'colorpin') {\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120 * 100;\\n\\t return tinycolor.mix('blue', 'red', amount = percent).toHexString();\\n\\t}\\n\\treturn 'blue';\\n}\\n\",\"useMarkerImageFunction\":true,\"markerImageFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'thermomether') {\\n\\tvar res = {\\n\\t url: images[0],\\n\\t size: 40\\n\\t}\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120;\\n\\t var index = Math.min(3, Math.floor(4 * percent));\\n\\t res.url = images[index];\\n\\t}\\n\\treturn res;\\n}\",\"markerImages\":[\"\",\"\",\"\",\"\"],\"labelFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<span style=\\\"color:orange;\\\">${entityName}, ${energy:2} kWt</span>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<span style=\\\"color:blue;\\\">${entityName}, ${temperature:2} °C</span>';\\r\\n }\\r\\n}\",\"tooltipFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Energy:</b> ${energy:2} kWt<br/>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Temperature:</b> ${temperature:2} °C<br/>';\\r\\n }\\r\\n}\",\"mapProviderHere\":\"HERE.normalDay\",\"provider\":\"tencent-map\",\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\",\"showPolygon\":false,\"polygonKeyName\":\"coordinates\",\"polygonOpacity\":0.5,\"polygonStrokeOpacity\":1,\"polygonStrokeWeight\":1},\"title\":\"Tencent Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
86 } 86 }
87 }, 87 },
88 { 88 {
@@ -102,22 +102,6 @@ @@ -102,22 +102,6 @@
102 } 102 }
103 }, 103 },
104 { 104 {
105 - "alias": "tencent_maps",  
106 - "name": "Tencent Maps",  
107 - "descriptor": {  
108 - "type": "latest",  
109 - "sizeX": 9,  
110 - "sizeY": 6,  
111 - "resources": [],  
112 - "templateHtml": "",  
113 - "templateCss": ".error {\n color: red;\n}\n.tb-labels {\n color: #222;\n font: 12px/1.5 \"Helvetica Neue\", Arial, Helvetica, sans-serif;\n text-align: center;\n width: 200px;\n white-space: nowrap;\n}",  
114 - "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('tencent-map', false, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('tencent-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('tencent-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}",  
115 - "settingsSchema": "{}",  
116 - "dataKeySettingsSchema": "{}\n",  
117 - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.24727730589425012,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.8437014651129422,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.7558240907832925,\"funcBody\":\"return \\\"colorpin\\\";\"}]},{\"type\":\"function\",\"name\":\"Second Point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.19266205227372524,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#9c27b0\",\"settings\":{},\"_hash\":0.7995830793603149,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#8bc34a\",\"settings\":{},\"_hash\":0.04902495467943502,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#3f51b5\",\"settings\":{},\"_hash\":0.44120841439482095,\"funcBody\":\"return \\\"thermomether\\\";\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"tmDefaultMapType\":\"roadmap\",\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"showTooltip\":true,\"autocloseTooltip\":true,\"tooltipPattern\":\"<div style='font-size: 13px;'><b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small></div>\",\"markerImageSize\":34,\"tmApiKey\":\"84d6d83e0e51e481e50454ccbe8986b\",\"color\":\"#fe7569\",\"useColorFunction\":true,\"colorFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'colorpin') {\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120 * 100;\\n\\t return tinycolor.mix('blue', 'red', amount = percent).toHexString();\\n\\t}\\n\\treturn 'blue';\\n}\\n\",\"useMarkerImageFunction\":true,\"markerImageFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'thermomether') {\\n\\tvar res = {\\n\\t url: images[0],\\n\\t size: 40\\n\\t}\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120;\\n\\t var index = Math.min(3, Math.floor(4 * percent));\\n\\t res.url = images[index];\\n\\t}\\n\\treturn res;\\n}\",\"markerImages\":[\"\",\"\",\"\",\"\"],\"labelFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<span style=\\\"color:orange;\\\">${entityName}, ${energy:2} kWt</span>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<span style=\\\"color:blue;\\\">${entityName}, ${temperature:2} °C</span>';\\r\\n }\\r\\n}\",\"tooltipFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Energy:</b> ${energy:2} kWt<br/>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Temperature:</b> ${temperature:2} °C<br/>';\\r\\n }\\r\\n}\",\"mapProviderHere\":\"HERE.normalDay\",\"provider\":\"tencent-map\",\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\",\"showPolygon\":false,\"polygonKeyName\":\"coordinates\",\"polygonOpacity\":0.5,\"polygonStrokeOpacity\":1,\"polygonStrokeWeight\":1},\"title\":\"Tencent Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"  
118 - }  
119 - },  
120 - {  
121 "alias": "image_map", 105 "alias": "image_map",
122 "name": "Image Map", 106 "name": "Image Map",
123 "descriptor": { 107 "descriptor": {
@@ -148,6 +132,22 @@ @@ -148,6 +132,22 @@
148 "dataKeySettingsSchema": "{}\n", 132 "dataKeySettingsSchema": "{}\n",
149 "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#9c27b0\",\"settings\":{},\"_hash\":0.9430343126300238,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#8bc34a\",\"settings\":{},\"_hash\":0.1784452363910778,\"funcBody\":\"return \\\"colorpin\\\";\"}]},{\"type\":\"function\",\"name\":\"Second point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#8bc34a\",\"settings\":{},\"_hash\":0.773875863339494,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#3f51b5\",\"settings\":{},\"_hash\":0.405822538899673,\"funcBody\":\"return \\\"thermomether\\\";\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"markerImageSize\":34,\"gmDefaultMapType\":\"roadmap\",\"gmApiKey\":\"AIzaSyDoEx2kaGz3PxwbI9T7ccTSg5xjdw8Nw8Q\",\"useColorFunction\":true,\"markerImages\":[\"\",\"\",\"\",\"\"],\"useMarkerImageFunction\":true,\"colorFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'colorpin') {\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120 * 100;\\n\\t return tinycolor.mix('blue', 'red', amount = percent).toHexString();\\n\\t}\\n\\treturn 'blue';\\n}\\n\",\"markerImageFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'thermomether') {\\n\\tvar res = {\\n\\t url: images[0],\\n\\t size: 40\\n\\t}\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120;\\n\\t var index = Math.min(3, Math.floor(4 * percent));\\n\\t res.url = images[index];\\n\\t}\\n\\treturn res;\\n}\",\"color\":\"#fe7568\",\"showTooltip\":true,\"autocloseTooltip\":true,\"labelFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<span style=\\\"color:orange;\\\">${entityName}, ${energy:2} kWt</span>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<span style=\\\"color:blue;\\\">${entityName}, ${temperature:2} °C</span>';\\r\\n }\\r\\n}\",\"tooltipFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Energy:</b> ${energy:2} kWt<br/>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Temperature:</b> ${temperature:2} °C<br/>';\\r\\n }\\r\\n}\",\"provider\":\"google-map\",\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\"},\"title\":\"Google Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}" 133 "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#9c27b0\",\"settings\":{},\"_hash\":0.9430343126300238,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#8bc34a\",\"settings\":{},\"_hash\":0.1784452363910778,\"funcBody\":\"return \\\"colorpin\\\";\"}]},{\"type\":\"function\",\"name\":\"Second point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#8bc34a\",\"settings\":{},\"_hash\":0.773875863339494,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#3f51b5\",\"settings\":{},\"_hash\":0.405822538899673,\"funcBody\":\"return \\\"thermomether\\\";\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"markerImageSize\":34,\"gmDefaultMapType\":\"roadmap\",\"gmApiKey\":\"AIzaSyDoEx2kaGz3PxwbI9T7ccTSg5xjdw8Nw8Q\",\"useColorFunction\":true,\"markerImages\":[\"\",\"\",\"\",\"\"],\"useMarkerImageFunction\":true,\"colorFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'colorpin') {\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120 * 100;\\n\\t return tinycolor.mix('blue', 'red', amount = percent).toHexString();\\n\\t}\\n\\treturn 'blue';\\n}\\n\",\"markerImageFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'thermomether') {\\n\\tvar res = {\\n\\t url: images[0],\\n\\t size: 40\\n\\t}\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120;\\n\\t var index = Math.min(3, Math.floor(4 * percent));\\n\\t res.url = images[index];\\n\\t}\\n\\treturn res;\\n}\",\"color\":\"#fe7568\",\"showTooltip\":true,\"autocloseTooltip\":true,\"labelFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<span style=\\\"color:orange;\\\">${entityName}, ${energy:2} kWt</span>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<span style=\\\"color:blue;\\\">${entityName}, ${temperature:2} °C</span>';\\r\\n }\\r\\n}\",\"tooltipFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Energy:</b> ${energy:2} kWt<br/>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Temperature:</b> ${temperature:2} °C<br/>';\\r\\n }\\r\\n}\",\"provider\":\"google-map\",\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\"},\"title\":\"Google Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
150 } 134 }
  135 + },
  136 + {
  137 + "alias": "openstreetmap",
  138 + "name": "OpenStreetMap",
  139 + "descriptor": {
  140 + "type": "latest",
  141 + "sizeX": 8.5,
  142 + "sizeY": 6,
  143 + "resources": [],
  144 + "templateHtml": "",
  145 + "templateCss": ".leaflet-zoom-box {\n\tz-index: 9;\n}\n\n.leaflet-pane { z-index: 4; }\n\n.leaflet-tile-pane { z-index: 2; }\n.leaflet-overlay-pane { z-index: 4; }\n.leaflet-shadow-pane { z-index: 5; }\n.leaflet-marker-pane { z-index: 6; }\n.leaflet-tooltip-pane { z-index: 7; }\n.leaflet-popup-pane { z-index: 8; }\n\n.leaflet-map-pane canvas { z-index: 1; }\n.leaflet-map-pane svg { z-index: 2; }\n\n.leaflet-control {\n\tz-index: 9;\n}\n.leaflet-top,\n.leaflet-bottom {\n\tz-index: 11;\n}\n\n.tb-marker-label {\n border: none;\n background: none;\n box-shadow: none;\n}\n\n.tb-marker-label:before {\n border: none;\n background: none;\n}\n",
  146 + "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('openstreet-map', false, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbMapWidgetV2.settingsSchema('openstreet-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('openstreet-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n\nself.typeParameters = function() {\n return {\n hasDataPageLink: true\n };\n}",
  147 + "settingsSchema": "{}",
  148 + "dataKeySettingsSchema": "{}\n",
  149 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#9c27b0\",\"settings\":{},\"_hash\":0.9430343126300238,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#8bc34a\",\"settings\":{},\"_hash\":0.1784452363910778,\"funcBody\":\"return \\\"colorpin\\\";\"}]},{\"type\":\"function\",\"name\":\"Second point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#8bc34a\",\"settings\":{},\"_hash\":0.773875863339494,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"type\",\"color\":\"#3f51b5\",\"settings\":{},\"_hash\":0.405822538899673,\"funcBody\":\"return \\\"thermomether\\\";\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"markerImageSize\":34,\"useColorFunction\":true,\"markerImages\":[\"\",\"\",\"\",\"\"],\"useMarkerImageFunction\":true,\"colorFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'colorpin') {\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120 * 100;\\n\\t return tinycolor.mix('blue', 'red', amount = percent).toHexString();\\n\\t}\\n\\treturn 'blue';\\n}\\n\",\"markerImageFunction\":\"var type = dsData[dsIndex]['type'];\\nif (type == 'thermomether') {\\n\\tvar res = {\\n\\t url: images[0],\\n\\t size: 40\\n\\t}\\n\\tvar temperature = dsData[dsIndex]['temperature'];\\n\\tif (typeof temperature !== undefined) {\\n\\t var percent = (temperature + 60)/120;\\n\\t var index = Math.min(3, Math.floor(4 * percent));\\n\\t res.url = images[index];\\n\\t}\\n\\treturn res;\\n}\",\"color\":\"#fe7569\",\"mapProvider\":\"OpenStreetMap.Mapnik\",\"showTooltip\":true,\"autocloseTooltip\":true,\"tooltipFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Energy:</b> ${energy:2} kWt<br/>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<b>${entityName}</b><br/><b>Temperature:</b> ${temperature:2} °C<br/>';\\r\\n }\\r\\n}\",\"labelFunction\":\"var deviceType = dsData[dsIndex]['deviceType'];\\r\\nif (typeof deviceType !== undefined) {\\r\\n if (deviceType == \\\"energy meter\\\") {\\r\\n return '<span style=\\\"color:orange;\\\">${entityName}, ${energy:2} kWt</span>';\\r\\n } else if (deviceType == \\\"thermometer\\\") {\\r\\n return '<span style=\\\"color:blue;\\\">${entityName}, ${temperature:2} °C</span>';\\r\\n }\\r\\n}\",\"provider\":\"openstreet-map\",\"defaultCenterPosition\":\"0,0\",\"showTooltipAction\":\"click\"},\"title\":\"OpenStreetMap\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
  150 + }
151 } 151 }
152 ] 152 ]
153 -}  
  153 +}
@@ -86,6 +86,7 @@ BEGIN @@ -86,6 +86,7 @@ BEGIN
86 DROP INDEX IF EXISTS idx_asset_customer_id; 86 DROP INDEX IF EXISTS idx_asset_customer_id;
87 DROP INDEX IF EXISTS idx_asset_customer_id_and_type; 87 DROP INDEX IF EXISTS idx_asset_customer_id_and_type;
88 DROP INDEX IF EXISTS idx_asset_type; 88 DROP INDEX IF EXISTS idx_asset_type;
  89 + DROP INDEX IF EXISTS idx_attribute_kv_by_key_and_last_update_ts;
89 END; 90 END;
90 $$; 91 $$;
91 92
@@ -105,6 +106,7 @@ BEGIN @@ -105,6 +106,7 @@ BEGIN
105 CREATE INDEX IF NOT EXISTS idx_asset_customer_id ON asset(tenant_id, customer_id); 106 CREATE INDEX IF NOT EXISTS idx_asset_customer_id ON asset(tenant_id, customer_id);
106 CREATE INDEX IF NOT EXISTS idx_asset_customer_id_and_type ON asset(tenant_id, customer_id, type); 107 CREATE INDEX IF NOT EXISTS idx_asset_customer_id_and_type ON asset(tenant_id, customer_id, type);
107 CREATE INDEX IF NOT EXISTS idx_asset_type ON asset(tenant_id, type); 108 CREATE INDEX IF NOT EXISTS idx_asset_type ON asset(tenant_id, type);
  109 + CREATE INDEX IF NOT EXISTS idx_attribute_kv_by_key_and_last_update_ts ON attribute_kv(entity_id, attribute_key, last_update_ts desc);
108 END; 110 END;
109 $$; 111 $$;
110 112
@@ -39,6 +39,7 @@ import org.thingsboard.server.common.data.query.EntityDataQuery; @@ -39,6 +39,7 @@ import org.thingsboard.server.common.data.query.EntityDataQuery;
39 import org.thingsboard.server.common.data.query.EntityDataSortOrder; 39 import org.thingsboard.server.common.data.query.EntityDataSortOrder;
40 import org.thingsboard.server.common.data.query.EntityFilter; 40 import org.thingsboard.server.common.data.query.EntityFilter;
41 import org.thingsboard.server.common.data.query.EntityFilterType; 41 import org.thingsboard.server.common.data.query.EntityFilterType;
  42 +import org.thingsboard.server.common.data.query.EntityKeyType;
42 import org.thingsboard.server.common.data.query.EntityListFilter; 43 import org.thingsboard.server.common.data.query.EntityListFilter;
43 import org.thingsboard.server.common.data.query.EntityNameFilter; 44 import org.thingsboard.server.common.data.query.EntityNameFilter;
44 import org.thingsboard.server.common.data.query.EntitySearchQueryFilter; 45 import org.thingsboard.server.common.data.query.EntitySearchQueryFilter;
@@ -200,6 +201,9 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository { @@ -200,6 +201,9 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
200 entityTableMap.put(EntityType.TENANT, "tenant"); 201 entityTableMap.put(EntityType.TENANT, "tenant");
201 } 202 }
202 203
  204 + public static EntityType[] RELATION_QUERY_ENTITY_TYPES = new EntityType[]{
  205 + EntityType.TENANT, EntityType.CUSTOMER, EntityType.USER, EntityType.DASHBOARD, EntityType.ASSET, EntityType.DEVICE, EntityType.ENTITY_VIEW};
  206 +
203 private static final String HIERARCHICAL_QUERY_TEMPLATE = " FROM (WITH RECURSIVE related_entities(from_id, from_type, to_id, to_type, relation_type, lvl) AS (" + 207 private static final String HIERARCHICAL_QUERY_TEMPLATE = " FROM (WITH RECURSIVE related_entities(from_id, from_type, to_id, to_type, relation_type, lvl) AS (" +
204 " SELECT from_id, from_type, to_id, to_type, relation_type, 1 as lvl" + 208 " SELECT from_id, from_type, to_id, to_type, relation_type, 1 as lvl" +
205 " FROM relation" + 209 " FROM relation" +
@@ -265,7 +269,8 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository { @@ -265,7 +269,8 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
265 269
266 270
267 String entityWhereClause = DefaultEntityQueryRepository.this.buildEntityWhere(ctx, query.getEntityFilter(), entityFieldsFiltersMapping); 271 String entityWhereClause = DefaultEntityQueryRepository.this.buildEntityWhere(ctx, query.getEntityFilter(), entityFieldsFiltersMapping);
268 - String latestJoins = EntityKeyMapping.buildLatestJoins(ctx, query.getEntityFilter(), entityType, allLatestMappings); 272 + String latestJoinsCnt = EntityKeyMapping.buildLatestJoins(ctx, query.getEntityFilter(), entityType, allLatestMappings, true);
  273 + String latestJoinsData = EntityKeyMapping.buildLatestJoins(ctx, query.getEntityFilter(), entityType, allLatestMappings, false);
269 String whereClause = DefaultEntityQueryRepository.this.buildWhere(ctx, latestFiltersMapping, query.getEntityFilter().getType()); 274 String whereClause = DefaultEntityQueryRepository.this.buildWhere(ctx, latestFiltersMapping, query.getEntityFilter().getType());
270 String textSearchQuery = DefaultEntityQueryRepository.this.buildTextSearchQuery(ctx, selectionMapping, pageLink.getTextSearch()); 275 String textSearchQuery = DefaultEntityQueryRepository.this.buildTextSearchQuery(ctx, selectionMapping, pageLink.getTextSearch());
271 String entityFieldsSelection = EntityKeyMapping.buildSelections(entityFieldsSelectionMapping, query.getEntityFilter().getType(), entityType); 276 String entityFieldsSelection = EntityKeyMapping.buildSelections(entityFieldsSelectionMapping, query.getEntityFilter().getType(), entityType);
@@ -286,29 +291,44 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository { @@ -286,29 +291,44 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
286 topSelection = topSelection + ", " + latestSelection; 291 topSelection = topSelection + ", " + latestSelection;
287 } 292 }
288 293
289 - String fromClause = String.format("from (select %s from (select %s from %s e where %s) entities %s %s) result %s", 294 + String fromClauseCount = String.format("from (select %s from (select %s from %s e where %s) entities %s %s) result %s",
  295 + "entities.*",
  296 + entityFieldsSelection,
  297 + addEntityTableQuery(ctx, query.getEntityFilter()),
  298 + entityWhereClause,
  299 + latestJoinsCnt,
  300 + whereClause,
  301 + textSearchQuery);
  302 +
  303 + String fromClauseData = String.format("from (select %s from (select %s from %s e where %s) entities %s %s) result %s",
290 topSelection, 304 topSelection,
291 entityFieldsSelection, 305 entityFieldsSelection,
292 addEntityTableQuery(ctx, query.getEntityFilter()), 306 addEntityTableQuery(ctx, query.getEntityFilter()),
293 entityWhereClause, 307 entityWhereClause,
294 - latestJoins, 308 + latestJoinsData,
295 whereClause, 309 whereClause,
296 textSearchQuery); 310 textSearchQuery);
297 311
298 - int totalElements = jdbcTemplate.queryForObject(String.format("select count(*) %s", fromClause), ctx, Integer.class); 312 + if (!StringUtils.isEmpty(pageLink.getTextSearch())) {
  313 + //Unfortunately, we need to sacrifice performance in case of full text search, because it is applied to all joined records.
  314 + fromClauseCount = fromClauseData;
  315 + }
  316 + String countQuery = String.format("select count(id) %s", fromClauseCount);
  317 + int totalElements = jdbcTemplate.queryForObject(countQuery, ctx, Integer.class);
299 318
300 - String dataQuery = String.format("select * %s", fromClause); 319 + String dataQuery = String.format("select * %s", fromClauseData);
301 320
302 EntityDataSortOrder sortOrder = pageLink.getSortOrder(); 321 EntityDataSortOrder sortOrder = pageLink.getSortOrder();
303 if (sortOrder != null) { 322 if (sortOrder != null) {
304 Optional<EntityKeyMapping> sortOrderMappingOpt = mappings.stream().filter(EntityKeyMapping::isSortOrder).findFirst(); 323 Optional<EntityKeyMapping> sortOrderMappingOpt = mappings.stream().filter(EntityKeyMapping::isSortOrder).findFirst();
305 if (sortOrderMappingOpt.isPresent()) { 324 if (sortOrderMappingOpt.isPresent()) {
306 EntityKeyMapping sortOrderMapping = sortOrderMappingOpt.get(); 325 EntityKeyMapping sortOrderMapping = sortOrderMappingOpt.get();
307 - dataQuery = String.format("%s order by %s", dataQuery, sortOrderMapping.getValueAlias());  
308 - if (sortOrder.getDirection() == EntityDataSortOrder.Direction.ASC) {  
309 - dataQuery += " asc"; 326 + String direction = sortOrder.getDirection() == EntityDataSortOrder.Direction.ASC ? "asc" : "desc";
  327 + if (sortOrderMapping.getEntityKey().getType() == EntityKeyType.ENTITY_FIELD) {
  328 + dataQuery = String.format("%s order by %s %s", dataQuery, sortOrderMapping.getValueAlias(), direction);
310 } else { 329 } else {
311 - dataQuery += " desc"; 330 + dataQuery = String.format("%s order by %s %s, %s %s", dataQuery,
  331 + sortOrderMapping.getSortOrderNumAlias(), direction, sortOrderMapping.getSortOrderStrAlias(), direction);
312 } 332 }
313 } 333 }
314 } 334 }
@@ -329,10 +349,10 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository { @@ -329,10 +349,10 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
329 String entityFieldsQuery = EntityKeyMapping.buildQuery(ctx, entityFieldsFilters, entityFilter.getType()); 349 String entityFieldsQuery = EntityKeyMapping.buildQuery(ctx, entityFieldsFilters, entityFilter.getType());
330 String result = permissionQuery; 350 String result = permissionQuery;
331 if (!entityFilterQuery.isEmpty()) { 351 if (!entityFilterQuery.isEmpty()) {
332 - result += " and " + entityFilterQuery; 352 + result += " and (" + entityFilterQuery + ")";
333 } 353 }
334 if (!entityFieldsQuery.isEmpty()) { 354 if (!entityFieldsQuery.isEmpty()) {
335 - result += " and " + entityFieldsQuery; 355 + result += " and (" + entityFieldsQuery + ")";
336 } 356 }
337 return result; 357 return result;
338 } 358 }
@@ -415,7 +435,12 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository { @@ -415,7 +435,12 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
415 String lvlFilter = getLvlFilter(entityFilter.getMaxLevel()); 435 String lvlFilter = getLvlFilter(entityFilter.getMaxLevel());
416 String selectFields = "SELECT tenant_id, customer_id, id, created_time, type, name, label FROM " + entityType.name() + " WHERE id in ( SELECT entity_id"; 436 String selectFields = "SELECT tenant_id, customer_id, id, created_time, type, name, label FROM " + entityType.name() + " WHERE id in ( SELECT entity_id";
417 String from = getQueryTemplate(entityFilter.getDirection()); 437 String from = getQueryTemplate(entityFilter.getDirection());
418 - String whereFilter = " WHERE re.relation_type = :where_relation_type AND re.to_type = :where_entity_type"; 438 + String whereFilter = " WHERE";
  439 + if (!StringUtils.isEmpty(entityFilter.getRelationType())) {
  440 + ctx.addStringParameter("where_relation_type", entityFilter.getRelationType());
  441 + whereFilter += " re.relation_type = :where_relation_type AND";
  442 + }
  443 + whereFilter += " re.to_type = :where_entity_type";
419 444
420 from = String.format(from, lvlFilter, whereFilter); 445 from = String.format(from, lvlFilter, whereFilter);
421 String query = "( " + selectFields + from + ")"; 446 String query = "( " + selectFields + from + ")";
@@ -426,7 +451,6 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository { @@ -426,7 +451,6 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
426 query += " )"; 451 query += " )";
427 ctx.addUuidParameter("relation_root_id", rootId.getId()); 452 ctx.addUuidParameter("relation_root_id", rootId.getId());
428 ctx.addStringParameter("relation_root_type", rootId.getEntityType().name()); 453 ctx.addStringParameter("relation_root_type", rootId.getEntityType().name());
429 - ctx.addStringParameter("where_relation_type", entityFilter.getRelationType());  
430 ctx.addStringParameter("where_entity_type", entityType.name()); 454 ctx.addStringParameter("where_entity_type", entityType.name());
431 return query; 455 return query;
432 } 456 }
@@ -446,46 +470,69 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository { @@ -446,46 +470,69 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
446 ctx.addUuidParameter("relation_root_id", rootId.getId()); 470 ctx.addUuidParameter("relation_root_id", rootId.getId());
447 ctx.addStringParameter("relation_root_type", rootId.getEntityType().name()); 471 ctx.addStringParameter("relation_root_type", rootId.getEntityType().name());
448 472
449 - StringBuilder whereFilter; 473 + StringBuilder whereFilter = new StringBuilder();
  474 + ;
  475 + boolean noConditions = true;
450 if (entityFilter.getFilters() != null && !entityFilter.getFilters().isEmpty()) { 476 if (entityFilter.getFilters() != null && !entityFilter.getFilters().isEmpty()) {
451 - whereFilter = new StringBuilder(" WHERE ");  
452 - boolean first = true;  
453 boolean single = entityFilter.getFilters().size() == 1; 477 boolean single = entityFilter.getFilters().size() == 1;
454 int entityTypeFilterIdx = 0; 478 int entityTypeFilterIdx = 0;
455 for (EntityTypeFilter etf : entityFilter.getFilters()) { 479 for (EntityTypeFilter etf : entityFilter.getFilters()) {
456 - if (first) {  
457 - first = false;  
458 - } else {  
459 - whereFilter.append(" AND ");  
460 - }  
461 - String relationType = etf.getRelationType();  
462 - if (!single) {  
463 - whereFilter.append(" (");  
464 - }  
465 - List<String> whereEntityTypes = etf.getEntityTypes().stream().map(EntityType::name).collect(Collectors.toList());  
466 - whereFilter  
467 - .append(" re.relation_type = :where_relation_type").append(entityTypeFilterIdx);  
468 - if (!whereEntityTypes.isEmpty()) {  
469 - whereFilter.append(" and re.")  
470 - .append(entityFilter.getDirection().equals(EntitySearchDirection.FROM) ? "to" : "from")  
471 - .append("_type in (:where_entity_types").append(entityTypeFilterIdx).append(")");  
472 - }  
473 - if (!single) {  
474 - whereFilter.append(" )");  
475 - }  
476 - ctx.addStringParameter("where_relation_type" + entityTypeFilterIdx, relationType);  
477 - if (!whereEntityTypes.isEmpty()) {  
478 - ctx.addStringListParameter("where_entity_types" + entityTypeFilterIdx, whereEntityTypes); 480 + String etfCondition = buildEtfCondition(ctx, etf, entityFilter.getDirection(), entityTypeFilterIdx++);
  481 + if (!etfCondition.isEmpty()) {
  482 + if (noConditions) {
  483 + whereFilter.append(" WHERE ");
  484 + noConditions = false;
  485 + } else {
  486 + whereFilter.append(" OR ");
  487 + }
  488 + if (!single) {
  489 + whereFilter.append(" (");
  490 + }
  491 + whereFilter.append(etfCondition);
  492 + if (!single) {
  493 + whereFilter.append(" )");
  494 + }
479 } 495 }
480 - entityTypeFilterIdx++;  
481 } 496 }
482 - } else {  
483 - whereFilter = new StringBuilder(); 497 + }
  498 + if (noConditions) {
  499 + whereFilter.append(" WHERE re.")
  500 + .append(entityFilter.getDirection().equals(EntitySearchDirection.FROM) ? "to" : "from")
  501 + .append("_type in (:where_entity_types").append(")");
  502 + ctx.addStringListParameter("where_entity_types", Arrays.stream(RELATION_QUERY_ENTITY_TYPES).map(EntityType::name).collect(Collectors.toList()));
484 } 503 }
485 from = String.format(from, lvlFilter, whereFilter); 504 from = String.format(from, lvlFilter, whereFilter);
486 return "( " + selectFields + from + ")"; 505 return "( " + selectFields + from + ")";
487 } 506 }
488 507
  508 + private String buildEtfCondition(QueryContext ctx, EntityTypeFilter etf, EntitySearchDirection direction, int entityTypeFilterIdx) {
  509 + StringBuilder whereFilter = new StringBuilder();
  510 + String relationType = etf.getRelationType();
  511 + List<EntityType> entityTypes = etf.getEntityTypes();
  512 + List<String> whereEntityTypes;
  513 + if (entityTypes == null || entityTypes.isEmpty()) {
  514 + whereEntityTypes = Collections.emptyList();
  515 + } else {
  516 + whereEntityTypes = etf.getEntityTypes().stream().map(EntityType::name).collect(Collectors.toList());
  517 + }
  518 + boolean hasRelationType = !StringUtils.isEmpty(relationType);
  519 + if (hasRelationType) {
  520 + ctx.addStringParameter("where_relation_type" + entityTypeFilterIdx, relationType);
  521 + whereFilter
  522 + .append("re.relation_type = :where_relation_type").append(entityTypeFilterIdx);
  523 + }
  524 + if (!whereEntityTypes.isEmpty()) {
  525 + if (hasRelationType) {
  526 + whereFilter.append(" and ");
  527 + }
  528 + whereFilter.append("re.")
  529 + .append(direction.equals(EntitySearchDirection.FROM) ? "to" : "from")
  530 + .append("_type in (:where_entity_types").append(entityTypeFilterIdx).append(")");
  531 + ctx.addStringListParameter("where_entity_types" + entityTypeFilterIdx, whereEntityTypes);
  532 + }
  533 + return whereFilter.toString();
  534 + }
  535 +
489 private String getLvlFilter(int maxLevel) { 536 private String getLvlFilter(int maxLevel) {
490 return maxLevel > 0 ? ("and lvl <= " + (maxLevel - 1)) : ""; 537 return maxLevel > 0 ? ("and lvl <= " + (maxLevel - 1)) : "";
491 } 538 }
@@ -270,9 +270,10 @@ public class EntityKeyMapping { @@ -270,9 +270,10 @@ public class EntityKeyMapping {
270 Collectors.joining(", ")); 270 Collectors.joining(", "));
271 } 271 }
272 272
273 - public static String buildLatestJoins(QueryContext ctx, EntityFilter entityFilter, EntityType entityType, List<EntityKeyMapping> latestMappings) {  
274 - return latestMappings.stream().map(mapping -> mapping.toLatestJoin(ctx, entityFilter, entityType)).collect(  
275 - Collectors.joining(" ")); 273 + public static String buildLatestJoins(QueryContext ctx, EntityFilter entityFilter, EntityType entityType, List<EntityKeyMapping> latestMappings, boolean countQuery) {
  274 + return latestMappings.stream().filter(mapping -> !countQuery || mapping.hasFilter())
  275 + .map(mapping -> mapping.toLatestJoin(ctx, entityFilter, entityType)).collect(
  276 + Collectors.joining(" "));
276 } 277 }
277 278
278 public static String buildQuery(QueryContext ctx, List<EntityKeyMapping> mappings, EntityFilterType filterType) { 279 public static String buildQuery(QueryContext ctx, List<EntityKeyMapping> mappings, EntityFilterType filterType) {
@@ -363,19 +364,14 @@ public class EntityKeyMapping { @@ -363,19 +364,14 @@ public class EntityKeyMapping {
363 } 364 }
364 365
365 private String buildAttributeSelection() { 366 private String buildAttributeSelection() {
366 - String attrValAlias = getValueAlias();  
367 - String attrTsAlias = getTsAlias();  
368 - String attrValSelection =  
369 - String.format("(coalesce(cast(%s.bool_v as varchar), '') || " +  
370 - "coalesce(%s.str_v, '') || " +  
371 - "coalesce(cast(%s.long_v as varchar), '') || " +  
372 - "coalesce(cast(%s.dbl_v as varchar), '') || " +  
373 - "coalesce(cast(%s.json_v as varchar), '')) as %s", alias, alias, alias, alias, alias, attrValAlias);  
374 - String attrTsSelection = String.format("%s.last_update_ts as %s", alias, attrTsAlias);  
375 - return String.join(", ", attrValSelection, attrTsSelection); 367 + return buildTimeSeriesOrAttrSelection(true);
376 } 368 }
377 369
378 private String buildTimeSeriesSelection() { 370 private String buildTimeSeriesSelection() {
  371 + return buildTimeSeriesOrAttrSelection(false);
  372 + }
  373 +
  374 + private String buildTimeSeriesOrAttrSelection(boolean attr) {
379 String attrValAlias = getValueAlias(); 375 String attrValAlias = getValueAlias();
380 String attrTsAlias = getTsAlias(); 376 String attrTsAlias = getTsAlias();
381 String attrValSelection = 377 String attrValSelection =
@@ -384,8 +380,25 @@ public class EntityKeyMapping { @@ -384,8 +380,25 @@ public class EntityKeyMapping {
384 "coalesce(cast(%s.long_v as varchar), '') || " + 380 "coalesce(cast(%s.long_v as varchar), '') || " +
385 "coalesce(cast(%s.dbl_v as varchar), '') || " + 381 "coalesce(cast(%s.dbl_v as varchar), '') || " +
386 "coalesce(cast(%s.json_v as varchar), '')) as %s", alias, alias, alias, alias, alias, attrValAlias); 382 "coalesce(cast(%s.json_v as varchar), '')) as %s", alias, alias, alias, alias, alias, attrValAlias);
387 - String attrTsSelection = String.format("%s.ts as %s", alias, attrTsAlias);  
388 - return String.join(", ", attrValSelection, attrTsSelection); 383 + String attrTsSelection = String.format("%s.%s as %s", alias, attr ? "last_update_ts" : "ts", attrTsAlias);
  384 + if (this.isSortOrder) {
  385 + String attrNumAlias = getSortOrderNumAlias();
  386 + String attrVarcharAlias = getSortOrderStrAlias();
  387 + String attrSortOrderSelection =
  388 + String.format("coalesce(%s.dbl_v, cast(%s.long_v as double precision), (case when %s.bool_v then 1 else 0 end)) %s," +
  389 + "coalesce(%s.str_v, cast(%s.json_v as varchar), '') %s", alias, alias, alias, attrNumAlias, alias, alias, attrVarcharAlias);
  390 + return String.join(", ", attrValSelection, attrTsSelection, attrSortOrderSelection);
  391 + } else {
  392 + return String.join(", ", attrValSelection, attrTsSelection);
  393 + }
  394 + }
  395 +
  396 + public String getSortOrderStrAlias() {
  397 + return getValueAlias() + "_so_varchar";
  398 + }
  399 +
  400 + public String getSortOrderNumAlias() {
  401 + return getValueAlias() + "_so_num";
389 } 402 }
390 403
391 private String buildKeyQuery(QueryContext ctx, String alias, KeyFilter keyFilter, 404 private String buildKeyQuery(QueryContext ctx, String alias, KeyFilter keyFilter,
@@ -425,7 +438,14 @@ public class EntityKeyMapping { @@ -425,7 +438,14 @@ public class EntityKeyMapping {
425 if (predicate.getType().equals(FilterPredicateType.NUMERIC)) { 438 if (predicate.getType().equals(FilterPredicateType.NUMERIC)) {
426 return this.buildNumericPredicateQuery(ctx, field, (NumericFilterPredicate) predicate); 439 return this.buildNumericPredicateQuery(ctx, field, (NumericFilterPredicate) predicate);
427 } else if (predicate.getType().equals(FilterPredicateType.STRING)) { 440 } else if (predicate.getType().equals(FilterPredicateType.STRING)) {
428 - return this.buildStringPredicateQuery(ctx, field, (StringFilterPredicate) predicate); 441 + if (key.getKey().equals("entityType") && !filterType.equals(EntityFilterType.RELATIONS_QUERY)) {
  442 + field = ctx.getEntityType().toString();
  443 + return this.buildStringPredicateQuery(ctx, field, (StringFilterPredicate) predicate)
  444 + .replace("lower(" + field, "lower('" + field + "'")
  445 + .replace(field + " ", "'" + field + "' ");
  446 + } else {
  447 + return this.buildStringPredicateQuery(ctx, field, (StringFilterPredicate) predicate);
  448 + }
429 } else { 449 } else {
430 return this.buildBooleanPredicateQuery(ctx, field, (BooleanFilterPredicate) predicate); 450 return this.buildBooleanPredicateQuery(ctx, field, (BooleanFilterPredicate) predicate);
431 } 451 }
@@ -460,30 +480,34 @@ public class EntityKeyMapping { @@ -460,30 +480,34 @@ public class EntityKeyMapping {
460 } 480 }
461 switch (stringFilterPredicate.getOperation()) { 481 switch (stringFilterPredicate.getOperation()) {
462 case EQUAL: 482 case EQUAL:
463 - stringOperationQuery = String.format("%s = :%s", operationField, paramName); 483 + stringOperationQuery = String.format("%s = :%s) or (%s is null and :%s = '')", operationField, paramName, operationField, paramName);
464 break; 484 break;
465 case NOT_EQUAL: 485 case NOT_EQUAL:
466 - stringOperationQuery = String.format("%s != :%s", operationField, paramName); 486 + stringOperationQuery = String.format("%s != :%s) or (%s is null and :%s != '')", operationField, paramName, operationField, paramName);
467 break; 487 break;
468 case STARTS_WITH: 488 case STARTS_WITH:
469 value += "%"; 489 value += "%";
470 - stringOperationQuery = String.format("%s like :%s", operationField, paramName); 490 + stringOperationQuery = String.format("%s like :%s) or (%s is null and :%s = '%%')", operationField, paramName, operationField, paramName);
471 break; 491 break;
472 case ENDS_WITH: 492 case ENDS_WITH:
473 value = "%" + value; 493 value = "%" + value;
474 - stringOperationQuery = String.format("%s like :%s", operationField, paramName); 494 + stringOperationQuery = String.format("%s like :%s) or (%s is null and :%s = '%%')", operationField, paramName, operationField, paramName);
475 break; 495 break;
476 case CONTAINS: 496 case CONTAINS:
477 - value = "%" + value + "%";  
478 - stringOperationQuery = String.format("%s like :%s", operationField, paramName); 497 + if (value.length() > 0) {
  498 + value = "%" + value + "%";
  499 + }
  500 + stringOperationQuery = String.format("%s like :%s) or (%s is null and :%s = '')", operationField, paramName, operationField, paramName);
479 break; 501 break;
480 case NOT_CONTAINS: 502 case NOT_CONTAINS:
481 - value = "%" + value + "%";  
482 - stringOperationQuery = String.format("%s not like :%s", operationField, paramName); 503 + if (value.length() > 0) {
  504 + value = "%" + value + "%";
  505 + }
  506 + stringOperationQuery = String.format("%s not like :%s) or (%s is null and :%s != '')", operationField, paramName, operationField, paramName);
483 break; 507 break;
484 } 508 }
485 ctx.addStringParameter(paramName, value); 509 ctx.addStringParameter(paramName, value);
486 - return String.format("(%s is not null and %s)", field, stringOperationQuery); 510 + return String.format("((%s is not null and %s)", field, stringOperationQuery);
487 } 511 }
488 512
489 private String buildNumericPredicateQuery(QueryContext ctx, String field, NumericFilterPredicate numericFilterPredicate) { 513 private String buildNumericPredicateQuery(QueryContext ctx, String field, NumericFilterPredicate numericFilterPredicate) {
@@ -15,6 +15,7 @@ @@ -15,6 +15,7 @@
15 */ 15 */
16 package org.thingsboard.server.dao.sql.query; 16 package org.thingsboard.server.dao.sql.query;
17 17
  18 +import lombok.extern.slf4j.Slf4j;
18 import org.hibernate.type.PostgresUUIDType; 19 import org.hibernate.type.PostgresUUIDType;
19 import org.springframework.jdbc.core.namedparam.SqlParameterSource; 20 import org.springframework.jdbc.core.namedparam.SqlParameterSource;
20 import org.thingsboard.server.common.data.EntityType; 21 import org.thingsboard.server.common.data.EntityType;
@@ -27,6 +28,7 @@ import java.util.List; @@ -27,6 +28,7 @@ import java.util.List;
27 import java.util.Map; 28 import java.util.Map;
28 import java.util.UUID; 29 import java.util.UUID;
29 30
  31 +@Slf4j
30 public class QueryContext implements SqlParameterSource { 32 public class QueryContext implements SqlParameterSource {
31 private static final PostgresUUIDType UUID_TYPE = new PostgresUUIDType(); 33 private static final PostgresUUIDType UUID_TYPE = new PostgresUUIDType();
32 34
@@ -41,10 +43,14 @@ public class QueryContext implements SqlParameterSource { @@ -41,10 +43,14 @@ public class QueryContext implements SqlParameterSource {
41 } 43 }
42 44
43 void addParameter(String name, Object value, int type, String typeName) { 45 void addParameter(String name, Object value, int type, String typeName) {
44 - Parameter existing = params.put(name, new Parameter(value, type, typeName));  
45 - if (existing != null) { 46 + Parameter newParam = new Parameter(value, type, typeName);
  47 + Parameter oldParam = params.put(name, newParam);
  48 + if (oldParam != null && oldParam.value != null && !oldParam.value.equals(newParam.value)) {
46 throw new RuntimeException("Parameter with name: " + name + " was already registered!"); 49 throw new RuntimeException("Parameter with name: " + name + " was already registered!");
47 } 50 }
  51 + if(value == null){
  52 + log.warn("[{}][{}][{}] Trying to set null value", getTenantId(), getCustomerId(), name);
  53 + }
48 } 54 }
49 55
50 public void append(String s) { 56 public void append(String s) {
@@ -36,4 +36,6 @@ CREATE INDEX IF NOT EXISTS idx_asset_customer_id ON asset(tenant_id, customer_id @@ -36,4 +36,6 @@ CREATE INDEX IF NOT EXISTS idx_asset_customer_id ON asset(tenant_id, customer_id
36 36
37 CREATE INDEX IF NOT EXISTS idx_asset_customer_id_and_type ON asset(tenant_id, customer_id, type); 37 CREATE INDEX IF NOT EXISTS idx_asset_customer_id_and_type ON asset(tenant_id, customer_id, type);
38 38
39 -CREATE INDEX IF NOT EXISTS idx_asset_type ON asset(tenant_id, type);  
  39 +CREATE INDEX IF NOT EXISTS idx_asset_type ON asset(tenant_id, type);
  40 +
  41 +CREATE INDEX IF NOT EXISTS idx_attribute_kv_by_key_and_last_update_ts ON attribute_kv(entity_id, attribute_key, last_update_ts desc);
@@ -15,12 +15,14 @@ @@ -15,12 +15,14 @@
15 */ 15 */
16 package org.thingsboard.server.dao.service; 16 package org.thingsboard.server.dao.service;
17 17
  18 +import com.fasterxml.jackson.core.JsonProcessingException;
  19 +import com.fasterxml.jackson.databind.JsonMappingException;
  20 +import com.fasterxml.jackson.databind.ObjectMapper;
18 import com.google.common.util.concurrent.Futures; 21 import com.google.common.util.concurrent.Futures;
19 import com.google.common.util.concurrent.ListenableFuture; 22 import com.google.common.util.concurrent.ListenableFuture;
20 import org.junit.After; 23 import org.junit.After;
21 import org.junit.Assert; 24 import org.junit.Assert;
22 import org.junit.Before; 25 import org.junit.Before;
23 -import org.junit.Ignore;  
24 import org.junit.Test; 26 import org.junit.Test;
25 import org.springframework.beans.factory.annotation.Autowired; 27 import org.springframework.beans.factory.annotation.Autowired;
26 import org.thingsboard.server.common.data.DataConstants; 28 import org.thingsboard.server.common.data.DataConstants;
@@ -28,42 +30,23 @@ import org.thingsboard.server.common.data.Device; @@ -28,42 +30,23 @@ import org.thingsboard.server.common.data.Device;
28 import org.thingsboard.server.common.data.EntityType; 30 import org.thingsboard.server.common.data.EntityType;
29 import org.thingsboard.server.common.data.Tenant; 31 import org.thingsboard.server.common.data.Tenant;
30 import org.thingsboard.server.common.data.asset.Asset; 32 import org.thingsboard.server.common.data.asset.Asset;
31 -import org.thingsboard.server.common.data.asset.AssetSearchQuery;  
32 -import org.thingsboard.server.common.data.id.CustomerId;  
33 -import org.thingsboard.server.common.data.id.DeviceId;  
34 -import org.thingsboard.server.common.data.id.EntityId;  
35 -import org.thingsboard.server.common.data.id.TenantId;  
36 -import org.thingsboard.server.common.data.kv.AttributeKvEntry;  
37 -import org.thingsboard.server.common.data.kv.BaseAttributeKvEntry;  
38 -import org.thingsboard.server.common.data.kv.KvEntry;  
39 -import org.thingsboard.server.common.data.kv.LongDataEntry; 33 +import org.thingsboard.server.common.data.id.*;
  34 +import org.thingsboard.server.common.data.kv.*;
40 import org.thingsboard.server.common.data.page.PageData; 35 import org.thingsboard.server.common.data.page.PageData;
41 -import org.thingsboard.server.common.data.query.AssetSearchQueryFilter;  
42 -import org.thingsboard.server.common.data.query.DeviceSearchQueryFilter;  
43 -import org.thingsboard.server.common.data.query.DeviceTypeFilter;  
44 -import org.thingsboard.server.common.data.query.EntityCountQuery;  
45 -import org.thingsboard.server.common.data.query.EntityData;  
46 -import org.thingsboard.server.common.data.query.EntityDataPageLink;  
47 -import org.thingsboard.server.common.data.query.EntityDataQuery;  
48 -import org.thingsboard.server.common.data.query.EntityDataSortOrder;  
49 -import org.thingsboard.server.common.data.query.EntityKey;  
50 -import org.thingsboard.server.common.data.query.EntityKeyType;  
51 -import org.thingsboard.server.common.data.query.EntityListFilter;  
52 -import org.thingsboard.server.common.data.query.FilterPredicateValue;  
53 -import org.thingsboard.server.common.data.query.KeyFilter;  
54 -import org.thingsboard.server.common.data.query.NumericFilterPredicate;  
55 -import org.thingsboard.server.common.data.query.RelationsQueryFilter; 36 +import org.thingsboard.server.common.data.query.*;
56 import org.thingsboard.server.common.data.relation.EntityRelation; 37 import org.thingsboard.server.common.data.relation.EntityRelation;
57 import org.thingsboard.server.common.data.relation.EntitySearchDirection; 38 import org.thingsboard.server.common.data.relation.EntitySearchDirection;
58 import org.thingsboard.server.common.data.relation.EntityTypeFilter; 39 import org.thingsboard.server.common.data.relation.EntityTypeFilter;
59 import org.thingsboard.server.common.data.relation.RelationTypeGroup; 40 import org.thingsboard.server.common.data.relation.RelationTypeGroup;
  41 +import org.thingsboard.server.common.data.rule.RuleChain;
  42 +import org.thingsboard.server.common.data.rule.RuleChainMetaData;
  43 +import org.thingsboard.server.common.data.rule.RuleNode;
60 import org.thingsboard.server.dao.attributes.AttributesService; 44 import org.thingsboard.server.dao.attributes.AttributesService;
  45 +import org.thingsboard.server.dao.model.sqlts.ts.TsKvEntity;
  46 +import org.thingsboard.server.dao.rule.RuleChainService;
  47 +import org.thingsboard.server.dao.timeseries.TimeseriesService;
61 48
62 -import java.util.ArrayList;  
63 -import java.util.Arrays;  
64 -import java.util.Collections;  
65 -import java.util.Comparator;  
66 -import java.util.List; 49 +import java.util.*;
67 import java.util.concurrent.ExecutionException; 50 import java.util.concurrent.ExecutionException;
68 import java.util.stream.Collectors; 51 import java.util.stream.Collectors;
69 52
@@ -72,6 +55,9 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -72,6 +55,9 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
72 @Autowired 55 @Autowired
73 private AttributesService attributesService; 56 private AttributesService attributesService;
74 57
  58 + @Autowired
  59 + private TimeseriesService timeseriesService;
  60 +
75 private TenantId tenantId; 61 private TenantId tenantId;
76 62
77 @Before 63 @Before
@@ -88,6 +74,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -88,6 +74,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
88 tenantService.deleteTenant(tenantId); 74 tenantService.deleteTenant(tenantId);
89 } 75 }
90 76
  77 +
91 @Test 78 @Test
92 public void testCountEntitiesByQuery() throws InterruptedException { 79 public void testCountEntitiesByQuery() throws InterruptedException {
93 List<Device> devices = new ArrayList<>(); 80 List<Device> devices = new ArrayList<>();
@@ -131,6 +118,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -131,6 +118,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
131 Assert.assertEquals(0, count); 118 Assert.assertEquals(0, count);
132 } 119 }
133 120
  121 +
134 @Test 122 @Test
135 public void testCountHierarchicalEntitiesByQuery() throws InterruptedException { 123 public void testCountHierarchicalEntitiesByQuery() throws InterruptedException {
136 List<Asset> assets = new ArrayList<>(); 124 List<Asset> assets = new ArrayList<>();
@@ -195,6 +183,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -195,6 +183,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
195 Assert.assertEquals(0, count); 183 Assert.assertEquals(0, count);
196 } 184 }
197 185
  186 +
198 @Test 187 @Test
199 public void testHierarchicalFindEntityDataWithAttributesByQuery() throws ExecutionException, InterruptedException { 188 public void testHierarchicalFindEntityDataWithAttributesByQuery() throws ExecutionException, InterruptedException {
200 List<Asset> assets = new ArrayList<>(); 189 List<Asset> assets = new ArrayList<>();
@@ -266,6 +255,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -266,6 +255,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
266 deviceService.deleteDevicesByTenantId(tenantId); 255 deviceService.deleteDevicesByTenantId(tenantId);
267 } 256 }
268 257
  258 +
269 @Test 259 @Test
270 public void testHierarchicalFindDevicesWithAttributesByQuery() throws ExecutionException, InterruptedException { 260 public void testHierarchicalFindDevicesWithAttributesByQuery() throws ExecutionException, InterruptedException {
271 List<Asset> assets = new ArrayList<>(); 261 List<Asset> assets = new ArrayList<>();
@@ -338,6 +328,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -338,6 +328,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
338 deviceService.deleteDevicesByTenantId(tenantId); 328 deviceService.deleteDevicesByTenantId(tenantId);
339 } 329 }
340 330
  331 +
341 @Test 332 @Test
342 public void testHierarchicalFindAssetsWithAttributesByQuery() throws ExecutionException, InterruptedException { 333 public void testHierarchicalFindAssetsWithAttributesByQuery() throws ExecutionException, InterruptedException {
343 List<Asset> assets = new ArrayList<>(); 334 List<Asset> assets = new ArrayList<>();
@@ -456,6 +447,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -456,6 +447,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
456 } 447 }
457 } 448 }
458 449
  450 +
459 @Test 451 @Test
460 public void testSimpleFindEntityDataByQuery() throws InterruptedException { 452 public void testSimpleFindEntityDataByQuery() throws InterruptedException {
461 List<Device> devices = new ArrayList<>(); 453 List<Device> devices = new ArrayList<>();
@@ -526,6 +518,8 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -526,6 +518,8 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
526 @Test 518 @Test
527 public void testFindEntityDataByQueryWithAttributes() throws ExecutionException, InterruptedException { 519 public void testFindEntityDataByQueryWithAttributes() throws ExecutionException, InterruptedException {
528 520
  521 + List<EntityKeyType> attributesEntityTypes = new ArrayList<>(Arrays.asList(EntityKeyType.CLIENT_ATTRIBUTE, EntityKeyType.SHARED_ATTRIBUTE, EntityKeyType.SERVER_ATTRIBUTE));
  522 +
529 List<Device> devices = new ArrayList<>(); 523 List<Device> devices = new ArrayList<>();
530 List<Long> temperatures = new ArrayList<>(); 524 List<Long> temperatures = new ArrayList<>();
531 List<Long> highTemperatures = new ArrayList<>(); 525 List<Long> highTemperatures = new ArrayList<>();
@@ -548,6 +542,108 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -548,6 +542,108 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
548 List<ListenableFuture<List<Void>>> attributeFutures = new ArrayList<>(); 542 List<ListenableFuture<List<Void>>> attributeFutures = new ArrayList<>();
549 for (int i = 0; i < devices.size(); i++) { 543 for (int i = 0; i < devices.size(); i++) {
550 Device device = devices.get(i); 544 Device device = devices.get(i);
  545 + for (String currentScope : DataConstants.allScopes()) {
  546 + attributeFutures.add(saveLongAttribute(device.getId(), "temperature", temperatures.get(i), currentScope));
  547 + }
  548 + }
  549 + Futures.successfulAsList(attributeFutures).get();
  550 +
  551 + DeviceTypeFilter filter = new DeviceTypeFilter();
  552 + filter.setDeviceType("default");
  553 + filter.setDeviceNameFilter("");
  554 +
  555 + EntityDataSortOrder sortOrder = new EntityDataSortOrder(
  556 + new EntityKey(EntityKeyType.ENTITY_FIELD, "createdTime"), EntityDataSortOrder.Direction.ASC
  557 + );
  558 + EntityDataPageLink pageLink = new EntityDataPageLink(10, 0, null, sortOrder);
  559 + List<EntityKey> entityFields = Collections.singletonList(new EntityKey(EntityKeyType.ENTITY_FIELD, "name"));
  560 + for (EntityKeyType currentAttributeKeyType : attributesEntityTypes) {
  561 + List<EntityKey> latestValues = Collections.singletonList(new EntityKey(currentAttributeKeyType, "temperature"));
  562 + EntityDataQuery query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, null);
  563 + PageData<EntityData> data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  564 + List<EntityData> loadedEntities = new ArrayList<>(data.getData());
  565 + while (data.hasNext()) {
  566 + query = query.next();
  567 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  568 + loadedEntities.addAll(data.getData());
  569 + }
  570 + Assert.assertEquals(67, loadedEntities.size());
  571 + List<String> loadedTemperatures = new ArrayList<>();
  572 + for (Device device : devices) {
  573 + loadedTemperatures.add(loadedEntities.stream().filter(entityData -> entityData.getEntityId().equals(device.getId())).findFirst().orElse(null)
  574 + .getLatest().get(currentAttributeKeyType).get("temperature").getValue());
  575 + }
  576 + List<String> deviceTemperatures = temperatures.stream().map(aLong -> Long.toString(aLong)).collect(Collectors.toList());
  577 + Assert.assertEquals(deviceTemperatures, loadedTemperatures);
  578 +
  579 + pageLink = new EntityDataPageLink(10, 0, null, sortOrder);
  580 + KeyFilter highTemperatureFilter = createNumericKeyFilter("temperature", currentAttributeKeyType, NumericFilterPredicate.NumericOperation.GREATER, 45);
  581 + List<KeyFilter> keyFiltersHighTemperature = Collections.singletonList(highTemperatureFilter);
  582 +
  583 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersHighTemperature);
  584 +
  585 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  586 +
  587 + loadedEntities = new ArrayList<>(data.getData());
  588 +
  589 + while (data.hasNext()) {
  590 + query = query.next();
  591 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  592 + loadedEntities.addAll(data.getData());
  593 + }
  594 + Assert.assertEquals(highTemperatures.size(), loadedEntities.size());
  595 +
  596 + List<String> loadedHighTemperatures = loadedEntities.stream().map(entityData ->
  597 + entityData.getLatest().get(currentAttributeKeyType).get("temperature").getValue()).collect(Collectors.toList());
  598 + List<String> deviceHighTemperatures = highTemperatures.stream().map(aLong -> Long.toString(aLong)).collect(Collectors.toList());
  599 +
  600 + Assert.assertEquals(deviceHighTemperatures, loadedHighTemperatures);
  601 +
  602 + }
  603 + deviceService.deleteDevicesByTenantId(tenantId);
  604 + }
  605 +
  606 + @Test
  607 + public void testBuildNumericPredicateQueryOperations() throws ExecutionException, InterruptedException{
  608 +
  609 + List<Device> devices = new ArrayList<>();
  610 + List<Long> temperatures = new ArrayList<>();
  611 + List<Long> equalTemperatures = new ArrayList<>();
  612 + List<Long> notEqualTemperatures = new ArrayList<>();
  613 + List<Long> greaterTemperatures = new ArrayList<>();
  614 + List<Long> greaterOrEqualTemperatures = new ArrayList<>();
  615 + List<Long> lessTemperatures = new ArrayList<>();
  616 + List<Long> lessOrEqualTemperatures = new ArrayList<>();
  617 +
  618 + for (int i = 0; i < 10; i++) {
  619 + Device device = new Device();
  620 + device.setTenantId(tenantId);
  621 + device.setName("Device" + i);
  622 + device.setType("default");
  623 + device.setLabel("testLabel" + (int) (Math.random() * 1000));
  624 + devices.add(deviceService.saveDevice(device));
  625 + //TO make sure devices have different created time
  626 + Thread.sleep(1);
  627 + long temperature = (long) (Math.random() * 100);
  628 + temperatures.add(temperature);
  629 + if (temperature == 45) {
  630 + greaterOrEqualTemperatures.add(temperature);
  631 + lessOrEqualTemperatures.add(temperature);
  632 + equalTemperatures.add(temperature);
  633 + } else if (temperature > 45) {
  634 + greaterTemperatures.add(temperature);
  635 + greaterOrEqualTemperatures.add(temperature);
  636 + notEqualTemperatures.add(temperature);
  637 + } else {
  638 + lessTemperatures.add(temperature);
  639 + lessOrEqualTemperatures.add(temperature);
  640 + notEqualTemperatures.add(temperature);
  641 + }
  642 + }
  643 +
  644 + List<ListenableFuture<List<Void>>> attributeFutures = new ArrayList<>();
  645 + for (int i = 0; i < devices.size(); i++) {
  646 + Device device = devices.get(i);
551 attributeFutures.add(saveLongAttribute(device.getId(), "temperature", temperatures.get(i), DataConstants.CLIENT_SCOPE)); 647 attributeFutures.add(saveLongAttribute(device.getId(), "temperature", temperatures.get(i), DataConstants.CLIENT_SCOPE));
552 } 648 }
553 Futures.successfulAsList(attributeFutures).get(); 649 Futures.successfulAsList(attributeFutures).get();
@@ -559,9 +655,155 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -559,9 +655,155 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
559 EntityDataSortOrder sortOrder = new EntityDataSortOrder( 655 EntityDataSortOrder sortOrder = new EntityDataSortOrder(
560 new EntityKey(EntityKeyType.ENTITY_FIELD, "createdTime"), EntityDataSortOrder.Direction.ASC 656 new EntityKey(EntityKeyType.ENTITY_FIELD, "createdTime"), EntityDataSortOrder.Direction.ASC
561 ); 657 );
  658 +
  659 + List<EntityKey> entityFields = Collections.singletonList(new EntityKey(EntityKeyType.ENTITY_FIELD, "name"));
  660 + List<EntityKey> latestValues = Collections.singletonList(new EntityKey(EntityKeyType.CLIENT_ATTRIBUTE, "temperature"));
  661 +
  662 + KeyFilter greaterTemperatureFilter = createNumericKeyFilter("temperature", EntityKeyType.CLIENT_ATTRIBUTE, NumericFilterPredicate.NumericOperation.GREATER, 45);
  663 + List<KeyFilter> keyFiltersGreaterTemperature = Collections.singletonList(greaterTemperatureFilter);
  664 +
  665 + KeyFilter greaterOrEqualTemperatureFilter = createNumericKeyFilter("temperature", EntityKeyType.CLIENT_ATTRIBUTE, NumericFilterPredicate.NumericOperation.GREATER_OR_EQUAL, 45);
  666 + List<KeyFilter> keyFiltersGreaterOrEqualTemperature = Collections.singletonList(greaterOrEqualTemperatureFilter);
  667 +
  668 + KeyFilter lessTemperatureFilter = createNumericKeyFilter("temperature", EntityKeyType.CLIENT_ATTRIBUTE, NumericFilterPredicate.NumericOperation.LESS, 45);
  669 + List<KeyFilter> keyFiltersLessTemperature = Collections.singletonList(lessTemperatureFilter);
  670 +
  671 + KeyFilter lessOrEqualTemperatureFilter = createNumericKeyFilter("temperature", EntityKeyType.CLIENT_ATTRIBUTE, NumericFilterPredicate.NumericOperation.LESS_OR_EQUAL, 45);
  672 + List<KeyFilter> keyFiltersLessOrEqualTemperature = Collections.singletonList(lessOrEqualTemperatureFilter);
  673 +
  674 + KeyFilter equalTemperatureFilter = createNumericKeyFilter("temperature", EntityKeyType.CLIENT_ATTRIBUTE, NumericFilterPredicate.NumericOperation.EQUAL, 45);
  675 + List<KeyFilter> keyFiltersEqualTemperature = Collections.singletonList(equalTemperatureFilter);
  676 +
  677 + KeyFilter notEqualTemperatureFilter = createNumericKeyFilter("temperature", EntityKeyType.CLIENT_ATTRIBUTE, NumericFilterPredicate.NumericOperation.NOT_EQUAL, 45);
  678 + List<KeyFilter> keyFiltersNotEqualTemperature = Collections.singletonList(notEqualTemperatureFilter);
  679 +
  680 + //Greater Operation
  681 +
  682 + EntityDataPageLink pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  683 + EntityDataQuery query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersGreaterTemperature);
  684 + PageData<EntityData> data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  685 + List<EntityData> loadedEntities = getLoadedEntities(data, query);
  686 + Assert.assertEquals(greaterTemperatures.size(), loadedEntities.size());
  687 +
  688 + List<String> loadedTemperatures = loadedEntities.stream().map(entityData ->
  689 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("temperature").getValue()).collect(Collectors.toList());
  690 + List<String> deviceTemperatures = greaterTemperatures.stream().map(aLong -> Long.toString(aLong)).collect(Collectors.toList());
  691 +
  692 + Assert.assertEquals(deviceTemperatures, loadedTemperatures);
  693 +
  694 + //Greater or equal Operation
  695 +
  696 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  697 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersGreaterOrEqualTemperature);
  698 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  699 + loadedEntities = getLoadedEntities(data, query);
  700 + Assert.assertEquals(greaterOrEqualTemperatures.size(), loadedEntities.size());
  701 +
  702 + loadedTemperatures = loadedEntities.stream().map(entityData ->
  703 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("temperature").getValue()).collect(Collectors.toList());
  704 + deviceTemperatures = greaterOrEqualTemperatures.stream().map(aLong -> Long.toString(aLong)).collect(Collectors.toList());
  705 +
  706 + Assert.assertEquals(deviceTemperatures, loadedTemperatures);
  707 +
  708 + //Less Operation
  709 +
  710 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  711 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersLessTemperature);
  712 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  713 + loadedEntities = getLoadedEntities(data, query);
  714 + Assert.assertEquals(lessTemperatures.size(), loadedEntities.size());
  715 +
  716 + loadedTemperatures = loadedEntities.stream().map(entityData ->
  717 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("temperature").getValue()).collect(Collectors.toList());
  718 + deviceTemperatures = lessTemperatures.stream().map(aLong -> Long.toString(aLong)).collect(Collectors.toList());
  719 +
  720 + Assert.assertEquals(deviceTemperatures, loadedTemperatures);
  721 +
  722 + //Less or equal Operation
  723 +
  724 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  725 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersLessOrEqualTemperature);
  726 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  727 + loadedEntities = getLoadedEntities(data, query);
  728 + Assert.assertEquals(lessOrEqualTemperatures.size(), loadedEntities.size());
  729 +
  730 + loadedTemperatures = loadedEntities.stream().map(entityData ->
  731 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("temperature").getValue()).collect(Collectors.toList());
  732 + deviceTemperatures = lessOrEqualTemperatures.stream().map(aLong -> Long.toString(aLong)).collect(Collectors.toList());
  733 +
  734 + Assert.assertEquals(deviceTemperatures, loadedTemperatures);
  735 +
  736 + //Equal Operation
  737 +
  738 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  739 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersEqualTemperature);
  740 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  741 + loadedEntities = getLoadedEntities(data, query);
  742 + Assert.assertEquals(equalTemperatures.size(), loadedEntities.size());
  743 +
  744 + loadedTemperatures = loadedEntities.stream().map(entityData ->
  745 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("temperature").getValue()).collect(Collectors.toList());
  746 + deviceTemperatures = equalTemperatures.stream().map(aLong -> Long.toString(aLong)).collect(Collectors.toList());
  747 +
  748 + Assert.assertEquals(deviceTemperatures, loadedTemperatures);
  749 +
  750 + //Not equal Operation
  751 +
  752 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  753 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersNotEqualTemperature);
  754 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  755 + loadedEntities = getLoadedEntities(data, query);
  756 + Assert.assertEquals(notEqualTemperatures.size(), loadedEntities.size());
  757 +
  758 + loadedTemperatures = loadedEntities.stream().map(entityData ->
  759 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("temperature").getValue()).collect(Collectors.toList());
  760 + deviceTemperatures = notEqualTemperatures.stream().map(aLong -> Long.toString(aLong)).collect(Collectors.toList());
  761 +
  762 + Assert.assertEquals(deviceTemperatures, loadedTemperatures);
  763 +
  764 +
  765 + deviceService.deleteDevicesByTenantId(tenantId);
  766 + }
  767 +
  768 + @Test
  769 + public void testFindEntityDataByQueryWithTimeseries() throws ExecutionException, InterruptedException {
  770 +
  771 + List<Device> devices = new ArrayList<>();
  772 + List<Double> temperatures = new ArrayList<>();
  773 + List<Double> highTemperatures = new ArrayList<>();
  774 + for (int i = 0; i < 67; i++) {
  775 + Device device = new Device();
  776 + device.setTenantId(tenantId);
  777 + device.setName("Device" + i);
  778 + device.setType("default");
  779 + device.setLabel("testLabel" + (int) (Math.random() * 1000));
  780 + devices.add(deviceService.saveDevice(device));
  781 + //TO make sure devices have different created time
  782 + Thread.sleep(1);
  783 + double temperature = (double) (Math.random() * 100.0);
  784 + temperatures.add(temperature);
  785 + if (temperature > 45.0) {
  786 + highTemperatures.add(temperature);
  787 + }
  788 + }
  789 +
  790 + List<ListenableFuture<List<Void>>> timeseriesFutures = new ArrayList<>();
  791 + for (int i = 0; i < devices.size(); i++) {
  792 + Device device = devices.get(i);
  793 + timeseriesFutures.add(saveLongTimeseries(device.getId(), "temperature", temperatures.get(i)));
  794 + }
  795 + Futures.successfulAsList(timeseriesFutures).get();
  796 +
  797 + DeviceTypeFilter filter = new DeviceTypeFilter();
  798 + filter.setDeviceType("default");
  799 + filter.setDeviceNameFilter("");
  800 +
  801 + EntityDataSortOrder sortOrder = new EntityDataSortOrder(
  802 + new EntityKey(EntityKeyType.ENTITY_FIELD, "createdTime"), EntityDataSortOrder.Direction.ASC
  803 + );
562 EntityDataPageLink pageLink = new EntityDataPageLink(10, 0, null, sortOrder); 804 EntityDataPageLink pageLink = new EntityDataPageLink(10, 0, null, sortOrder);
563 List<EntityKey> entityFields = Collections.singletonList(new EntityKey(EntityKeyType.ENTITY_FIELD, "name")); 805 List<EntityKey> entityFields = Collections.singletonList(new EntityKey(EntityKeyType.ENTITY_FIELD, "name"));
564 - List<EntityKey> latestValues = Collections.singletonList(new EntityKey(EntityKeyType.ATTRIBUTE, "temperature")); 806 + List<EntityKey> latestValues = Collections.singletonList(new EntityKey(EntityKeyType.TIME_SERIES, "temperature"));
565 807
566 EntityDataQuery query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, null); 808 EntityDataQuery query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, null);
567 PageData<EntityData> data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query); 809 PageData<EntityData> data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
@@ -576,14 +818,14 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -576,14 +818,14 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
576 List<String> loadedTemperatures = new ArrayList<>(); 818 List<String> loadedTemperatures = new ArrayList<>();
577 for (Device device : devices) { 819 for (Device device : devices) {
578 loadedTemperatures.add(loadedEntities.stream().filter(entityData -> entityData.getEntityId().equals(device.getId())).findFirst().orElse(null) 820 loadedTemperatures.add(loadedEntities.stream().filter(entityData -> entityData.getEntityId().equals(device.getId())).findFirst().orElse(null)
579 - .getLatest().get(EntityKeyType.ATTRIBUTE).get("temperature").getValue()); 821 + .getLatest().get(EntityKeyType.TIME_SERIES).get("temperature").getValue());
580 } 822 }
581 - List<String> deviceTemperatures = temperatures.stream().map(aLong -> Long.toString(aLong)).collect(Collectors.toList()); 823 + List<String> deviceTemperatures = temperatures.stream().map(aDouble -> Double.toString(aDouble)).collect(Collectors.toList());
582 Assert.assertEquals(deviceTemperatures, loadedTemperatures); 824 Assert.assertEquals(deviceTemperatures, loadedTemperatures);
583 825
584 pageLink = new EntityDataPageLink(10, 0, null, sortOrder); 826 pageLink = new EntityDataPageLink(10, 0, null, sortOrder);
585 KeyFilter highTemperatureFilter = new KeyFilter(); 827 KeyFilter highTemperatureFilter = new KeyFilter();
586 - highTemperatureFilter.setKey(new EntityKey(EntityKeyType.ATTRIBUTE, "temperature")); 828 + highTemperatureFilter.setKey(new EntityKey(EntityKeyType.TIME_SERIES, "temperature"));
587 NumericFilterPredicate predicate = new NumericFilterPredicate(); 829 NumericFilterPredicate predicate = new NumericFilterPredicate();
588 predicate.setValue(FilterPredicateValue.fromDouble(45)); 830 predicate.setValue(FilterPredicateValue.fromDouble(45));
589 predicate.setOperation(NumericFilterPredicate.NumericOperation.GREATER); 831 predicate.setOperation(NumericFilterPredicate.NumericOperation.GREATER);
@@ -603,17 +845,425 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest { @@ -603,17 +845,425 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
603 Assert.assertEquals(highTemperatures.size(), loadedEntities.size()); 845 Assert.assertEquals(highTemperatures.size(), loadedEntities.size());
604 846
605 List<String> loadedHighTemperatures = loadedEntities.stream().map(entityData -> 847 List<String> loadedHighTemperatures = loadedEntities.stream().map(entityData ->
606 - entityData.getLatest().get(EntityKeyType.ATTRIBUTE).get("temperature").getValue()).collect(Collectors.toList());  
607 - List<String> deviceHighTemperatures = highTemperatures.stream().map(aLong -> Long.toString(aLong)).collect(Collectors.toList()); 848 + entityData.getLatest().get(EntityKeyType.TIME_SERIES).get("temperature").getValue()).collect(Collectors.toList());
  849 + List<String> deviceHighTemperatures = highTemperatures.stream().map(aDouble -> Double.toString(aDouble)).collect(Collectors.toList());
608 850
609 Assert.assertEquals(deviceHighTemperatures, loadedHighTemperatures); 851 Assert.assertEquals(deviceHighTemperatures, loadedHighTemperatures);
610 852
611 deviceService.deleteDevicesByTenantId(tenantId); 853 deviceService.deleteDevicesByTenantId(tenantId);
612 } 854 }
613 855
  856 + @Test
  857 + public void testBuildStringPredicateQueryOperations() throws ExecutionException, InterruptedException{
  858 +
  859 + List<Device> devices = new ArrayList<>();
  860 + List<String> attributeStrings = new ArrayList<>();
  861 + List<String> equalStrings = new ArrayList<>();
  862 + List<String> notEqualStrings = new ArrayList<>();
  863 + List<String> startsWithStrings = new ArrayList<>();
  864 + List<String> endsWithStrings = new ArrayList<>();
  865 + List<String> containsStrings = new ArrayList<>();
  866 + List<String> notContainsStrings = new ArrayList<>();
  867 +
  868 + for (int i = 0; i < 10; i++) {
  869 + Device device = new Device();
  870 + device.setTenantId(tenantId);
  871 + device.setName("Device" + i);
  872 + device.setType("default");
  873 + device.setLabel("testLabel" + (int) (Math.random() * 1000));
  874 + devices.add(deviceService.saveDevice(device));
  875 + //TO make sure devices have different created time
  876 + Thread.sleep(1);
  877 + List<StringFilterPredicate.StringOperation> operationValues= Arrays.asList(StringFilterPredicate.StringOperation.values());
  878 + StringFilterPredicate.StringOperation operation = operationValues.get(new Random().nextInt(operationValues.size()));
  879 + String operationName = operation.name();
  880 + attributeStrings.add(operationName);
  881 + switch(operation){
  882 + case EQUAL:
  883 + equalStrings.add(operationName);
  884 + notContainsStrings.add(operationName);
  885 + notEqualStrings.add(operationName);
  886 + break;
  887 + case NOT_EQUAL:
  888 + notContainsStrings.add(operationName);
  889 + break;
  890 + case STARTS_WITH:
  891 + notEqualStrings.add(operationName);
  892 + startsWithStrings.add(operationName);
  893 + endsWithStrings.add(operationName);
  894 + notContainsStrings.add(operationName);
  895 + break;
  896 + case ENDS_WITH:
  897 + notEqualStrings.add(operationName);
  898 + endsWithStrings.add(operationName);
  899 + notContainsStrings.add(operationName);
  900 + break;
  901 + case CONTAINS:
  902 + notEqualStrings.add(operationName);
  903 + notContainsStrings.add(operationName);
  904 + containsStrings.add(operationName);
  905 + break;
  906 + case NOT_CONTAINS:
  907 + notEqualStrings.add(operationName);
  908 + containsStrings.add(operationName);
  909 + break;
  910 + }
  911 + }
  912 +
  913 + List<ListenableFuture<List<Void>>> attributeFutures = new ArrayList<>();
  914 + for (int i = 0; i < devices.size(); i++) {
  915 + Device device = devices.get(i);
  916 + attributeFutures.add(saveStringAttribute(device.getId(), "attributeString", attributeStrings.get(i), DataConstants.CLIENT_SCOPE));
  917 + }
  918 + Futures.successfulAsList(attributeFutures).get();
  919 +
  920 + DeviceTypeFilter filter = new DeviceTypeFilter();
  921 + filter.setDeviceType("default");
  922 + filter.setDeviceNameFilter("");
  923 +
  924 + EntityDataSortOrder sortOrder = new EntityDataSortOrder(
  925 + new EntityKey(EntityKeyType.ENTITY_FIELD, "createdTime"), EntityDataSortOrder.Direction.DESC
  926 + );
  927 +
  928 + List<EntityKey> entityFields = Arrays.asList(new EntityKey(EntityKeyType.ENTITY_FIELD, "name"),
  929 + new EntityKey(EntityKeyType.ENTITY_FIELD, "entityType"));
  930 +
  931 + List<EntityKey> latestValues = Collections.singletonList(new EntityKey(EntityKeyType.CLIENT_ATTRIBUTE, "attributeString"));
  932 +
  933 + List<KeyFilter> keyFiltersEqualString = createStringKeyFilters("attributeString", EntityKeyType.CLIENT_ATTRIBUTE, StringFilterPredicate.StringOperation.EQUAL, "equal");
  934 +
  935 + List<KeyFilter> keyFiltersNotEqualString = createStringKeyFilters("attributeString", EntityKeyType.CLIENT_ATTRIBUTE, StringFilterPredicate.StringOperation.NOT_EQUAL, "NOT_EQUAL");
  936 +
  937 + List<KeyFilter> keyFiltersStartsWithString = createStringKeyFilters("attributeString", EntityKeyType.CLIENT_ATTRIBUTE, StringFilterPredicate.StringOperation.STARTS_WITH, "starts_");
  938 +
  939 + List<KeyFilter> keyFiltersEndsWithString = createStringKeyFilters("attributeString", EntityKeyType.CLIENT_ATTRIBUTE, StringFilterPredicate.StringOperation.ENDS_WITH, "_WITH");
  940 +
  941 + List<KeyFilter> keyFiltersContainsString = createStringKeyFilters("attributeString", EntityKeyType.CLIENT_ATTRIBUTE, StringFilterPredicate.StringOperation.CONTAINS, "contains");
  942 +
  943 + List<KeyFilter> keyFiltersNotContainsString = createStringKeyFilters("attributeString", EntityKeyType.CLIENT_ATTRIBUTE, StringFilterPredicate.StringOperation.NOT_CONTAINS, "NOT_CONTAINS");
  944 +
  945 + List<KeyFilter> deviceTypeFilters = createStringKeyFilters("entityType", EntityKeyType.ENTITY_FIELD, StringFilterPredicate.StringOperation.NOT_EQUAL, "NOT_EQUAL");
  946 +
  947 + // Equal Operation
  948 +
  949 + EntityDataPageLink pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  950 + EntityDataQuery query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersEqualString);
  951 + PageData<EntityData> data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  952 + List<EntityData> loadedEntities = getLoadedEntities(data, query);
  953 + Assert.assertEquals(equalStrings.size(), loadedEntities.size());
  954 +
  955 + List<String> loadedStrings = loadedEntities.stream().map(entityData ->
  956 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("attributeString").getValue()).collect(Collectors.toList());
  957 +
  958 + Assert.assertTrue(listEqualWithoutOrder(equalStrings, loadedStrings));
  959 +
  960 + // Not equal Operation
  961 +
  962 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  963 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersNotEqualString);
  964 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  965 + loadedEntities = getLoadedEntities(data, query);
  966 + Assert.assertEquals(notEqualStrings.size(), loadedEntities.size());
  967 +
  968 + loadedStrings = loadedEntities.stream().map(entityData ->
  969 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("attributeString").getValue()).collect(Collectors.toList());
  970 +
  971 + Assert.assertTrue(listEqualWithoutOrder(notEqualStrings, loadedStrings));
  972 +
  973 + // Starts with Operation
  974 +
  975 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  976 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersStartsWithString);
  977 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  978 + loadedEntities = getLoadedEntities(data, query);
  979 + Assert.assertEquals(startsWithStrings.size(), loadedEntities.size());
  980 +
  981 + loadedStrings = loadedEntities.stream().map(entityData ->
  982 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("attributeString").getValue()).collect(Collectors.toList());
  983 +
  984 + Assert.assertTrue(listEqualWithoutOrder(startsWithStrings, loadedStrings));
  985 +
  986 + // Ends with Operation
  987 +
  988 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  989 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersEndsWithString);
  990 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  991 + loadedEntities = getLoadedEntities(data, query);
  992 + Assert.assertEquals(endsWithStrings.size(), loadedEntities.size());
  993 +
  994 + loadedStrings = loadedEntities.stream().map(entityData ->
  995 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("attributeString").getValue()).collect(Collectors.toList());
  996 +
  997 + Assert.assertTrue(listEqualWithoutOrder(endsWithStrings, loadedStrings));
  998 +
  999 + // Contains Operation
  1000 +
  1001 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1002 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersContainsString);
  1003 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1004 + loadedEntities = getLoadedEntities(data, query);
  1005 + Assert.assertEquals(containsStrings.size(), loadedEntities.size());
  1006 +
  1007 + loadedStrings = loadedEntities.stream().map(entityData ->
  1008 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("attributeString").getValue()).collect(Collectors.toList());
  1009 +
  1010 + Assert.assertTrue(listEqualWithoutOrder(containsStrings, loadedStrings));
  1011 +
  1012 + // Not contains Operation
  1013 +
  1014 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1015 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, keyFiltersNotContainsString);
  1016 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1017 + loadedEntities = getLoadedEntities(data, query);
  1018 + Assert.assertEquals(notContainsStrings.size(), loadedEntities.size());
  1019 +
  1020 + loadedStrings = loadedEntities.stream().map(entityData ->
  1021 + entityData.getLatest().get(EntityKeyType.CLIENT_ATTRIBUTE).get("attributeString").getValue()).collect(Collectors.toList());
  1022 +
  1023 + Assert.assertTrue(listEqualWithoutOrder(notContainsStrings, loadedStrings));
  1024 +
  1025 + // Device type filters Operation
  1026 +
  1027 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1028 + query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, deviceTypeFilters);
  1029 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1030 + loadedEntities = getLoadedEntities(data, query);
  1031 + Assert.assertEquals(devices.size(), loadedEntities.size());
  1032 +
  1033 + deviceService.deleteDevicesByTenantId(tenantId);
  1034 + }
  1035 +
  1036 + @Test
  1037 + public void testBuildStringPredicateQueryOperationsForEntityType() throws ExecutionException, InterruptedException{
  1038 +
  1039 + List<Device> devices = new ArrayList<>();
  1040 +
  1041 + for (int i = 0; i < 10; i++) {
  1042 + Device device = new Device();
  1043 + device.setTenantId(tenantId);
  1044 + device.setName("Device" + i);
  1045 + device.setType("default");
  1046 + device.setLabel("testLabel" + (int) (Math.random() * 1000));
  1047 + devices.add(deviceService.saveDevice(device));
  1048 + //TO make sure devices have different created time
  1049 + Thread.sleep(1);
  1050 + }
  1051 +
  1052 + DeviceTypeFilter filter = new DeviceTypeFilter();
  1053 + filter.setDeviceType("default");
  1054 + filter.setDeviceNameFilter("");
  1055 +
  1056 + EntityDataSortOrder sortOrder = new EntityDataSortOrder(
  1057 + new EntityKey(EntityKeyType.ENTITY_FIELD, "createdTime"), EntityDataSortOrder.Direction.DESC
  1058 + );
  1059 +
  1060 + List<EntityKey> entityFields = Arrays.asList(new EntityKey(EntityKeyType.ENTITY_FIELD, "name"),
  1061 + new EntityKey(EntityKeyType.ENTITY_FIELD, "entityType"));
  1062 +
  1063 + List<KeyFilter> keyFiltersEqualString = createStringKeyFilters("entityType", EntityKeyType.ENTITY_FIELD, StringFilterPredicate.StringOperation.EQUAL, "device");
  1064 + List<KeyFilter> keyFiltersNotEqualString = createStringKeyFilters("entityType", EntityKeyType.ENTITY_FIELD, StringFilterPredicate.StringOperation.NOT_EQUAL, "asset");
  1065 + List<KeyFilter> keyFiltersStartsWithString = createStringKeyFilters("entityType", EntityKeyType.ENTITY_FIELD, StringFilterPredicate.StringOperation.STARTS_WITH, "dev");
  1066 + List<KeyFilter> keyFiltersEndsWithString = createStringKeyFilters("entityType", EntityKeyType.ENTITY_FIELD, StringFilterPredicate.StringOperation.ENDS_WITH, "ice");
  1067 + List<KeyFilter> keyFiltersContainsString = createStringKeyFilters("entityType", EntityKeyType.ENTITY_FIELD, StringFilterPredicate.StringOperation.CONTAINS, "vic");
  1068 + List<KeyFilter> keyFiltersNotContainsString = createStringKeyFilters("entityType", EntityKeyType.ENTITY_FIELD, StringFilterPredicate.StringOperation.NOT_CONTAINS, "dolphin");
  1069 +
  1070 + // Equal Operation
  1071 +
  1072 + EntityDataPageLink pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1073 + EntityDataQuery query = new EntityDataQuery(filter, pageLink, entityFields, null, keyFiltersEqualString);
  1074 + PageData<EntityData> data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1075 + List<EntityData> loadedEntities = getLoadedEntities(data, query);
  1076 + Assert.assertEquals(devices.size(), loadedEntities.size());
  1077 +
  1078 + List<String> loadedStrings = loadedEntities.stream().map(entityData ->
  1079 + entityData.getLatest().get(EntityKeyType.ENTITY_FIELD).get("name").getValue()).collect(Collectors.toList());
  1080 +
  1081 + List<String> devicesNames = devices.stream().map(Device::getName).collect(Collectors.toList());
  1082 +
  1083 + Assert.assertTrue(listEqualWithoutOrder(devicesNames, loadedStrings));
  1084 +
  1085 + // Not equal Operation
  1086 +
  1087 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1088 + query = new EntityDataQuery(filter, pageLink, entityFields, null, keyFiltersNotEqualString);
  1089 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1090 + loadedEntities = getLoadedEntities(data, query);
  1091 + Assert.assertEquals(devices.size(), loadedEntities.size());
  1092 +
  1093 + loadedStrings = loadedEntities.stream().map(entityData ->
  1094 + entityData.getLatest().get(EntityKeyType.ENTITY_FIELD).get("name").getValue()).collect(Collectors.toList());
  1095 +
  1096 + Assert.assertTrue(listEqualWithoutOrder(devicesNames, loadedStrings));
  1097 +
  1098 + // Starts with Operation
  1099 +
  1100 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1101 + query = new EntityDataQuery(filter, pageLink, entityFields, null, keyFiltersStartsWithString);
  1102 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1103 + loadedEntities = getLoadedEntities(data, query);
  1104 + Assert.assertEquals(devices.size(), loadedEntities.size());
  1105 +
  1106 + loadedStrings = loadedEntities.stream().map(entityData ->
  1107 + entityData.getLatest().get(EntityKeyType.ENTITY_FIELD).get("name").getValue()).collect(Collectors.toList());
  1108 +
  1109 + Assert.assertTrue(listEqualWithoutOrder(devicesNames, loadedStrings));
  1110 +
  1111 + // Ends with Operation
  1112 +
  1113 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1114 + query = new EntityDataQuery(filter, pageLink, entityFields, null, keyFiltersEndsWithString);
  1115 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1116 + loadedEntities = getLoadedEntities(data, query);
  1117 + Assert.assertEquals(devices.size(), loadedEntities.size());
  1118 +
  1119 + loadedStrings = loadedEntities.stream().map(entityData ->
  1120 + entityData.getLatest().get(EntityKeyType.ENTITY_FIELD).get("name").getValue()).collect(Collectors.toList());
  1121 +
  1122 + Assert.assertTrue(listEqualWithoutOrder(devicesNames, loadedStrings));
  1123 +
  1124 + // Contains Operation
  1125 +
  1126 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1127 + query = new EntityDataQuery(filter, pageLink, entityFields, null, keyFiltersContainsString);
  1128 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1129 + loadedEntities = getLoadedEntities(data, query);
  1130 + Assert.assertEquals(devices.size(), loadedEntities.size());
  1131 +
  1132 + loadedStrings = loadedEntities.stream().map(entityData ->
  1133 + entityData.getLatest().get(EntityKeyType.ENTITY_FIELD).get("name").getValue()).collect(Collectors.toList());
  1134 +
  1135 + Assert.assertTrue(listEqualWithoutOrder(devicesNames, loadedStrings));
  1136 +
  1137 + // Not contains Operation
  1138 +
  1139 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1140 + query = new EntityDataQuery(filter, pageLink, entityFields, null, keyFiltersNotContainsString);
  1141 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1142 + loadedEntities = getLoadedEntities(data, query);
  1143 + Assert.assertEquals(devices.size(), loadedEntities.size());
  1144 +
  1145 + loadedStrings = loadedEntities.stream().map(entityData ->
  1146 + entityData.getLatest().get(EntityKeyType.ENTITY_FIELD).get("name").getValue()).collect(Collectors.toList());
  1147 +
  1148 + Assert.assertTrue(listEqualWithoutOrder(devicesNames, loadedStrings));
  1149 +
  1150 + deviceService.deleteDevicesByTenantId(tenantId);
  1151 + }
  1152 +
  1153 + @Test
  1154 + public void testBuildSimplePredicateQueryOperations() throws InterruptedException{
  1155 +
  1156 + List<Device> devices = new ArrayList<>();
  1157 +
  1158 + for (int i = 0; i < 10; i++) {
  1159 + Device device = new Device();
  1160 + device.setTenantId(tenantId);
  1161 + device.setName("Device" + i);
  1162 + device.setType("default");
  1163 + device.setLabel("testLabel" + (int) (Math.random() * 1000));
  1164 + devices.add(deviceService.saveDevice(device));
  1165 + //TO make sure devices have different created time
  1166 + Thread.sleep(1);
  1167 + }
  1168 +
  1169 + DeviceTypeFilter filter = new DeviceTypeFilter();
  1170 + filter.setDeviceType("default");
  1171 + filter.setDeviceNameFilter("");
  1172 +
  1173 + EntityDataSortOrder sortOrder = new EntityDataSortOrder(new EntityKey(EntityKeyType.ENTITY_FIELD, "name"), EntityDataSortOrder.Direction.DESC);
  1174 +
  1175 + List<KeyFilter> deviceTypeFilters = createStringKeyFilters("type", EntityKeyType.ENTITY_FIELD, StringFilterPredicate.StringOperation.EQUAL, "default");
  1176 +
  1177 + KeyFilter createdTimeFilter = createNumericKeyFilter("createdTime", EntityKeyType.ENTITY_FIELD, NumericFilterPredicate.NumericOperation.GREATER, 1L);
  1178 + List<KeyFilter> createdTimeFilters = Collections.singletonList(createdTimeFilter);
  1179 +
  1180 + List<KeyFilter> nameFilters = createStringKeyFilters("name", EntityKeyType.ENTITY_FIELD, StringFilterPredicate.StringOperation.CONTAINS, "Device");
  1181 +
  1182 + List<EntityKey> entityFields = Arrays.asList(new EntityKey(EntityKeyType.ENTITY_FIELD, "name"),
  1183 + new EntityKey(EntityKeyType.ENTITY_FIELD, "type"));
  1184 +
  1185 + // Device type filters
  1186 +
  1187 + EntityDataPageLink pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1188 + EntityDataQuery query = new EntityDataQuery(filter, pageLink, entityFields, null, deviceTypeFilters);
  1189 + PageData data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1190 + List<EntityData> loadedEntities = getLoadedEntities(data, query);
  1191 + Assert.assertEquals(devices.size(), loadedEntities.size());
  1192 +
  1193 + // Device create time filters
  1194 +
  1195 + pageLink = new EntityDataPageLink(100, 0, null, sortOrder);
  1196 + query = new EntityDataQuery(filter, pageLink, entityFields, null, createdTimeFilters);
  1197 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1198 + loadedEntities = getLoadedEntities(data, query);
  1199 + Assert.assertEquals(devices.size(), loadedEntities.size());
  1200 +
  1201 + // Device name filters
  1202 +
  1203 + pageLink = new EntityDataPageLink(100, 0, null, null);
  1204 + query = new EntityDataQuery(filter, pageLink, entityFields, null, nameFilters);
  1205 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1206 + loadedEntities = getLoadedEntities(data, query);
  1207 + Assert.assertEquals(devices.size(), loadedEntities.size());
  1208 +
  1209 + deviceService.deleteDevicesByTenantId(tenantId);
  1210 + }
  1211 +
  1212 + private Boolean listEqualWithoutOrder(List<String> A, List<String> B) {
  1213 + return A.containsAll(B) && B.containsAll(A);
  1214 + }
  1215 +
  1216 + private List<EntityData> getLoadedEntities(PageData data, EntityDataQuery query) {
  1217 + List<EntityData> loadedEntities = new ArrayList<>(data.getData());
  1218 +
  1219 + while (data.hasNext()) {
  1220 + query = query.next();
  1221 + data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
  1222 + loadedEntities.addAll(data.getData());
  1223 + }
  1224 + return loadedEntities;
  1225 + }
  1226 +
  1227 + private List<KeyFilter> createStringKeyFilters(String key, EntityKeyType keyType, StringFilterPredicate.StringOperation operation, String value){
  1228 + KeyFilter filter = new KeyFilter();
  1229 + filter.setKey(new EntityKey(keyType, key));
  1230 + StringFilterPredicate predicate = new StringFilterPredicate();
  1231 + predicate.setValue(FilterPredicateValue.fromString(value));
  1232 + predicate.setOperation(operation);
  1233 + predicate.setIgnoreCase(true);
  1234 + filter.setPredicate(predicate);
  1235 + return Collections.singletonList(filter);
  1236 + }
  1237 +
  1238 + private KeyFilter createNumericKeyFilter(String key, EntityKeyType keyType, NumericFilterPredicate.NumericOperation operation, double value){
  1239 + KeyFilter filter = new KeyFilter();
  1240 + filter.setKey(new EntityKey(keyType, key));
  1241 + NumericFilterPredicate predicate = new NumericFilterPredicate();
  1242 + predicate.setValue(FilterPredicateValue.fromDouble(value));
  1243 + predicate.setOperation(operation);
  1244 + filter.setPredicate(predicate);
  1245 +
  1246 + return filter;
  1247 + }
  1248 +
614 private ListenableFuture<List<Void>> saveLongAttribute(EntityId entityId, String key, long value, String scope) { 1249 private ListenableFuture<List<Void>> saveLongAttribute(EntityId entityId, String key, long value, String scope) {
615 KvEntry attrValue = new LongDataEntry(key, value); 1250 KvEntry attrValue = new LongDataEntry(key, value);
616 AttributeKvEntry attr = new BaseAttributeKvEntry(attrValue, 42L); 1251 AttributeKvEntry attr = new BaseAttributeKvEntry(attrValue, 42L);
617 return attributesService.save(SYSTEM_TENANT_ID, entityId, scope, Collections.singletonList(attr)); 1252 return attributesService.save(SYSTEM_TENANT_ID, entityId, scope, Collections.singletonList(attr));
618 } 1253 }
  1254 +
  1255 + private ListenableFuture<List<Void>> saveStringAttribute(EntityId entityId, String key, String value, String scope) {
  1256 + KvEntry attrValue = new StringDataEntry(key, value);
  1257 + AttributeKvEntry attr = new BaseAttributeKvEntry(attrValue, 42L);
  1258 + return attributesService.save(SYSTEM_TENANT_ID, entityId, scope, Collections.singletonList(attr));
  1259 + }
  1260 +
  1261 + private ListenableFuture<List<Void>> saveLongTimeseries(EntityId entityId, String key, Double value) {
  1262 + TsKvEntity tsKv = new TsKvEntity();
  1263 + tsKv.setStrKey(key);
  1264 + tsKv.setDoubleValue(value);
  1265 + KvEntry telemetryValue = new DoubleDataEntry(key, value);
  1266 + BasicTsKvEntry timeseries = new BasicTsKvEntry(42L, telemetryValue);
  1267 + return timeseriesService.save(SYSTEM_TENANT_ID, entityId, timeseries);
  1268 + }
619 } 1269 }
@@ -75,7 +75,7 @@ export class AddAttributeDialogComponent extends DialogComponent<AddAttributeDia @@ -75,7 +75,7 @@ export class AddAttributeDialogComponent extends DialogComponent<AddAttributeDia
75 this.submitted = true; 75 this.submitted = true;
76 const attribute: AttributeData = { 76 const attribute: AttributeData = {
77 lastUpdateTs: null, 77 lastUpdateTs: null,
78 - key: this.attributeFormGroup.get('key').value, 78 + key: this.attributeFormGroup.get('key').value.trim(),
79 value: this.attributeFormGroup.get('value').value 79 value: this.attributeFormGroup.get('value').value
80 }; 80 };
81 this.attributeService.saveEntityAttributes(this.data.entityId, 81 this.attributeService.saveEntityAttributes(this.data.entityId,
@@ -129,7 +129,7 @@ export abstract class EntityComponent<T extends BaseData<HasId>, @@ -129,7 +129,7 @@ export abstract class EntityComponent<T extends BaseData<HasId>,
129 acc[curr] = obj[curr]; 129 acc[curr] = obj[curr];
130 } 130 }
131 return acc; 131 return acc;
132 - }, {}); 132 + }, Array.isArray(obj) ? [] : {});
133 } 133 }
134 134
135 protected setEntitiesTableConfig(entitiesTableConfig: C) { 135 protected setEntitiesTableConfig(entitiesTableConfig: C) {
@@ -29,13 +29,6 @@ @@ -29,13 +29,6 @@
29 <fieldset [disabled]="isLoading$ | async" fxLayout="column"> 29 <fieldset [disabled]="isLoading$ | async" fxLayout="column">
30 <section fxLayout="row" fxLayoutGap="8px" class="entity-key"> 30 <section fxLayout="row" fxLayoutGap="8px" class="entity-key">
31 <section fxFlex="70" fxLayout="row" formGroupName="key" fxLayoutGap="8px"> 31 <section fxFlex="70" fxLayout="row" formGroupName="key" fxLayoutGap="8px">
32 - <mat-form-field fxFlex="60" class="mat-block">  
33 - <mat-label translate>filter.key-name</mat-label>  
34 - <input matInput required formControlName="key">  
35 - <mat-error *ngIf="keyFilterFormGroup.get('key.key').hasError('required')">  
36 - {{ 'filter.key-name-required' | translate }}  
37 - </mat-error>  
38 - </mat-form-field>  
39 <mat-form-field fxFlex="40" class="mat-block"> 32 <mat-form-field fxFlex="40" class="mat-block">
40 <mat-label translate>filter.key-type.key-type</mat-label> 33 <mat-label translate>filter.key-type.key-type</mat-label>
41 <mat-select required formControlName="type"> 34 <mat-select required formControlName="type">
@@ -44,10 +37,24 @@ @@ -44,10 +37,24 @@
44 </mat-option> 37 </mat-option>
45 </mat-select> 38 </mat-select>
46 </mat-form-field> 39 </mat-form-field>
  40 + <mat-form-field fxFlex="60" class="mat-block">
  41 + <mat-label translate>filter.key-name</mat-label>
  42 + <input matInput required formControlName="key"
  43 + [matAutocomplete]="auto"
  44 + [matAutocompleteDisabled]="keyFilterFormGroup.get('key.type').value !== entityField">
  45 + <mat-autocomplete autoActiveFirstOption #auto="matAutocomplete">
  46 + <mat-option *ngFor="let option of filteredEntityFields | async" [value]="option">
  47 + {{option}}
  48 + </mat-option>
  49 + </mat-autocomplete>
  50 + <mat-error *ngIf="keyFilterFormGroup.get('key.key').hasError('required')">
  51 + {{ 'filter.key-name-required' | translate }}
  52 + </mat-error>
  53 + </mat-form-field>
47 </section> 54 </section>
48 <mat-form-field fxFlex="30" class="mat-block"> 55 <mat-form-field fxFlex="30" class="mat-block">
49 <mat-label translate>filter.value-type.value-type</mat-label> 56 <mat-label translate>filter.value-type.value-type</mat-label>
50 - <mat-select matInput formControlName="valueType"> 57 + <mat-select formControlName="valueType">
51 <mat-select-trigger> 58 <mat-select-trigger>
52 <mat-icon class="tb-mat-18" svgIcon="{{ entityKeyValueTypes.get(keyFilterFormGroup.get('valueType').value)?.icon }}"></mat-icon> 59 <mat-icon class="tb-mat-18" svgIcon="{{ entityKeyValueTypes.get(keyFilterFormGroup.get('valueType').value)?.icon }}"></mat-icon>
53 <span>{{ entityKeyValueTypes.get(keyFilterFormGroup.get('valueType').value)?.name | translate }}</span> 60 <span>{{ entityKeyValueTypes.get(keyFilterFormGroup.get('valueType').value)?.name | translate }}</span>
@@ -27,10 +27,14 @@ import { @@ -27,10 +27,14 @@ import {
27 entityKeyTypeTranslationMap, 27 entityKeyTypeTranslationMap,
28 EntityKeyValueType, 28 EntityKeyValueType,
29 entityKeyValueTypesMap, 29 entityKeyValueTypesMap,
30 - KeyFilterInfo, KeyFilterPredicate 30 + KeyFilterInfo,
  31 + KeyFilterPredicate
31 } from '@shared/models/query/query.models'; 32 } from '@shared/models/query/query.models';
32 import { DialogService } from '@core/services/dialog.service'; 33 import { DialogService } from '@core/services/dialog.service';
33 import { TranslateService } from '@ngx-translate/core'; 34 import { TranslateService } from '@ngx-translate/core';
  35 +import { EntityField, entityFields } from '@shared/models/entity.models';
  36 +import { Observable } from 'rxjs';
  37 +import { filter, map, startWith } from 'rxjs/operators';
34 38
35 export interface KeyFilterDialogData { 39 export interface KeyFilterDialogData {
36 keyFilter: KeyFilterInfo; 40 keyFilter: KeyFilterInfo;
@@ -61,6 +65,14 @@ export class KeyFilterDialogComponent extends @@ -61,6 +65,14 @@ export class KeyFilterDialogComponent extends
61 65
62 submitted = false; 66 submitted = false;
63 67
  68 + entityFields: { [fieldName: string]: EntityField };
  69 +
  70 + entityFieldsList: string[];
  71 +
  72 + readonly entityField = EntityKeyType.ENTITY_FIELD;
  73 +
  74 + filteredEntityFields: Observable<string[]>;
  75 +
64 constructor(protected store: Store<AppState>, 76 constructor(protected store: Store<AppState>,
65 protected router: Router, 77 protected router: Router,
66 @Inject(MAT_DIALOG_DATA) public data: KeyFilterDialogData, 78 @Inject(MAT_DIALOG_DATA) public data: KeyFilterDialogData,
@@ -99,9 +111,28 @@ export class KeyFilterDialogComponent extends @@ -99,9 +111,28 @@ export class KeyFilterDialogComponent extends
99 ); 111 );
100 } 112 }
101 }); 113 });
  114 +
  115 + this.keyFilterFormGroup.get('key.key').valueChanges.pipe(
  116 + filter((keyName) => this.keyFilterFormGroup.get('key.type').value === this.entityField && this.entityFields.hasOwnProperty(keyName))
  117 + ).subscribe((keyName: string) => {
  118 + const prevValueType: EntityKeyValueType = this.keyFilterFormGroup.value.valueType;
  119 + const newValueType = this.entityFields[keyName]?.time ? EntityKeyValueType.DATE_TIME : EntityKeyValueType.STRING;
  120 + if (prevValueType !== newValueType) {
  121 + this.keyFilterFormGroup.get('valueType').patchValue(newValueType, {emitEvent: false});
  122 + }
  123 + });
  124 +
  125 + this.entityFields = entityFields;
  126 + this.entityFieldsList = Object.values(entityFields).map(entityField => entityField.keyName).sort();
102 } 127 }
103 128
104 ngOnInit(): void { 129 ngOnInit(): void {
  130 + this.filteredEntityFields = this.keyFilterFormGroup.get('key.key').valueChanges.pipe(
  131 + startWith(''),
  132 + map(value => {
  133 + return this.entityFieldsList.filter(option => option.startsWith(value));
  134 + })
  135 + );
105 } 136 }
106 137
107 isErrorState(control: FormControl | null, form: FormGroupDirective | NgForm | null): boolean { 138 isErrorState(control: FormControl | null, form: FormGroupDirective | NgForm | null): boolean {
@@ -118,16 +118,18 @@ @@ -118,16 +118,18 @@
118 </mat-form-field> 118 </mat-form-field>
119 <mat-form-field class="mat-block"> 119 <mat-form-field class="mat-block">
120 <mat-label translate>admin.proxy-password</mat-label> 120 <mat-label translate>admin.proxy-password</mat-label>
121 - <input matInput formControlName="proxyPassword"> 121 + <input matInput type="password" formControlName="proxyPassword" autocomplete="new-proxy-password">
122 </mat-form-field> 122 </mat-form-field>
123 </div> 123 </div>
124 <mat-form-field class="mat-block"> 124 <mat-form-field class="mat-block">
125 <mat-label translate>common.username</mat-label> 125 <mat-label translate>common.username</mat-label>
126 - <input matInput formControlName="username" placeholder="{{ 'common.enter-username' | translate }}"/> 126 + <input matInput formControlName="username" placeholder="{{ 'common.enter-username' | translate }}"
  127 + autocomplete="new-username"/>
127 </mat-form-field> 128 </mat-form-field>
128 <mat-form-field class="mat-block"> 129 <mat-form-field class="mat-block">
129 <mat-label translate>common.password</mat-label> 130 <mat-label translate>common.password</mat-label>
130 - <input matInput formControlName="password" type="password" placeholder="{{ 'common.enter-password' | translate }}"/> 131 + <input matInput formControlName="password" type="password"
  132 + placeholder="{{ 'common.enter-password' | translate }}" autocomplete="new-password"/>
131 </mat-form-field> 133 </mat-form-field>
132 <div fxLayout="row" fxLayoutAlign="end center" fxLayout.xs="column" fxLayoutAlign.xs="end" fxLayoutGap="16px"> 134 <div fxLayout="row" fxLayoutAlign="end center" fxLayout.xs="column" fxLayoutAlign.xs="end" fxLayoutGap="16px">
133 <button mat-raised-button type="button" 135 <button mat-raised-button type="button"
@@ -92,7 +92,7 @@ export class MailServerComponent extends PageComponent implements OnInit, HasCon @@ -92,7 +92,7 @@ export class MailServerComponent extends PageComponent implements OnInit, HasCon
92 } 92 }
93 93
94 enableProxyChanged(): void { 94 enableProxyChanged(): void {
95 - let enableProxy: boolean = this.mailSettings.get('enableProxy').value; 95 + const enableProxy: boolean = this.mailSettings.get('enableProxy').value;
96 if (enableProxy) { 96 if (enableProxy) {
97 this.mailSettings.get('proxyHost').enable(); 97 this.mailSettings.get('proxyHost').enable();
98 this.mailSettings.get('proxyPort').enable(); 98 this.mailSettings.get('proxyPort').enable();
@@ -203,7 +203,7 @@ export enum StringOperation { @@ -203,7 +203,7 @@ export enum StringOperation {
203 STARTS_WITH = 'STARTS_WITH', 203 STARTS_WITH = 'STARTS_WITH',
204 ENDS_WITH = 'ENDS_WITH', 204 ENDS_WITH = 'ENDS_WITH',
205 CONTAINS = 'CONTAINS', 205 CONTAINS = 'CONTAINS',
206 - NOT_CONTAIN = 'NOT_CONTAIN' 206 + NOT_CONTAINS = 'NOT_CONTAINS'
207 } 207 }
208 208
209 export const stringOperationTranslationMap = new Map<StringOperation, string>( 209 export const stringOperationTranslationMap = new Map<StringOperation, string>(
@@ -213,7 +213,7 @@ export const stringOperationTranslationMap = new Map<StringOperation, string>( @@ -213,7 +213,7 @@ export const stringOperationTranslationMap = new Map<StringOperation, string>(
213 [StringOperation.STARTS_WITH, 'filter.operation.starts-with'], 213 [StringOperation.STARTS_WITH, 'filter.operation.starts-with'],
214 [StringOperation.ENDS_WITH, 'filter.operation.ends-with'], 214 [StringOperation.ENDS_WITH, 'filter.operation.ends-with'],
215 [StringOperation.CONTAINS, 'filter.operation.contains'], 215 [StringOperation.CONTAINS, 'filter.operation.contains'],
216 - [StringOperation.NOT_CONTAIN, 'filter.operation.not-contain'] 216 + [StringOperation.NOT_CONTAINS, 'filter.operation.not-contains']
217 ] 217 ]
218 ); 218 );
219 219
@@ -1189,7 +1189,7 @@ @@ -1189,7 +1189,7 @@
1189 "starts-with": "starts with", 1189 "starts-with": "starts with",
1190 "ends-with": "ends with", 1190 "ends-with": "ends with",
1191 "contains": "contains", 1191 "contains": "contains",
1192 - "not-contain": "not contain", 1192 + "not-contains": "not contains",
1193 "greater": "greater than", 1193 "greater": "greater than",
1194 "less": "less than", 1194 "less": "less than",
1195 "greater-or-equal": "greater or equal", 1195 "greater-or-equal": "greater or equal",