Commit fad04b0dd00fefc64772272cddfb4e1ad3d47037

Authored by Volodymyr Babak
2 parents 23b21b29 ae7d92b1

Merge remote-tracking branch 'upstream/master'

Showing 40 changed files with 1139 additions and 196 deletions
... ... @@ -18,7 +18,23 @@
18 18 "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('google-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('google-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('google-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\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,\"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.floor(4 * percent);\\n\\t res.url = images[index];\\n\\t}\\n\\treturn res;\\n}\",\"color\":\"#fe7569\"},\"title\":\"Google Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
  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,\"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\":\"#fe7569\",\"showTooltip\":true,\"autocloseTooltip\":true},\"title\":\"Google Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
  22 + }
  23 + },
  24 + {
  25 + "alias": "image_map",
  26 + "name": "Image Map",
  27 + "descriptor": {
  28 + "type": "latest",
  29 + "sizeX": 8.5,
  30 + "sizeY": 6.5,
  31 + "resources": [],
  32 + "templateHtml": "",
  33 + "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",
  34 + "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('image-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('image-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('image-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n",
  35 + "settingsSchema": "{}",
  36 + "dataKeySettingsSchema": "{}\n",
  37 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"xPos\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 0.2;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"yPos\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || 0.3;\\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\":\"xPos\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 0.6;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"yPos\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || 0.7;\\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\":{\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>X Pos:</b> ${xPos:2}<br/><b>Y Pos:</b> ${yPos:2}<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\",\"mapImageUrl\":\"\",\"xPosKeyName\":\"xPos\",\"yPosKeyName\":\"yPos\",\"posFunction\":\"return {x: origXPos, y: origYPos};\",\"markerOffsetX\":0.5,\"markerOffsetY\":1,\"showTooltip\":true,\"autocloseTooltip\":true},\"title\":\"Image Map\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
22 38 }
23 39 },
24 40 {
... ... @@ -34,7 +50,7 @@
34 50 "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",
35 51 "settingsSchema": "{}",
36 52 "dataKeySettingsSchema": "{}\n",
37   - "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.floor(4 * percent);\\n\\t res.url = images[index];\\n\\t}\\n\\treturn res;\\n}\",\"color\":\"#fe7569\"},\"title\":\"OpenStreetMap\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
  53 + "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},\"title\":\"OpenStreetMap\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
38 54 }
39 55 },
40 56 {
... ... @@ -50,7 +66,7 @@
50 66 "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",
51 67 "settingsSchema": "{}",
52 68 "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\":\"<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.floor(3 * percent);\\n res.url = images[index];\\n}\\nreturn res;\",\"strokeWeight\":4,\"strokeOpacity\":0.65,\"color\":\"#1976d2\"},\"title\":\"Route Map\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
  69 + "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},\"title\":\"Route Map\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
54 70 }
55 71 },
56 72 {
... ... @@ -66,23 +82,39 @@
66 82 "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('openstreet-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('openstreet-map', true);\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('openstreet-map', true);\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n",
67 83 "settingsSchema": "{}",
68 84 "dataKeySettingsSchema": "{}\n",
69   - "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,\"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.floor(3 * percent);\\n res.url = images[index];\\n}\\nreturn res;\",\"strokeWeight\":4,\"strokeOpacity\":0.65,\"color\":\"#1976d2\"},\"title\":\"Route Map - OpenStreetMap\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
  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,\"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\",\"mapProvider\":\"OpenStreetMap.Mapnik\",\"showTooltip\":true,\"autocloseTooltip\":true},\"title\":\"Route Map - OpenStreetMap\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
70 86 }
71 87 },
72 88 {
73   - "alias": "image_map",
74   - "name": "Image Map",
  89 + "alias": "route_map_tencent_maps",
  90 + "name": "Route Map - Tencent Maps",
75 91 "descriptor": {
76   - "type": "latest",
  92 + "type": "timeseries",
77 93 "sizeX": 8.5,
78   - "sizeY": 6.5,
  94 + "sizeY": 6,
79 95 "resources": [],
80 96 "templateHtml": "",
81   - "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",
82   - "controllerScript": "self.onInit = function() {\n self.ctx.map = new TbMapWidgetV2('image-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('image-map');\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbMapWidgetV2.dataKeySettingsSchema('image-map');\n}\n\nself.actionSources = function() {\n return TbMapWidgetV2.actionSources();\n}\n\nself.onDestroy = function() {\n}\n",
  97 + "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}",
  98 + "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",
  99 + "settingsSchema": "{}",
  100 + "dataKeySettingsSchema": "{}\n",
  101 + "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\":\"#1976d2\",\"tmDefaultMapType\":\"roadmap\",\"showTooltip\":true,\"autocloseTooltip\":true,\"tmApiKey\":\"84d6d83e0e51e481e50454ccbe8986b\"},\"title\":\"Route Map - Tencent Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
  102 + }
  103 + },
  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",
83 115 "settingsSchema": "{}",
84 116 "dataKeySettingsSchema": "{}\n",
85   - "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"First point\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"xPos\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 0.2;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"yPos\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || 0.3;\\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\":\"xPos\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 0.6;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"yPos\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || 0.7;\\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\":{\"showLabel\":true,\"label\":\"${entityName}\",\"tooltipPattern\":\"<b>${entityName}</b><br/><br/><b>X Pos:</b> ${xPos:2}<br/><b>Y Pos:</b> ${yPos:2}<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.floor(4 * percent);\\n\\t res.url = images[index];\\n\\t}\\n\\treturn res;\\n}\",\"color\":\"#fe7569\",\"mapImageUrl\":\"\",\"xPosKeyName\":\"xPos\",\"yPosKeyName\":\"yPos\",\"posFunction\":\"return {x: origXPos, y: origYPos};\",\"markerOffsetX\":0.5,\"markerOffsetY\":1},\"title\":\"Image Map\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{}}"
  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\":[\"\",\"\",\"\",\"\"]},\"title\":\"Tencent Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
86 118 }
87 119 }
88 120 ]
... ...
... ... @@ -66,6 +66,8 @@ public class CassandraDatabaseUpgradeService implements DatabaseUpgradeService {
66 66
67 67 //Dump devices, assets and relations
68 68
  69 + cluster.getSession();
  70 +
69 71 KeyspaceMetadata ks = cluster.getCluster().getMetadata().getKeyspace(cluster.getKeyspaceName());
70 72
71 73 log.info("Dumping devices ...");
... ...
... ... @@ -77,6 +77,8 @@ http:
77 77
78 78 # MQTT server parameters
79 79 mqtt:
  80 + # Enable/disable mqtt transport protocol.
  81 + enabled: "${MQTT_ENABLED:true}"
80 82 bind_address: "${MQTT_BIND_ADDRESS:0.0.0.0}"
81 83 bind_port: "${MQTT_BIND_PORT:1883}"
82 84 adaptor: "${MQTT_ADAPTOR_NAME:JsonMqttAdaptor}"
... ... @@ -102,6 +104,8 @@ mqtt:
102 104
103 105 # CoAP server parameters
104 106 coap:
  107 + # Enable/disable coap transport protocol.
  108 + enabled: "${COAP_ENABLED:true}"
105 109 bind_address: "${COAP_BIND_ADDRESS:0.0.0.0}"
106 110 bind_port: "${COAP_BIND_PORT:5683}"
107 111 adaptor: "${COAP_ADAPTOR_NAME:JsonCoapAdaptor}"
... ... @@ -157,6 +161,13 @@ cassandra:
157 161 # Specify partitioning size for timestamp key-value storage. Example MINUTES, HOURS, DAYS, MONTHS
158 162 ts_key_value_partitioning: "${TS_KV_PARTITIONING:MONTHS}"
159 163
  164 +# SQL configuration parameters
  165 +sql:
  166 + # Specify executor service type used to perform timeseries insert tasks: SINGLE FIXED CACHED
  167 + ts_inserts_executor_type: "${SQL_TS_INSERTS_EXECUTOR_TYPE:fixed}"
  168 + # Specify thread pool size for FIXED executor service type
  169 + ts_inserts_fixed_thread_pool_size: "${SQL_TS_INSERTS_FIXED_THREAD_POOL_SIZE:10}"
  170 +
160 171 # Actor system parameters
161 172 actors:
162 173 tenant:
... ... @@ -201,6 +212,18 @@ cache:
201 212 policy: "${CACHE_DEVICE_CREDENTIAL_MAX_SIZE_POLICY:PER_NODE}"
202 213 size: "${CACHE_DEVICE_CREDENTIAL_MAX_SIZE_SIZE:1000000}"
203 214
  215 +caching:
  216 + specs:
  217 + relations:
  218 + timeToLiveInMinutes: 1440
  219 + maxSize: 100000
  220 + deviceCredentials:
  221 + timeToLiveInMinutes: 1440
  222 + maxSize: 100000
  223 + devices:
  224 + timeToLiveInMinutes: 1440
  225 + maxSize: 100000
  226 +
204 227 # Check new version updates parameters
205 228 updates:
206 229 # Enable/disable updates checking.
... ...
... ... @@ -17,4 +17,6 @@ package org.thingsboard.server.common.data;
17 17
18 18 public class CacheConstants {
19 19 public static final String DEVICE_CREDENTIALS_CACHE = "deviceCredentials";
  20 + public static final String RELATIONS_CACHE = "relations";
  21 + public static final String DEVICE_CACHE = "devices";
20 22 }
... ...
... ... @@ -149,6 +149,10 @@
149 149 <artifactId>hazelcast</artifactId>
150 150 </dependency>
151 151 <dependency>
  152 + <groupId>com.github.ben-manes.caffeine</groupId>
  153 + <artifactId>caffeine</artifactId>
  154 + </dependency>
  155 + <dependency>
152 156 <groupId>com.hazelcast</groupId>
153 157 <artifactId>hazelcast-spring</artifactId>
154 158 </dependency>
... ... @@ -174,6 +178,10 @@
174 178 <artifactId>hsqldb</artifactId>
175 179 <scope>test</scope>
176 180 </dependency>
  181 + <dependency>
  182 + <groupId>org.springframework</groupId>
  183 + <artifactId>spring-context-support</artifactId>
  184 + </dependency>
177 185 </dependencies>
178 186 <build>
179 187 <plugins>
... ...
  1 +/**
  2 + * Copyright © 2016-2017 The Thingsboard Authors
  3 + *
  4 + * Licensed under the Apache License, Version 2.0 (the "License");
  5 + * you may not use this file except in compliance with the License.
  6 + * You may obtain a copy of the License at
  7 + *
  8 + * http://www.apache.org/licenses/LICENSE-2.0
  9 + *
  10 + * Unless required by applicable law or agreed to in writing, software
  11 + * distributed under the License is distributed on an "AS IS" BASIS,
  12 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 + * See the License for the specific language governing permissions and
  14 + * limitations under the License.
  15 + */
  16 +package org.thingsboard.server.dao.cache;
  17 +
  18 +import lombok.Data;
  19 +
  20 +@Data
  21 +public class CacheSpecs {
  22 + private Integer timeToLiveInMinutes;
  23 + private Integer maxSize;
  24 +}
... ...
... ... @@ -23,6 +23,8 @@ import java.lang.reflect.Method;
23 23
24 24 public class PreviousDeviceCredentialsIdKeyGenerator implements KeyGenerator {
25 25
  26 + private static final String NOT_VALID_DEVICE = "notValidDeviceCredentialsId";
  27 +
26 28 @Override
27 29 public Object generate(Object o, Method method, Object... objects) {
28 30 DeviceCredentialsService deviceCredentialsService = (DeviceCredentialsService) o;
... ... @@ -33,6 +35,6 @@ public class PreviousDeviceCredentialsIdKeyGenerator implements KeyGenerator {
33 35 return oldDeviceCredentials.getCredentialsId();
34 36 }
35 37 }
36   - return null;
  38 + return NOT_VALID_DEVICE;
37 39 }
38 40 }
... ...
... ... @@ -15,76 +15,57 @@
15 15 */
16 16 package org.thingsboard.server.dao.cache;
17 17
18   -import com.hazelcast.config.*;
19   -import com.hazelcast.core.Hazelcast;
20   -import com.hazelcast.core.HazelcastInstance;
21   -import com.hazelcast.instance.GroupProperty;
22   -import com.hazelcast.spring.cache.HazelcastCacheManager;
23   -import com.hazelcast.zookeeper.ZookeeperDiscoveryProperties;
24   -import com.hazelcast.zookeeper.ZookeeperDiscoveryStrategyFactory;
25   -import org.springframework.beans.factory.annotation.Value;
26   -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
  18 +import com.github.benmanes.caffeine.cache.Caffeine;
  19 +import com.github.benmanes.caffeine.cache.Ticker;
  20 +import lombok.Data;
  21 +import org.springframework.boot.context.properties.ConfigurationProperties;
27 22 import org.springframework.cache.CacheManager;
28 23 import org.springframework.cache.annotation.EnableCaching;
  24 +import org.springframework.cache.caffeine.CaffeineCache;
29 25 import org.springframework.cache.interceptor.KeyGenerator;
  26 +import org.springframework.cache.support.SimpleCacheManager;
30 27 import org.springframework.context.annotation.Bean;
31 28 import org.springframework.context.annotation.Configuration;
32   -import org.thingsboard.server.common.data.CacheConstants;
  29 +
  30 +import java.util.List;
  31 +import java.util.Map;
  32 +import java.util.concurrent.TimeUnit;
  33 +import java.util.stream.Collectors;
33 34
34 35 @Configuration
  36 +@ConfigurationProperties(prefix = "caching")
35 37 @EnableCaching
36   -@ConditionalOnProperty(prefix = "cache", value = "enabled", havingValue = "true")
  38 +@Data
