Commit 9c4d094075bac3dbd6cba30397fa829d5249d28c
Merge remote-tracking branch 'upstream/master' into bug/string-to-number-auto-convertion
Showing
16 changed files
with
928 additions
and
169 deletions
.travis.yml
deleted
100644 → 0
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", | ... | ... |