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 6 },
7 7 "widgetTypes": [
8 8 {
9   - "alias": "openstreetmap",
10   - "name": "OpenStreetMap",
  9 + "alias": "route_map_tencent_maps",
  10 + "name": "Route Map - Tencent Maps",
11 11 "descriptor": {
12   - "type": "latest",
  12 + "type": "timeseries",
13 13 "sizeX": 8.5,
14 14 "sizeY": 6,
15 15 "resources": [],
16 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 19 "settingsSchema": "{}",
20 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 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 43 "descriptor": {
44 44 "type": "timeseries",
45 45 "sizeX": 8.5,
... ... @@ -47,10 +47,10 @@
47 47 "resources": [],
48 48 "templateHtml": "",
49 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 51 "settingsSchema": "{}",
52 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 70 }
71 71 },
72 72 {
73   - "alias": "route_map",
74   - "name": "Route Map",
  73 + "alias": "tencent_maps",
  74 + "name": "Tencent Maps",
75 75 "descriptor": {
76   - "type": "timeseries",
77   - "sizeX": 8.5,
  76 + "type": "latest",
  77 + "sizeX": 9,
78 78 "sizeY": 6,
79 79 "resources": [],
80 80 "templateHtml": "",
81 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 83 "settingsSchema": "{}",
84 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 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 105 "alias": "image_map",
122 106 "name": "Image Map",
123 107 "descriptor": {
... ... @@ -148,6 +132,22 @@
148 132 "dataKeySettingsSchema": "{}\n",
149 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   -}
\ No newline at end of file
  153 +}
... ...
... ... @@ -86,6 +86,7 @@ BEGIN
86 86 DROP INDEX IF EXISTS idx_asset_customer_id;
87 87 DROP INDEX IF EXISTS idx_asset_customer_id_and_type;
88 88 DROP INDEX IF EXISTS idx_asset_type;
  89 + DROP INDEX IF EXISTS idx_attribute_kv_by_key_and_last_update_ts;
89 90 END;
90 91 $$;
91 92
... ... @@ -105,6 +106,7 @@ BEGIN
105 106 CREATE INDEX IF NOT EXISTS idx_asset_customer_id ON asset(tenant_id, customer_id);
106 107 CREATE INDEX IF NOT EXISTS idx_asset_customer_id_and_type ON asset(tenant_id, customer_id, type);
107 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 110 END;
109 111 $$;
110 112
... ...
... ... @@ -39,6 +39,7 @@ import org.thingsboard.server.common.data.query.EntityDataQuery;
39 39 import org.thingsboard.server.common.data.query.EntityDataSortOrder;
40 40 import org.thingsboard.server.common.data.query.EntityFilter;
41 41 import org.thingsboard.server.common.data.query.EntityFilterType;
  42 +import org.thingsboard.server.common.data.query.EntityKeyType;
42 43 import org.thingsboard.server.common.data.query.EntityListFilter;
43 44 import org.thingsboard.server.common.data.query.EntityNameFilter;
44 45 import org.thingsboard.server.common.data.query.EntitySearchQueryFilter;
... ... @@ -200,6 +201,9 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
200 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 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 208 " SELECT from_id, from_type, to_id, to_type, relation_type, 1 as lvl" +
205 209 " FROM relation" +
... ... @@ -265,7 +269,8 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
265 269
266 270
267 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 274 String whereClause = DefaultEntityQueryRepository.this.buildWhere(ctx, latestFiltersMapping, query.getEntityFilter().getType());
270 275 String textSearchQuery = DefaultEntityQueryRepository.this.buildTextSearchQuery(ctx, selectionMapping, pageLink.getTextSearch());
271 276 String entityFieldsSelection = EntityKeyMapping.buildSelections(entityFieldsSelectionMapping, query.getEntityFilter().getType(), entityType);
... ... @@ -286,29 +291,44 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
286 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 304 topSelection,
291 305 entityFieldsSelection,
292 306 addEntityTableQuery(ctx, query.getEntityFilter()),
293 307 entityWhereClause,
294   - latestJoins,
  308 + latestJoinsData,
295 309 whereClause,
296 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 321 EntityDataSortOrder sortOrder = pageLink.getSortOrder();
303 322 if (sortOrder != null) {
304 323 Optional<EntityKeyMapping> sortOrderMappingOpt = mappings.stream().filter(EntityKeyMapping::isSortOrder).findFirst();
305 324 if (sortOrderMappingOpt.isPresent()) {
306 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 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 349 String entityFieldsQuery = EntityKeyMapping.buildQuery(ctx, entityFieldsFilters, entityFilter.getType());
330 350 String result = permissionQuery;
331 351 if (!entityFilterQuery.isEmpty()) {
332   - result += " and " + entityFilterQuery;
  352 + result += " and (" + entityFilterQuery + ")";
333 353 }
334 354 if (!entityFieldsQuery.isEmpty()) {
335   - result += " and " + entityFieldsQuery;
  355 + result += " and (" + entityFieldsQuery + ")";
336 356 }
337 357 return result;
338 358 }
... ... @@ -415,7 +435,12 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
415 435 String lvlFilter = getLvlFilter(entityFilter.getMaxLevel());
416 436 String selectFields = "SELECT tenant_id, customer_id, id, created_time, type, name, label FROM " + entityType.name() + " WHERE id in ( SELECT entity_id";
417 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 445 from = String.format(from, lvlFilter, whereFilter);
421 446 String query = "( " + selectFields + from + ")";
... ... @@ -426,7 +451,6 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
426 451 query += " )";
427 452 ctx.addUuidParameter("relation_root_id", rootId.getId());
428 453 ctx.addStringParameter("relation_root_type", rootId.getEntityType().name());
429   - ctx.addStringParameter("where_relation_type", entityFilter.getRelationType());
430 454 ctx.addStringParameter("where_entity_type", entityType.name());
431 455 return query;
432 456 }
... ... @@ -446,46 +470,69 @@ public class DefaultEntityQueryRepository implements EntityQueryRepository {
446 470 ctx.addUuidParameter("relation_root_id", rootId.getId());
447 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 476 if (entityFilter.getFilters() != null && !entityFilter.getFilters().isEmpty()) {
451   - whereFilter = new StringBuilder(" WHERE ");
452   - boolean first = true;
453 477 boolean single = entityFilter.getFilters().size() == 1;
454 478 int entityTypeFilterIdx = 0;
455 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 504 from = String.format(from, lvlFilter, whereFilter);
486 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 536 private String getLvlFilter(int maxLevel) {
490 537 return maxLevel > 0 ? ("and lvl <= " + (maxLevel - 1)) : "";
491 538 }
... ...
... ... @@ -270,9 +270,10 @@ public class EntityKeyMapping {
270 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 279 public static String buildQuery(QueryContext ctx, List<EntityKeyMapping> mappings, EntityFilterType filterType) {
... ... @@ -363,19 +364,14 @@ public class EntityKeyMapping {
363 364 }
364 365
365 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 370 private String buildTimeSeriesSelection() {
  371 + return buildTimeSeriesOrAttrSelection(false);
  372 + }
  373 +
  374 + private String buildTimeSeriesOrAttrSelection(boolean attr) {
379 375 String attrValAlias = getValueAlias();
380 376 String attrTsAlias = getTsAlias();
381 377 String attrValSelection =
... ... @@ -384,8 +380,25 @@ public class EntityKeyMapping {
384 380 "coalesce(cast(%s.long_v as varchar), '') || " +
385 381 "coalesce(cast(%s.dbl_v as varchar), '') || " +
386 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 404 private String buildKeyQuery(QueryContext ctx, String alias, KeyFilter keyFilter,
... ... @@ -425,7 +438,14 @@ public class EntityKeyMapping {
425 438 if (predicate.getType().equals(FilterPredicateType.NUMERIC)) {
426 439 return this.buildNumericPredicateQuery(ctx, field, (NumericFilterPredicate) predicate);
427 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 449 } else {
430 450 return this.buildBooleanPredicateQuery(ctx, field, (BooleanFilterPredicate) predicate);
431 451 }
... ... @@ -460,30 +480,34 @@ public class EntityKeyMapping {
460 480 }
461 481 switch (stringFilterPredicate.getOperation()) {
462 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 484 break;
465 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 487 break;
468 488 case STARTS_WITH:
469 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 491 break;
472 492 case ENDS_WITH:
473 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 495 break;
476 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 501 break;
480 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 507 break;
484 508 }
485 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 513 private String buildNumericPredicateQuery(QueryContext ctx, String field, NumericFilterPredicate numericFilterPredicate) {
... ...
... ... @@ -15,6 +15,7 @@
15 15 */
16 16 package org.thingsboard.server.dao.sql.query;
17 17
  18 +import lombok.extern.slf4j.Slf4j;
18 19 import org.hibernate.type.PostgresUUIDType;
19 20 import org.springframework.jdbc.core.namedparam.SqlParameterSource;
20 21 import org.thingsboard.server.common.data.EntityType;
... ... @@ -27,6 +28,7 @@ import java.util.List;
27 28 import java.util.Map;
28 29 import java.util.UUID;
29 30
  31 +@Slf4j
30 32 public class QueryContext implements SqlParameterSource {
31 33 private static final PostgresUUIDType UUID_TYPE = new PostgresUUIDType();
32 34
... ... @@ -41,10 +43,14 @@ public class QueryContext implements SqlParameterSource {
41 43 }
42 44
43 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 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 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 36
37 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);
\ No newline at end of file
  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);
\ No newline at end of file
... ...
... ... @@ -15,12 +15,14 @@
15 15 */
16 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 21 import com.google.common.util.concurrent.Futures;
19 22 import com.google.common.util.concurrent.ListenableFuture;
20 23 import org.junit.After;
21 24 import org.junit.Assert;
22 25 import org.junit.Before;
23   -import org.junit.Ignore;
24 26 import org.junit.Test;
25 27 import org.springframework.beans.factory.annotation.Autowired;
26 28 import org.thingsboard.server.common.data.DataConstants;
... ... @@ -28,42 +30,23 @@ import org.thingsboard.server.common.data.Device;
28 30 import org.thingsboard.server.common.data.EntityType;
29 31 import org.thingsboard.server.common.data.Tenant;
30 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 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 37 import org.thingsboard.server.common.data.relation.EntityRelation;
57 38 import org.thingsboard.server.common.data.relation.EntitySearchDirection;
58 39 import org.thingsboard.server.common.data.relation.EntityTypeFilter;
59 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 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 50 import java.util.concurrent.ExecutionException;
68 51 import java.util.stream.Collectors;
69 52
... ... @@ -72,6 +55,9 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
72 55 @Autowired
73 56 private AttributesService attributesService;
74 57
  58 + @Autowired
  59 + private TimeseriesService timeseriesService;
  60 +
75 61 private TenantId tenantId;
76 62
77 63 @Before
... ... @@ -88,6 +74,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
88 74 tenantService.deleteTenant(tenantId);
89 75 }
90 76
  77 +
91 78 @Test
92 79 public void testCountEntitiesByQuery() throws InterruptedException {
93 80 List<Device> devices = new ArrayList<>();
... ... @@ -131,6 +118,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
131 118 Assert.assertEquals(0, count);
132 119 }
133 120
  121 +
134 122 @Test
135 123 public void testCountHierarchicalEntitiesByQuery() throws InterruptedException {
136 124 List<Asset> assets = new ArrayList<>();
... ... @@ -195,6 +183,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
195 183 Assert.assertEquals(0, count);
196 184 }
197 185
  186 +
198 187 @Test
199 188 public void testHierarchicalFindEntityDataWithAttributesByQuery() throws ExecutionException, InterruptedException {
200 189 List<Asset> assets = new ArrayList<>();
... ... @@ -266,6 +255,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
266 255 deviceService.deleteDevicesByTenantId(tenantId);
267 256 }
268 257
  258 +
269 259 @Test
270 260 public void testHierarchicalFindDevicesWithAttributesByQuery() throws ExecutionException, InterruptedException {
271 261 List<Asset> assets = new ArrayList<>();
... ... @@ -338,6 +328,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
338 328 deviceService.deleteDevicesByTenantId(tenantId);
339 329 }
340 330
  331 +
341 332 @Test
342 333 public void testHierarchicalFindAssetsWithAttributesByQuery() throws ExecutionException, InterruptedException {
343 334 List<Asset> assets = new ArrayList<>();
... ... @@ -456,6 +447,7 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
456 447 }
457 448 }
458 449
  450 +
459 451 @Test
460 452 public void testSimpleFindEntityDataByQuery() throws InterruptedException {
461 453 List<Device> devices = new ArrayList<>();
... ... @@ -526,6 +518,8 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
526 518 @Test
527 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 523 List<Device> devices = new ArrayList<>();
530 524 List<Long> temperatures = new ArrayList<>();
531 525 List<Long> highTemperatures = new ArrayList<>();
... ... @@ -548,6 +542,108 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
548 542 List<ListenableFuture<List<Void>>> attributeFutures = new ArrayList<>();
549 543 for (int i = 0; i < devices.size(); i++) {
550 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 647 attributeFutures.add(saveLongAttribute(device.getId(), "temperature", temperatures.get(i), DataConstants.CLIENT_SCOPE));
552 648 }
553 649 Futures.successfulAsList(attributeFutures).get();
... ... @@ -559,9 +655,155 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
559 655 EntityDataSortOrder sortOrder = new EntityDataSortOrder(
560 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 804 EntityDataPageLink pageLink = new EntityDataPageLink(10, 0, null, sortOrder);
563 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 808 EntityDataQuery query = new EntityDataQuery(filter, pageLink, entityFields, latestValues, null);
567 809 PageData<EntityData> data = entityService.findEntityDataByQuery(tenantId, new CustomerId(CustomerId.NULL_UUID), query);
... ... @@ -576,14 +818,14 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
576 818 List<String> loadedTemperatures = new ArrayList<>();
577 819 for (Device device : devices) {
578 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 824 Assert.assertEquals(deviceTemperatures, loadedTemperatures);
583 825
584 826 pageLink = new EntityDataPageLink(10, 0, null, sortOrder);
585 827 KeyFilter highTemperatureFilter = new KeyFilter();
586   - highTemperatureFilter.setKey(new EntityKey(EntityKeyType.ATTRIBUTE, "temperature"));
  828 + highTemperatureFilter.setKey(new EntityKey(EntityKeyType.TIME_SERIES, "temperature"));
587 829 NumericFilterPredicate predicate = new NumericFilterPredicate();
588 830 predicate.setValue(FilterPredicateValue.fromDouble(45));
589 831 predicate.setOperation(NumericFilterPredicate.NumericOperation.GREATER);
... ... @@ -603,17 +845,425 @@ public abstract class BaseEntityServiceTest extends AbstractServiceTest {
603 845 Assert.assertEquals(highTemperatures.size(), loadedEntities.size());
604 846
605 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 851 Assert.assertEquals(deviceHighTemperatures, loadedHighTemperatures);
610 852
611 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 1249 private ListenableFuture<List<Void>> saveLongAttribute(EntityId entityId, String key, long value, String scope) {
615 1250 KvEntry attrValue = new LongDataEntry(key, value);
616 1251 AttributeKvEntry attr = new BaseAttributeKvEntry(attrValue, 42L);
617 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 75 this.submitted = true;
76 76 const attribute: AttributeData = {
77 77 lastUpdateTs: null,
78   - key: this.attributeFormGroup.get('key').value,
  78 + key: this.attributeFormGroup.get('key').value.trim(),
79 79 value: this.attributeFormGroup.get('value').value
80 80 };
81 81 this.attributeService.saveEntityAttributes(this.data.entityId,
... ...
... ... @@ -129,7 +129,7 @@ export abstract class EntityComponent<T extends BaseData<HasId>,
129 129 acc[curr] = obj[curr];
130 130 }
131 131 return acc;
132   - }, {});
  132 + }, Array.isArray(obj) ? [] : {});
133 133 }
134 134
135 135 protected setEntitiesTableConfig(entitiesTableConfig: C) {
... ...
... ... @@ -29,13 +29,6 @@
29 29 <fieldset [disabled]="isLoading$ | async" fxLayout="column">
30 30 <section fxLayout="row" fxLayoutGap="8px" class="entity-key">
31 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 32 <mat-form-field fxFlex="40" class="mat-block">
40 33 <mat-label translate>filter.key-type.key-type</mat-label>
41 34 <mat-select required formControlName="type">
... ... @@ -44,10 +37,24 @@
44 37 </mat-option>
45 38 </mat-select>
46 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 54 </section>
48 55 <mat-form-field fxFlex="30" class="mat-block">
49 56 <mat-label translate>filter.value-type.value-type</mat-label>
50   - <mat-select matInput formControlName="valueType">
  57 + <mat-select formControlName="valueType">
51 58 <mat-select-trigger>
52 59 <mat-icon class="tb-mat-18" svgIcon="{{ entityKeyValueTypes.get(keyFilterFormGroup.get('valueType').value)?.icon }}"></mat-icon>
53 60 <span>{{ entityKeyValueTypes.get(keyFilterFormGroup.get('valueType').value)?.name | translate }}</span>
... ...
... ... @@ -27,10 +27,14 @@ import {
27 27 entityKeyTypeTranslationMap,
28 28 EntityKeyValueType,
29 29 entityKeyValueTypesMap,
30   - KeyFilterInfo, KeyFilterPredicate
  30 + KeyFilterInfo,
  31 + KeyFilterPredicate
31 32 } from '@shared/models/query/query.models';
32 33 import { DialogService } from '@core/services/dialog.service';
33 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 39 export interface KeyFilterDialogData {
36 40 keyFilter: KeyFilterInfo;
... ... @@ -61,6 +65,14 @@ export class KeyFilterDialogComponent extends
61 65
62 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 76 constructor(protected store: Store<AppState>,
65 77 protected router: Router,
66 78 @Inject(MAT_DIALOG_DATA) public data: KeyFilterDialogData,
... ... @@ -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 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 138 isErrorState(control: FormControl | null, form: FormGroupDirective | NgForm | null): boolean {
... ...
... ... @@ -118,16 +118,18 @@
118 118 </mat-form-field>
119 119 <mat-form-field class="mat-block">
120 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 122 </mat-form-field>
123 123 </div>
124 124 <mat-form-field class="mat-block">
125 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 128 </mat-form-field>
128 129 <mat-form-field class="mat-block">
129 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 133 </mat-form-field>
132 134 <div fxLayout="row" fxLayoutAlign="end center" fxLayout.xs="column" fxLayoutAlign.xs="end" fxLayoutGap="16px">
133 135 <button mat-raised-button type="button"
... ...
... ... @@ -92,7 +92,7 @@ export class MailServerComponent extends PageComponent implements OnInit, HasCon
92 92 }
93 93
94 94 enableProxyChanged(): void {
95   - let enableProxy: boolean = this.mailSettings.get('enableProxy').value;
  95 + const enableProxy: boolean = this.mailSettings.get('enableProxy').value;
96 96 if (enableProxy) {
97 97 this.mailSettings.get('proxyHost').enable();
98 98 this.mailSettings.get('proxyPort').enable();
... ...
... ... @@ -203,7 +203,7 @@ export enum StringOperation {
203 203 STARTS_WITH = 'STARTS_WITH',
204 204 ENDS_WITH = 'ENDS_WITH',
205 205 CONTAINS = 'CONTAINS',
206   - NOT_CONTAIN = 'NOT_CONTAIN'
  206 + NOT_CONTAINS = 'NOT_CONTAINS'
207 207 }
208 208
209 209 export const stringOperationTranslationMap = new Map<StringOperation, string>(
... ... @@ -213,7 +213,7 @@ export const stringOperationTranslationMap = new Map<StringOperation, string>(
213 213 [StringOperation.STARTS_WITH, 'filter.operation.starts-with'],
214 214 [StringOperation.ENDS_WITH, 'filter.operation.ends-with'],
215 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 1189 "starts-with": "starts with",
1190 1190 "ends-with": "ends with",
1191 1191 "contains": "contains",
1192   - "not-contain": "not contain",
  1192 + "not-contains": "not contains",
1193 1193 "greater": "greater than",
1194 1194 "less": "less than",
1195 1195 "greater-or-equal": "greater or equal",
... ...