37 39 public class ServiceCacheConfiguration {
38 40
39   - private static final String HAZELCAST_CLUSTER_NAME = "hazelcast";
40   -
41   - @Value("${cache.device_credentials.max_size.size}")
42   - private Integer cacheDeviceCredentialsMaxSizeSize;
43   - @Value("${cache.device_credentials.max_size.policy}")
44   - private String cacheDeviceCredentialsMaxSizePolicy;
45   - @Value("${cache.device_credentials.time_to_live}")
46   - private Integer cacheDeviceCredentialsTTL;
47   -
48   - @Value("${zk.enabled}")
49   - private boolean zkEnabled;
50   - @Value("${zk.url}")
51   - private String zkUrl;
52   - @Value("${zk.zk_dir}")
53   - private String zkDir;
  41 + private Map<String, CacheSpecs> specs;
54 42
55 43 @Bean
56   - public HazelcastInstance hazelcastInstance() {
57   - Config config = new Config();
58   -
59   - if (zkEnabled) {
60   - addZkConfig(config);
  44 + public CacheManager cacheManager() {
  45 + SimpleCacheManager manager = new SimpleCacheManager();
  46 + if (specs != null) {
  47 + List<CaffeineCache> caches =
  48 + specs.entrySet().stream()
  49 + .map(entry -> buildCache(entry.getKey(),
  50 + entry.getValue()))
  51 + .collect(Collectors.toList());
  52 + manager.setCaches(caches);
61 53 }
62   -
63   - config.addMapConfig(createDeviceCredentialsCacheConfig());
64   -
65   - return Hazelcast.newHazelcastInstance(config);
  54 + return manager;
66 55 }
67 56
68   - private void addZkConfig(Config config) {
69   - config.getNetworkConfig().getJoin().getMulticastConfig().setEnabled(false);
70   - config.setProperty(GroupProperty.DISCOVERY_SPI_ENABLED.getName(), Boolean.TRUE.toString());
71   - DiscoveryStrategyConfig discoveryStrategyConfig = new DiscoveryStrategyConfig(new ZookeeperDiscoveryStrategyFactory());
72   - discoveryStrategyConfig.addProperty(ZookeeperDiscoveryProperties.ZOOKEEPER_URL.key(), zkUrl);
73   - discoveryStrategyConfig.addProperty(ZookeeperDiscoveryProperties.ZOOKEEPER_PATH.key(), zkDir);
74   - discoveryStrategyConfig.addProperty(ZookeeperDiscoveryProperties.GROUP.key(), HAZELCAST_CLUSTER_NAME);
75   - config.getNetworkConfig().getJoin().getDiscoveryConfig().addDiscoveryStrategyConfig(discoveryStrategyConfig);
  57 + private CaffeineCache buildCache(String name, CacheSpecs cacheSpec) {
  58 + final Caffeine<Object, Object> caffeineBuilder
  59 + = Caffeine.newBuilder()
  60 + .expireAfterWrite(cacheSpec.getTimeToLiveInMinutes(), TimeUnit.MINUTES)
  61 + .maximumSize(cacheSpec.getMaxSize())
  62 + .ticker(ticker());
  63 + return new CaffeineCache(name, caffeineBuilder.build());
76 64 }
77 65
78   - private MapConfig createDeviceCredentialsCacheConfig() {
79   - MapConfig deviceCredentialsCacheConfig = new MapConfig(CacheConstants.DEVICE_CREDENTIALS_CACHE);
80   - deviceCredentialsCacheConfig.setTimeToLiveSeconds(cacheDeviceCredentialsTTL);
81   - deviceCredentialsCacheConfig.setEvictionPolicy(EvictionPolicy.LRU);
82   - deviceCredentialsCacheConfig.setMaxSizeConfig(
83   - new MaxSizeConfig(
84   - cacheDeviceCredentialsMaxSizeSize,
85   - MaxSizeConfig.MaxSizePolicy.valueOf(cacheDeviceCredentialsMaxSizePolicy))
86   - );
87   - return deviceCredentialsCacheConfig;
  66 + @Bean
  67 + public Ticker ticker() {
  68 + return Ticker.systemTicker();
88 69 }
89 70
90 71 @Bean
... ... @@ -92,8 +73,4 @@ public class ServiceCacheConfiguration {
92 73 return new PreviousDeviceCredentialsIdKeyGenerator();
93 74 }
94 75
95   - @Bean
96   - public CacheManager cacheManager() {
97   - return new HazelcastCacheManager(hazelcastInstance());
98   - }
99 76 }
... ...
... ... @@ -75,6 +75,7 @@ public abstract class AbstractCassandraCluster {
75 75 private Environment environment;
76 76
77 77 private Cluster cluster;
  78 + private Cluster.Builder clusterBuilder;
78 79
79 80 @Getter(AccessLevel.NONE) private Session session;
80 81
... ... @@ -88,29 +89,27 @@ public abstract class AbstractCassandraCluster {
88 89
89 90 protected void init(String keyspaceName) {
90 91 this.keyspaceName = keyspaceName;
91   - Cluster.Builder builder = Cluster.builder()
  92 + this.clusterBuilder = Cluster.builder()
92 93 .addContactPointsWithPorts(getContactPoints(url))
93 94 .withClusterName(clusterName)
94 95 .withSocketOptions(socketOpts.getOpts())
95 96 .withPoolingOptions(new PoolingOptions()
96 97 .setMaxRequestsPerConnection(HostDistance.LOCAL, 32768)
97 98 .setMaxRequestsPerConnection(HostDistance.REMOTE, 32768));
98   - builder.withQueryOptions(queryOpts.getOpts());
99   - builder.withCompression(StringUtils.isEmpty(compression) ? Compression.NONE : Compression.valueOf(compression.toUpperCase()));
  99 + this.clusterBuilder.withQueryOptions(queryOpts.getOpts());
  100 + this.clusterBuilder.withCompression(StringUtils.isEmpty(compression) ? Compression.NONE : Compression.valueOf(compression.toUpperCase()));
100 101 if (ssl) {
101   - builder.withSSL();
  102 + this.clusterBuilder.withSSL();
102 103 }
103 104 if (!jmx) {
104   - builder.withoutJMXReporting();
  105 + this.clusterBuilder.withoutJMXReporting();
105 106 }
106 107 if (!metrics) {
107   - builder.withoutMetrics();
  108 + this.clusterBuilder.withoutMetrics();
108 109 }
109 110 if (credentials) {
110   - builder.withCredentials(username, password);
  111 + this.clusterBuilder.withCredentials(username, password);
111 112 }
112   - cluster = builder.build();
113   - cluster.init();
114 113 if (!isInstall()) {
115 114 initSession();
116 115 }
... ... @@ -139,7 +138,8 @@ public abstract class AbstractCassandraCluster {
139 138 long endTime = System.currentTimeMillis() + initTimeout;
140 139 while (System.currentTimeMillis() < endTime) {
141 140 try {
142   -
  141 + cluster = clusterBuilder.build();
  142 + cluster.init();
143 143 if (this.keyspaceName != null) {
144 144 session = cluster.connect(keyspaceName);
145 145 } else {
... ...
... ... @@ -34,7 +34,7 @@ public interface DeviceService {
34 34
35 35 ListenableFuture<Device> findDeviceByIdAsync(DeviceId deviceId);
36 36
37   - Optional<Device> findDeviceByTenantIdAndName(TenantId tenantId, String name);
  37 + Device findDeviceByTenantIdAndName(TenantId tenantId, String name);
38 38
39 39 Device saveDevice(Device device);
40 40
... ...
... ... @@ -22,6 +22,10 @@ import com.google.common.util.concurrent.ListenableFuture;
22 22 import lombok.extern.slf4j.Slf4j;
23 23 import org.apache.commons.lang3.RandomStringUtils;
24 24 import org.springframework.beans.factory.annotation.Autowired;
  25 +import org.springframework.cache.Cache;
  26 +import org.springframework.cache.CacheManager;
  27 +import org.springframework.cache.annotation.CacheEvict;
  28 +import org.springframework.cache.annotation.Cacheable;
25 29 import org.springframework.stereotype.Service;
26 30 import org.springframework.util.StringUtils;
27 31 import org.thingsboard.server.common.data.*;
... ... @@ -33,12 +37,12 @@ import org.thingsboard.server.common.data.id.TenantId;
33 37 import org.thingsboard.server.common.data.page.TextPageData;
34 38 import org.thingsboard.server.common.data.page.TextPageLink;
35 39 import org.thingsboard.server.common.data.relation.EntityRelation;
  40 +import org.thingsboard.server.common.data.relation.EntitySearchDirection;
36 41 import org.thingsboard.server.common.data.security.DeviceCredentials;
37 42 import org.thingsboard.server.common.data.security.DeviceCredentialsType;
38 43 import org.thingsboard.server.dao.customer.CustomerDao;
39 44 import org.thingsboard.server.dao.entity.AbstractEntityService;
40 45 import org.thingsboard.server.dao.exception.DataValidationException;
41   -import org.thingsboard.server.common.data.relation.EntitySearchDirection;
42 46 import org.thingsboard.server.dao.service.DataValidator;
43 47 import org.thingsboard.server.dao.service.PaginatedRemover;
44 48 import org.thingsboard.server.dao.tenant.TenantDao;
... ... @@ -47,6 +51,7 @@ import javax.annotation.Nullable;
47 51 import java.util.*;
48 52 import java.util.stream.Collectors;
49 53
  54 +import static org.thingsboard.server.common.data.CacheConstants.DEVICE_CACHE;
50 55 import static org.thingsboard.server.dao.DaoUtil.toUUIDs;
51 56 import static org.thingsboard.server.dao.model.ModelConstants.NULL_UUID;
52 57 import static org.thingsboard.server.dao.service.Validator.*;
... ... @@ -71,6 +76,9 @@ public class DeviceServiceImpl extends AbstractEntityService implements DeviceSe
71 76 @Autowired
72 77 private DeviceCredentialsService deviceCredentialsService;
73 78
  79 + @Autowired
  80 + private CacheManager cacheManager;
  81 +
74 82 @Override
75 83 public Device findDeviceById(DeviceId deviceId) {
76 84 log.trace("Executing findDeviceById [{}]", deviceId);
... ... @@ -85,18 +93,16 @@ public class DeviceServiceImpl extends AbstractEntityService implements DeviceSe
85 93 return deviceDao.findByIdAsync(deviceId.getId());
86 94 }
87 95
  96 + @Cacheable(cacheNames = DEVICE_CACHE, key = "{#tenantId, #name}")
88 97 @Override
89   - public Optional<Device> findDeviceByTenantIdAndName(TenantId tenantId, String name) {
  98 + public Device findDeviceByTenantIdAndName(TenantId tenantId, String name) {
90 99 log.trace("Executing findDeviceByTenantIdAndName [{}][{}]", tenantId, name);
91 100 validateId(tenantId, INCORRECT_TENANT_ID + tenantId);
92 101 Optional<Device> deviceOpt = deviceDao.findDeviceByTenantIdAndName(tenantId.getId(), name);
93   - if (deviceOpt.isPresent()) {
94   - return Optional.of(deviceOpt.get());
95   - } else {
96   - return Optional.empty();
97   - }
  102 + return deviceOpt.orElse(null);
98 103 }
99 104
  105 + @CacheEvict(cacheNames = DEVICE_CACHE, key = "{#device.tenantId, #device.name}")
100 106 @Override
101 107 public Device saveDevice(Device device) {
102 108 log.trace("Executing saveDevice [{}]", device);
... ... @@ -129,12 +135,18 @@ public class DeviceServiceImpl extends AbstractEntityService implements DeviceSe
129 135 @Override
130 136 public void deleteDevice(DeviceId deviceId) {
131 137 log.trace("Executing deleteDevice [{}]", deviceId);
  138 + Cache cache = cacheManager.getCache(DEVICE_CACHE);
132 139 validateId(deviceId, INCORRECT_DEVICE_ID + deviceId);
133 140 DeviceCredentials deviceCredentials = deviceCredentialsService.findDeviceCredentialsByDeviceId(deviceId);
134 141 if (deviceCredentials != null) {
135 142 deviceCredentialsService.deleteDeviceCredentials(deviceCredentials);
136 143 }
137 144 deleteEntityRelations(deviceId);
  145 + Device device = deviceDao.findById(deviceId.getId());
  146 + List<Object> list = new ArrayList<>();
  147 + list.add(device.getTenantId());
  148 + list.add(device.getName());
  149 + cache.evict(list);
138 150 deviceDao.removeById(deviceId.getId());
139 151 }
140 152
... ... @@ -190,7 +202,7 @@ public class DeviceServiceImpl extends AbstractEntityService implements DeviceSe
190 202 validateId(customerId, INCORRECT_CUSTOMER_ID + customerId);
191 203 validateString(type, "Incorrect type " + type);
192 204 validatePageLink(pageLink, INCORRECT_PAGE_LINK + pageLink);
193   - List<Device> devices = deviceDao.findDevicesByTenantIdAndCustomerIdAndType(tenantId.getId(), customerId.getId(), type, pageLink);
  205 + List<Device> devices = deviceDao.findDevicesByTenantIdAndCustomerIdAndType(tenantId.getId(), customerId.getId(), type, pageLink);
194 206 return new TextPageData<>(devices, pageLink);
195 207 }
196 208
... ... @@ -244,10 +256,10 @@ public class DeviceServiceImpl extends AbstractEntityService implements DeviceSe
244 256 validateId(tenantId, INCORRECT_TENANT_ID + tenantId);
245 257 ListenableFuture<List<EntitySubtype>> tenantDeviceTypes = deviceDao.findTenantDeviceTypesAsync(tenantId.getId());
246 258 return Futures.transform(tenantDeviceTypes,
247   - (Function<List<EntitySubtype>, List<EntitySubtype>>) deviceTypes -> {
248   - deviceTypes.sort(Comparator.comparing(EntitySubtype::getType));
249   - return deviceTypes;
250   - });
  259 + (Function<List<EntitySubtype>, List<EntitySubtype>>) deviceTypes -> {
  260 + deviceTypes.sort(Comparator.comparing(EntitySubtype::getType));
  261 + return deviceTypes;
  262 + });
251 263 }
252 264
253 265 private DataValidator<Device> deviceValidator =
... ...
... ... @@ -21,6 +21,11 @@ import com.google.common.util.concurrent.Futures;
21 21 import com.google.common.util.concurrent.ListenableFuture;
22 22 import lombok.extern.slf4j.Slf4j;
23 23 import org.springframework.beans.factory.annotation.Autowired;
  24 +import org.springframework.cache.Cache;
  25 +import org.springframework.cache.CacheManager;
  26 +import org.springframework.cache.annotation.CacheEvict;
  27 +import org.springframework.cache.annotation.Cacheable;
  28 +import org.springframework.cache.annotation.Caching;
24 29 import org.springframework.stereotype.Service;
25 30 import org.springframework.util.StringUtils;
26 31 import org.thingsboard.server.common.data.id.EntityId;
... ... @@ -34,6 +39,8 @@ import java.util.concurrent.ConcurrentHashMap;
34 39 import java.util.concurrent.ExecutionException;
35 40 import java.util.function.BiConsumer;
36 41
  42 +import static org.thingsboard.server.common.data.CacheConstants.RELATIONS_CACHE;
  43 +
37 44 /**
38 45 * Created by ashvayka on 28.04.17.
39 46 */
... ... @@ -47,6 +54,9 @@ public class BaseRelationService implements RelationService {
47 54 @Autowired
48 55 private EntityService entityService;
49 56
  57 + @Autowired
  58 + private CacheManager cacheManager;
  59 +
50 60 @Override
51 61 public ListenableFuture<Boolean> checkRelation(EntityId from, EntityId to, String relationType, RelationTypeGroup typeGroup) {
52 62 log.trace("Executing checkRelation [{}][{}][{}][{}]", from, to, relationType, typeGroup);
... ... @@ -54,6 +64,7 @@ public class BaseRelationService implements RelationService {
54 64 return relationDao.checkRelation(from, to, relationType, typeGroup);
55 65 }
56 66
  67 + @Cacheable(cacheNames = RELATIONS_CACHE, key = "{#from, #to, #relationType}")
57 68 @Override
58 69 public ListenableFuture<EntityRelation> getRelation(EntityId from, EntityId to, String relationType, RelationTypeGroup typeGroup) {
59 70 log.trace("Executing EntityRelation [{}][{}][{}][{}]", from, to, relationType, typeGroup);
... ... @@ -61,6 +72,12 @@ public class BaseRelationService implements RelationService {
61 72 return relationDao.getRelation(from, to, relationType, typeGroup);
62 73 }
63 74
  75 + @Caching(evict = {
  76 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#relation.from"),
  77 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#relation.from, #relation.type}"),
  78 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#relation.to"),
  79 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#relation.to, #relation.type}")
  80 + })
64 81 @Override
65 82 public boolean saveRelation(EntityRelation relation) {
66 83 log.trace("Executing saveRelation [{}]", relation);
... ... @@ -68,6 +85,12 @@ public class BaseRelationService implements RelationService {
68 85 return relationDao.saveRelation(relation);
69 86 }
70 87
  88 + @Caching(evict = {
  89 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#relation.from"),
  90 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#relation.from, #relation.type}"),
  91 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#relation.to"),
  92 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#relation.to, #relation.type}")
  93 + })
71 94 @Override
72 95 public ListenableFuture<Boolean> saveRelationAsync(EntityRelation relation) {
73 96 log.trace("Executing saveRelationAsync [{}]", relation);
... ... @@ -75,6 +98,13 @@ public class BaseRelationService implements RelationService {
75 98 return relationDao.saveRelationAsync(relation);
76 99 }
77 100
  101 + @Caching(evict = {
  102 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#relation.from"),
  103 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#relation.from, #relation.type}"),
  104 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#relation.to"),
  105 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#relation.to, #relation.type}"),
  106 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#relation.from, #relation.to, #relation.type}")
  107 + })
78 108 @Override
79 109 public boolean deleteRelation(EntityRelation relation) {
80 110 log.trace("Executing deleteRelation [{}]", relation);
... ... @@ -82,6 +112,13 @@ public class BaseRelationService implements RelationService {
82 112 return relationDao.deleteRelation(relation);
83 113 }
84 114
  115 + @Caching(evict = {
  116 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#relation.from"),
  117 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#relation.from, #relation.type}"),
  118 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#relation.to"),
  119 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#relation.to, #relation.type}"),
  120 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#relation.from, #relation.to, #relation.type}")
  121 + })
85 122 @Override
86 123 public ListenableFuture<Boolean> deleteRelationAsync(EntityRelation relation) {
87 124 log.trace("Executing deleteRelationAsync [{}]", relation);
... ... @@ -89,6 +126,13 @@ public class BaseRelationService implements RelationService {
89 126 return relationDao.deleteRelationAsync(relation);
90 127 }
91 128
  129 + @Caching(evict = {
  130 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#from"),
  131 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#from, #relationType}"),
  132 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#to"),
  133 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#to, #relationType}"),
  134 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#from, #to, #relationType}")
  135 + })
92 136 @Override
93 137 public boolean deleteRelation(EntityId from, EntityId to, String relationType, RelationTypeGroup typeGroup) {
94 138 log.trace("Executing deleteRelation [{}][{}][{}][{}]", from, to, relationType, typeGroup);
... ... @@ -96,6 +140,13 @@ public class BaseRelationService implements RelationService {
96 140 return relationDao.deleteRelation(from, to, relationType, typeGroup);
97 141 }
98 142
  143 + @Caching(evict = {
  144 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#from"),
  145 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#from, #relationType}"),
  146 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "#to"),
  147 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#to, #relationType}"),
  148 + @CacheEvict(cacheNames = RELATIONS_CACHE, key = "{#from, #to, #relationType}")
  149 + })
99 150 @Override
100 151 public ListenableFuture<Boolean> deleteRelationAsync(EntityId from, EntityId to, String relationType, RelationTypeGroup typeGroup) {
101 152 log.trace("Executing deleteRelationAsync [{}][{}][{}][{}]", from, to, relationType, typeGroup);
... ... @@ -105,23 +156,17 @@ public class BaseRelationService implements RelationService {
105 156
106 157 @Override
107 158 public boolean deleteEntityRelations(EntityId entity) {
  159 + Cache cache = cacheManager.getCache(RELATIONS_CACHE);
108 160 log.trace("Executing deleteEntityRelations [{}]", entity);
109 161 validate(entity);
110   - List<ListenableFuture<List<EntityRelation>>> inboundRelationsList = new ArrayList<>();
  162 + List<ListenableFuture<List<EntityRelation>>> inboundRelationsListTo = new ArrayList<>();
111 163 for (RelationTypeGroup typeGroup : RelationTypeGroup.values()) {
112   - inboundRelationsList.add(relationDao.findAllByTo(entity, typeGroup));
  164 + inboundRelationsListTo.add(relationDao.findAllByTo(entity, typeGroup));
113 165 }
114   - ListenableFuture<List<List<EntityRelation>>> inboundRelations = Futures.allAsList(inboundRelationsList);
115   - ListenableFuture<List<Boolean>> inboundDeletions = Futures.transform(inboundRelations, new Function<List<List<EntityRelation>>, List<Boolean>>() {
116   - @Override
117   - public List<Boolean> apply(List<List<EntityRelation>> relations) {
118   - List<Boolean> results = new ArrayList<>();
119   - for (List<EntityRelation> relationList : relations) {
120   - relationList.stream().forEach(relation -> results.add(relationDao.deleteRelation(relation)));
121   - }
122   - return results;
123   - }
124   - });
  166 + ListenableFuture<List<List<EntityRelation>>> inboundRelationsTo = Futures.allAsList(inboundRelationsListTo);
  167 + ListenableFuture<List<Boolean>> inboundDeletions = Futures.transform(inboundRelationsTo, (List<List<EntityRelation>> relations) ->
  168 + getBooleans(relations, cache, true));
  169 +
125 170 ListenableFuture<Boolean> inboundFuture = Futures.transform(inboundDeletions, getListToBooleanFunction());
126 171 boolean inboundDeleteResult = false;
127 172 try {
... ... @@ -129,37 +174,105 @@ public class BaseRelationService implements RelationService {
129 174 } catch (InterruptedException | ExecutionException e) {
130 175 log.error("Error deleting entity inbound relations", e);
131 176 }
  177 +
  178 + List<ListenableFuture<List<EntityRelation>>> inboundRelationsListFrom = new ArrayList<>();
  179 + for (RelationTypeGroup typeGroup : RelationTypeGroup.values()) {
  180 + inboundRelationsListFrom.add(relationDao.findAllByFrom(entity, typeGroup));
  181 + }
  182 + ListenableFuture<List<List<EntityRelation>>> inboundRelationsFrom = Futures.allAsList(inboundRelationsListFrom);
  183 + Futures.transform(inboundRelationsFrom, (Function<List<List<EntityRelation>>, List<Boolean>>) relations ->
  184 + getBooleans(relations, cache, false));
  185 +
132 186 boolean outboundDeleteResult = relationDao.deleteOutboundRelations(entity);
133 187 return inboundDeleteResult && outboundDeleteResult;
134 188 }
135 189
  190 + private List<Boolean> getBooleans(List<List<EntityRelation>> relations, Cache cache, boolean isRemove) {
  191 + List<Boolean> results = new ArrayList<>();
  192 + for (List<EntityRelation> relationList : relations) {
  193 + relationList.stream().forEach(relation -> {
  194 + checkFromDeleteSync(cache, results, relation, isRemove);
  195 + });
  196 + }
  197 + return results;
  198 + }
  199 +
  200 + private void checkFromDeleteSync(Cache cache, List<Boolean> results, EntityRelation relation, boolean isRemove) {
  201 + if (isRemove) {
  202 + results.add(relationDao.deleteRelation(relation));
  203 + cacheEviction(relation, relation.getTo(), cache);
  204 + } else {
  205 + cacheEviction(relation, relation.getFrom(), cache);
  206 + }
  207 + }
  208 +
136 209 @Override
137 210 public ListenableFuture<Boolean> deleteEntityRelationsAsync(EntityId entity) {
  211 + Cache cache = cacheManager.getCache(RELATIONS_CACHE);
138 212 log.trace("Executing deleteEntityRelationsAsync [{}]", entity);
139 213 validate(entity);
140   - List<ListenableFuture<List<EntityRelation>>> inboundRelationsList = new ArrayList<>();
  214 + List<ListenableFuture<List<EntityRelation>>> inboundRelationsListTo = new ArrayList<>();
141 215 for (RelationTypeGroup typeGroup : RelationTypeGroup.values()) {
142   - inboundRelationsList.add(relationDao.findAllByTo(entity, typeGroup));
  216 + inboundRelationsListTo.add(relationDao.findAllByTo(entity, typeGroup));
143 217 }
144   - ListenableFuture<List<List<EntityRelation>>> inboundRelations = Futures.allAsList(inboundRelationsList);
145   - ListenableFuture<List<Boolean>> inboundDeletions = Futures.transform(inboundRelations, new AsyncFunction<List<List<EntityRelation>>, List<Boolean>>() {
146   - @Override
147   - public ListenableFuture<List<Boolean>> apply(List<List<EntityRelation>> relations) throws Exception {
148   - List<ListenableFuture<Boolean>> results = new ArrayList<>();
149   - for (List<EntityRelation> relationList : relations) {
150   - relationList.stream().forEach(relation -> results.add(relationDao.deleteRelationAsync(relation)));
151   - }
152   - return Futures.allAsList(results);
153   - }
  218 + ListenableFuture<List<List<EntityRelation>>> inboundRelationsTo = Futures.allAsList(inboundRelationsListTo);
  219 + ListenableFuture<List<Boolean>> inboundDeletions = Futures.transform(inboundRelationsTo,
  220 + (AsyncFunction<List<List<EntityRelation>>, List<Boolean>>) relations -> {
  221 + List<ListenableFuture<Boolean>> results = getListenableFutures(relations, cache, true);
  222 + return Futures.allAsList(results);
154 223 });
155 224
156 225 ListenableFuture<Boolean> inboundFuture = Futures.transform(inboundDeletions, getListToBooleanFunction());
157 226
158   - ListenableFuture<Boolean> outboundFuture = relationDao.deleteOutboundRelationsAsync(entity);
  227 + List<ListenableFuture<List<EntityRelation>>> inboundRelationsListFrom = new ArrayList<>();
  228 + for (RelationTypeGroup typeGroup : RelationTypeGroup.values()) {
  229 + inboundRelationsListFrom.add(relationDao.findAllByTo(entity, typeGroup));
  230 + }
  231 + ListenableFuture<List<List<EntityRelation>>> inboundRelationsFrom = Futures.allAsList(inboundRelationsListFrom);
  232 + Futures.transform(inboundRelationsFrom, (AsyncFunction<List<List<EntityRelation>>, List<Boolean>>) relations -> {
  233 + List<ListenableFuture<Boolean>> results = getListenableFutures(relations, cache, false);
  234 + return Futures.allAsList(results);
  235 + });
159 236
  237 + ListenableFuture<Boolean> outboundFuture = relationDao.deleteOutboundRelationsAsync(entity);
160 238 return Futures.transform(Futures.allAsList(Arrays.asList(inboundFuture, outboundFuture)), getListToBooleanFunction());
161 239 }
162 240
  241 + private List<ListenableFuture<Boolean>> getListenableFutures(List<List<EntityRelation>> relations, Cache cache, boolean isRemove) {
  242 + List<ListenableFuture<Boolean>> results = new ArrayList<>();
  243 + for (List<EntityRelation> relationList : relations) {
  244 + relationList.stream().forEach(relation -> {
  245 + checkFromDeleteAsync(cache, results, relation, isRemove);
  246 + });
  247 + }
  248 + return results;
  249 + }
  250 +
  251 + private void checkFromDeleteAsync(Cache cache, List<ListenableFuture<Boolean>> results, EntityRelation relation, boolean isRemove) {
  252 + if (isRemove) {
  253 + results.add(relationDao.deleteRelationAsync(relation));
  254 + cacheEviction(relation, relation.getTo(), cache);
  255 + } else {
  256 + cacheEviction(relation, relation.getFrom(), cache);
  257 + }
  258 + }
  259 +
  260 + private void cacheEviction(EntityRelation relation, EntityId entityId, Cache cache) {
  261 + cache.evict(entityId);
  262 +
  263 + List<Object> toAndType = new ArrayList<>();
  264 + toAndType.add(entityId);
  265 + toAndType.add(relation.getType());
  266 + cache.evict(toAndType);
  267 +
  268 + List<Object> fromToAndType = new ArrayList<>();
  269 + fromToAndType.add(relation.getFrom());
  270 + fromToAndType.add(relation.getTo());
  271 + fromToAndType.add(relation.getType());
  272 + cache.evict(fromToAndType);
  273 + }
  274 +
  275 + @Cacheable(cacheNames = RELATIONS_CACHE, key = "#from")
163 276 @Override
164 277 public ListenableFuture<List<EntityRelation>> findByFrom(EntityId from, RelationTypeGroup typeGroup) {
165 278 log.trace("Executing findByFrom [{}][{}]", from, typeGroup);
... ... @@ -176,17 +289,18 @@ public class BaseRelationService implements RelationService {
176 289 ListenableFuture<List<EntityRelation>> relations = relationDao.findAllByFrom(from, typeGroup);
177 290 ListenableFuture<List<EntityRelationInfo>> relationsInfo = Futures.transform(relations,
178 291 (AsyncFunction<List<EntityRelation>, List<EntityRelationInfo>>) relations1 -> {
179   - List<ListenableFuture<EntityRelationInfo>> futures = new ArrayList<>();
  292 + List<ListenableFuture<EntityRelationInfo>> futures = new ArrayList<>();
180 293 relations1.stream().forEach(relation ->
181 294 futures.add(fetchRelationInfoAsync(relation,
182 295 relation2 -> relation2.getTo(),
183 296 (EntityRelationInfo relationInfo, String entityName) -> relationInfo.setToName(entityName)))
184 297 );
185 298 return Futures.successfulAsList(futures);
186   - });
  299 + });
187 300 return relationsInfo;
188 301 }
189 302
  303 + @Cacheable(cacheNames = RELATIONS_CACHE, key = "{#from, #relationType}")
190 304 @Override
191 305 public ListenableFuture<List<EntityRelation>> findByFromAndType(EntityId from, String relationType, RelationTypeGroup typeGroup) {
192 306 log.trace("Executing findByFromAndType [{}][{}][{}]", from, relationType, typeGroup);
... ... @@ -196,6 +310,7 @@ public class BaseRelationService implements RelationService {
196 310 return relationDao.findAllByFromAndType(from, relationType, typeGroup);
197 311 }
198 312
  313 + @Cacheable(cacheNames = RELATIONS_CACHE, key = "#to")
199 314 @Override
200 315 public ListenableFuture<List<EntityRelation>> findByTo(EntityId to, RelationTypeGroup typeGroup) {
201 316 log.trace("Executing findByTo [{}][{}]", to, typeGroup);
... ... @@ -214,9 +329,9 @@ public class BaseRelationService implements RelationService {
214 329 (AsyncFunction<List<EntityRelation>, List<EntityRelationInfo>>) relations1 -> {
215 330 List<ListenableFuture<EntityRelationInfo>> futures = new ArrayList<>();
216 331 relations1.stream().forEach(relation ->
217   - futures.add(fetchRelationInfoAsync(relation,
218   - relation2 -> relation2.getFrom(),
219   - (EntityRelationInfo relationInfo, String entityName) -> relationInfo.setFromName(entityName)))
  332 + futures.add(fetchRelationInfoAsync(relation,
  333 + relation2 -> relation2.getFrom(),
  334 + (EntityRelationInfo relationInfo, String entityName) -> relationInfo.setFromName(entityName)))
220 335 );
221 336 return Futures.successfulAsList(futures);
222 337 });
... ... @@ -236,6 +351,7 @@ public class BaseRelationService implements RelationService {
236 351 return entityRelationInfo;
237 352 }
238 353
  354 + @Cacheable(cacheNames = RELATIONS_CACHE, key = "{#to, #relationType}")
239 355 @Override
240 356 public ListenableFuture<List<EntityRelation>> findByToAndType(EntityId to, String relationType, RelationTypeGroup typeGroup) {
241 357 log.trace("Executing findByToAndType [{}][{}][{}]", to, relationType, typeGroup);
... ... @@ -417,5 +533,4 @@ public class BaseRelationService implements RelationService {
417 533 }
418 534 return relations;
419 535 }
420   -
421 536 }
... ...
... ... @@ -20,6 +20,7 @@ import com.google.common.collect.Lists;
20 20 import com.google.common.util.concurrent.*;
21 21 import lombok.extern.slf4j.Slf4j;
22 22 import org.springframework.beans.factory.annotation.Autowired;
  23 +import org.springframework.beans.factory.annotation.Value;
23 24 import org.springframework.data.domain.PageRequest;
24 25 import org.springframework.stereotype.Component;
25 26 import org.thingsboard.server.common.data.UUIDConverter;
... ... @@ -31,14 +32,17 @@ import org.thingsboard.server.dao.model.sql.TsKvLatestCompositeKey;
31 32 import org.thingsboard.server.dao.model.sql.TsKvLatestEntity;
32 33 import org.thingsboard.server.dao.sql.JpaAbstractDaoListeningExecutorService;
33 34 import org.thingsboard.server.dao.timeseries.TimeseriesDao;
  35 +import org.thingsboard.server.dao.timeseries.TsInsertExecutorType;
34 36 import org.thingsboard.server.dao.util.SqlDao;
35 37
36 38 import javax.annotation.Nullable;
  39 +import javax.annotation.PostConstruct;
37 40 import javax.annotation.PreDestroy;
38 41 import java.util.ArrayList;
39 42 import java.util.List;
40 43 import java.util.Optional;
41 44 import java.util.concurrent.CompletableFuture;
  45 +import java.util.concurrent.ExecutorService;
42 46 import java.util.concurrent.Executors;
43 47 import java.util.stream.Collectors;
44 48
... ... @@ -50,7 +54,13 @@ import static org.thingsboard.server.common.data.UUIDConverter.fromTimeUUID;
50 54 @SqlDao
51 55 public class JpaTimeseriesDao extends JpaAbstractDaoListeningExecutorService implements TimeseriesDao {
52 56
53   - private ListeningExecutorService insertService = MoreExecutors.listeningDecorator(Executors.newSingleThreadExecutor());
  57 + @Value("${sql.ts_inserts_executor_type}")
  58 + private String insertExecutorType;
  59 +
  60 + @Value("${sql.ts_inserts_fixed_thread_pool_size}")
  61 + private int insertFixedThreadPoolSize;
  62 +
  63 + private ListeningExecutorService insertService;
54 64
55 65 @Autowired
56 66 private TsKvRepository tsKvRepository;
... ... @@ -58,6 +68,32 @@ public class JpaTimeseriesDao extends JpaAbstractDaoListeningExecutorService imp
58 68 @Autowired
59 69 private TsKvLatestRepository tsKvLatestRepository;
60 70
  71 + @PostConstruct
  72 + public void init() {
  73 + Optional<TsInsertExecutorType> executorTypeOptional = TsInsertExecutorType.parse(insertExecutorType);
  74 + TsInsertExecutorType executorType;
  75 + if (executorTypeOptional.isPresent()) {
  76 + executorType = executorTypeOptional.get();
  77 + } else {
  78 + executorType = TsInsertExecutorType.FIXED;
  79 + }
  80 + switch (executorType) {
  81 + case SINGLE:
  82 + insertService = MoreExecutors.listeningDecorator(Executors.newSingleThreadExecutor());
  83 + break;
  84 + case FIXED:
  85 + int poolSize = insertFixedThreadPoolSize;
  86 + if (poolSize <= 0) {
  87 + poolSize = 10;
  88 + }
  89 + insertService = MoreExecutors.listeningDecorator(Executors.newFixedThreadPool(poolSize));
  90 + break;
  91 + case CACHED:
  92 + insertService = MoreExecutors.listeningDecorator(Executors.newCachedThreadPool());
  93 + break;
  94 + }
  95 + }
  96 +
61 97 @Override
62 98 public ListenableFuture<List<TsKvEntry>> findAllAsync(EntityId entityId, List<TsKvQuery> queries) {
63 99 List<ListenableFuture<List<TsKvEntry>>> futures = queries
... ... @@ -234,7 +270,7 @@ public class JpaTimeseriesDao extends JpaAbstractDaoListeningExecutorService imp
234 270 entity.setDoubleValue(tsKvEntry.getDoubleValue().orElse(null));
235 271 entity.setLongValue(tsKvEntry.getLongValue().orElse(null));
236 272 entity.setBooleanValue(tsKvEntry.getBooleanValue().orElse(null));
237   - log.trace("Saving entity: " + entity);
  273 + log.trace("Saving entity: {}", entity);
238 274 return insertService.submit(() -> {
239 275 tsKvRepository.save(entity);
240 276 return null;
... ... @@ -265,7 +301,9 @@ public class JpaTimeseriesDao extends JpaAbstractDaoListeningExecutorService imp
265 301
266 302 @PreDestroy
267 303 void onDestroy() {
268   - insertService.shutdown();
  304 + if (insertService != null) {
  305 + insertService.shutdown();
  306 + }
269 307 }
270 308
271 309 }
... ...
  1 +/**
  2 + * Copyright © 2016-2017 The Thingsboard Authors
  3 + *
  4 + * Licensed under the Apache License, Version 2.0 (the "License");
  5 + * you may not use this file except in compliance with the License.
  6 + * You may obtain a copy of the License at
  7 + *
  8 + * http://www.apache.org/licenses/LICENSE-2.0
  9 + *
  10 + * Unless required by applicable law or agreed to in writing, software
  11 + * distributed under the License is distributed on an "AS IS" BASIS,
  12 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 + * See the License for the specific language governing permissions and
  14 + * limitations under the License.
  15 + */
  16 +package org.thingsboard.server.dao.timeseries;
  17 +
  18 +import java.util.Optional;
  19 +
  20 +public enum TsInsertExecutorType {
  21 + SINGLE,
  22 + FIXED,
  23 + CACHED;
  24 +
  25 + public static Optional<TsInsertExecutorType> parse(String name) {
  26 + TsInsertExecutorType executorType = null;
  27 + if (name != null) {
  28 + for (TsInsertExecutorType type : TsInsertExecutorType.values()) {
  29 + if (type.name().equalsIgnoreCase(name)) {
  30 + executorType = type;
  31 + break;
  32 + }
  33 + }
  34 + }
  35 + return Optional.of(executorType);
  36 + }
  37 +}
... ...
... ... @@ -15,16 +15,14 @@
15 15 */
16 16 package org.thingsboard.server.dao.service;
17 17
18   -import com.hazelcast.core.HazelcastInstance;
19 18 import org.apache.commons.lang3.RandomStringUtils;
20 19 import org.junit.After;
21   -import org.junit.Assert;
22 20 import org.junit.Before;
23 21 import org.junit.Test;
24 22 import org.springframework.aop.framework.Advised;
25 23 import org.springframework.aop.support.AopUtils;
26 24 import org.springframework.beans.factory.annotation.Autowired;
27   -import org.springframework.test.context.TestPropertySource;
  25 +import org.springframework.cache.CacheManager;
28 26 import org.springframework.test.util.ReflectionTestUtils;
29 27 import org.thingsboard.server.common.data.CacheConstants;
30 28 import org.thingsboard.server.common.data.Device;
... ... @@ -40,7 +38,6 @@ import java.util.UUID;
40 38
41 39 import static org.mockito.Mockito.*;
42 40
43   -@TestPropertySource(properties = {"cache.enabled = true"})
44 41 public abstract class BaseDeviceCredentialsCacheTest extends AbstractServiceTest {
45 42
46 43 private static final String CREDENTIALS_ID_1 = RandomStringUtils.randomAlphanumeric(20);
... ... @@ -53,7 +50,7 @@ public abstract class BaseDeviceCredentialsCacheTest extends AbstractServiceTest
53 50 private DeviceService deviceService;
54 51
55 52 @Autowired
56   - private HazelcastInstance hazelcastInstance;
  53 + private CacheManager cacheManager;
57 54
58 55 private UUID deviceId = UUID.randomUUID();
59 56
... ... @@ -67,7 +64,7 @@ public abstract class BaseDeviceCredentialsCacheTest extends AbstractServiceTest
67 64
68 65 @After
69 66 public void cleanup() {
70   - hazelcastInstance.getMap(CacheConstants.DEVICE_CREDENTIALS_CACHE).evictAll();
  67 + cacheManager.getCache(CacheConstants.DEVICE_CREDENTIALS_CACHE).clear();
71 68 }
72 69
73 70 @Test
... ... @@ -77,7 +74,6 @@ public abstract class BaseDeviceCredentialsCacheTest extends AbstractServiceTest
77 74 deviceCredentialsService.findDeviceCredentialsByCredentialsId(CREDENTIALS_ID_1);
78 75 deviceCredentialsService.findDeviceCredentialsByCredentialsId(CREDENTIALS_ID_1);
79 76
80   - Assert.assertEquals(1, hazelcastInstance.getMap(CacheConstants.DEVICE_CREDENTIALS_CACHE).size());
81 77 verify(deviceCredentialsDao, times(1)).findByCredentialsId(CREDENTIALS_ID_1);
82 78 }
83 79
... ... @@ -88,17 +84,13 @@ public abstract class BaseDeviceCredentialsCacheTest extends AbstractServiceTest
88 84 deviceCredentialsService.findDeviceCredentialsByCredentialsId(CREDENTIALS_ID_1);
89 85 deviceCredentialsService.findDeviceCredentialsByCredentialsId(CREDENTIALS_ID_1);
90 86
91   - Assert.assertEquals(1, hazelcastInstance.getMap(CacheConstants.DEVICE_CREDENTIALS_CACHE).size());
92 87 verify(deviceCredentialsDao, times(1)).findByCredentialsId(CREDENTIALS_ID_1);
93 88
94 89 deviceCredentialsService.deleteDeviceCredentials(createDummyDeviceCredentials(CREDENTIALS_ID_1, deviceId));
95 90
96   - Assert.assertEquals(0, hazelcastInstance.getMap(CacheConstants.DEVICE_CREDENTIALS_CACHE).size());
97   -
98 91 deviceCredentialsService.findDeviceCredentialsByCredentialsId(CREDENTIALS_ID_1);
99 92 deviceCredentialsService.findDeviceCredentialsByCredentialsId(CREDENTIALS_ID_1);
100 93
101   - Assert.assertEquals(1, hazelcastInstance.getMap(CacheConstants.DEVICE_CREDENTIALS_CACHE).size());
102 94 verify(deviceCredentialsDao, times(2)).findByCredentialsId(CREDENTIALS_ID_1);
103 95 }
104 96
... ... @@ -109,7 +101,6 @@ public abstract class BaseDeviceCredentialsCacheTest extends AbstractServiceTest
109 101 deviceCredentialsService.findDeviceCredentialsByCredentialsId(CREDENTIALS_ID_1);
110 102 deviceCredentialsService.findDeviceCredentialsByCredentialsId(CREDENTIALS_ID_1);
111 103
112   - Assert.assertEquals(1, hazelcastInstance.getMap(CacheConstants.DEVICE_CREDENTIALS_CACHE).size());
113 104 verify(deviceCredentialsDao, times(1)).findByCredentialsId(CREDENTIALS_ID_1);
114 105
115 106 when(deviceCredentialsDao.findByDeviceId(deviceId)).thenReturn(createDummyDeviceCredentialsEntity(CREDENTIALS_ID_1));
... ... @@ -119,13 +110,11 @@ public abstract class BaseDeviceCredentialsCacheTest extends AbstractServiceTest
119 110 when(deviceService.findDeviceById(new DeviceId(deviceId))).thenReturn(new Device());
120 111
121 112 deviceCredentialsService.updateDeviceCredentials(createDummyDeviceCredentials(deviceCredentialsId, CREDENTIALS_ID_2, deviceId));
122   - Assert.assertEquals(0, hazelcastInstance.getMap(CacheConstants.DEVICE_CREDENTIALS_CACHE).size());
123 113
124 114 when(deviceCredentialsDao.findByCredentialsId(CREDENTIALS_ID_1)).thenReturn(null);
125 115
126 116 deviceCredentialsService.findDeviceCredentialsByCredentialsId(CREDENTIALS_ID_1);
127 117 deviceCredentialsService.findDeviceCredentialsByCredentialsId(CREDENTIALS_ID_1);
128   - Assert.assertEquals(0, hazelcastInstance.getMap(CacheConstants.DEVICE_CREDENTIALS_CACHE).size());
129 118
130 119 verify(deviceCredentialsDao, times(3)).findByCredentialsId(CREDENTIALS_ID_1);
131 120 }
... ...
  1 +/**
  2 + * Copyright © 2016-2017 The Thingsboard Authors
  3 + *
  4 + * Licensed under the Apache License, Version 2.0 (the "License");
  5 + * you may not use this file except in compliance with the License.
  6 + * You may obtain a copy of the License at
  7 + *
  8 + * http://www.apache.org/licenses/LICENSE-2.0
  9 + *
  10 + * Unless required by applicable law or agreed to in writing, software
  11 + * distributed under the License is distributed on an "AS IS" BASIS,
  12 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 + * See the License for the specific language governing permissions and
  14 + * limitations under the License.
  15 + */
  16 +package org.thingsboard.server.dao.service;
  17 +
  18 +import com.google.common.util.concurrent.Futures;
  19 +import org.junit.After;
  20 +import org.junit.Before;
  21 +import org.junit.Test;
  22 +import org.springframework.aop.framework.Advised;
  23 +import org.springframework.aop.support.AopUtils;
  24 +import org.springframework.beans.factory.annotation.Autowired;
  25 +import org.springframework.cache.CacheManager;
  26 +import org.springframework.test.util.ReflectionTestUtils;
  27 +import org.thingsboard.server.common.data.id.DeviceId;
  28 +import org.thingsboard.server.common.data.id.EntityId;
  29 +import org.thingsboard.server.common.data.relation.EntityRelation;
  30 +import org.thingsboard.server.common.data.relation.RelationTypeGroup;
  31 +import org.thingsboard.server.dao.relation.RelationDao;
  32 +import org.thingsboard.server.dao.relation.RelationService;
  33 +
  34 +import java.util.UUID;
  35 +import java.util.concurrent.ExecutionException;
  36 +
  37 +import static org.mockito.Mockito.*;
  38 +import static org.thingsboard.server.common.data.CacheConstants.RELATIONS_CACHE;
  39 +
  40 +public abstract class BaseRelationCacheTest extends AbstractServiceTest {
  41 +
  42 + private static final EntityId ENTITY_ID_FROM = new DeviceId(UUID.randomUUID());
  43 + private static final EntityId ENTITY_ID_TO = new DeviceId(UUID.randomUUID());
  44 + private static final String RELATION_TYPE = "Contains";
  45 +
  46 + @Autowired
  47 + private RelationService relationService;
  48 + @Autowired
  49 + private CacheManager cacheManager;
  50 +
  51 + private RelationDao relationDao;
  52 +
  53 + @Before
  54 + public void setup() throws Exception {
  55 + relationDao = mock(RelationDao.class);
  56 + ReflectionTestUtils.setField(unwrapRelationService(), "relationDao", relationDao);
  57 + }
  58 +
  59 + @After
  60 + public void cleanup() {
  61 + cacheManager.getCache(RELATIONS_CACHE).clear();
  62 + }
  63 +
  64 + private RelationService unwrapRelationService() throws Exception {
  65 + if (AopUtils.isAopProxy(relationService) && relationService instanceof Advised) {
  66 + Object target = ((Advised) relationService).getTargetSource().getTarget();
  67 + return (RelationService) target;
  68 + }
  69 + return null;
  70 + }
  71 +
  72 + @Test
  73 + public void testFindRelationByFrom_Cached() throws ExecutionException, InterruptedException {
  74 + when(relationDao.getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON))
  75 + .thenReturn(Futures.immediateFuture(new EntityRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE)));
  76 +
  77 + relationService.getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON);
  78 + relationService.getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON);
  79 +
  80 + verify(relationDao, times(1)).getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON);
  81 + }
  82 +
  83 + @Test
  84 + public void testDeleteRelations_EvictsCache() {
  85 + when(relationDao.getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON))
  86 + .thenReturn(Futures.immediateFuture(new EntityRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE)));
  87 +
  88 + relationService.getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON);
  89 + relationService.getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON);
  90 +
  91 + verify(relationDao, times(1)).getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON);
  92 +
  93 + relationService.deleteRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON);
  94 +
  95 + relationService.getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON);
  96 + relationService.getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON);
  97 +
  98 + verify(relationDao, times(2)).getRelation(ENTITY_ID_FROM, ENTITY_ID_TO, RELATION_TYPE, RelationTypeGroup.COMMON);
  99 +
  100 + }
  101 +}
... ...
  1 +/**
  2 + * Copyright © 2016-2017 The Thingsboard Authors
  3 + *
  4 + * Licensed under the Apache License, Version 2.0 (the "License");
  5 + * you may not use this file except in compliance with the License.
  6 + * You may obtain a copy of the License at
  7 + *
  8 + * http://www.apache.org/licenses/LICENSE-2.0
  9 + *
  10 + * Unless required by applicable law or agreed to in writing, software
  11 + * distributed under the License is distributed on an "AS IS" BASIS,
  12 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 + * See the License for the specific language governing permissions and
  14 + * limitations under the License.
  15 + */
  16 +package org.thingsboard.server.dao.service.nosql;
  17 +
  18 +import org.thingsboard.server.dao.service.BaseRelationCacheTest;
  19 +import org.thingsboard.server.dao.service.DaoNoSqlTest;
  20 +
  21 +@DaoNoSqlTest
  22 +public class RelationCacheNoSqlTest extends BaseRelationCacheTest {
  23 +}
... ...
  1 +/**
  2 + * Copyright © 2016-2017 The Thingsboard Authors
  3 + *
  4 + * Licensed under the Apache License, Version 2.0 (the "License");
  5 + * you may not use this file except in compliance with the License.
  6 + * You may obtain a copy of the License at
  7 + *
  8 + * http://www.apache.org/licenses/LICENSE-2.0
  9 + *
  10 + * Unless required by applicable law or agreed to in writing, software
  11 + * distributed under the License is distributed on an "AS IS" BASIS,
  12 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 + * See the License for the specific language governing permissions and
  14 + * limitations under the License.
  15 + */
  16 +package org.thingsboard.server.dao.service.sql;
  17 +
  18 +import org.thingsboard.server.dao.service.BaseRelationCacheTest;
  19 +import org.thingsboard.server.dao.service.DaoSqlTest;
  20 +
  21 +@DaoSqlTest
  22 +public class RelationCacheSqlTest extends BaseRelationCacheTest {
  23 +}
... ...
... ... @@ -9,4 +9,13 @@ zk.zk_dir=/thingsboard
9 9
10 10 updates.enabled=false
11 11
12   -audit_log.enabled=true
\ No newline at end of file
  12 +audit_log.enabled=true
  13 +
  14 +caching.specs.relations.timeToLiveInMinutes=1440
  15 +caching.specs.relations.maxSize=100000
  16 +
  17 +caching.specs.deviceCredentials.timeToLiveInMinutes=1440
  18 +caching.specs.deviceCredentials.maxSize=100000
  19 +
  20 +caching.specs.devices.timeToLiveInMinutes=1440
  21 +caching.specs.devices.maxSize=100000
... ...
1   - database.type=sql
  1 +database.type=sql
  2 +
  3 +sql.ts_inserts_executor_type=fixed
  4 +sql.ts_inserts_fixed_thread_pool_size=10
2 5
3 6 spring.jpa.show-sql=false
4 7 spring.jpa.hibernate.ddl-auto=validate
... ...
... ... @@ -108,12 +108,12 @@ public class MailPlugin extends AbstractPlugin<MailPluginConfiguration> implemen
108 108 MimeMessage mailMsg = mailSender.createMimeMessage();
109 109 MimeMessageHelper helper = new MimeMessageHelper(mailMsg, "UTF-8");
110 110 helper.setFrom(msg.getFrom());
111   - helper.setTo(msg.getTo());
  111 + helper.setTo(msg.getTo().split("\\s*,\\s*"));
112 112 if (!StringUtils.isEmpty(msg.getCc())) {
113   - helper.setCc(msg.getCc());
  113 + helper.setCc(msg.getCc().split("\\s*,\\s*"));
114 114 }
115 115 if (!StringUtils.isEmpty(msg.getBcc())) {
116   - helper.setBcc(msg.getBcc());
  116 + helper.setBcc(msg.getBcc().split("\\s*,\\s*"));
117 117 }
118 118 helper.setSubject(msg.getSubject());
119 119 helper.setText(msg.getBody());
... ...
... ... @@ -33,6 +33,7 @@ import org.thingsboard.server.extensions.core.plugin.telemetry.sub.SubscriptionU
33 33
34 34 import java.util.*;
35 35 import java.util.function.Function;
  36 +import java.util.function.Predicate;
36 37
37 38 /**
38 39 * @author Andrew Shvayka
... ... @@ -174,9 +175,13 @@ public class SubscriptionManager {
174 175 }
175 176
176 177 public void onLocalSubscriptionUpdate(PluginContext ctx, EntityId entityId, SubscriptionType type, Function<Subscription, List<TsKvEntry>> f) {
  178 + onLocalSubscriptionUpdate(ctx, entityId, s -> type == s.getType(), f);
  179 + }
  180 +
  181 + public void onLocalSubscriptionUpdate(PluginContext ctx, EntityId entityId, Predicate<Subscription> filter, Function<Subscription, List<TsKvEntry>> f) {
177 182 Set<Subscription> deviceSubscriptions = subscriptionsByEntityId.get(entityId);
178 183 if (deviceSubscriptions != null) {
179   - deviceSubscriptions.stream().filter(s -> type == s.getType()).forEach(s -> {
  184 + deviceSubscriptions.stream().filter(filter).forEach(s -> {
180 185 String sessionId = s.getWsSessionId();
181 186 List<TsKvEntry> subscriptionUpdate = f.apply(s);
182 187 if (!subscriptionUpdate.isEmpty()) {
... ... @@ -206,7 +211,7 @@ public class SubscriptionManager {
206 211 public void onAttributesUpdateFromServer(PluginContext ctx, EntityId entityId, String scope, List<AttributeKvEntry> attributes) {
207 212 Optional<ServerAddress> serverAddress = ctx.resolve(entityId);
208 213 if (!serverAddress.isPresent()) {
209   - onLocalSubscriptionUpdate(ctx, entityId, SubscriptionType.ATTRIBUTES, s -> {
  214 + onLocalSubscriptionUpdate(ctx, entityId, s -> SubscriptionType.ATTRIBUTES == s.getType() && scope.equals(s.getScope()), s -> {
210 215 List<TsKvEntry> subscriptionUpdate = new ArrayList<TsKvEntry>();
211 216 for (AttributeKvEntry kv : attributes) {
212 217 if (s.isAllKeys() || s.getKeyStates().containsKey(kv.getKey())) {
... ...
... ... @@ -114,7 +114,7 @@ public class TelemetryRpcMsgHandler implements RpcMsgHandler {
114 114 }
115 115 Map<String, Long> statesMap = proto.getKeyStatesList().stream().collect(Collectors.toMap(SubscriptionKetStateProto::getKey, SubscriptionKetStateProto::getTs));
116 116 Subscription subscription = new Subscription(
117   - new SubscriptionState(proto.getSessionId(), proto.getSubscriptionId(), EntityIdFactory.getByTypeAndId(proto.getEntityType(), proto.getEntityId()), SubscriptionType.valueOf(proto.getType()), proto.getAllKeys(), statesMap),
  117 + new SubscriptionState(proto.getSessionId(), proto.getSubscriptionId(), EntityIdFactory.getByTypeAndId(proto.getEntityType(), proto.getEntityId()), SubscriptionType.valueOf(proto.getType()), proto.getAllKeys(), statesMap, proto.getScope()),
118 118 false, msg.getServerAddress());
119 119 subscriptionManager.addRemoteWsSubscription(ctx, msg.getServerAddress(), proto.getSessionId(), subscription);
120 120 }
... ... @@ -127,6 +127,7 @@ public class TelemetryRpcMsgHandler implements RpcMsgHandler {
127 127 builder.setEntityId(cmd.getEntityId().getId().toString());
128 128 builder.setType(cmd.getType().name());
129 129 builder.setAllKeys(cmd.isAllKeys());
  130 + builder.setScope(cmd.getScope());
130 131 cmd.getKeyStates().entrySet().forEach(e -> builder.addKeyStates(SubscriptionKetStateProto.newBuilder().setKey(e.getKey()).setTs(e.getValue()).build()));
131 132 ctx.sendPluginRpcMsg(new RpcMsg(address, SUBSCRIPTION_CLAZZ, builder.build().toByteArray()));
132 133 }
... ...
... ... @@ -131,7 +131,7 @@ public class TelemetryWebsocketMsgHandler extends DefaultWebsocketMsgHandler {
131 131 keys.forEach(key -> subState.put(key, 0L));
132 132 attributesData.forEach(v -> subState.put(v.getKey(), v.getTs()));
133 133
134   - SubscriptionState sub = new SubscriptionState(sessionId, cmd.getCmdId(), entityId, SubscriptionType.ATTRIBUTES, false, subState);
  134 + SubscriptionState sub = new SubscriptionState(sessionId, cmd.getCmdId(), entityId, SubscriptionType.ATTRIBUTES, false, subState, cmd.getScope());
135 135 subscriptionManager.addLocalWsSubscription(ctx, sessionId, entityId, sub);
136 136 }
137 137
... ... @@ -168,7 +168,7 @@ public class TelemetryWebsocketMsgHandler extends DefaultWebsocketMsgHandler {
168 168 Map<String, Long> subState = new HashMap<>(attributesData.size());
169 169 attributesData.forEach(v -> subState.put(v.getKey(), v.getTs()));
170 170
171   - SubscriptionState sub = new SubscriptionState(sessionId, cmd.getCmdId(), entityId, SubscriptionType.ATTRIBUTES, true, subState);
  171 + SubscriptionState sub = new SubscriptionState(sessionId, cmd.getCmdId(), entityId, SubscriptionType.ATTRIBUTES, true, subState, cmd.getScope());
172 172 subscriptionManager.addLocalWsSubscription(ctx, sessionId, entityId, sub);
173 173 }
174 174
... ... @@ -234,7 +234,7 @@ public class TelemetryWebsocketMsgHandler extends DefaultWebsocketMsgHandler {
234 234 sendWsMsg(ctx, sessionRef, new SubscriptionUpdate(cmd.getCmdId(), data));
235 235 Map<String, Long> subState = new HashMap<>(data.size());
236 236 data.forEach(v -> subState.put(v.getKey(), v.getTs()));
237   - SubscriptionState sub = new SubscriptionState(sessionId, cmd.getCmdId(), entityId, SubscriptionType.TIMESERIES, true, subState);
  237 + SubscriptionState sub = new SubscriptionState(sessionId, cmd.getCmdId(), entityId, SubscriptionType.TIMESERIES, true, subState, cmd.getScope());
238 238 subscriptionManager.addLocalWsSubscription(ctx, sessionId, entityId, sub);
239 239 }
240 240
... ... @@ -262,7 +262,7 @@ public class TelemetryWebsocketMsgHandler extends DefaultWebsocketMsgHandler {
262 262 Map<String, Long> subState = new HashMap<>(keys.size());
263 263 keys.forEach(key -> subState.put(key, startTs));
264 264 data.forEach(v -> subState.put(v.getKey(), v.getTs()));
265   - SubscriptionState sub = new SubscriptionState(sessionId, cmd.getCmdId(), entityId, SubscriptionType.TIMESERIES, false, subState);
  265 + SubscriptionState sub = new SubscriptionState(sessionId, cmd.getCmdId(), entityId, SubscriptionType.TIMESERIES, false, subState, cmd.getScope());
266 266 subscriptionManager.addLocalWsSubscription(ctx, sessionId, entityId, sub);
267 267 }
268 268
... ...
... ... @@ -51,6 +51,10 @@ public class Subscription {
51 51 return getSub().getType();
52 52 }
53 53
  54 + public String getScope() {
  55 + return getSub().getScope();
  56 + }
  57 +
54 58 public boolean isAllKeys() {
55 59 return getSub().isAllKeys();
56 60 }
... ...
... ... @@ -33,6 +33,7 @@ public class SubscriptionState {
33 33 @Getter private final SubscriptionType type;
34 34 @Getter private final boolean allKeys;
35 35 @Getter private final Map<String, Long> keyStates;
  36 + @Getter private final String scope;
36 37
37 38 @Override
38 39 public boolean equals(Object o) {
... ...
... ... @@ -27,6 +27,7 @@ message SubscriptionProto {
27 27 string type = 5;
28 28 bool allKeys = 6;
29 29 repeated SubscriptionKetStateProto keyStates = 7;
  30 + string scope = 8;
30 31 }
31 32
32 33 message SubscriptionUpdateProto {
... ...
... ... @@ -43,6 +43,7 @@
43 43 <cassandra-unit.version>3.0.0.1</cassandra-unit.version>
44 44 <takari-cpsuite.version>1.2.7</takari-cpsuite.version>
45 45 <guava.version>18.0</guava.version>
  46 + <caffeine.version>2.6.1</caffeine.version>
46 47 <commons-lang3.version>3.4</commons-lang3.version>
47 48 <commons-validator.version>1.5.0</commons-validator.version>
48 49 <commons-io.version>2.5</commons-io.version>
... ... @@ -645,6 +646,11 @@
645 646 <version>${guava.version}</version>
646 647 </dependency>
647 648 <dependency>
  649 + <groupId>com.github.ben-manes.caffeine</groupId>
  650 + <artifactId>caffeine</artifactId>
  651 + <version>${caffeine.version}</version>
  652 + </dependency>
  653 + <dependency>
648 654 <groupId>com.google.protobuf</groupId>
649 655 <artifactId>protobuf-java</artifactId>
650 656 <version>${protobuf.version}</version>
... ...
... ... @@ -26,17 +26,17 @@ import lombok.extern.slf4j.Slf4j;
26 26 import org.eclipse.californium.core.CoapResource;
27 27 import org.eclipse.californium.core.CoapServer;
28 28 import org.eclipse.californium.core.network.CoapEndpoint;
  29 +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
29 30 import org.thingsboard.server.common.transport.SessionMsgProcessor;
30 31 import org.thingsboard.server.common.transport.auth.DeviceAuthService;
31 32 import org.thingsboard.server.transport.coap.adaptors.CoapTransportAdaptor;
32   -import org.slf4j.Logger;
33   -import org.slf4j.LoggerFactory;
34 33 import org.springframework.beans.factory.annotation.Autowired;
35 34 import org.springframework.beans.factory.annotation.Value;
36 35 import org.springframework.context.ApplicationContext;
37 36 import org.springframework.stereotype.Service;
38 37
39 38 @Service("CoapTransportService")
  39 +@ConditionalOnProperty(prefix = "coap", value = "enabled", havingValue = "true", matchIfMissing = true)
40 40 @Slf4j
41 41 public class CoapTransportService {
42 42
... ...
... ... @@ -24,6 +24,7 @@ import io.netty.util.ResourceLeakDetector;
24 24 import lombok.extern.slf4j.Slf4j;
25 25 import org.springframework.beans.factory.annotation.Autowired;
26 26 import org.springframework.beans.factory.annotation.Value;
  27 +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
27 28 import org.springframework.context.ApplicationContext;
28 29 import org.springframework.stereotype.Service;
29 30 import org.thingsboard.server.common.transport.SessionMsgProcessor;
... ... @@ -39,6 +40,7 @@ import javax.annotation.PreDestroy;
39 40 * @author Andrew Shvayka
40 41 */
41 42 @Service("MqttTransportService")
  43 +@ConditionalOnProperty(prefix = "mqtt", value = "enabled", havingValue = "true", matchIfMissing = false)
42 44 @Slf4j
43 45 public class MqttTransportService {
44 46
... ...
... ... @@ -84,16 +84,15 @@ public class GatewaySessionCtx {
84 84
85 85 private void onDeviceConnect(String deviceName, String deviceType) {
86 86 if (!devices.containsKey(deviceName)) {
87   - Optional<Device> deviceOpt = deviceService.findDeviceByTenantIdAndName(gateway.getTenantId(), deviceName);
88   - Device device = deviceOpt.orElseGet(() -> {
89   - Device newDevice = new Device();
90   - newDevice.setTenantId(gateway.getTenantId());
91   - newDevice.setName(deviceName);
92   - newDevice.setType(deviceType);
93   - newDevice = deviceService.saveDevice(newDevice);
94   - relationService.saveRelationAsync(new EntityRelation(gateway.getId(), newDevice.getId(), "Created"));
95   - return newDevice;
96   - });
  87 + Device device = deviceService.findDeviceByTenantIdAndName(gateway.getTenantId(), deviceName);
  88 + if (device == null) {
  89 + device = new Device();
  90 + device.setTenantId(gateway.getTenantId());
  91 + device.setName(deviceName);
  92 + device.setType(deviceType);
  93 + device = deviceService.saveDevice(device);
  94 + relationService.saveRelationAsync(new EntityRelation(gateway.getId(), device.getId(), "Created"));
  95 + }
97 96 GatewayDeviceSessionCtx ctx = new GatewayDeviceSessionCtx(this, device);
98 97 devices.put(deviceName, ctx);
99 98 log.debug("[{}] Added device [{}] to the gateway session", gatewaySessionId, deviceName);
... ...
... ... @@ -20,7 +20,13 @@ import UrlHandler from './url.handler';
20 20 export default function AppRun($rootScope, $window, $injector, $location, $log, $state, $mdDialog, $filter, loginService, userService, $translate) {
21 21
22 22 $window.Flow = Flow;
23   - var frame = $window.frameElement;
  23 + var frame = null;
  24 + try {
  25 + frame = $window.frameElement;
  26 + } catch(e) {
  27 + // ie11 fix
  28 + }
  29 +
24 30 var unauthorizedDialog = null;
25 31 var forbiddenDialog = null;
26 32
... ...
... ... @@ -553,7 +553,9 @@ function Utils($mdColorPalette, $rootScope, $window, $translate, $q, $timeout, t
553 553 var aspect = imageAspectMap[urlHashCode];
554 554 if (angular.isUndefined(aspect)) {
555 555 var testImage = document.createElement('img'); // eslint-disable-line
556   - testImage.style.visibility = 'hidden';
  556 + testImage.style.position = 'absolute';
  557 + testImage.style.left = '-99999px';
  558 + testImage.style.top = '-99999px';
557 559 testImage.onload = function() {
558 560 aspect = testImage.width / testImage.height;
559 561 document.body.removeChild(testImage); //eslint-disable-line
... ...
... ... @@ -1214,6 +1214,7 @@ export default angular.module('thingsboard.locale', [])
1214 1214 "remove-widget-text": "After the confirmation the widget and all related data will become unrecoverable.",
1215 1215 "timeseries": "Time series",
1216 1216 "search-data": "Search data",
  1217 + "no-data-found": "No data found",
1217 1218 "latest-values": "Latest values",
1218 1219 "rpc": "Control widget",
1219 1220 "alarm": "Alarm widget",
... ...
... ... @@ -340,7 +340,11 @@ function EntitiesTableWidgetController($element, $scope, $filter, $mdMedia, $tra
340 340 }
341 341 if (!style.width) {
342 342 var columnWidth = vm.columnWidth[key.label];
343   - style.width = columnWidth;
  343 + if(columnWidth !== "0px") {
  344 + style.width = columnWidth;
  345 + } else {
  346 + style.width = "auto";
  347 + }
344 348 }
345 349 return style;
346 350 }
... ...
... ... @@ -238,6 +238,7 @@ export default class TbFlot {
238 238 if (this.ticksFormatterFunction) {
239 239 return this.ticksFormatterFunction(value);
240 240 }
  241 +
241 242 var factor = this.tickDecimals ? Math.pow(10, this.tickDecimals) : 1,
242 243 formatted = "" + Math.round(value * factor) / factor;
243 244 if (this.tickDecimals != null) {
... ... @@ -248,9 +249,12 @@ export default class TbFlot {
248 249 formatted = (precision ? formatted : formatted + ".") + ("" + factor).substr(1, this.tickDecimals - precision);
249 250 }
250 251 }
251   - formatted += ' ' + this.tickUnits;
  252 + if (this.tickUnits) {
  253 + formatted += ' ' + this.tickUnits;
  254 + }
  255 +
252 256 return formatted;
253   - }
  257 + };
254 258
255 259 this.yaxis.tickFormatter = ctx.yAxisTickFormatter;
256 260
... ... @@ -262,6 +266,16 @@ export default class TbFlot {
262 266 this.yaxis.labelFont.color = this.yaxis.font.color;
263 267 this.yaxis.labelFont.size = this.yaxis.font.size+2;
264 268 this.yaxis.labelFont.weight = "bold";
  269 + if (angular.isNumber(settings.yaxis.tickSize)) {
  270 + this.yaxis.tickSize = settings.yaxis.tickSize;
  271 + } else {
  272 + this.yaxis.tickSize = null;
  273 + }
  274 + if (angular.isNumber(settings.yaxis.tickDecimals)) {
  275 + this.yaxis.tickDecimals = settings.yaxis.tickDecimals
  276 + } else {
  277 + this.yaxis.tickDecimals = null;
  278 + }
265 279 if (settings.yaxis.ticksFormatter && settings.yaxis.ticksFormatter.length) {
266 280 try {
267 281 this.yaxis.ticksFormatterFunction = new Function('value', settings.yaxis.ticksFormatter);
... ... @@ -405,9 +419,13 @@ export default class TbFlot {
405 419 }
406 420 }
407 421 series.lines = {
408   - fill: keySettings.fillLines === true,
409   - show: this.chartType === 'line' ? keySettings.showLines !== false : keySettings.showLines === true
  422 + fill: keySettings.fillLines === true
410 423 };
  424 + if (this.chartType === 'line' || this.chartType === 'state') {
  425 + series.lines.show = keySettings.showLines !== false
  426 + } else {
  427 + series.lines.show = keySettings.showLines === true;
  428 + }
411 429
412 430 if (angular.isDefined(keySettings.lineWidth)) {
413 431 series.lines.lineWidth = keySettings.lineWidth;
... ... @@ -487,9 +505,19 @@ export default class TbFlot {
487 505
488 506 createYAxis(keySettings, units) {
489 507 var yaxis = angular.copy(this.yaxis);
  508 + var tickDecimals, tickSize;
490 509
491 510 var label = keySettings.axisTitle && keySettings.axisTitle.length ? keySettings.axisTitle : yaxis.label;
492   - var tickDecimals = angular.isDefined(keySettings.axisTickDecimals) ? keySettings.axisTickDecimals : 0;
  511 + if (angular.isNumber(keySettings.axisTickDecimals)) {
  512 + tickDecimals = keySettings.axisTickDecimals;
  513 + } else {
  514 + tickDecimals = yaxis.tickDecimals;
  515 + }
  516 + if (angular.isNumber(keySettings.axisTickSize)) {
  517 + tickSize = keySettings.axisTickSize;
  518 + } else {
  519 + tickSize = yaxis.tickSize;
  520 + }
493 521 var position = keySettings.axisPosition && keySettings.axisPosition.length ? keySettings.axisPosition : "left";
494 522
495 523 var min = angular.isDefined(keySettings.axisMin) ? keySettings.axisMin : yaxis.min;
... ... @@ -500,6 +528,7 @@ export default class TbFlot {
500 528 yaxis.max = max;
501 529 yaxis.tickUnits = units;
502 530 yaxis.tickDecimals = tickDecimals;
  531 + yaxis.tickSize = tickSize;
503 532 yaxis.alignTicksWithAxis = position == "right" ? 1 : null;
504 533 yaxis.position = position;
505 534
... ... @@ -545,7 +574,7 @@ export default class TbFlot {
545 574 }
546 575 }
547 576 yaxis.hidden = hidden;
548   - var newIndex = -1;
  577 + var newIndex = 1;
549 578 if (!yaxis.hidden) {
550 579 this.options.yaxes.push(yaxis);
551 580 newIndex = this.options.yaxes.length;
... ... @@ -928,6 +957,16 @@ export default class TbFlot {
928 957 "title": "Ticks formatter function, f(value)",
929 958 "type": "string",
930 959 "default": ""
  960 + },
  961 + "tickDecimals": {
  962 + "title": "The number of decimals to display",
  963 + "type": "number",
  964 + "default": 0
  965 + },
  966 + "tickSize": {
  967 + "title": "Step size between ticks",
  968 + "type": "number",
  969 + "default": null
931 970 }
932 971 }
933 972 }
... ... @@ -986,6 +1025,8 @@ export default class TbFlot {
986 1025 "items": [
987 1026 "yaxis.min",
988 1027 "yaxis.max",
  1028 + "yaxis.tickDecimals",
  1029 + "yaxis.tickSize",
989 1030 "yaxis.showLabels",
990 1031 "yaxis.title",
991 1032 "yaxis.titleAngle",
... ... @@ -1010,24 +1051,24 @@ export default class TbFlot {
1010 1051
1011 1052 static datakeySettingsSchema(defaultShowLines) {
1012 1053 return {
1013   - "schema": {
  1054 + "schema": {
1014 1055 "type": "object",
1015   - "title": "DataKeySettings",
1016   - "properties": {
  1056 + "title": "DataKeySettings",
  1057 + "properties": {
1017 1058 "showLines": {
1018 1059 "title": "Show lines",
1019   - "type": "boolean",
1020   - "default": defaultShowLines
  1060 + "type": "boolean",
  1061 + "default": defaultShowLines
1021 1062 },
1022 1063 "fillLines": {
1023 1064 "title": "Fill lines",
1024   - "type": "boolean",
1025   - "default": false
  1065 + "type": "boolean",
  1066 + "default": false
1026 1067 },
1027 1068 "showPoints": {
1028 1069 "title": "Show points",
1029   - "type": "boolean",
1030   - "default": false
  1070 + "type": "boolean",
  1071 + "default": false
1031 1072 },
1032 1073 "tooltipValueFormatter": {
1033 1074 "title": "Tooltip value format function, f(value)",
... ... @@ -1059,6 +1100,11 @@ export default class TbFlot {
1059 1100 "type": "number",
1060 1101 "default": 0
1061 1102 },
  1103 + "axisTickSize": {
  1104 + "title": "Axis step size between ticks",
  1105 + "type": "number",
  1106 + "default": null
  1107 + },
1062 1108 "axisPosition": {
1063 1109 "title": "Axis position",
1064 1110 "type": "string",
... ... @@ -1072,7 +1118,7 @@ export default class TbFlot {
1072 1118 },
1073 1119 "required": ["showLines", "fillLines", "showPoints"]
1074 1120 },
1075   - "form": [
  1121 + "form": [
1076 1122 "showLines",
1077 1123 "fillLines",
1078 1124 "showPoints",
... ... @@ -1085,6 +1131,7 @@ export default class TbFlot {
1085 1131 "axisMax",
1086 1132 "axisTitle",
1087 1133 "axisTickDecimals",
  1134 + "axisTickSize",
1088 1135 {
1089 1136 "key": "axisPosition",
1090 1137 "type": "rc-select",
... ... @@ -1401,4 +1448,4 @@ export default class TbFlot {
1401 1448 }
1402 1449 }
1403 1450
1404   -/* eslint-enable angular/angularelement */
\ No newline at end of file
  1451 +/* eslint-enable angular/angularelement */
... ...
... ... @@ -276,7 +276,7 @@ export default class TbMapWidget {
276 276 this.locationsSettings[i].useMarkerImage = true;
277 277 var url = this.ctx.settings.markerImage;
278 278 var size = this.ctx.settings.markerImageSize || 34;
279   - this.locationSettings.currentImage = {
  279 + this.locationsSettings[i].currentImage = {
280 280 url: url,
281 281 size: size
282 282 };
... ...
... ... @@ -18,6 +18,7 @@ import tinycolor from 'tinycolor2';
18 18 import TbGoogleMap from './google-map';
19 19 import TbOpenStreetMap from './openstreet-map';
20 20 import TbImageMap from './image-map';
  21 +import TbTencentMap from './tencent-map';
21 22
22 23 import {processPattern, arraysEqual, toLabelValueMap, fillPattern, fillPatternWithActions} from './widget-utils';
23 24
... ... @@ -83,6 +84,8 @@ export default class TbMapWidgetV2 {
83 84 settings.posFunction,
84 85 settings.imageEntityAlias,
85 86 settings.imageUrlAttribute);
  87 + } else if (mapProvider === 'tencent-map') {
  88 + this.map = new TbTencentMap($element,this.utils, initCallback, this.defaultZoomLevel, this.dontFitMapBounds, minZoomLevel, settings.tmApiKey, settings.tmDefaultMapType);
86 89 }
87 90 }
88 91
... ... @@ -466,6 +469,8 @@ export default class TbMapWidgetV2 {
466 469 schema = angular.copy(openstreetMapSettingsSchema);
467 470 } else if (mapProvider === 'image-map') {
468 471 return imageMapSettingsSchema;
  472 + } else if (mapProvider === 'tencent-map') {
  473 + schema = angular.copy(tencentMapSettingsSchema);
469 474 }
470 475 angular.merge(schema.schema.properties, commonMapSettingsSchema.schema.properties);
471 476 schema.schema.required = schema.schema.required.concat(commonMapSettingsSchema.schema.required);
... ... @@ -544,7 +549,51 @@ const googleMapSettingsSchema =
544 549 }
545 550 ]
546 551 };
547   -
  552 +
  553 +const tencentMapSettingsSchema =
  554 + {
  555 + "schema":{
  556 + "title":"Tencent Map Configuration",
  557 + "type":"object",
  558 + "properties":{
  559 + "tmApiKey":{
  560 + "title":"Tencent Maps API Key",
  561 + "type":"string"
  562 + },
  563 + "tmDefaultMapType":{
  564 + "title":"Default map type",
  565 + "type":"string",
  566 + "default":"roadmap"
  567 + }
  568 + },
  569 + "required":[
  570 + "tmApiKey"
  571 + ]
  572 + },
  573 + "form":[
  574 + "tmApiKey",
  575 + {
  576 + "key":"tmDefaultMapType",
  577 + "type":"rc-select",
  578 + "multiple":false,
  579 + "items":[
  580 + {
  581 + "value":"roadmap",
  582 + "label":"Roadmap"
  583 + },
  584 + {
  585 + "value":"satellite",
  586 + "label":"Satellite"
  587 + },
  588 + {
  589 + "value":"hybrid",
  590 + "label":"Hybrid"
  591 + },
  592 + ]
  593 + }
  594 + ]
  595 + };
  596 +
548 597 const openstreetMapSettingsSchema =
549 598 {
550 599 "schema":{
... ...
  1 +/*
  2 + * Copyright © 2016-2017 The Thingsboard Authors
  3 + *
  4 + * Licensed under the Apache License, Version 2.0 (the "License");
  5 + * you may not use this file except in compliance with the License.
  6 + * You may obtain a copy of the License at
  7 + *
  8 + * http://www.apache.org/licenses/LICENSE-2.0
  9 + *
  10 + * Unless required by applicable law or agreed to in writing, software
  11 + * distributed under the License is distributed on an "AS IS" BASIS,
  12 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 + * See the License for the specific language governing permissions and
  14 + * limitations under the License.
  15 + */
  16 +
  17 +var tmGlobals = {
  18 + loadingTmId: null,
  19 + tmApiKeys: {}
  20 +}
  21 +
  22 +export default class TbTencentMap {
  23 + constructor($containerElement,utils, initCallback, defaultZoomLevel, dontFitMapBounds, minZoomLevel, tmApiKey, tmDefaultMapType) {
  24 + var tbMap = this;
  25 + this.utils = utils;
  26 + this.defaultZoomLevel = defaultZoomLevel;
  27 + this.dontFitMapBounds = dontFitMapBounds;
  28 + this.minZoomLevel = minZoomLevel;
  29 + this.tooltips = [];
  30 + this.defaultMapType = tmDefaultMapType;
  31 +
  32 + function clearGlobalId() {
  33 + if (tmGlobals.loadingTmId && tmGlobals.loadingTmId === tbMap.mapId) {
  34 + tmGlobals.loadingTmId = null;
  35 + }
  36 + }
  37 +
  38 + function displayError(message) {
  39 + $containerElement.html( // eslint-disable-line angular/angularelement
  40 + "<div class='error'>"+ message + "</div>"
  41 + );
  42 + }
  43 +
  44 + function initTencentMap() {
  45 + tbMap.map = new qq.maps.Map($containerElement[0], { // eslint-disable-line no-undef
  46 + scrollwheel: true,
  47 + mapTypeId: getTencentMapTypeId(tbMap.defaultMapType),
  48 + zoom: tbMap.defaultZoomLevel || 8
  49 + });
  50 +
  51 + if (initCallback) {
  52 + initCallback();
  53 + }
  54 + }
  55 +
  56 + /* eslint-disable no-undef */
  57 +
  58 + function getTencentMapTypeId(mapType) {
  59 + var mapTypeId =qq.maps.MapTypeId.ROADMAP;
  60 + if (mapType) {
  61 + if (mapType === 'hybrid') {
  62 + mapTypeId = qq.maps.MapTypeId.HYBRID;
  63 + } else if (mapType === 'satellite') {
  64 + mapTypeId = qq.maps.MapTypeId.SATELLITE;
  65 + } else if (mapType === 'terrain') {
  66 + mapTypeId = qq.maps.MapTypeId.ROADMAP;
  67 + }
  68 + }
  69 + return mapTypeId;
  70 + }
  71 +
  72 + /* eslint-enable no-undef */
  73 +
  74 + this.mapId = '' + Math.random().toString(36).substr(2, 9);
  75 + this.apiKey = tmApiKey || '84d6d83e0e51e481e50454ccbe8986b';
  76 +
  77 + window.tm_authFailure = function() { // eslint-disable-line no-undef, angular/window-service
  78 + if (tmGlobals.loadingTmId && tmGlobals.loadingTmId === tbMap.mapId) {
  79 + tmGlobals.loadingTmId = null;
  80 + tmGlobals.tmApiKeys[tbMap.apiKey].error = 'Unable to authentificate for tencent Map API.</br>Please check your API key.';
  81 + displayError(tmGlobals.tmApiKeys[tbMap.apiKey].error);
  82 + }
  83 + };
  84 +
  85 + this.initMapFunctionName = 'initTencentMap_' + this.mapId;
  86 +
  87 + window[this.initMapFunctionName] = function() { // eslint-disable-line no-undef, angular/window-service
  88 + tmGlobals.tmApiKeys[tbMap.apiKey].loaded = true;
  89 + initTencentMap();
  90 + for (var p = 0; p < tmGlobals.tmApiKeys[tbMap.apiKey].pendingInits.length; p++) {
  91 + var pendingInit = tmGlobals.tmApiKeys[tbMap.apiKey].pendingInits[p];
  92 + pendingInit();
  93 + }
  94 + tmGlobals.tmApiKeys[tbMap.apiKey].pendingInits = [];
  95 + };
  96 + if (this.apiKey && this.apiKey.length > 0) {
  97 + if (tmGlobals.tmApiKeys[this.apiKey]) {
  98 + if (tmGlobals.tmApiKeys[this.apiKey].error) {
  99 + displayError(tmGlobals.tmApiKeys[this.apiKey].error);
  100 + } else if (tmGlobals.tmApiKeys[this.apiKey].loaded) {
  101 + initTencentMap();
  102 + } else {
  103 + tmGlobals.tmApiKeys[this.apiKey].pendingInits.push(initTencentMap);
  104 + }
  105 + } else {
  106 + tmGlobals.tmApiKeys[this.apiKey] = {
  107 + loaded: false,
  108 + pendingInits: []
  109 + };
  110 + var tencentMapScriptRes = 'https://map.qq.com/api/js?v=2.exp&key='+this.apiKey+'&callback='+this.initMapFunctionName;
  111 +
  112 + tmGlobals.loadingTmId = this.mapId;
  113 + lazyLoad.load({ type: 'js', path: tencentMapScriptRes }).then( // eslint-disable-line no-undef
  114 + function success() {
  115 + setTimeout(clearGlobalId, 2000); // eslint-disable-line no-undef, angular/timeout-service
  116 + },
  117 + function fail(e) {
  118 + clearGlobalId();
  119 + tmGlobals.tmApiKeys[tbMap.apiKey].error = 'tencent map api load failed!</br>'+e;
  120 + displayError(tmGlobals.tmApiKeys[tbMap.apiKey].error);
  121 + }
  122 + );
  123 + }
  124 + } else {
  125 + displayError('No tencent Map Api Key provided!');
  126 + }
  127 + }
  128 +
  129 + inited() {
  130 + return angular.isDefined(this.map);
  131 + }
  132 +
  133 + createMarkerLabelStyle(settings) {
  134 + return {
  135 + width: "200px",
  136 + textAlign: "center",
  137 + color: settings.labelColor,
  138 + background: "none",
  139 + border: "none",
  140 + fontSize: "12px",
  141 + fontFamily: "\"Helvetica Neue\", Arial, Helvetica, sans-serif",
  142 + fontWeight: "bold"
  143 + };
  144 + }
  145 +
  146 + /* eslint-disable no-undef,no-unused-vars*/
  147 + updateMarkerLabel(marker, settings) {
  148 + if (marker.label) {
  149 + marker.label.setContent(settings.labelText);
  150 + marker.label.setStyle(this.createMarkerLabelStyle(settings));
  151 + }
  152 + }
  153 + /* eslint-enable no-undef,no-unused-vars */
  154 +
  155 + /* eslint-disable no-undef,no-unused-vars */
  156 + updateMarkerColor(marker, color) {
  157 + this.createDefaultMarkerIcon(marker, color, (iconInfo) => {
  158 + marker.setIcon(iconInfo.icon);
  159 + });
  160 + }
  161 + /* eslint-enable no-undef,,no-unused-vars */
  162 +
  163 + /* eslint-disable no-undef */
  164 + updateMarkerIcon(marker, settings) {
  165 + this.createMarkerIcon(marker, settings, (iconInfo) => {
  166 + marker.setIcon(iconInfo.icon);
  167 + if (marker.label) {
  168 + marker.label.setOffset(new qq.maps.Size(-100, -iconInfo.size[1]-20));
  169 + }
  170 + });
  171 + }
  172 + /* eslint-disable no-undef */
  173 +
  174 + /* eslint-disable no-undef */
  175 + createMarkerIcon(marker, settings, onMarkerIconReady) {
  176 + var currentImage = settings.currentImage;
  177 + var tMap = this;
  178 + if (currentImage && currentImage.url) {
  179 + this.utils.loadImageAspect(currentImage.url).then(
  180 + (aspect) => {
  181 + if (aspect) {
  182 + var width;
  183 + var height;
  184 + if (aspect > 1) {
  185 + width = currentImage.size;
  186 + height = currentImage.size / aspect;
  187 + } else {
  188 + width = currentImage.size * aspect;
  189 + height = currentImage.size;
  190 + }
  191 + var icon = new qq.maps.MarkerImage(currentImage.url,
  192 + new qq.maps.Size(width, height),
  193 + new qq.maps.Point(0,0),
  194 + new qq.maps.Point(width/2, height),
  195 + new qq.maps.Size(width, height));
  196 + var iconInfo = {
  197 + size: [width, height],
  198 + icon: icon
  199 + };
  200 + onMarkerIconReady(iconInfo);
  201 + } else {
  202 + tMap.createDefaultMarkerIcon(marker, settings.color, onMarkerIconReady);
  203 + }
  204 + }
  205 + );
  206 + } else {
  207 + this.createDefaultMarkerIcon(marker, settings.color, onMarkerIconReady);
  208 + }
  209 + }
  210 + /* eslint-enable no-undef */
  211 +
  212 + /* eslint-disable no-undef */
  213 + createDefaultMarkerIcon(marker, color, onMarkerIconReady) {
  214 + var pinColor = color.substr(1);
  215 + var icon = new qq.maps.MarkerImage("https://chart.apis.google.com/chart?chst=d_map_pin_letter_withshadow&chld=%E2%80%A2|" + pinColor,
  216 + new qq.maps.Size(40, 37),
  217 + new qq.maps.Point(0,0),
  218 + new qq.maps.Point(10, 37));
  219 + var iconInfo = {
  220 + size: [40, 37],
  221 + icon: icon
  222 + };
  223 + onMarkerIconReady(iconInfo);
  224 + }
  225 + /* eslint-enable no-undef */
  226 +
  227 + /* eslint-disable no-undef */
  228 + createMarker(location, settings, onClickListener, markerArgs) {
  229 + var marker = new qq.maps.Marker({
  230 + position: location
  231 + });
  232 + var tMap = this;
  233 + this.createMarkerIcon(marker, settings, (iconInfo) => {
  234 + marker.setIcon(iconInfo.icon);
  235 + marker.setMap(tMap.map);
  236 + if (settings.showLabel) {
  237 + marker.label = new qq.maps.Label({
  238 + clickable: false,
  239 + content: settings.labelText,
  240 + offset: new qq.maps.Size(-100, -iconInfo.size[1]-20),
  241 + style: tMap.createMarkerLabelStyle(settings),
  242 + visible: true,
  243 + position: location,
  244 + map: tMap.map,
  245 + zIndex: 1000
  246 + });
  247 + }
  248 + });
  249 +
  250 + if (settings.displayTooltip) {
  251 + this.createTooltip(marker, settings.tooltipPattern, settings.tooltipReplaceInfo, settings.autocloseTooltip, markerArgs);
  252 + }
  253 +
  254 + if (onClickListener) {
  255 + qq.maps.event.addListener(marker, 'click', onClickListener);
  256 + }
  257 +
  258 + return marker;
  259 + }
  260 +
  261 + /* eslint-disable no-undef */
  262 + removeMarker(marker) {
  263 + marker.setMap(null);
  264 + if (marker.label) {
  265 + marker.label.setMap(null);
  266 + }
  267 + }
  268 +
  269 + /* eslint-enable no-undef */
  270 +
  271 + /* eslint-disable no-undef */
  272 + createTooltip(marker, pattern, replaceInfo, autoClose, markerArgs) {
  273 + var popup = new qq.maps.InfoWindow({
  274 + map :this.map
  275 + });
  276 + var map = this;
  277 + qq.maps.event.addListener(marker, 'click', function() {
  278 + if (autoClose) {
  279 + map.tooltips.forEach((tooltip) => {
  280 + tooltip.popup.close();
  281 + });
  282 + }
  283 + popup.open();
  284 + popup.setPosition(marker);
  285 + });
  286 + this.tooltips.push( {
  287 + markerArgs: markerArgs,
  288 + popup: popup,
  289 + pattern: pattern,
  290 + replaceInfo: replaceInfo
  291 + });
  292 + }
  293 + /* eslint-enable no-undef */
  294 +
  295 + /* eslint-disable no-undef */
  296 + updatePolylineColor(polyline, settings, color) {
  297 + var options = {
  298 + path: polyline.getPath(),
  299 + strokeColor: color,
  300 + strokeOpacity: settings.strokeOpacity,
  301 + strokeWeight: settings.strokeWeight,
  302 + map: this.map
  303 + };
  304 + polyline.setOptions(options);
  305 + }
  306 + /* eslint-enable no-undef */
  307 +
  308 + /* eslint-disable no-undef */
  309 + createPolyline(locations, settings) {
  310 + var polyline = new qq.maps.Polyline({
  311 + path: locations,
  312 + strokeColor: settings.color,
  313 + strokeOpacity: settings.strokeOpacity,
  314 + strokeWeight: settings.strokeWeight,
  315 + map: this.map
  316 + });
  317 +
  318 + return polyline;
  319 + }
  320 + /* eslint-enable no-undef */
  321 +
  322 + removePolyline(polyline) {
  323 + polyline.setMap(null);
  324 + }
  325 +
  326 + /* eslint-disable no-undef ,no-unused-vars*/
  327 + fitBounds(bounds) {
  328 + if (this.dontFitMapBounds && this.defaultZoomLevel) {
  329 + this.map.setZoom(this.defaultZoomLevel);
  330 + this.map.setCenter(bounds.getCenter());
  331 + } else {
  332 + var tbMap = this;
  333 + qq.maps.event.addListenerOnce(this.map, 'bounds_changed', function() { // eslint-disable-line no-undef
  334 + if (!tbMap.defaultZoomLevel && tbMap.map.getZoom() > tbMap.minZoomLevel) {
  335 + tbMap.map.setZoom(tbMap.minZoomLevel);
  336 + }
  337 + });
  338 + this.map.fitBounds(bounds);
  339 + }
  340 + }
  341 + /* eslint-enable no-undef,no-unused-vars */
  342 +
  343 + createLatLng(lat, lng) {
  344 + return new qq.maps.LatLng(lat, lng); // eslint-disable-line no-undef
  345 + }
  346 +
  347 + extendBoundsWithMarker(bounds, marker) {
  348 + bounds.extend(marker.getPosition());
  349 + }
  350 +
  351 + getMarkerPosition(marker) {
  352 + return marker.getPosition();
  353 + }
  354 +
  355 + setMarkerPosition(marker, latLng) {
  356 + marker.setPosition(latLng);
  357 + if (marker.label) {
  358 + marker.label.setPosition(latLng);
  359 + }
  360 + }
  361 +
  362 + getPolylineLatLngs(polyline) {
  363 + return polyline.getPath().getArray();
  364 + }
  365 +
  366 + setPolylineLatLngs(polyline, latLngs) {
  367 + polyline.setPath(latLngs);
  368 + }
  369 +
  370 + createBounds() {
  371 + return new qq.maps.LatLngBounds(); // eslint-disable-line no-undef
  372 + }
  373 +
  374 + extendBounds(bounds, polyline) {
  375 + if (polyline && polyline.getPath()) {
  376 + var locations = polyline.getPath();
  377 + for (var i = 0; i < locations.getLength(); i++) {
  378 + bounds.extend(locations.getAt(i));
  379 + }
  380 + }
  381 + }
  382 +
  383 + invalidateSize() {
  384 + qq.maps.event.trigger(this.map, "resize"); // eslint-disable-line no-undef
  385 + }
  386 +
  387 + getTooltips() {
  388 + return this.tooltips;
  389 + }
  390 +
  391 +}
... ...
... ... @@ -41,7 +41,7 @@
41 41 <md-tabs flex md-selected="vm.sourceIndex" ng-class="{'tb-headless': vm.sources.length === 1}"
42 42 id="tabs" md-border-bottom flex>
43 43 <md-tab ng-repeat="source in vm.sources" label="{{ source.datasource.name }}">
44   - <md-table-container>
  44 + <md-table-container class="flex">
45 45 <table md-table>
46 46 <thead md-head md-order="source.query.order" md-on-reorder="vm.onReorder(source)">
47 47 <tr md-row>
... ... @@ -70,16 +70,20 @@
70 70 </tr>
71 71 </tbody>
72 72 </table>
  73 + <md-divider></md-divider>
  74 + <span ng-show="!vm.sources[vm.sourceIndex].data.length"
  75 + layout-align="center center"
  76 + class="no-data-found" translate>widget.no-data-found</span>
73 77 </md-table-container>
74   - <md-table-pagination ng-if="vm.displayPagination"
75   - md-limit="source.query.limit"
76   - md-limit-options="vm.limitOptions"
77   - md-page="source.query.page"
78   - md-total="{{source.data.length}}"
79   - md-on-paginate="vm.onPaginate(source)"
80   - md-page-select>
81   - </md-table-pagination>
82 78 </md-tab>
83 79 </md-tabs>
  80 + <md-table-pagination ng-if="vm.displayPagination"
  81 + md-limit="vm.sources[vm.sourceIndex].query.limit"
  82 + md-limit-options="vm.limitOptions"
  83 + md-page="vm.sources[vm.sourceIndex].query.page"
  84 + md-total="{{vm.sources[vm.sourceIndex].data.length}}"
  85 + md-on-paginate="vm.onPaginate(vm.sources[vm.sourceIndex])"
  86 + md-page-select>
  87 + </md-table-pagination>
84 88 </div>
85 89 </div>
\ No newline at end of file
... ...