Commit 38f81df839e7039693bd21891f9b75bf9a761a65
Committed by
GitHub
Merge branch 'master' into dashboard_logo
Showing
72 changed files
with
3100 additions
and
439 deletions
Too many changes to show.
To preserve performance only 72 of 259 files are displayed.
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "alarm_widgets", |
4 | 4 | "title": "Alarm widgets", |
5 | 5 | "image": "", |
6 | - "description": "Useful for visualization of alarms for specific entities both in real-time and history mode." | |
6 | + "description": "Visualization of alarms for devices, assets and other entities." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "alarms_table", |
11 | 11 | "name": "Alarms table", |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Displays alarms based on defined time window and other filters.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "alarm", |
16 | 16 | "sizeX": 10.5, |
... | ... | @@ -19,8 +19,8 @@ |
19 | 19 | "templateHtml": "<tb-alarms-table-widget \n [ctx]=\"ctx\">\n</tb-alarms-table-widget>", |
20 | 20 | "templateCss": "", |
21 | 21 | "controllerScript": "self.onInit = function() {\n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.alarmsTableWidget.onDataUpdated();\n}\n\nself.actionSources = function() {\n return {\n 'actionCellButton': {\n name: 'widget-action.action-cell-button',\n multiple: true\n },\n 'rowClick': {\n name: 'widget-action.row-click',\n multiple: false\n }\n };\n}\n\nself.onDestroy = function() {\n}\n", |
22 | - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"AlarmTableSettings\",\n \"properties\": {\n \"alarmsTitle\": {\n \"title\": \"Alarms table title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"enableSelection\": {\n \"title\": \"Enable alarms selection\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableSearch\": {\n \"title\": \"Enable alarms search\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableSelectColumnDisplay\": {\n \"title\": \"Enable select columns to display\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableFilter\": {\n \"title\": \"Enable alarm filter\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyAction\": {\n \"title\": \"Always display actions column\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"displayDetails\": {\n \"title\": \"Display alarm details\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"allowAcknowledgment\": {\n \"title\": \"Allow alarms acknowledgment\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"allowClear\": {\n \"title\": \"Allow alarms clear\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayPagination\": {\n \"title\": \"Display pagination\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"defaultPageSize\": {\n \"title\": \"Default page size\",\n \"type\": \"number\",\n \"default\": 10\n },\n \"defaultSortOrder\": {\n \"title\": \"Default sort order\",\n \"type\": \"string\",\n \"default\": \"-createdTime\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"alarmsTitle\",\n \"enableSelection\",\n \"enableSearch\",\n \"enableSelectColumnDisplay\",\n \"enableFilter\",\n \"enableStickyAction\",\n \"displayDetails\",\n \"allowAcknowledgment\",\n \"allowClear\",\n \"displayPagination\",\n \"defaultPageSize\",\n \"defaultSortOrder\"\n ]\n}", | |
23 | - "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"columnWidth\": {\n \"title\": \"Column width (px or %)\",\n \"type\": \"string\",\n \"default\": \"0px\"\n },\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, alarm, filter)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"columnWidth\",\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\"\n }\n ]\n}", | |
22 | + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"AlarmTableSettings\",\n \"properties\": {\n \"alarmsTitle\": {\n \"title\": \"Alarms table title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"enableSelection\": {\n \"title\": \"Enable alarms selection\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableSearch\": {\n \"title\": \"Enable alarms search\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableSelectColumnDisplay\": {\n \"title\": \"Enable select columns to display\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableFilter\": {\n \"title\": \"Enable alarm filter\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyHeader\": {\n \"title\": \"Always display header\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyAction\": {\n \"title\": \"Always display actions column\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"displayDetails\": {\n \"title\": \"Display alarm details\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"allowAcknowledgment\": {\n \"title\": \"Allow alarms acknowledgment\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"allowClear\": {\n \"title\": \"Allow alarms clear\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayPagination\": {\n \"title\": \"Display pagination\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"defaultPageSize\": {\n \"title\": \"Default page size\",\n \"type\": \"number\",\n \"default\": 10\n },\n \"defaultSortOrder\": {\n \"title\": \"Default sort order\",\n \"type\": \"string\",\n \"default\": \"-createdTime\"\n },\n \"useRowStyleFunction\": {\n \"title\": \"Use row style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"rowStyleFunction\": {\n \"title\": \"Row style function: f(alarm, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"alarmsTitle\",\n \"enableSelection\",\n \"enableSearch\",\n \"enableSelectColumnDisplay\",\n \"enableFilter\",\n \"enableStickyHeader\",\n \"enableStickyAction\",\n \"displayDetails\",\n \"allowAcknowledgment\",\n \"allowClear\",\n \"displayPagination\",\n \"defaultPageSize\",\n \"defaultSortOrder\",\n \"useRowStyleFunction\",\n {\n \"key\": \"rowStyleFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useRowStyleFunction === true\"\n }\n ]\n}", | |
23 | + "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"columnWidth\": {\n \"title\": \"Column width (px or %)\",\n \"type\": \"string\",\n \"default\": \"0px\"\n },\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value, alarm, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, alarm, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"defaultColumnVisibility\": {\n \"title\": \"Default column visibility\",\n \"type\": \"string\",\n \"default\": \"visible\"\n },\n \"columnSelectionToDisplay\": {\n \"title\": \"Column selection in 'Columns to Display'\",\n \"type\": \"string\",\n \"default\": \"enabled\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"columnWidth\",\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useCellStyleFunction === true\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useCellContentFunction === true\"\n },\n {\n \"key\": \"defaultColumnVisibility\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"visible\",\n \"label\": \"Visible\"\n },\n {\n \"value\": \"hidden\",\n \"label\": \"Hidden\"\n } \n ]\n },\n {\n \"key\": \"columnSelectionToDisplay\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"enabled\",\n \"label\": \"Enabled\"\n },\n {\n \"value\": \"disabled\",\n \"label\": \"Disabled\"\n } \n ]\n }\n ]\n}", | |
24 | 24 | "defaultConfig": "{\"timewindow\":{\"realtime\":{\"interval\":1000,\"timewindowMs\":86400000},\"aggregation\":{\"type\":\"NONE\",\"limit\":200}},\"showTitle\":true,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"4px\",\"settings\":{\"enableSelection\":true,\"enableSearch\":true,\"displayDetails\":true,\"allowAcknowledgment\":true,\"allowClear\":true,\"displayPagination\":true,\"defaultPageSize\":10,\"defaultSortOrder\":\"-createdTime\",\"enableSelectColumnDisplay\":true,\"enableStickyAction\":false,\"enableFilter\":true},\"title\":\"Alarms table\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400,\"padding\":\"5px 10px 5px 10px\"},\"useDashboardTimewindow\":false,\"showLegend\":false,\"alarmSource\":{\"type\":\"function\",\"dataKeys\":[{\"name\":\"createdTime\",\"type\":\"alarm\",\"label\":\"Created time\",\"color\":\"#2196f3\",\"settings\":{\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.021092237451093787},{\"name\":\"originator\",\"type\":\"alarm\",\"label\":\"Originator\",\"color\":\"#4caf50\",\"settings\":{\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.2780007688856758},{\"name\":\"type\",\"type\":\"alarm\",\"label\":\"Type\",\"color\":\"#f44336\",\"settings\":{\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.7323586880398418},{\"name\":\"severity\",\"type\":\"alarm\",\"label\":\"Severity\",\"color\":\"#ffc107\",\"settings\":{\"useCellStyleFunction\":false,\"useCellContentFunction\":false},\"_hash\":0.09927019860088193},{\"name\":\"status\",\"type\":\"alarm\",\"label\":\"Status\",\"color\":\"#607d8b\",\"settings\":{\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.6588418951443418}],\"entityAliasId\":null,\"name\":\"alarms\"},\"alarmSearchStatus\":\"ANY\",\"alarmsPollingInterval\":5,\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"widgetStyle\":{},\"displayTimewindow\":true,\"actions\":{},\"alarmStatusList\":[],\"alarmSeverityList\":[],\"alarmTypeList\":[],\"searchPropagatedAlarms\":false}" |
25 | 25 | } |
26 | 26 | } | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "analogue_gauges", |
4 | 4 | "title": "Analogue gauges", |
5 | 5 | "image": "", |
6 | - "description": "Similar to digital gauges, but have a different style." | |
6 | + "description": "Display temperature, humidity, speed, and other latest values on various analog gauge widgets." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "analogue_compass", |
11 | - "name": "Analogue Compass", | |
11 | + "name": "Compass", | |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Displays latest value of the attribute or timeseries key on the compass. Expects value to be in range of 0 to 360.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "latest", |
16 | 16 | "sizeX": 6, |
... | ... | @@ -26,9 +26,9 @@ |
26 | 26 | }, |
27 | 27 | { |
28 | 28 | "alias": "temperature_gauge_canvas_gauges", |
29 | - "name": "Temperature gauge - Canvas Gauges", | |
29 | + "name": "Thermometer scale", | |
30 | 30 | "image": "", |
31 | - "description": null, | |
31 | + "description": "Preconfigured widget to display temperature. Allows to configure temperature range, gradient colors and other settings.", | |
32 | 32 | "descriptor": { |
33 | 33 | "type": "latest", |
34 | 34 | "sizeX": 7, |
... | ... | @@ -44,9 +44,9 @@ |
44 | 44 | }, |
45 | 45 | { |
46 | 46 | "alias": "temperature_radial_gauge_canvas_gauges", |
47 | - "name": "Temperature radial gauge - Canvas Gauges", | |
47 | + "name": "Temperature radial gauge", | |
48 | 48 | "image": "", |
49 | - "description": null, | |
49 | + "description": "Preconfigured gauge to display temperature. Allows to configure temperature range, gradient colors and other settings.", | |
50 | 50 | "descriptor": { |
51 | 51 | "type": "latest", |
52 | 52 | "sizeX": 6, |
... | ... | @@ -62,9 +62,9 @@ |
62 | 62 | }, |
63 | 63 | { |
64 | 64 | "alias": "speed_gauge_canvas_gauges", |
65 | - "name": "Speed gauge - Canvas Gauges", | |
65 | + "name": "Speed gauge", | |
66 | 66 | "image": "", |
67 | - "description": null, | |
67 | + "description": "Preconfigured gauge to display speed. Allows to configure speed range, gradient colors and other settings.", | |
68 | 68 | "descriptor": { |
69 | 69 | "type": "latest", |
70 | 70 | "sizeX": 7, |
... | ... | @@ -80,9 +80,9 @@ |
80 | 80 | }, |
81 | 81 | { |
82 | 82 | "alias": "radial_gauge_canvas_gauges", |
83 | - "name": "Radial gauge - Canvas Gauges", | |
83 | + "name": "Radial gauge", | |
84 | 84 | "image": "", |
85 | - "description": null, | |
85 | + "description": "Preconfigured gauge to display any value reading. Allows to configure value range, gradient colors and other settings.", | |
86 | 86 | "descriptor": { |
87 | 87 | "type": "latest", |
88 | 88 | "sizeX": 6, | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "cards", |
4 | 4 | "title": "Cards", |
5 | 5 | "image": "", |
6 | - "description": "Useful for visualization of timeseries data or attributes in a table or card widget." | |
6 | + "description": "Tables and cards to display latest and historical values for multiple entities simultaneously." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "attributes_card", |
11 | 11 | "name": "Attributes card", |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Displays one or more latest values of the entity. Supports multiple entities.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "latest", |
16 | 16 | "sizeX": 7.5, |
... | ... | @@ -28,7 +28,7 @@ |
28 | 28 | "alias": "html_card", |
29 | 29 | "name": "HTML Card", |
30 | 30 | "image": "", |
31 | - "description": null, | |
31 | + "description": "Useful to inject custom HTML code. Designed to displays static information only.", | |
32 | 32 | "descriptor": { |
33 | 33 | "type": "static", |
34 | 34 | "sizeX": 7.5, |
... | ... | @@ -46,7 +46,7 @@ |
46 | 46 | "alias": "timeseries_table", |
47 | 47 | "name": "Timeseries table", |
48 | 48 | "image": "", |
49 | - "description": null, | |
49 | + "description": "Displays time series data for one or more entities. Data for each entity is displayed in a separate tab.", | |
50 | 50 | "descriptor": { |
51 | 51 | "type": "timeseries", |
52 | 52 | "sizeX": 8, |
... | ... | @@ -55,8 +55,8 @@ |
55 | 55 | "templateHtml": "<tb-timeseries-table-widget \n [ctx]=\"ctx\">\n</tb-timeseries-table-widget>", |
56 | 56 | "templateCss": "", |
57 | 57 | "controllerScript": "self.onInit = function() {\n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.timeseriesTableWidget.onDataUpdated();\n}\n\nself.typeParameters = function() {\n return {\n ignoreDataUpdateOnIntervalTick: true\n };\n}\n\nself.actionSources = function() {\n return {\n 'actionCellButton': {\n name: 'widget-action.action-cell-button',\n multiple: true\n },\n 'rowClick': {\n name: 'widget-action.row-click',\n multiple: false\n }\n };\n}\n\nself.onDestroy = function() {\n}", |
58 | - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"TimeseriesTableSettings\",\n \"properties\": {\n \"showTimestamp\": {\n \"title\": \"Display timestamp column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"showMilliseconds\": {\n \"title\": \"Display timestamp milliseconds\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"displayPagination\": {\n \"title\": \"Display pagination\",\n \"type\": \"boolean\",\n \"default\": true\n }, \n \"defaultPageSize\": {\n \"title\": \"Default page size\",\n \"type\": \"number\",\n \"default\": 10\n },\n \"hideEmptyLines\": {\n \"title\": \"Hide empty lines\",\n \"type\": \"boolean\",\n \"default\": false\n }\n },\n \"required\": []\n },\n \"form\": [\n \"showTimestamp\",\n \"showMilliseconds\",\n \"displayPagination\",\n \"defaultPageSize\",\n \"hideEmptyLines\"\n ]\n}", | |
59 | - "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, rowData, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\"\n }\n ]\n}", | |
58 | + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"TimeseriesTableSettings\",\n \"properties\": {\n \"enableSearch\": {\n \"title\": \"Enable search\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyHeader\": {\n \"title\": \"Always display header\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyAction\": {\n \"title\": \"Always display actions column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"showTimestamp\": {\n \"title\": \"Display timestamp column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"showMilliseconds\": {\n \"title\": \"Display timestamp milliseconds\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"displayPagination\": {\n \"title\": \"Display pagination\",\n \"type\": \"boolean\",\n \"default\": true\n }, \n \"defaultPageSize\": {\n \"title\": \"Default page size\",\n \"type\": \"number\",\n \"default\": 10\n },\n \"hideEmptyLines\": {\n \"title\": \"Hide empty lines\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"disableStickyHeader\": {\n \"title\": \"Disable sticky header\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"useRowStyleFunction\": {\n \"title\": \"Use row style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"rowStyleFunction\": {\n \"title\": \"Row style function: f(rowData, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"enableSearch\",\n \"enableStickyHeader\",\n \"enableStickyAction\",\n \"showTimestamp\",\n \"showMilliseconds\",\n \"displayPagination\",\n \"defaultPageSize\",\n \"hideEmptyLines\",\n \"useRowStyleFunction\",\n {\n \"key\": \"rowStyleFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useRowStyleFunction === true\"\n }\n ]\n}", | |
59 | + "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value, rowData, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, rowData, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useCellStyleFunction === true\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useCellContentFunction === true\"\n }\n ]\n}", | |
60 | 60 | "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temperature °C\",\"color\":\"#2196f3\",\"settings\":{\"useCellStyleFunction\":true,\"cellStyleFunction\":\"if (value) {\\n var percent = (value + 60)/120 * 100;\\n var color = tinycolor.mix('blue', 'red', amount = percent);\\n color.setAlpha(.5);\\n return {\\n paddingLeft: '20px',\\n color: '#ffffff',\\n background: color.toRgbString(),\\n fontSize: '18px'\\n };\\n} else {\\n return {};\\n}\"},\"_hash\":0.8587686344902596,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nvar multiplier = Math.pow(10, 1 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Humidity, %\",\"color\":\"#ffc107\",\"settings\":{\"useCellStyleFunction\":true,\"cellStyleFunction\":\"if (value) {\\n var percent = value;\\n var backgroundColor = tinycolor('blue');\\n backgroundColor.setAlpha(value/100);\\n var color = 'blue';\\n if (value > 50) {\\n color = 'white';\\n }\\n \\n return {\\n paddingLeft: '20px',\\n color: color,\\n background: backgroundColor.toRgbString(),\\n fontSize: '18px'\\n };\\n} else {\\n return {};\\n}\",\"useCellContentFunction\":false},\"_hash\":0.12775350966079668,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nvar multiplier = Math.pow(10, 1 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < 5) {\\n\\tvalue = 5;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"interval\":1000,\"timewindowMs\":60000},\"aggregation\":{\"type\":\"NONE\",\"limit\":200}},\"showTitle\":true,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"showTimestamp\":true,\"displayPagination\":true,\"defaultPageSize\":10},\"title\":\"Timeseries table\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400,\"padding\":\"5px 10px 5px 10px\"},\"useDashboardTimewindow\":false,\"showLegend\":false,\"widgetStyle\":{},\"actions\":{},\"showTitleIcon\":false,\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\"}" |
61 | 61 | } |
62 | 62 | }, |
... | ... | @@ -64,7 +64,7 @@ |
64 | 64 | "alias": "html_value_card", |
65 | 65 | "name": "HTML Value Card", |
66 | 66 | "image": "", |
67 | - "description": null, | |
67 | + "description": "Displays configurable HTML with ability to inject values from the selected datasource. For example, display single or multiple attribute values.", | |
68 | 68 | "descriptor": { |
69 | 69 | "type": "latest", |
70 | 70 | "sizeX": 7.5, |
... | ... | @@ -82,7 +82,7 @@ |
82 | 82 | "alias": "simple_card", |
83 | 83 | "name": "Simple card", |
84 | 84 | "image": "", |
85 | - "description": null, | |
85 | + "description": "Designed to display single value of the selected attribute or timeseries data. Widget styles are customizable.", | |
86 | 86 | "descriptor": { |
87 | 87 | "type": "latest", |
88 | 88 | "sizeX": 5, |
... | ... | @@ -100,7 +100,7 @@ |
100 | 100 | "alias": "label_widget", |
101 | 101 | "name": "Label widget", |
102 | 102 | "image": "", |
103 | - "description": null, | |
103 | + "description": "Displays static image and multiple values of selected attributes or timeseries keys on top of it. Position of the values on the image is configurable using advanced settings.", | |
104 | 104 | "descriptor": { |
105 | 105 | "type": "latest", |
106 | 106 | "sizeX": 4.5, |
... | ... | @@ -118,7 +118,7 @@ |
118 | 118 | "alias": "entities_table", |
119 | 119 | "name": "Entities table", |
120 | 120 | "image": "", |
121 | - "description": null, | |
121 | + "description": "Displays list of entities that match selected alias and filter with ability of additional full text search and pagination. Highly customizable using widget styles, data source keys and widget actions.", | |
122 | 122 | "descriptor": { |
123 | 123 | "type": "latest", |
124 | 124 | "sizeX": 7.5, |
... | ... | @@ -127,8 +127,8 @@ |
127 | 127 | "templateHtml": "<tb-entities-table-widget \n [ctx]=\"ctx\">\n</tb-entities-table-widget>", |
128 | 128 | "templateCss": "", |
129 | 129 | "controllerScript": "self.onInit = function() {\n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.entitiesTableWidget.onDataUpdated();\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n hasDataPageLink: true,\n warnOnPageDataOverflow: false,\n dataKeysOptional: true\n };\n}\n\nself.actionSources = function() {\n return {\n 'actionCellButton': {\n name: 'widget-action.action-cell-button',\n multiple: true\n },\n 'rowClick': {\n name: 'widget-action.row-click',\n multiple: false\n },\n 'rowDoubleClick': {\n name: 'widget-action.row-double-click',\n multiple: false\n }\n };\n}\n\nself.onDestroy = function() {\n}\n", |
130 | - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"entitiesTitle\": {\n \"title\": \"Entities table title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"enableSearch\": {\n \"title\": \"Enable entities search\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableSelectColumnDisplay\": {\n \"title\": \"Enable select columns to display\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayEntityName\": {\n \"title\": \"Display entity name column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"entityNameColumnTitle\": {\n \"title\": \"Entity name column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityLabel\": {\n \"title\": \"Display entity label column\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"entityLabelColumnTitle\": {\n \"title\": \"Entity label column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityType\": {\n \"title\": \"Display entity type column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayPagination\": {\n \"title\": \"Display pagination\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"defaultPageSize\": {\n \"title\": \"Default page size\",\n \"type\": \"number\",\n \"default\": 10\n },\n \"defaultSortOrder\": {\n \"title\": \"Default sort order\",\n \"type\": \"string\",\n \"default\": \"entityName\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"entitiesTitle\",\n \"enableSearch\",\n \"enableSelectColumnDisplay\",\n \"displayEntityName\",\n \"entityNameColumnTitle\",\n \"displayEntityLabel\",\n \"entityLabelColumnTitle\",\n \"displayEntityType\",\n \"displayPagination\",\n \"defaultPageSize\",\n \"defaultSortOrder\"\n ]\n}", | |
131 | - "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"columnWidth\": {\n \"title\": \"Column width (px or %)\",\n \"type\": \"string\",\n \"default\": \"0px\"\n },\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, entity, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"columnWidth\",\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\"\n }\n ]\n}", | |
130 | + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"entitiesTitle\": {\n \"title\": \"Entities table title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"enableSearch\": {\n \"title\": \"Enable entities search\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableSelectColumnDisplay\": {\n \"title\": \"Enable select columns to display\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyHeader\": {\n \"title\": \"Always display header\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyAction\": {\n \"title\": \"Always display actions column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayEntityName\": {\n \"title\": \"Display entity name column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"entityNameColumnTitle\": {\n \"title\": \"Entity name column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityLabel\": {\n \"title\": \"Display entity label column\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"entityLabelColumnTitle\": {\n \"title\": \"Entity label column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityType\": {\n \"title\": \"Display entity type column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayPagination\": {\n \"title\": \"Display pagination\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"defaultPageSize\": {\n \"title\": \"Default page size\",\n \"type\": \"number\",\n \"default\": 10\n },\n \"defaultSortOrder\": {\n \"title\": \"Default sort order\",\n \"type\": \"string\",\n \"default\": \"entityName\"\n },\n \"useRowStyleFunction\": {\n \"title\": \"Use row style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"rowStyleFunction\": {\n \"title\": \"Row style function: f(entity, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"entitiesTitle\",\n \"enableSearch\",\n \"enableSelectColumnDisplay\",\n \"enableStickyHeader\",\n \"enableStickyAction\",\n \"displayEntityName\",\n \"entityNameColumnTitle\",\n \"displayEntityLabel\",\n \"entityLabelColumnTitle\",\n \"displayEntityType\",\n \"displayPagination\",\n \"defaultPageSize\",\n \"defaultSortOrder\",\n \"useRowStyleFunction\",\n {\n \"key\": \"rowStyleFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useRowStyleFunction === true\"\n }\n ]\n}", | |
131 | + "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"columnWidth\": {\n \"title\": \"Column width (px or %)\",\n \"type\": \"string\",\n \"default\": \"0px\"\n },\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value, entity, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, entity, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"defaultColumnVisibility\": {\n \"title\": \"Default column visibility\",\n \"type\": \"string\",\n \"default\": \"visible\"\n },\n \"columnSelectionToDisplay\": {\n \"title\": \"Column selection in 'Columns to Display'\",\n \"type\": \"string\",\n \"default\": \"enabled\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"columnWidth\",\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useCellStyleFunction === true\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useCellContentFunction === true\"\n },\n {\n \"key\": \"defaultColumnVisibility\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"visible\",\n \"label\": \"Visible\"\n },\n {\n \"value\": \"hidden\",\n \"label\": \"Hidden\"\n } \n ]\n },\n {\n \"key\": \"columnSelectionToDisplay\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"enabled\",\n \"label\": \"Enabled\"\n },\n {\n \"value\": \"disabled\",\n \"label\": \"Disabled\"\n } \n ]\n }\n ]\n}", | |
132 | 132 | "defaultConfig": "{\"timewindow\":{\"realtime\":{\"interval\":1000,\"timewindowMs\":86400000},\"aggregation\":{\"type\":\"NONE\",\"limit\":200}},\"showTitle\":true,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"4px\",\"settings\":{\"enableSelection\":true,\"enableSearch\":true,\"displayDetails\":true,\"displayPagination\":true,\"defaultPageSize\":10,\"defaultSortOrder\":\"entityName\",\"displayEntityName\":true,\"displayEntityType\":true},\"title\":\"Entities table\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400,\"padding\":\"5px 10px 5px 10px\"},\"useDashboardTimewindow\":false,\"showLegend\":false,\"datasources\":[{\"type\":\"function\",\"name\":\"Simulated\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Sin\",\"color\":\"#2196f3\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.472295003170325,\"funcBody\":\"return Math.round(1000*Math.sin(time/5000));\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Cos\",\"color\":\"#4caf50\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.8926244886945558,\"funcBody\":\"return Math.round(1000*Math.cos(time/5000));\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#f44336\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.6401141393938932,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}]}" |
133 | 133 | } |
134 | 134 | }, |
... | ... | @@ -136,7 +136,7 @@ |
136 | 136 | "alias": "entities_hierarchy", |
137 | 137 | "name": "Entities hierarchy", |
138 | 138 | "image": "", |
139 | - "description": null, | |
139 | + "description": "Displays hierarchy of entities based on their relations. The root of the hierarchy is defined using entity alias. By default, displays entities related using \"Contains\" relation. You may change the behaviour using advanced settings.", | |
140 | 140 | "descriptor": { |
141 | 141 | "type": "latest", |
142 | 142 | "sizeX": 7.5, | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "charts", |
4 | 4 | "title": "Charts", |
5 | 5 | "image": "", |
6 | - "description": "Useful for visualization of historical or real-time data with a time window." | |
6 | + "description": "Display timeseries data using customizable line and bar charts. Use various pie charts to display latest values." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "bars", |
11 | - "name": "Bars - Chart.js", | |
11 | + "name": "Bars", | |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Displays latest values of the attributes or timeseries data for multiple entities as separate bars.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "latest", |
16 | 16 | "sizeX": 7, |
... | ... | @@ -30,9 +30,9 @@ |
30 | 30 | }, |
31 | 31 | { |
32 | 32 | "alias": "doughnut_chart_js", |
33 | - "name": "Doughnut - Chart.js", | |
33 | + "name": "Doughnut", | |
34 | 34 | "image": "", |
35 | - "description": null, | |
35 | + "description": "Displays latest values of the attributes or timeseries data for multiple entities in a doughnut chart. Supports numeric values only.", | |
36 | 36 | "descriptor": { |
37 | 37 | "type": "latest", |
38 | 38 | "sizeX": 7, |
... | ... | @@ -52,9 +52,9 @@ |
52 | 52 | }, |
53 | 53 | { |
54 | 54 | "alias": "pie", |
55 | - "name": "Pie - Flot", | |
55 | + "name": "Pie- Flot", | |
56 | 56 | "image": "", |
57 | - "description": null, | |
57 | + "description": "Displays latest values of the attributes or timeseries data for multiple entities in a pie chart. Supports numeric values only.", | |
58 | 58 | "descriptor": { |
59 | 59 | "type": "latest", |
60 | 60 | "sizeX": 8, |
... | ... | @@ -72,7 +72,7 @@ |
72 | 72 | "alias": "pie_chart_js", |
73 | 73 | "name": "Pie - Chart.js", |
74 | 74 | "image": "", |
75 | - "description": null, | |
75 | + "description": "Displays latest values of the attributes or timeseries data for multiple entities in a pie chart. Supports numeric values only.", | |
76 | 76 | "descriptor": { |
77 | 77 | "type": "latest", |
78 | 78 | "sizeX": 8, |
... | ... | @@ -92,9 +92,9 @@ |
92 | 92 | }, |
93 | 93 | { |
94 | 94 | "alias": "polar_area_chart_js", |
95 | - "name": "Polar Area - Chart.js", | |
95 | + "name": "Polar Area", | |
96 | 96 | "image": "", |
97 | - "description": null, | |
97 | + "description": "Displays latest values of the attributes or timeseries data for multiple entities in a polar area chart. Supports numeric values only.", | |
98 | 98 | "descriptor": { |
99 | 99 | "type": "latest", |
100 | 100 | "sizeX": 7, |
... | ... | @@ -114,9 +114,9 @@ |
114 | 114 | }, |
115 | 115 | { |
116 | 116 | "alias": "radar_chart_js", |
117 | - "name": "Radar - Chart.js", | |
117 | + "name": "Radar", | |
118 | 118 | "image": "", |
119 | - "description": null, | |
119 | + "description": "Displays latest values of the attributes or timeseries data for multiple entities in a radar chart. Supports numeric values only.", | |
120 | 120 | "descriptor": { |
121 | 121 | "type": "latest", |
122 | 122 | "sizeX": 7, |
... | ... | @@ -138,7 +138,7 @@ |
138 | 138 | "alias": "state_chart", |
139 | 139 | "name": "State Chart", |
140 | 140 | "image": "", |
141 | - "description": null, | |
141 | + "description": "Displays changes to the state of the entity over time. For example, online and offline.", | |
142 | 142 | "descriptor": { |
143 | 143 | "type": "timeseries", |
144 | 144 | "sizeX": 8, |
... | ... | @@ -154,9 +154,9 @@ |
154 | 154 | }, |
155 | 155 | { |
156 | 156 | "alias": "basic_timeseries", |
157 | - "name": "Timeseries - Flot", | |
157 | + "name": "Timeseries Line Chart", | |
158 | 158 | "image": "", |
159 | - "description": null, | |
159 | + "description": "Displays changes to timeseries data over time. For example, temperature or humidity readings.", | |
160 | 160 | "descriptor": { |
161 | 161 | "type": "timeseries", |
162 | 162 | "sizeX": 8, |
... | ... | @@ -172,9 +172,9 @@ |
172 | 172 | }, |
173 | 173 | { |
174 | 174 | "alias": "timeseries_bars_flot", |
175 | - "name": "Timeseries Bars - Flot", | |
175 | + "name": "Timeseries Bar Chart", | |
176 | 176 | "image": "", |
177 | - "description": null, | |
177 | + "description": "Displays changes to timeseries data over time. For example, daily water consumption for last month.", | |
178 | 178 | "descriptor": { |
179 | 179 | "type": "timeseries", |
180 | 180 | "sizeX": 8, | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "control_widgets", |
4 | 4 | "title": "Control widgets", |
5 | 5 | "image": "", |
6 | - "description": "Useful for visualization of current state and sending RPC commands to target devices." | |
6 | + "description": "Send commands to devices." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "rpc_debug_terminal", |
11 | 11 | "name": "RPC debug terminal", |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Allows to send any RPC command using it's name and parameters to device. Useful for debug.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "rpc", |
16 | 16 | "sizeX": 9.5, |
... | ... | @@ -28,7 +28,7 @@ |
28 | 28 | "alias": "rpc_remote_shell", |
29 | 29 | "name": "RPC remote shell", |
30 | 30 | "image": "", |
31 | - "description": null, | |
31 | + "description": "Allows to send emulate remote shell. Requires custom implementation on the target device to work properly.", | |
32 | 32 | "descriptor": { |
33 | 33 | "type": "rpc", |
34 | 34 | "sizeX": 9.5, |
... | ... | @@ -46,7 +46,7 @@ |
46 | 46 | "alias": "knob_control", |
47 | 47 | "name": "Knob Control", |
48 | 48 | "image": "", |
49 | - "description": null, | |
49 | + "description": "Sends the command to device with specified value each time user changes the value. Uses 'setValue' and 'getValue' RPC calls by default. The name of the RPC calls is configurable in advanced settings.", | |
50 | 50 | "descriptor": { |
51 | 51 | "type": "rpc", |
52 | 52 | "sizeX": 5, |
... | ... | @@ -64,7 +64,7 @@ |
64 | 64 | "alias": "switch_control", |
65 | 65 | "name": "Switch Control", |
66 | 66 | "image": "", |
67 | - "description": null, | |
67 | + "description": "Allows to send the RPC call to device when user toggle the switch. Advanced widget settings allow you to configure how to fetch the initial value of the switch.", | |
68 | 68 | "descriptor": { |
69 | 69 | "type": "rpc", |
70 | 70 | "sizeX": 4, |
... | ... | @@ -82,7 +82,7 @@ |
82 | 82 | "alias": "round_switch", |
83 | 83 | "name": "Round switch", |
84 | 84 | "image": "", |
85 | - "description": null, | |
85 | + "description": "Allows to send the RPC call to device when user toggle the switch. Advanced widget settings allow you to configure how to fetch the initial value of the switch.", | |
86 | 86 | "descriptor": { |
87 | 87 | "type": "rpc", |
88 | 88 | "sizeX": 2.5, |
... | ... | @@ -100,7 +100,7 @@ |
100 | 100 | "alias": "led_indicator", |
101 | 101 | "name": "Led indicator", |
102 | 102 | "image": "", |
103 | - "description": null, | |
103 | + "description": "Visualize the state of the device. Fetches the value from device using RPC or using attribute subscription.", | |
104 | 104 | "descriptor": { |
105 | 105 | "type": "rpc", |
106 | 106 | "sizeX": 2.5, |
... | ... | @@ -118,7 +118,7 @@ |
118 | 118 | "alias": "rpcbutton", |
119 | 119 | "name": "RPC Button", |
120 | 120 | "image": "", |
121 | - "description": null, | |
121 | + "description": "Allows to send RPC command when user press the button.", | |
122 | 122 | "descriptor": { |
123 | 123 | "type": "rpc", |
124 | 124 | "sizeX": 4, |
... | ... | @@ -136,7 +136,7 @@ |
136 | 136 | "alias": "update_attributes", |
137 | 137 | "name": "Update device attribute", |
138 | 138 | "image": "", |
139 | - "description": null, | |
139 | + "description": "Allows to send shared attribute update when user press the button.", | |
140 | 140 | "descriptor": { |
141 | 141 | "type": "rpc", |
142 | 142 | "sizeX": 4, | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "date", |
4 | 4 | "title": "Date", |
5 | 5 | "image": "", |
6 | - "description": null | |
6 | + "description": "Contains widgets to change the data range for other widgets on the dashboard." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "date_range_navigator", |
11 | 11 | "name": "Date-range-navigator", |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Allows to change the data range for other widgets on the dashboard.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "static", |
16 | 16 | "sizeX": 5, | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "digital_gauges", |
4 | 4 | "title": "Digital gauges", |
5 | 5 | "image": "", |
6 | - "description": "Useful for visualization of temperature, humidity, speed, and other integer or float values." | |
6 | + "description": "Display temperature, humidity, speed, and other latest values on various digital gauge widgets." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "gauge_justgage", |
11 | - "name": "Gauge - justGage", | |
11 | + "name": "Gauge", | |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Preconfigured gauge to display any value reading. Allows to configure value range, gradient colors and other settings.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "latest", |
16 | 16 | "sizeX": 4, |
... | ... | @@ -28,7 +28,7 @@ |
28 | 28 | "alias": "digital_thermometer", |
29 | 29 | "name": "Digital thermometer", |
30 | 30 | "image": "", |
31 | - "description": null, | |
31 | + "description": "Preconfigured gauge to display temperature. Allows to configure temperature range, gradient colors and other settings.", | |
32 | 32 | "descriptor": { |
33 | 33 | "type": "latest", |
34 | 34 | "sizeX": 3, |
... | ... | @@ -46,7 +46,7 @@ |
46 | 46 | "alias": "digital_speedometer", |
47 | 47 | "name": "Digital speedometer", |
48 | 48 | "image": "", |
49 | - "description": null, | |
49 | + "description": "Preconfigured gauge to display speed. Allows to configure speed range, gradient colors and other settings.", | |
50 | 50 | "descriptor": { |
51 | 51 | "type": "latest", |
52 | 52 | "sizeX": 5, |
... | ... | @@ -62,9 +62,9 @@ |
62 | 62 | }, |
63 | 63 | { |
64 | 64 | "alias": "neon_gauge_justgage", |
65 | - "name": "Neon gauge - justGage", | |
65 | + "name": "Neon gauge", | |
66 | 66 | "image": "", |
67 | - "description": null, | |
67 | + "description": "Preconfigured gauge to display any value reading as an arc. Allows to configure value range, gradient colors and other settings.", | |
68 | 68 | "descriptor": { |
69 | 69 | "type": "latest", |
70 | 70 | "sizeX": 5, |
... | ... | @@ -82,7 +82,7 @@ |
82 | 82 | "alias": "lcd_gauge", |
83 | 83 | "name": "LCD gauge", |
84 | 84 | "image": "", |
85 | - "description": null, | |
85 | + "description": "Preconfigured gauge to display any value reading as an arc. Allows to configure value range, gradient colors and other settings.", | |
86 | 86 | "descriptor": { |
87 | 87 | "type": "latest", |
88 | 88 | "sizeX": 5, |
... | ... | @@ -100,7 +100,7 @@ |
100 | 100 | "alias": "lcd_bar_gauge", |
101 | 101 | "name": "LCD bar gauge", |
102 | 102 | "image": "", |
103 | - "description": null, | |
103 | + "description": "Preconfigured gauge to display any value reading as an bar. Allows to configure value range, gradient colors and other settings.", | |
104 | 104 | "descriptor": { |
105 | 105 | "type": "latest", |
106 | 106 | "sizeX": 2, |
... | ... | @@ -116,9 +116,9 @@ |
116 | 116 | }, |
117 | 117 | { |
118 | 118 | "alias": "simple_neon_gauge_justgage", |
119 | - "name": "Simple neon gauge - justGage", | |
119 | + "name": "Simple neon gauge", | |
120 | 120 | "image": "", |
121 | - "description": null, | |
121 | + "description": "Preconfigured gauge to display any value reading as a doughnut. Allows to configure value range, gradient colors and other settings.", | |
122 | 122 | "descriptor": { |
123 | 123 | "type": "latest", |
124 | 124 | "sizeX": 3, |
... | ... | @@ -134,9 +134,9 @@ |
134 | 134 | }, |
135 | 135 | { |
136 | 136 | "alias": "vertical_bar_justgage", |
137 | - "name": "Vertical bar - justGage", | |
137 | + "name": "Vertical bar", | |
138 | 138 | "image": "", |
139 | - "description": null, | |
139 | + "description": "Preconfigured gauge to display any value reading as a bar. Allows to configure value range, gradient colors and other settings.", | |
140 | 140 | "descriptor": { |
141 | 141 | "type": "latest", |
142 | 142 | "sizeX": 2, |
... | ... | @@ -152,9 +152,9 @@ |
152 | 152 | }, |
153 | 153 | { |
154 | 154 | "alias": "simple_gauge_justgage", |
155 | - "name": "Simple gauge - justGage", | |
155 | + "name": "Simple gauge", | |
156 | 156 | "image": "", |
157 | - "description": null, | |
157 | + "description": "Preconfigured gauge to display any value reading as a circle. Allows to configure value range, gradient colors and other settings.", | |
158 | 158 | "descriptor": { |
159 | 159 | "type": "latest", |
160 | 160 | "sizeX": 2, |
... | ... | @@ -172,7 +172,7 @@ |
172 | 172 | "alias": "digital_bar", |
173 | 173 | "name": "Digital horizontal bar", |
174 | 174 | "image": "", |
175 | - "description": null, | |
175 | + "description": "Preconfigured gauge to display any value reading as a horizontal bar. Allows to configure value range, gradient colors and other settings.", | |
176 | 176 | "descriptor": { |
177 | 177 | "type": "latest", |
178 | 178 | "sizeX": 6, |
... | ... | @@ -188,9 +188,9 @@ |
188 | 188 | }, |
189 | 189 | { |
190 | 190 | "alias": "mini_gauge_justgage", |
191 | - "name": "Mini gauge - justGage", | |
191 | + "name": "Mini gauge", | |
192 | 192 | "image": "", |
193 | - "description": null, | |
193 | + "description": "Preconfigured gauge to display any value reading as a circle. Allows to configure value range, gradient colors and other settings.", | |
194 | 194 | "descriptor": { |
195 | 195 | "type": "latest", |
196 | 196 | "sizeX": 2, |
... | ... | @@ -206,9 +206,9 @@ |
206 | 206 | }, |
207 | 207 | { |
208 | 208 | "alias": "horizontal_bar_justgage", |
209 | - "name": "Horizontal bar - justGage", | |
209 | + "name": "Horizontal bar", | |
210 | 210 | "image": "", |
211 | - "description": null, | |
211 | + "description": "Preconfigured gauge to display any value reading as a horizontal bar. Allows to configure value range, gradient colors and other settings.", | |
212 | 212 | "descriptor": { |
213 | 213 | "type": "latest", |
214 | 214 | "sizeX": 7, |
... | ... | @@ -226,7 +226,7 @@ |
226 | 226 | "alias": "digital_vertical_bar", |
227 | 227 | "name": "Digital vertical bar", |
228 | 228 | "image": "", |
229 | - "description": null, | |
229 | + "description": "Preconfigured gauge to display any value reading as a vertical bar. Allows to configure value range, gradient colors and other settings.", | |
230 | 230 | "descriptor": { |
231 | 231 | "type": "latest", |
232 | 232 | "sizeX": 2.5, | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "entity_admin_widgets", |
4 | 4 | "title": "Entity admin widgets", |
5 | 5 | "image": "", |
6 | - "description": null | |
6 | + "description": "Templates of complex widgets that allow to list and create/update/delete devices and assets." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "device_admin_table", |
11 | 11 | "name": "Device admin table", |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Customized entity table widget with preconfigured actions to create update and delete devices.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "latest", |
16 | 16 | "sizeX": 7.5, |
... | ... | @@ -19,8 +19,8 @@ |
19 | 19 | "templateHtml": "<tb-entities-table-widget \n [ctx]=\"ctx\">\n</tb-entities-table-widget>", |
20 | 20 | "templateCss": "", |
21 | 21 | "controllerScript": "self.onInit = function() {\n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.entitiesTableWidget.onDataUpdated();\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n hasDataPageLink: true,\n warnOnPageDataOverflow: false,\n dataKeysOptional: true\n };\n}\n\nself.actionSources = function() {\n return {\n 'actionCellButton': {\n name: 'widget-action.action-cell-button',\n multiple: true\n },\n 'rowClick': {\n name: 'widget-action.row-click',\n multiple: false\n },\n 'rowDoubleClick': {\n name: 'widget-action.row-double-click',\n multiple: false\n }\n };\n}\n\nself.onDestroy = function() {\n}\n", |
22 | - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"entitiesTitle\": {\n \"title\": \"Entities table title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"enableSearch\": {\n \"title\": \"Enable entities search\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableSelectColumnDisplay\": {\n \"title\": \"Enable select columns to display\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayEntityName\": {\n \"title\": \"Display entity name column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"entityNameColumnTitle\": {\n \"title\": \"Entity name column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityLabel\": {\n \"title\": \"Display entity label column\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"entityLabelColumnTitle\": {\n \"title\": \"Entity label column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityType\": {\n \"title\": \"Display entity type column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayPagination\": {\n \"title\": \"Display pagination\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"defaultPageSize\": {\n \"title\": \"Default page size\",\n \"type\": \"number\",\n \"default\": 10\n },\n \"defaultSortOrder\": {\n \"title\": \"Default sort order\",\n \"type\": \"string\",\n \"default\": \"entityName\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"entitiesTitle\",\n \"enableSearch\",\n \"enableSelectColumnDisplay\",\n \"displayEntityName\",\n \"entityNameColumnTitle\",\n \"displayEntityLabel\",\n \"entityLabelColumnTitle\",\n \"displayEntityType\",\n \"displayPagination\",\n \"defaultPageSize\",\n \"defaultSortOrder\"\n ]\n}", | |
23 | - "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"columnWidth\": {\n \"title\": \"Column width (px or %)\",\n \"type\": \"string\",\n \"default\": \"0px\"\n },\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, entity, filter)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"columnWidth\",\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\"\n }\n ]\n}", | |
22 | + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"entitiesTitle\": {\n \"title\": \"Entities table title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"enableSearch\": {\n \"title\": \"Enable entities search\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableSelectColumnDisplay\": {\n \"title\": \"Enable select columns to display\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyHeader\": {\n \"title\": \"Always display header\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyAction\": {\n \"title\": \"Always display actions column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayEntityName\": {\n \"title\": \"Display entity name column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"entityNameColumnTitle\": {\n \"title\": \"Entity name column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityLabel\": {\n \"title\": \"Display entity label column\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"entityLabelColumnTitle\": {\n \"title\": \"Entity label column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityType\": {\n \"title\": \"Display entity type column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayPagination\": {\n \"title\": \"Display pagination\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"defaultPageSize\": {\n \"title\": \"Default page size\",\n \"type\": \"number\",\n \"default\": 10\n },\n \"defaultSortOrder\": {\n \"title\": \"Default sort order\",\n \"type\": \"string\",\n \"default\": \"entityName\"\n },\n \"useRowStyleFunction\": {\n \"title\": \"Use row style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"rowStyleFunction\": {\n \"title\": \"Row style function: f(entity, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"entitiesTitle\",\n \"enableSearch\",\n \"enableSelectColumnDisplay\",\n \"enableStickyHeader\",\n \"enableStickyAction\",\n \"displayEntityName\",\n \"entityNameColumnTitle\",\n \"displayEntityLabel\",\n \"entityLabelColumnTitle\",\n \"displayEntityType\",\n \"displayPagination\",\n \"defaultPageSize\",\n \"defaultSortOrder\",\n \"useRowStyleFunction\",\n {\n \"key\": \"rowStyleFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useRowStyleFunction === true\"\n }\n ]\n}", | |
23 | + "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"columnWidth\": {\n \"title\": \"Column width (px or %)\",\n \"type\": \"string\",\n \"default\": \"0px\"\n },\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value, entity, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, entity, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"defaultColumnVisibility\": {\n \"title\": \"Default column visibility\",\n \"type\": \"string\",\n \"default\": \"visible\"\n },\n \"columnSelectionToDisplay\": {\n \"title\": \"Column selection in 'Columns to Display'\",\n \"type\": \"string\",\n \"default\": \"enabled\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"columnWidth\",\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useCellStyleFunction === true\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useCellContentFunction === true\"\n },\n {\n \"key\": \"defaultColumnVisibility\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"visible\",\n \"label\": \"Visible\"\n },\n {\n \"value\": \"hidden\",\n \"label\": \"Hidden\"\n } \n ]\n },\n {\n \"key\": \"columnSelectionToDisplay\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"enabled\",\n \"label\": \"Enabled\"\n },\n {\n \"value\": \"disabled\",\n \"label\": \"Disabled\"\n } \n ]\n }\n ]\n}", | |
24 | 24 | "defaultConfig": "{\"timewindow\":{\"realtime\":{\"interval\":1000,\"timewindowMs\":86400000},\"aggregation\":{\"type\":\"NONE\",\"limit\":200}},\"showTitle\":true,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"4px\",\"settings\":{\"enableSearch\":true,\"displayPagination\":true,\"defaultPageSize\":10,\"defaultSortOrder\":\"entityName\",\"displayEntityName\":true,\"displayEntityType\":true,\"entitiesTitle\":\"Device admin table\",\"enableSelectColumnDisplay\":true},\"title\":\"Device admin table\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400,\"padding\":\"5px 10px 5px 10px\"},\"useDashboardTimewindow\":false,\"showLegend\":false,\"datasources\":[{\"type\":\"function\",\"name\":\"Simulated\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#f44336\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.6401141393938932,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"widgetStyle\":{},\"displayTimewindow\":true,\"actions\":{\"headerButton\":[{\"name\":\"Add device\",\"icon\":\"add\",\"type\":\"customPretty\",\"customHtml\":\"<form #addDeviceForm=\\\"ngForm\\\" [formGroup]=\\\"addDeviceFormGroup\\\"\\n (ngSubmit)=\\\"save()\\\" style=\\\"width: 480px;\\\">\\n <mat-toolbar fxLayout=\\\"row\\\" color=\\\"primary\\\">\\n <h2>Add device</h2>\\n <span fxFlex></span>\\n <button mat-button mat-icon-button\\n (click)=\\\"cancel()\\\"\\n type=\\\"button\\\">\\n <mat-icon class=\\\"material-icons\\\">close</mat-icon>\\n </button>\\n </mat-toolbar>\\n <mat-progress-bar color=\\\"warn\\\" mode=\\\"indeterminate\\\" *ngIf=\\\"isLoading$ | async\\\">\\n </mat-progress-bar>\\n <div style=\\\"height: 4px;\\\" *ngIf=\\\"!(isLoading$ | async)\\\"></div>\\n <div mat-dialog-content>\\n <div class=\\\"mat-padding\\\" fxLayout=\\\"column\\\">\\n <mat-form-field class=\\\"mat-block\\\">\\n <mat-label>Device name</mat-label>\\n <input matInput formControlName=\\\"deviceName\\\" required>\\n <mat-error *ngIf=\\\"addDeviceFormGroup.get('deviceName').hasError('required')\\\">\\n Device name is required.\\n </mat-error>\\n </mat-form-field>\\n <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\">\\n <tb-entity-subtype-autocomplete\\n fxFlex=\\\"50\\\"\\n formControlName=\\\"deviceType\\\"\\n [required]=\\\"true\\\"\\n [entityType]=\\\"'DEVICE'\\\"\\n ></tb-entity-subtype-autocomplete>\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Label</mat-label>\\n <input matInput formControlName=\\\"deviceLabel\\\">\\n </mat-form-field>\\n </div>\\n <div formGroupName=\\\"attributes\\\" fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\">\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Latitude</mat-label>\\n <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"latitude\\\">\\n </mat-form-field>\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Longitude</mat-label>\\n <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"longitude\\\">\\n </mat-form-field>\\n </div>\\n </div> \\n </div>\\n <div mat-dialog-actions fxLayout=\\\"row\\\">\\n <span fxFlex></span>\\n <button mat-button color=\\\"primary\\\"\\n type=\\\"button\\\"\\n [disabled]=\\\"(isLoading$ | async)\\\"\\n (click)=\\\"cancel()\\\" cdkFocusInitial>\\n Cancel\\n </button>\\n <button mat-button mat-raised-button color=\\\"primary\\\"\\n style=\\\"margin-right: 20px;\\\"\\n type=\\\"submit\\\"\\n [disabled]=\\\"(isLoading$ | async) || addDeviceForm.invalid || !addDeviceForm.dirty\\\">\\n Create\\n </button>\\n </div>\\n</form>\\n\",\"customCss\":\"\",\"customFunction\":\"let $injector = widgetContext.$scope.$injector;\\nlet customDialog = $injector.get(widgetContext.servicesMap.get('customDialog'));\\nlet deviceService = $injector.get(widgetContext.servicesMap.get('deviceService'));\\nlet attributeService = $injector.get(widgetContext.servicesMap.get('attributeService'));\\n\\nopenAddDeviceDialog();\\n\\nfunction openAddDeviceDialog() {\\n customDialog.customDialog(htmlTemplate, AddDeviceDialogController).subscribe();\\n}\\n\\nfunction AddDeviceDialogController(instance) {\\n let vm = instance;\\n \\n vm.addDeviceFormGroup = vm.fb.group({\\n deviceName: ['', [vm.validators.required]],\\n deviceType: ['', [vm.validators.required]],\\n deviceLabel: [''],\\n attributes: vm.fb.group({\\n latitude: [null],\\n longitude: [null]\\n }) \\n });\\n \\n vm.cancel = function() {\\n vm.dialogRef.close(null);\\n };\\n \\n vm.save = function() {\\n vm.addDeviceFormGroup.markAsPristine();\\n let device = {\\n name: vm.addDeviceFormGroup.get('deviceName').value,\\n type: vm.addDeviceFormGroup.get('deviceType').value,\\n label: vm.addDeviceFormGroup.get('deviceLabel').value\\n };\\n deviceService.saveDevice(device).subscribe(\\n function (device) {\\n saveAttributes(device.id).subscribe(\\n function () {\\n widgetContext.updateAliases();\\n vm.dialogRef.close(null);\\n }\\n );\\n }\\n );\\n };\\n \\n function saveAttributes(entityId) {\\n let attributes = vm.addDeviceFormGroup.get('attributes').value;\\n let attributesArray = [];\\n for (let key in attributes) {\\n attributesArray.push({key: key, value: attributes[key]});\\n }\\n if (attributesArray.length > 0) {\\n return attributeService.saveEntityAttributes(entityId, \\\"SERVER_SCOPE\\\", attributesArray);\\n } else {\\n return widgetContext.rxjs.of([]);\\n }\\n }\\n}\",\"customResources\":[],\"id\":\"70837a9d-c3de-a9a7-03c5-dccd14998758\"}],\"actionCellButton\":[{\"name\":\"Edit device\",\"icon\":\"edit\",\"type\":\"customPretty\",\"customHtml\":\"<form #editDeviceForm=\\\"ngForm\\\" [formGroup]=\\\"editDeviceFormGroup\\\"\\n (ngSubmit)=\\\"save()\\\" style=\\\"width: 480px;\\\">\\n <mat-toolbar fxLayout=\\\"row\\\" color=\\\"primary\\\">\\n <h2>Edit device</h2>\\n <span fxFlex></span>\\n <button mat-button mat-icon-button\\n (click)=\\\"cancel()\\\"\\n type=\\\"button\\\">\\n <mat-icon class=\\\"material-icons\\\">close</mat-icon>\\n </button>\\n </mat-toolbar>\\n <mat-progress-bar color=\\\"warn\\\" mode=\\\"indeterminate\\\" *ngIf=\\\"isLoading$ | async\\\">\\n </mat-progress-bar>\\n <div style=\\\"height: 4px;\\\" *ngIf=\\\"!(isLoading$ | async)\\\"></div>\\n <div mat-dialog-content>\\n <div class=\\\"mat-padding\\\" fxLayout=\\\"column\\\">\\n <mat-form-field class=\\\"mat-block\\\">\\n <mat-label>Device name</mat-label>\\n <input matInput formControlName=\\\"deviceName\\\" required>\\n <mat-error *ngIf=\\\"editDeviceFormGroup.get('deviceName').hasError('required')\\\">\\n Device name is required.\\n </mat-error>\\n </mat-form-field>\\n <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\">\\n <tb-entity-subtype-autocomplete\\n fxFlex=\\\"50\\\"\\n formControlName=\\\"deviceType\\\"\\n [required]=\\\"true\\\"\\n [entityType]=\\\"'DEVICE'\\\"\\n ></tb-entity-subtype-autocomplete>\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Label</mat-label>\\n <input matInput formControlName=\\\"deviceLabel\\\">\\n </mat-form-field>\\n </div>\\n <div formGroupName=\\\"attributes\\\" fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\">\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Latitude</mat-label>\\n <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"latitude\\\">\\n </mat-form-field>\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Longitude</mat-label>\\n <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"longitude\\\">\\n </mat-form-field>\\n </div>\\n </div> \\n </div>\\n <div mat-dialog-actions fxLayout=\\\"row\\\">\\n <span fxFlex></span>\\n <button mat-button color=\\\"primary\\\"\\n type=\\\"button\\\"\\n [disabled]=\\\"(isLoading$ | async)\\\"\\n (click)=\\\"cancel()\\\" cdkFocusInitial>\\n Cancel\\n </button>\\n <button mat-button mat-raised-button color=\\\"primary\\\"\\n style=\\\"margin-right: 20px;\\\"\\n type=\\\"submit\\\"\\n [disabled]=\\\"(isLoading$ | async) || editDeviceForm.invalid || !editDeviceForm.dirty\\\">\\n Update\\n </button>\\n </div>\\n</form>\\n\",\"customCss\":\"\",\"customFunction\":\"let $injector = widgetContext.$scope.$injector;\\nlet customDialog = $injector.get(widgetContext.servicesMap.get('customDialog'));\\nlet deviceService = $injector.get(widgetContext.servicesMap.get('deviceService'));\\nlet attributeService = $injector.get(widgetContext.servicesMap.get('attributeService'));\\n\\nopenEditDeviceDialog();\\n\\nfunction openEditDeviceDialog() {\\n customDialog.customDialog(htmlTemplate, EditDeviceDialogController).subscribe();\\n}\\n\\nfunction EditDeviceDialogController(instance) {\\n let vm = instance;\\n \\n vm.device = null;\\n vm.attributes = {};\\n \\n vm.editDeviceFormGroup = vm.fb.group({\\n deviceName: ['', [vm.validators.required]],\\n deviceType: ['', [vm.validators.required]],\\n deviceLabel: [''],\\n attributes: vm.fb.group({\\n latitude: [null],\\n longitude: [null]\\n }) \\n });\\n \\n vm.cancel = function() {\\n vm.dialogRef.close(null);\\n };\\n \\n vm.save = function() {\\n vm.editDeviceFormGroup.markAsPristine();\\n vm.device.name = vm.editDeviceFormGroup.get('deviceName').value,\\n vm.device.type = vm.editDeviceFormGroup.get('deviceType').value,\\n vm.device.label = vm.editDeviceFormGroup.get('deviceLabel').value\\n deviceService.saveDevice(vm.device).subscribe(\\n function () {\\n saveAttributes().subscribe(\\n function () {\\n widgetContext.updateAliases();\\n vm.dialogRef.close(null);\\n }\\n );\\n }\\n );\\n };\\n \\n getEntityInfo();\\n \\n function getEntityInfo() {\\n deviceService.getDevice(entityId.id).subscribe(\\n function (device) {\\n attributeService.getEntityAttributes(entityId, 'SERVER_SCOPE',\\n ['latitude', 'longitude']).subscribe(\\n function (attributes) {\\n for (let i = 0; i < attributes.length; i++) {\\n vm.attributes[attributes[i].key] = attributes[i].value; \\n }\\n vm.device = device;\\n vm.editDeviceFormGroup.patchValue(\\n {\\n deviceName: vm.device.name,\\n deviceType: vm.device.type,\\n deviceLabel: vm.device.label,\\n attributes: {\\n latitude: vm.attributes.latitude,\\n longitude: vm.attributes.longitude\\n }\\n }, {emitEvent: false}\\n );\\n } \\n );\\n }\\n ); \\n }\\n \\n function saveAttributes() {\\n let attributes = vm.editDeviceFormGroup.get('attributes').value;\\n let attributesArray = [];\\n for (let key in attributes) {\\n attributesArray.push({key: key, value: attributes[key]});\\n }\\n if (attributesArray.length > 0) {\\n return attributeService.saveEntityAttributes(entityId, 'SERVER_SCOPE', attributesArray);\\n } else {\\n return widgetContext.rxjs.of([]);\\n }\\n }\\n}\",\"customResources\":[],\"id\":\"93931e52-5d7c-903e-67aa-b9435df44ff4\"},{\"name\":\"Delete device\",\"icon\":\"delete\",\"type\":\"custom\",\"customFunction\":\"let $injector = widgetContext.$scope.$injector;\\nlet dialogs = $injector.get(widgetContext.servicesMap.get('dialogs'));\\nlet deviceService = $injector.get(widgetContext.servicesMap.get('deviceService'));\\n\\nopenDeleteDeviceDialog();\\n\\nfunction openDeleteDeviceDialog() {\\n let title = \\\"Are you sure you want to delete the device \\\" + entityName + \\\"?\\\";\\n let content = \\\"Be careful, after the confirmation, the device and all related data will become unrecoverable!\\\";\\n dialogs.confirm(title, content, 'Cancel', 'Delete').subscribe(\\n function (result) {\\n if (result) {\\n deleteDevice();\\n }\\n }\\n );\\n}\\n\\nfunction deleteDevice() {\\n deviceService.deleteDevice(entityId.id).subscribe(\\n function () {\\n widgetContext.updateAliases();\\n }\\n );\\n}\\n\",\"id\":\"ec2708f6-9ff0-186b-e4fc-7635ebfa3074\"}]}}" |
25 | 25 | } |
26 | 26 | }, |
... | ... | @@ -28,7 +28,7 @@ |
28 | 28 | "alias": "asset_admin_table", |
29 | 29 | "name": "Asset admin table", |
30 | 30 | "image": "", |
31 | - "description": null, | |
31 | + "description": "Customized entity table widget with preconfigured actions to create update and delete assets.", | |
32 | 32 | "descriptor": { |
33 | 33 | "type": "latest", |
34 | 34 | "sizeX": 7.5, |
... | ... | @@ -37,8 +37,8 @@ |
37 | 37 | "templateHtml": "<tb-entities-table-widget \n [ctx]=\"ctx\">\n</tb-entities-table-widget>", |
38 | 38 | "templateCss": "", |
39 | 39 | "controllerScript": "self.onInit = function() {\n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.entitiesTableWidget.onDataUpdated();\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n hasDataPageLink: true,\n warnOnPageDataOverflow: false,\n dataKeysOptional: true\n };\n}\n\nself.actionSources = function() {\n return {\n 'actionCellButton': {\n name: 'widget-action.action-cell-button',\n multiple: true\n },\n 'rowClick': {\n name: 'widget-action.row-click',\n multiple: false\n },\n 'rowDoubleClick': {\n name: 'widget-action.row-double-click',\n multiple: false\n }\n };\n}\n\nself.onDestroy = function() {\n}\n", |
40 | - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"entitiesTitle\": {\n \"title\": \"Entities table title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"enableSearch\": {\n \"title\": \"Enable entities search\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableSelectColumnDisplay\": {\n \"title\": \"Enable select columns to display\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayEntityName\": {\n \"title\": \"Display entity name column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"entityNameColumnTitle\": {\n \"title\": \"Entity name column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityLabel\": {\n \"title\": \"Display entity label column\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"entityLabelColumnTitle\": {\n \"title\": \"Entity label column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityType\": {\n \"title\": \"Display entity type column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayPagination\": {\n \"title\": \"Display pagination\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"defaultPageSize\": {\n \"title\": \"Default page size\",\n \"type\": \"number\",\n \"default\": 10\n },\n \"defaultSortOrder\": {\n \"title\": \"Default sort order\",\n \"type\": \"string\",\n \"default\": \"entityName\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"entitiesTitle\",\n \"enableSearch\",\n \"enableSelectColumnDisplay\",\n \"displayEntityName\",\n \"entityNameColumnTitle\",\n \"displayEntityLabel\",\n \"entityLabelColumnTitle\",\n \"displayEntityType\",\n \"displayPagination\",\n \"defaultPageSize\",\n \"defaultSortOrder\"\n ]\n}", | |
41 | - "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"columnWidth\": {\n \"title\": \"Column width (px or %)\",\n \"type\": \"string\",\n \"default\": \"0px\"\n },\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, entity, filter)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"columnWidth\",\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\"\n }\n ]\n}", | |
40 | + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"EntitiesTableSettings\",\n \"properties\": {\n \"entitiesTitle\": {\n \"title\": \"Entities table title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"enableSearch\": {\n \"title\": \"Enable entities search\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableSelectColumnDisplay\": {\n \"title\": \"Enable select columns to display\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyHeader\": {\n \"title\": \"Always display header\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"enableStickyAction\": {\n \"title\": \"Always display actions column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayEntityName\": {\n \"title\": \"Display entity name column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"entityNameColumnTitle\": {\n \"title\": \"Entity name column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityLabel\": {\n \"title\": \"Display entity label column\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"entityLabelColumnTitle\": {\n \"title\": \"Entity label column title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"displayEntityType\": {\n \"title\": \"Display entity type column\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"displayPagination\": {\n \"title\": \"Display pagination\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"defaultPageSize\": {\n \"title\": \"Default page size\",\n \"type\": \"number\",\n \"default\": 10\n },\n \"defaultSortOrder\": {\n \"title\": \"Default sort order\",\n \"type\": \"string\",\n \"default\": \"entityName\"\n },\n \"useRowStyleFunction\": {\n \"title\": \"Use row style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"rowStyleFunction\": {\n \"title\": \"Row style function: f(entity, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"entitiesTitle\",\n \"enableSearch\",\n \"enableSelectColumnDisplay\",\n \"enableStickyHeader\",\n \"enableStickyAction\",\n \"displayEntityName\",\n \"entityNameColumnTitle\",\n \"displayEntityLabel\",\n \"entityLabelColumnTitle\",\n \"displayEntityType\",\n \"displayPagination\",\n \"defaultPageSize\",\n \"defaultSortOrder\",\n \"useRowStyleFunction\",\n {\n \"key\": \"rowStyleFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useRowStyleFunction === true\"\n }\n ]\n}", | |
41 | + "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"columnWidth\": {\n \"title\": \"Column width (px or %)\",\n \"type\": \"string\",\n \"default\": \"0px\"\n },\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value, entity, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, entity, ctx)\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"defaultColumnVisibility\": {\n \"title\": \"Default column visibility\",\n \"type\": \"string\",\n \"default\": \"visible\"\n },\n \"columnSelectionToDisplay\": {\n \"title\": \"Column selection in 'Columns to Display'\",\n \"type\": \"string\",\n \"default\": \"enabled\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"columnWidth\",\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useCellStyleFunction === true\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\",\n \"condition\": \"model.useCellContentFunction === true\"\n },\n {\n \"key\": \"defaultColumnVisibility\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"visible\",\n \"label\": \"Visible\"\n },\n {\n \"value\": \"hidden\",\n \"label\": \"Hidden\"\n } \n ]\n },\n {\n \"key\": \"columnSelectionToDisplay\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"enabled\",\n \"label\": \"Enabled\"\n },\n {\n \"value\": \"disabled\",\n \"label\": \"Disabled\"\n } \n ]\n }\n ]\n}", | |
42 | 42 | "defaultConfig": "{\"timewindow\":{\"realtime\":{\"interval\":1000,\"timewindowMs\":86400000},\"aggregation\":{\"type\":\"NONE\",\"limit\":200}},\"showTitle\":true,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"4px\",\"settings\":{\"enableSearch\":true,\"displayPagination\":true,\"defaultPageSize\":10,\"defaultSortOrder\":\"entityName\",\"displayEntityName\":true,\"displayEntityType\":true,\"entitiesTitle\":\"Asset admin table\",\"enableSelectColumnDisplay\":true},\"title\":\"Asset admin table\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400,\"padding\":\"5px 10px 5px 10px\"},\"useDashboardTimewindow\":false,\"showLegend\":false,\"datasources\":[{\"type\":\"function\",\"name\":\"Simulated\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#f44336\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.6401141393938932,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"widgetStyle\":{},\"displayTimewindow\":true,\"actions\":{\"headerButton\":[{\"name\":\"Add asset\",\"icon\":\"add\",\"type\":\"customPretty\",\"customHtml\":\"<form #addAssetForm=\\\"ngForm\\\" [formGroup]=\\\"addAssetFormGroup\\\"\\n (ngSubmit)=\\\"save()\\\" style=\\\"width: 480px;\\\">\\n <mat-toolbar fxLayout=\\\"row\\\" color=\\\"primary\\\">\\n <h2>Add asset</h2>\\n <span fxFlex></span>\\n <button mat-button mat-icon-button\\n (click)=\\\"cancel()\\\"\\n type=\\\"button\\\">\\n <mat-icon class=\\\"material-icons\\\">close</mat-icon>\\n </button>\\n </mat-toolbar>\\n <mat-progress-bar color=\\\"warn\\\" mode=\\\"indeterminate\\\" *ngIf=\\\"isLoading$ | async\\\">\\n </mat-progress-bar>\\n <div style=\\\"height: 4px;\\\" *ngIf=\\\"!(isLoading$ | async)\\\"></div>\\n <div mat-dialog-content>\\n <div class=\\\"mat-padding\\\" fxLayout=\\\"column\\\">\\n <mat-form-field class=\\\"mat-block\\\">\\n <mat-label>Asset name</mat-label>\\n <input matInput formControlName=\\\"assetName\\\" required>\\n <mat-error *ngIf=\\\"addAssetFormGroup.get('assetName').hasError('required')\\\">\\n Asset name is required.\\n </mat-error>\\n </mat-form-field>\\n <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\">\\n <tb-entity-subtype-autocomplete\\n fxFlex=\\\"50\\\"\\n formControlName=\\\"assetType\\\"\\n [required]=\\\"true\\\"\\n [entityType]=\\\"'ASSET'\\\"\\n ></tb-entity-subtype-autocomplete>\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Label</mat-label>\\n <input matInput formControlName=\\\"assetLabel\\\">\\n </mat-form-field>\\n </div>\\n <div formGroupName=\\\"attributes\\\" fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\">\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Latitude</mat-label>\\n <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"latitude\\\">\\n </mat-form-field>\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Longitude</mat-label>\\n <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"longitude\\\">\\n </mat-form-field>\\n </div>\\n </div> \\n </div>\\n <div mat-dialog-actions fxLayout=\\\"row\\\">\\n <span fxFlex></span>\\n <button mat-button color=\\\"primary\\\"\\n type=\\\"button\\\"\\n [disabled]=\\\"(isLoading$ | async)\\\"\\n (click)=\\\"cancel()\\\" cdkFocusInitial>\\n Cancel\\n </button>\\n <button mat-button mat-raised-button color=\\\"primary\\\"\\n style=\\\"margin-right: 20px;\\\"\\n type=\\\"submit\\\"\\n [disabled]=\\\"(isLoading$ | async) || addAssetForm.invalid || !addAssetForm.dirty\\\">\\n Create\\n </button>\\n </div>\\n</form>\\n\",\"customCss\":\"\",\"customFunction\":\"let $injector = widgetContext.$scope.$injector;\\nlet customDialog = $injector.get(widgetContext.servicesMap.get('customDialog'));\\nlet assetService = $injector.get(widgetContext.servicesMap.get('assetService'));\\nlet attributeService = $injector.get(widgetContext.servicesMap.get('attributeService'));\\n\\nopenAddAssetDialog();\\n\\nfunction openAddAssetDialog() {\\n customDialog.customDialog(htmlTemplate, AddAssetDialogController).subscribe();\\n}\\n\\nfunction AddAssetDialogController(instance) {\\n let vm = instance;\\n \\n vm.addAssetFormGroup = vm.fb.group({\\n assetName: ['', [vm.validators.required]],\\n assetType: ['', [vm.validators.required]],\\n assetLabel: [''],\\n attributes: vm.fb.group({\\n latitude: [null],\\n longitude: [null]\\n }) \\n });\\n \\n vm.cancel = function() {\\n vm.dialogRef.close(null);\\n };\\n \\n vm.save = function() {\\n vm.addAssetFormGroup.markAsPristine();\\n let asset = {\\n name: vm.addAssetFormGroup.get('assetName').value,\\n type: vm.addAssetFormGroup.get('assetType').value,\\n label: vm.addAssetFormGroup.get('assetLabel').value\\n };\\n assetService.saveAsset(asset).subscribe(\\n function (asset) {\\n saveAttributes(asset.id).subscribe(\\n function () {\\n widgetContext.updateAliases();\\n vm.dialogRef.close(null);\\n }\\n );\\n }\\n );\\n };\\n \\n function saveAttributes(entityId) {\\n let attributes = vm.addAssetFormGroup.get('attributes').value;\\n let attributesArray = [];\\n for (let key in attributes) {\\n attributesArray.push({key: key, value: attributes[key]});\\n }\\n if (attributesArray.length > 0) {\\n return attributeService.saveEntityAttributes(entityId, \\\"SERVER_SCOPE\\\", attributesArray);\\n } else {\\n return widgetContext.rxjs.of([]);\\n }\\n }\\n}\",\"customResources\":[],\"id\":\"70837a9d-c3de-a9a7-03c5-dccd14998758\"}],\"actionCellButton\":[{\"name\":\"Edit asset\",\"icon\":\"edit\",\"type\":\"customPretty\",\"customHtml\":\"<form #editAssetForm=\\\"ngForm\\\" [formGroup]=\\\"editAssetFormGroup\\\"\\n (ngSubmit)=\\\"save()\\\" style=\\\"width: 480px;\\\">\\n <mat-toolbar fxLayout=\\\"row\\\" color=\\\"primary\\\">\\n <h2>Edit asset</h2>\\n <span fxFlex></span>\\n <button mat-button mat-icon-button\\n (click)=\\\"cancel()\\\"\\n type=\\\"button\\\">\\n <mat-icon class=\\\"material-icons\\\">close</mat-icon>\\n </button>\\n </mat-toolbar>\\n <mat-progress-bar color=\\\"warn\\\" mode=\\\"indeterminate\\\" *ngIf=\\\"isLoading$ | async\\\">\\n </mat-progress-bar>\\n <div style=\\\"height: 4px;\\\" *ngIf=\\\"!(isLoading$ | async)\\\"></div>\\n <div mat-dialog-content>\\n <div class=\\\"mat-padding\\\" fxLayout=\\\"column\\\">\\n <mat-form-field class=\\\"mat-block\\\">\\n <mat-label>Asset name</mat-label>\\n <input matInput formControlName=\\\"assetName\\\" required>\\n <mat-error *ngIf=\\\"editAssetFormGroup.get('assetName').hasError('required')\\\">\\n Asset name is required.\\n </mat-error>\\n </mat-form-field>\\n <div fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\">\\n <tb-entity-subtype-autocomplete\\n fxFlex=\\\"50\\\"\\n formControlName=\\\"assetType\\\"\\n [required]=\\\"true\\\"\\n [entityType]=\\\"'ASSET'\\\"\\n ></tb-entity-subtype-autocomplete>\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Label</mat-label>\\n <input matInput formControlName=\\\"assetLabel\\\">\\n </mat-form-field>\\n </div>\\n <div formGroupName=\\\"attributes\\\" fxLayout=\\\"row\\\" fxLayoutGap=\\\"8px\\\">\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Latitude</mat-label>\\n <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"latitude\\\">\\n </mat-form-field>\\n <mat-form-field fxFlex=\\\"50\\\" class=\\\"mat-block\\\">\\n <mat-label>Longitude</mat-label>\\n <input type=\\\"number\\\" step=\\\"any\\\" matInput formControlName=\\\"longitude\\\">\\n </mat-form-field>\\n </div>\\n </div> \\n </div>\\n <div mat-dialog-actions fxLayout=\\\"row\\\">\\n <span fxFlex></span>\\n <button mat-button color=\\\"primary\\\"\\n type=\\\"button\\\"\\n [disabled]=\\\"(isLoading$ | async)\\\"\\n (click)=\\\"cancel()\\\" cdkFocusInitial>\\n Cancel\\n </button>\\n <button mat-button mat-raised-button color=\\\"primary\\\"\\n type=\\\"submit\\\"\\n style=\\\"margin-right: 20px;\\\"\\n [disabled]=\\\"(isLoading$ | async) || editAssetForm.invalid || !editAssetForm.dirty\\\">\\n Update\\n </button>\\n </div>\\n</form>\\n\",\"customCss\":\"\",\"customFunction\":\"let $injector = widgetContext.$scope.$injector;\\nlet customDialog = $injector.get(widgetContext.servicesMap.get('customDialog'));\\nlet assetService = $injector.get(widgetContext.servicesMap.get('assetService'));\\nlet attributeService = $injector.get(widgetContext.servicesMap.get('attributeService'));\\n\\nopenEditAssetDialog();\\n\\nfunction openEditAssetDialog() {\\n customDialog.customDialog(htmlTemplate, EditAssetDialogController).subscribe();\\n}\\n\\nfunction EditAssetDialogController(instance) {\\n let vm = instance;\\n \\n vm.asset = null;\\n vm.attributes = {};\\n \\n vm.editAssetFormGroup = vm.fb.group({\\n assetName: ['', [vm.validators.required]],\\n assetType: ['', [vm.validators.required]],\\n assetLabel: [''],\\n attributes: vm.fb.group({\\n latitude: [null],\\n longitude: [null]\\n }) \\n });\\n \\n vm.cancel = function() {\\n vm.dialogRef.close(null);\\n };\\n \\n vm.save = function() {\\n vm.editAssetFormGroup.markAsPristine();\\n vm.asset.name = vm.editAssetFormGroup.get('assetName').value,\\n vm.asset.type = vm.editAssetFormGroup.get('assetType').value,\\n vm.asset.label = vm.editAssetFormGroup.get('assetLabel').value\\n assetService.saveAsset(vm.asset).subscribe(\\n function () {\\n saveAttributes().subscribe(\\n function () {\\n widgetContext.updateAliases();\\n vm.dialogRef.close(null);\\n }\\n );\\n }\\n );\\n };\\n \\n getEntityInfo();\\n \\n function getEntityInfo() {\\n assetService.getAsset(entityId.id).subscribe(\\n function (asset) {\\n attributeService.getEntityAttributes(entityId, 'SERVER_SCOPE',\\n ['latitude', 'longitude']).subscribe(\\n function (attributes) {\\n for (let i = 0; i < attributes.length; i++) {\\n vm.attributes[attributes[i].key] = attributes[i].value; \\n }\\n vm.asset = asset;\\n vm.editAssetFormGroup.patchValue(\\n {\\n assetName: vm.asset.name,\\n assetType: vm.asset.type,\\n assetLabel: vm.asset.label,\\n attributes: {\\n latitude: vm.attributes.latitude,\\n longitude: vm.attributes.longitude\\n }\\n }, {emitEvent: false}\\n );\\n } \\n );\\n }\\n ); \\n }\\n \\n function saveAttributes() {\\n let attributes = vm.editAssetFormGroup.get('attributes').value;\\n let attributesArray = [];\\n for (let key in attributes) {\\n attributesArray.push({key: key, value: attributes[key]});\\n }\\n if (attributesArray.length > 0) {\\n return attributeService.saveEntityAttributes(entityId, 'SERVER_SCOPE', attributesArray);\\n } else {\\n return widgetContext.rxjs.of([]);\\n }\\n }\\n}\",\"customResources\":[],\"id\":\"93931e52-5d7c-903e-67aa-b9435df44ff4\"},{\"name\":\"Delete asset\",\"icon\":\"delete\",\"type\":\"custom\",\"customFunction\":\"let $injector = widgetContext.$scope.$injector;\\nlet dialogs = $injector.get(widgetContext.servicesMap.get('dialogs'));\\nlet assetService = $injector.get(widgetContext.servicesMap.get('assetService'));\\n\\nopenDeleteAssetDialog();\\n\\nfunction openDeleteAssetDialog() {\\n let title = \\\"Are you sure you want to delete the asset \\\" + entityName + \\\"?\\\";\\n let content = \\\"Be careful, after the confirmation, the asset and all related data will become unrecoverable!\\\";\\n dialogs.confirm(title, content, 'Cancel', 'Delete').subscribe(\\n function (result) {\\n if (result) {\\n deleteAsset();\\n }\\n }\\n );\\n}\\n\\nfunction deleteAsset() {\\n assetService.deleteAsset(entityId.id).subscribe(\\n function () {\\n widgetContext.updateAliases();\\n }\\n );\\n}\\n\",\"id\":\"ec2708f6-9ff0-186b-e4fc-7635ebfa3074\"}]}}" |
43 | 43 | } |
44 | 44 | } | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "gateway_widgets", |
4 | 4 | "title": "Gateway widgets", |
5 | 5 | "image": "", |
6 | - "description": null | |
6 | + "description": "Widgets to manage ThingsBoard IoT Gateway." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "gateway_configuration", |
11 | 11 | "name": "Gateway Configuration", |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Allows to define widget gateway configuration for a single selected device.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "static", |
16 | 16 | "sizeX": 8, |
... | ... | @@ -28,7 +28,7 @@ |
28 | 28 | "alias": "attributes_card", |
29 | 29 | "name": "Gateway events", |
30 | 30 | "image": "", |
31 | - "description": null, | |
31 | + "description": "Allows to browse events from the gateway.", | |
32 | 32 | "descriptor": { |
33 | 33 | "type": "latest", |
34 | 34 | "sizeX": 7.5, |
... | ... | @@ -46,7 +46,7 @@ |
46 | 46 | "alias": "config_form_latest", |
47 | 47 | "name": "Gateway configuration (Single device)", |
48 | 48 | "image": "", |
49 | - "description": null, | |
49 | + "description": "Allows to create or choose the gateway device and edit the configuration.", | |
50 | 50 | "descriptor": { |
51 | 51 | "type": "latest", |
52 | 52 | "sizeX": 7.5, | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "gpio_widgets", |
4 | 4 | "title": "GPIO widgets", |
5 | 5 | "image": "", |
6 | - "description": "Useful for visualization and control of GPIO state for target devices." | |
6 | + "description": "Visualization and control of GPIO state for target devices." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "basic_gpio_control", |
11 | 11 | "name": "Basic GPIO Control", |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Allows to change state of the GPIO for target device using RPC commands. Requires handling of the RPC commands in the device firmware. Uses 'getGpioStatus' and 'setGpioStatus' RPC calls", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "rpc", |
16 | 16 | "sizeX": 4, |
... | ... | @@ -28,7 +28,7 @@ |
28 | 28 | "alias": "gpio_panel", |
29 | 29 | "name": "Basic GPIO Panel", |
30 | 30 | "image": "", |
31 | - "description": null, | |
31 | + "description": "Allows to display state of the GPIO for target device using latest attribute values. You should set the label of the selected data key to GPIO pin number (e.g. '1') and use boolean values for widget to display the data.", | |
32 | 32 | "descriptor": { |
33 | 33 | "type": "latest", |
34 | 34 | "sizeX": 5, |
... | ... | @@ -46,7 +46,7 @@ |
46 | 46 | "alias": "raspberry_pi_gpio_panel", |
47 | 47 | "name": "Raspberry Pi GPIO Panel", |
48 | 48 | "image": "", |
49 | - "description": null, | |
49 | + "description": "Allows to change state of the GPIO for Raspberry Pi device using RPC commands. Requires handling of the RPC commands in the device firmware. Uses 'getGpioStatus' and 'setGpioStatus' RPC calls", | |
50 | 50 | "descriptor": { |
51 | 51 | "type": "latest", |
52 | 52 | "sizeX": 7, |
... | ... | @@ -64,7 +64,7 @@ |
64 | 64 | "alias": "raspberry_pi_gpio_control", |
65 | 65 | "name": "Raspberry Pi GPIO Control", |
66 | 66 | "image": "", |
67 | - "description": null, | |
67 | + "description": "Allows to display state of the GPIO for target Raspberry Pi device using latest attribute values. You should set the label of the selected data key to GPIO pin number (e.g. '1') and use boolean values for widget to display the data.", | |
68 | 68 | "descriptor": { |
69 | 69 | "type": "rpc", |
70 | 70 | "sizeX": 6, | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "input_widgets", |
4 | 4 | "title": "Input widgets", |
5 | 5 | "image": "", |
6 | - "description": null | |
6 | + "description": "Various input forms that allow users to set location, image and other configuration parameters of the target entity" | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "markers_placement_openstreetmap", |
11 | 11 | "name": "Markers Placement - OpenStreetMap", |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Allows configuring location of the selected entities on the OpenStreetMap.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "latest", |
16 | 16 | "sizeX": 8.5, |
... | ... | @@ -28,7 +28,7 @@ |
28 | 28 | "alias": "update_multiple_attributes", |
29 | 29 | "name": "Update Multiple Attributes", |
30 | 30 | "image": "", |
31 | - "description": null, | |
31 | + "description": "Allows to create an input form and set values for multiple attributes simultaniously.", | |
32 | 32 | "descriptor": { |
33 | 33 | "type": "latest", |
34 | 34 | "sizeX": 7.5, |
... | ... | @@ -46,7 +46,7 @@ |
46 | 46 | "alias": "device_claiming_widget", |
47 | 47 | "name": "Device claiming widget", |
48 | 48 | "image": "", |
49 | - "description": null, | |
49 | + "description": "Allows to claim the device using name and optional secret key.", | |
50 | 50 | "descriptor": { |
51 | 51 | "type": "static", |
52 | 52 | "sizeX": 7.5, |
... | ... | @@ -64,7 +64,7 @@ |
64 | 64 | "alias": "markers_placement_image_map", |
65 | 65 | "name": "Markers Placement - Image Map", |
66 | 66 | "image": "", |
67 | - "description": null, | |
67 | + "description": "Allows configuring location of the selected entities on the image map.", | |
68 | 68 | "descriptor": { |
69 | 69 | "type": "latest", |
70 | 70 | "sizeX": 8.5, |
... | ... | @@ -82,7 +82,7 @@ |
82 | 82 | "alias": "update_integer_timeseries", |
83 | 83 | "name": "Update integer timeseries", |
84 | 84 | "image": "", |
85 | - "description": null, | |
85 | + "description": "Simple form to input new integer value for pre-defined timeseries key.", | |
86 | 86 | "descriptor": { |
87 | 87 | "type": "latest", |
88 | 88 | "sizeX": 7.5, |
... | ... | @@ -100,7 +100,7 @@ |
100 | 100 | "alias": "update_double_timeseries", |
101 | 101 | "name": "Update double timeseries", |
102 | 102 | "image": "", |
103 | - "description": null, | |
103 | + "description": "Simple form to input new double value for pre-defined timeseries key.", | |
104 | 104 | "descriptor": { |
105 | 105 | "type": "latest", |
106 | 106 | "sizeX": 7.5, |
... | ... | @@ -118,7 +118,7 @@ |
118 | 118 | "alias": "update_boolean_timeseries", |
119 | 119 | "name": "Update boolean timeseries", |
120 | 120 | "image": "", |
121 | - "description": null, | |
121 | + "description": "Simple form to input new boolean value for pre-defined timeseries key.", | |
122 | 122 | "descriptor": { |
123 | 123 | "type": "latest", |
124 | 124 | "sizeX": 7.5, |
... | ... | @@ -136,7 +136,7 @@ |
136 | 136 | "alias": "update_string_timeseries", |
137 | 137 | "name": "Update string timeseries", |
138 | 138 | "image": "", |
139 | - "description": null, | |
139 | + "description": "Simple form to input new string value for pre-defined timeseries key.", | |
140 | 140 | "descriptor": { |
141 | 141 | "type": "latest", |
142 | 142 | "sizeX": 7.5, |
... | ... | @@ -154,7 +154,7 @@ |
154 | 154 | "alias": "update_server_image_attribute", |
155 | 155 | "name": "Update server image attribute", |
156 | 156 | "image": "", |
157 | - "description": null, | |
157 | + "description": "Simple form to input new image for pre-defined server-side attribute key.", | |
158 | 158 | "descriptor": { |
159 | 159 | "type": "latest", |
160 | 160 | "sizeX": 7.5, |
... | ... | @@ -172,7 +172,7 @@ |
172 | 172 | "alias": "update_server_date_attribute", |
173 | 173 | "name": "Update server date attribute", |
174 | 174 | "image": "", |
175 | - "description": null, | |
175 | + "description": "Simple form to input new date value for pre-defined server-side attribute key.", | |
176 | 176 | "descriptor": { |
177 | 177 | "type": "latest", |
178 | 178 | "sizeX": 7.5, |
... | ... | @@ -190,7 +190,7 @@ |
190 | 190 | "alias": "update_server_boolean_attribute", |
191 | 191 | "name": "Update server boolean attribute", |
192 | 192 | "image": "", |
193 | - "description": null, | |
193 | + "description": "Simple form to input new boolean value for pre-defined server-side attribute key.", | |
194 | 194 | "descriptor": { |
195 | 195 | "type": "latest", |
196 | 196 | "sizeX": 7.5, |
... | ... | @@ -208,7 +208,7 @@ |
208 | 208 | "alias": "update_server_double_attribute", |
209 | 209 | "name": "Update server double attribute", |
210 | 210 | "image": "", |
211 | - "description": null, | |
211 | + "description": "Simple form to input new double value for pre-defined server-side attribute key.", | |
212 | 212 | "descriptor": { |
213 | 213 | "type": "latest", |
214 | 214 | "sizeX": 7.5, |
... | ... | @@ -226,7 +226,7 @@ |
226 | 226 | "alias": "update_shared_image_attribute", |
227 | 227 | "name": "Update shared image attribute", |
228 | 228 | "image": "", |
229 | - "description": null, | |
229 | + "description": "Simple form to input new image for pre-defined shared attribute key.", | |
230 | 230 | "descriptor": { |
231 | 231 | "type": "latest", |
232 | 232 | "sizeX": 7.5, |
... | ... | @@ -244,7 +244,7 @@ |
244 | 244 | "alias": "update_shared_date_attribute", |
245 | 245 | "name": "Update shared date attribute", |
246 | 246 | "image": "", |
247 | - "description": null, | |
247 | + "description": "Simple form to input new date for pre-defined shared attribute key.", | |
248 | 248 | "descriptor": { |
249 | 249 | "type": "latest", |
250 | 250 | "sizeX": 7.5, |
... | ... | @@ -262,7 +262,7 @@ |
262 | 262 | "alias": "update_shared_boolean_attribute", |
263 | 263 | "name": "Update shared boolean attribute", |
264 | 264 | "image": "", |
265 | - "description": null, | |
265 | + "description": "Simple form to input new boolean value for pre-defined shared attribute key.", | |
266 | 266 | "descriptor": { |
267 | 267 | "type": "latest", |
268 | 268 | "sizeX": 7.5, |
... | ... | @@ -280,7 +280,7 @@ |
280 | 280 | "alias": "update_server_integer_attribute", |
281 | 281 | "name": "Update server integer attribute", |
282 | 282 | "image": "", |
283 | - "description": null, | |
283 | + "description": "Simple form to input new integer value for pre-defined server attribute key.", | |
284 | 284 | "descriptor": { |
285 | 285 | "type": "latest", |
286 | 286 | "sizeX": 7.5, |
... | ... | @@ -298,7 +298,7 @@ |
298 | 298 | "alias": "update_server_string_attribute", |
299 | 299 | "name": "Update server string attribute", |
300 | 300 | "image": "", |
301 | - "description": null, | |
301 | + "description": "Simple form to input new string value for pre-defined server-side attribute key.", | |
302 | 302 | "descriptor": { |
303 | 303 | "type": "latest", |
304 | 304 | "sizeX": 7.5, |
... | ... | @@ -316,7 +316,7 @@ |
316 | 316 | "alias": "update_location_timeseries", |
317 | 317 | "name": "Update location timeseries", |
318 | 318 | "image": "", |
319 | - "description": null, | |
319 | + "description": "Simple form to input new location for pre-defined timeseries keys.", | |
320 | 320 | "descriptor": { |
321 | 321 | "type": "latest", |
322 | 322 | "sizeX": 7.5, |
... | ... | @@ -334,7 +334,7 @@ |
334 | 334 | "alias": "update_shared_double_attribute", |
335 | 335 | "name": "Update shared double attribute", |
336 | 336 | "image": "", |
337 | - "description": null, | |
337 | + "description": "Simple form to input new double value for pre-defined shared attribute key.", | |
338 | 338 | "descriptor": { |
339 | 339 | "type": "latest", |
340 | 340 | "sizeX": 7.5, |
... | ... | @@ -352,7 +352,7 @@ |
352 | 352 | "alias": "update_shared_integer_attribute", |
353 | 353 | "name": "Update shared integer attribute", |
354 | 354 | "image": "", |
355 | - "description": null, | |
355 | + "description": "Simple form to input new integer value for pre-defined shared attribute key.", | |
356 | 356 | "descriptor": { |
357 | 357 | "type": "latest", |
358 | 358 | "sizeX": 7.5, |
... | ... | @@ -370,7 +370,7 @@ |
370 | 370 | "alias": "update_shared_string_attribute", |
371 | 371 | "name": "Update shared string attribute", |
372 | 372 | "image": "", |
373 | - "description": null, | |
373 | + "description": "Simple form to input new string value for pre-defined shared attribute key.", | |
374 | 374 | "descriptor": { |
375 | 375 | "type": "latest", |
376 | 376 | "sizeX": 7.5, |
... | ... | @@ -388,7 +388,7 @@ |
388 | 388 | "alias": "update_server_location_attribute", |
389 | 389 | "name": "Update server location attribute", |
390 | 390 | "image": "", |
391 | - "description": null, | |
391 | + "description": "Simple form to input new location for pre-defined server attribute key.", | |
392 | 392 | "descriptor": { |
393 | 393 | "type": "latest", |
394 | 394 | "sizeX": 7.5, |
... | ... | @@ -406,7 +406,7 @@ |
406 | 406 | "alias": "markers_placement_google_maps", |
407 | 407 | "name": "Markers Placement - Google Maps", |
408 | 408 | "image": "", |
409 | - "description": null, | |
409 | + "description": "Allows configuring location of the selected entities on the Google Map.", | |
410 | 410 | "descriptor": { |
411 | 411 | "type": "latest", |
412 | 412 | "sizeX": 8.5, |
... | ... | @@ -424,7 +424,7 @@ |
424 | 424 | "alias": "update_shared_location_attribute", |
425 | 425 | "name": "Update shared location attribute", |
426 | 426 | "image": "", |
427 | - "description": null, | |
427 | + "description": "Simple form to input new location for pre-defined shared attribute key.", | |
428 | 428 | "descriptor": { |
429 | 429 | "type": "latest", |
430 | 430 | "sizeX": 7.5, |
... | ... | @@ -442,7 +442,7 @@ |
442 | 442 | "alias": "web_camera_input", |
443 | 443 | "name": "Photo camera input", |
444 | 444 | "image": "", |
445 | - "description": null, | |
445 | + "description": "Simple form to take webcamera image or upload photo.", | |
446 | 446 | "descriptor": { |
447 | 447 | "type": "latest", |
448 | 448 | "sizeX": 7.5, | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "maps_v2", |
4 | 4 | "title": "Maps", |
5 | 5 | "image": "", |
6 | - "description": null | |
6 | + "description": "Visualize latest location or trip animation of the devices or other entities on the indoor or outdoor maps." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "route_map_tencent_maps", |
11 | 11 | "name": "Route Map - Tencent Maps", |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Trip animation on the Tencent maps. Allows to visualize location change over time. Use Trip Animation widget for advanced features.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "timeseries", |
16 | 16 | "sizeX": 8.5, |
... | ... | @@ -28,7 +28,7 @@ |
28 | 28 | "alias": "test", |
29 | 29 | "name": "Trip Animation", |
30 | 30 | "image": "", |
31 | - "description": null, | |
31 | + "description": "Trip animation on the OpenStreetMap or other map providers. Allows to visualize location and other timeseries data for each point in time.", | |
32 | 32 | "descriptor": { |
33 | 33 | "type": "timeseries", |
34 | 34 | "sizeX": 10, |
... | ... | @@ -46,7 +46,7 @@ |
46 | 46 | "alias": "route_map", |
47 | 47 | "name": "Route Map", |
48 | 48 | "image": "", |
49 | - "description": null, | |
49 | + "description": "TTrip animation on the Google maps. Allows to visualize location change over time. Use Trip Animation widget for advanced features.", | |
50 | 50 | "descriptor": { |
51 | 51 | "type": "timeseries", |
52 | 52 | "sizeX": 8.5, |
... | ... | @@ -64,7 +64,7 @@ |
64 | 64 | "alias": "route_map_openstreetmap", |
65 | 65 | "name": "Route Map - OpenStreetMap", |
66 | 66 | "image": "", |
67 | - "description": null, | |
67 | + "description": "Trip animation on the OpenStreetMap. Allows to visualize location change over time. Use Trip Animation widget for advanced features.", | |
68 | 68 | "descriptor": { |
69 | 69 | "type": "timeseries", |
70 | 70 | "sizeX": 8.5, |
... | ... | @@ -82,7 +82,7 @@ |
82 | 82 | "alias": "tencent_maps", |
83 | 83 | "name": "Tencent Maps", |
84 | 84 | "image": "", |
85 | - "description": null, | |
85 | + "description": "Show latest values and location of the entities on Tencent maps.", | |
86 | 86 | "descriptor": { |
87 | 87 | "type": "latest", |
88 | 88 | "sizeX": 9, |
... | ... | @@ -100,7 +100,7 @@ |
100 | 100 | "alias": "here_map", |
101 | 101 | "name": "HERE Map", |
102 | 102 | "image": "", |
103 | - "description": null, | |
103 | + "description": "Show latest values and location of the entities on HERE maps.", | |
104 | 104 | "descriptor": { |
105 | 105 | "type": "latest", |
106 | 106 | "sizeX": 9.5, |
... | ... | @@ -118,7 +118,7 @@ |
118 | 118 | "alias": "image_map", |
119 | 119 | "name": "Image Map", |
120 | 120 | "image": "", |
121 | - "description": null, | |
121 | + "description": "Show latest values and location of the entities on image map. Uses configurable background image and coordintates in range from 0 to 1.", | |
122 | 122 | "descriptor": { |
123 | 123 | "type": "latest", |
124 | 124 | "sizeX": 8.5, |
... | ... | @@ -136,7 +136,7 @@ |
136 | 136 | "alias": "google_maps", |
137 | 137 | "name": "Google Maps", |
138 | 138 | "image": "", |
139 | - "description": null, | |
139 | + "description": "Show latest values and location of the entities on Google Maps.", | |
140 | 140 | "descriptor": { |
141 | 141 | "type": "latest", |
142 | 142 | "sizeX": 8.5, |
... | ... | @@ -154,7 +154,7 @@ |
154 | 154 | "alias": "openstreetmap", |
155 | 155 | "name": "OpenStreetMap", |
156 | 156 | "image": "", |
157 | - "description": null, | |
157 | + "description": "Show latest values and location of the entities on OpenStreetMap.", | |
158 | 158 | "descriptor": { |
159 | 159 | "type": "latest", |
160 | 160 | "sizeX": 8.5, | ... | ... |
... | ... | @@ -3,14 +3,14 @@ |
3 | 3 | "alias": "navigation_widgets", |
4 | 4 | "title": "Navigation widgets", |
5 | 5 | "image": "", |
6 | - "description": null | |
6 | + "description": "Useful to define home dashboard of the user. Contains widgets that enable navigation to other dashboards and menu items." | |
7 | 7 | }, |
8 | 8 | "widgetTypes": [ |
9 | 9 | { |
10 | 10 | "alias": "navigation_cards", |
11 | 11 | "name": "Navigation cards", |
12 | 12 | "image": "", |
13 | - "description": null, | |
13 | + "description": "Allows to open one or more specific pages using filter defined in the advanced settings of the widget.", | |
14 | 14 | "descriptor": { |
15 | 15 | "type": "static", |
16 | 16 | "sizeX": 7, |
... | ... | @@ -19,7 +19,7 @@ |
19 | 19 | "templateHtml": "<tb-navigation-cards-widget [ctx]=\"ctx\"></tb-navigation-cards-widget>", |
20 | 20 | "templateCss": "/*#widget-container {\n overflow-y: auto;\n box-sizing: content-box !important;\n cursor: auto;\n}*/\n\n#widget-container #container {\n overflow-y: auto;\n box-sizing: content-box;\n cursor: auto;\n}", |
21 | 21 | "controllerScript": "self.onInit = function() {\n self.ctx.$scope.navigationCardsWidget.resize();\n}\n\nself.onResize = function() {\n self.ctx.$scope.navigationCardsWidget.resize();\n}\n\nself.onDestroy = function() {\n}\n", |
22 | - "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"filterType\": {\n \"title\": \"Filter type\",\n \"type\": \"string\",\n \"default\": \"all\"\n },\n \"filter\": {\n \"title\": \"Items\",\n \"type\": \"array\"\n }\n },\n \"required\": []\n },\n \"form\": [\n {\n \"key\": \"filterType\",\n \"type\": \"radios\",\n \"direction\": \"row\",\n \"titleMap\": [\n {\n \"value\": \"all\",\n \"name\": \"All items\"\n },\n {\n \"value\": \"include\",\n \"name\": \"Include items\"\n },\n {\n \"value\": \"exclude\",\n \"name\": \"Exclude items\"\n }\n ]\n },\n {\n \"key\": \"filter\",\n \"type\": \"rc-select\",\n \"condition\": \"model.filterType !== 'all'\",\n \"tags\": true,\n \"placeholder\": \"Enter urls to filter\",\n \"items\": [{\"value\": \"/devices\", \"label\": \"/devices\"}, {\"value\": \"/assets\", \"label\": \"/assets\"}, {\"value\": \"/deviceProfies\", \"label\": \"/deviceProfies\"}]\n }\n ]\n}\n", | |
22 | + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"filterType\": {\n \"title\": \"Filter type\",\n \"type\": \"string\",\n \"default\": \"all\"\n },\n \"filter\": {\n \"title\": \"Items\",\n \"type\": \"array\"\n }\n },\n \"required\": []\n },\n \"form\": [\n {\n \"key\": \"filterType\",\n \"type\": \"radios\",\n \"direction\": \"row\",\n \"titleMap\": [\n {\n \"value\": \"all\",\n \"name\": \"All items\"\n },\n {\n \"value\": \"include\",\n \"name\": \"Include items\"\n },\n {\n \"value\": \"exclude\",\n \"name\": \"Exclude items\"\n }\n ]\n },\n {\n \"key\": \"filter\",\n \"type\": \"rc-select\",\n \"condition\": \"model.filterType !== 'all'\",\n \"tags\": true,\n \"placeholder\": \"Enter urls to filter\",\n \"items\": [{\"value\": \"/devices\", \"label\": \"/devices\"}, {\"value\": \"/assets\", \"label\": \"/assets\"}, {\"value\": \"/deviceProfiles\", \"label\": \"/deviceProfiles\"}]\n }\n ]\n}\n", | |
23 | 23 | "dataKeySettingsSchema": "{}\n", |
24 | 24 | "defaultConfig": "{\"datasources\":[{\"type\":\"static\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"rgba(255,255,255,0)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"filterType\":\"all\"},\"title\":\"Navigation cards\",\"dropShadow\":false,\"showTitleIcon\":false,\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"showLegend\":false}" |
25 | 25 | } |
... | ... | @@ -28,7 +28,7 @@ |
28 | 28 | "alias": "navigation_card", |
29 | 29 | "name": "Navigation card", |
30 | 30 | "image": "", |
31 | - "description": null, | |
31 | + "description": "Allows to open specific page using relative path defined in the advanced settings of the widget.", | |
32 | 32 | "descriptor": { |
33 | 33 | "type": "static", |
34 | 34 | "sizeX": 2.5, |
... | ... | @@ -43,4 +43,4 @@ |
43 | 43 | } |
44 | 44 | } |
45 | 45 | ] |
46 | -} | |
\ No newline at end of file | ||
46 | +} | ... | ... |
1 | +-- | |
2 | +-- Copyright © 2016-2021 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 | +CREATE OR REPLACE PROCEDURE cleanup_timeseries_by_ttl(IN null_uuid uuid, | |
18 | + IN system_ttl bigint, INOUT deleted bigint) | |
19 | + LANGUAGE plpgsql AS | |
20 | +$$ | |
21 | +DECLARE | |
22 | +tenant_cursor CURSOR FOR select tenant.id as tenant_id | |
23 | + from tenant; | |
24 | + tenant_id_record uuid; | |
25 | + customer_id_record uuid; | |
26 | + tenant_ttl bigint; | |
27 | + customer_ttl bigint; | |
28 | + deleted_for_entities bigint; | |
29 | + tenant_ttl_ts bigint; | |
30 | + customer_ttl_ts bigint; | |
31 | +BEGIN | |
32 | +OPEN tenant_cursor; | |
33 | +FETCH tenant_cursor INTO tenant_id_record; | |
34 | +WHILE FOUND | |
35 | + LOOP | |
36 | + EXECUTE format( | |
37 | + 'select attribute_kv.long_v from attribute_kv where attribute_kv.entity_id = %L and attribute_kv.attribute_key = %L', | |
38 | + tenant_id_record, 'TTL') INTO tenant_ttl; | |
39 | + if tenant_ttl IS NULL THEN | |
40 | + tenant_ttl := system_ttl; | |
41 | +END IF; | |
42 | + IF tenant_ttl > 0 THEN | |
43 | + tenant_ttl_ts := (EXTRACT(EPOCH FROM current_timestamp) * 1000 - tenant_ttl::bigint * 1000)::bigint; | |
44 | + deleted_for_entities := delete_device_records_from_ts_kv(tenant_id_record, null_uuid, tenant_ttl_ts); | |
45 | + deleted := deleted + deleted_for_entities; | |
46 | + RAISE NOTICE '% telemetry removed for devices where tenant_id = %', deleted_for_entities, tenant_id_record; | |
47 | + deleted_for_entities := delete_asset_records_from_ts_kv(tenant_id_record, null_uuid, tenant_ttl_ts); | |
48 | + deleted := deleted + deleted_for_entities; | |
49 | + RAISE NOTICE '% telemetry removed for assets where tenant_id = %', deleted_for_entities, tenant_id_record; | |
50 | +END IF; | |
51 | +FOR customer_id_record IN | |
52 | +SELECT customer.id AS customer_id FROM customer WHERE customer.tenant_id = tenant_id_record | |
53 | + LOOP | |
54 | + EXECUTE format( | |
55 | + 'select attribute_kv.long_v from attribute_kv where attribute_kv.entity_id = %L and attribute_kv.attribute_key = %L', | |
56 | + customer_id_record, 'TTL') INTO customer_ttl; | |
57 | +IF customer_ttl IS NULL THEN | |
58 | + customer_ttl_ts := tenant_ttl_ts; | |
59 | +ELSE | |
60 | + IF customer_ttl > 0 THEN | |
61 | + customer_ttl_ts := | |
62 | + (EXTRACT(EPOCH FROM current_timestamp) * 1000 - | |
63 | + customer_ttl::bigint * 1000)::bigint; | |
64 | +END IF; | |
65 | +END IF; | |
66 | + IF customer_ttl_ts IS NOT NULL AND customer_ttl_ts > 0 THEN | |
67 | + deleted_for_entities := | |
68 | + delete_customer_records_from_ts_kv(tenant_id_record, customer_id_record, | |
69 | + customer_ttl_ts); | |
70 | + deleted := deleted + deleted_for_entities; | |
71 | + RAISE NOTICE '% telemetry removed for customer with id = % where tenant_id = %', deleted_for_entities, customer_id_record, tenant_id_record; | |
72 | + deleted_for_entities := | |
73 | + delete_device_records_from_ts_kv(tenant_id_record, customer_id_record, | |
74 | + customer_ttl_ts); | |
75 | + deleted := deleted + deleted_for_entities; | |
76 | + RAISE NOTICE '% telemetry removed for devices where tenant_id = % and customer_id = %', deleted_for_entities, tenant_id_record, customer_id_record; | |
77 | + deleted_for_entities := delete_asset_records_from_ts_kv(tenant_id_record, | |
78 | + customer_id_record, | |
79 | + customer_ttl_ts); | |
80 | + deleted := deleted + deleted_for_entities; | |
81 | + RAISE NOTICE '% telemetry removed for assets where tenant_id = % and customer_id = %', deleted_for_entities, tenant_id_record, customer_id_record; | |
82 | +END IF; | |
83 | +END LOOP; | |
84 | +FETCH tenant_cursor INTO tenant_id_record; | |
85 | +END LOOP; | |
86 | +END | |
87 | +$$; | ... | ... |
... | ... | @@ -18,6 +18,7 @@ package org.thingsboard.server.controller; |
18 | 18 | import com.fasterxml.jackson.core.type.TypeReference; |
19 | 19 | import com.fasterxml.jackson.databind.JsonNode; |
20 | 20 | import com.fasterxml.jackson.databind.ObjectMapper; |
21 | +import com.fasterxml.jackson.databind.node.ArrayNode; | |
21 | 22 | import com.fasterxml.jackson.databind.node.ObjectNode; |
22 | 23 | import lombok.extern.slf4j.Slf4j; |
23 | 24 | import org.springframework.beans.factory.annotation.Autowired; |
... | ... | @@ -422,6 +423,25 @@ public class RuleChainController extends BaseController { |
422 | 423 | } |
423 | 424 | |
424 | 425 | private String msgToOutput(TbMsg msg) throws Exception { |
426 | + JsonNode resultNode = convertMsgToOut(msg); | |
427 | + return objectMapper.writeValueAsString(resultNode); | |
428 | + } | |
429 | + | |
430 | + private String msgToOutput(List<TbMsg> msgs) throws Exception { | |
431 | + JsonNode resultNode; | |
432 | + if (msgs.size() > 1) { | |
433 | + resultNode = objectMapper.createArrayNode(); | |
434 | + for (TbMsg msg : msgs) { | |
435 | + JsonNode convertedData = convertMsgToOut(msg); | |
436 | + ((ArrayNode) resultNode).add(convertedData); | |
437 | + } | |
438 | + } else { | |
439 | + resultNode = convertMsgToOut(msgs.get(0)); | |
440 | + } | |
441 | + return objectMapper.writeValueAsString(resultNode); | |
442 | + } | |
443 | + | |
444 | + private JsonNode convertMsgToOut(TbMsg msg) throws Exception{ | |
425 | 445 | ObjectNode msgData = objectMapper.createObjectNode(); |
426 | 446 | if (!StringUtils.isEmpty(msg.getData())) { |
427 | 447 | msgData.set("msg", objectMapper.readTree(msg.getData())); |
... | ... | @@ -429,7 +449,8 @@ public class RuleChainController extends BaseController { |
429 | 449 | Map<String, String> metadata = msg.getMetaData().getData(); |
430 | 450 | msgData.set("metadata", objectMapper.valueToTree(metadata)); |
431 | 451 | msgData.put("msgType", msg.getType()); |
432 | - return objectMapper.writeValueAsString(msgData); | |
452 | + return msgData; | |
433 | 453 | } |
434 | 454 | |
455 | + | |
435 | 456 | } | ... | ... |
application/src/main/java/org/thingsboard/server/service/apiusage/DefaultTbApiUsageStateService.java
... | ... | @@ -72,6 +72,7 @@ import java.util.concurrent.ConcurrentHashMap; |
72 | 72 | import java.util.concurrent.ExecutionException; |
73 | 73 | import java.util.concurrent.ExecutorService; |
74 | 74 | import java.util.concurrent.Executors; |
75 | +import java.util.concurrent.Future; | |
75 | 76 | import java.util.concurrent.TimeUnit; |
76 | 77 | import java.util.concurrent.locks.Lock; |
77 | 78 | import java.util.concurrent.locks.ReentrantLock; |
... | ... | @@ -421,20 +422,33 @@ public class DefaultTbApiUsageStateService extends TbApplicationEventListener<Pa |
421 | 422 | private void initStatesFromDataBase() { |
422 | 423 | try { |
423 | 424 | log.info("Initializing tenant states."); |
424 | - PageDataIterable<Tenant> tenantIterator = new PageDataIterable<>(tenantService::findTenants, 1024); | |
425 | - for (Tenant tenant : tenantIterator) { | |
426 | - if (!myTenantStates.containsKey(tenant.getId()) && partitionService.resolve(ServiceType.TB_CORE, tenant.getId(), tenant.getId()).isMyPartition()) { | |
427 | - log.debug("[{}] Initializing tenant state.", tenant.getId()); | |
428 | - updateLock.lock(); | |
429 | - try { | |
430 | - updateTenantState(getOrFetchState(tenant.getId()), tenantProfileCache.get(tenant.getTenantProfileId())); | |
431 | - log.debug("[{}] Initialized tenant state.", tenant.getId()); | |
432 | - } catch (Exception e) { | |
433 | - log.warn("[{}] Failed to initialize tenant API state", tenant.getId(), e); | |
434 | - } finally { | |
435 | - updateLock.unlock(); | |
425 | + updateLock.lock(); | |
426 | + try { | |
427 | + ExecutorService tmpInitExecutor = Executors.newWorkStealingPool(20); | |
428 | + try { | |
429 | + PageDataIterable<Tenant> tenantIterator = new PageDataIterable<>(tenantService::findTenants, 1024); | |
430 | + List<Future<?>> futures = new ArrayList<>(); | |
431 | + for (Tenant tenant : tenantIterator) { | |
432 | + if (!myTenantStates.containsKey(tenant.getId()) && partitionService.resolve(ServiceType.TB_CORE, tenant.getId(), tenant.getId()).isMyPartition()) { | |
433 | + log.debug("[{}] Initializing tenant state.", tenant.getId()); | |
434 | + futures.add(tmpInitExecutor.submit(() -> { | |
435 | + try { | |
436 | + updateTenantState(getOrFetchState(tenant.getId()), tenantProfileCache.get(tenant.getTenantProfileId())); | |
437 | + log.debug("[{}] Initialized tenant state.", tenant.getId()); | |
438 | + } catch (Exception e) { | |
439 | + log.warn("[{}] Failed to initialize tenant API state", tenant.getId(), e); | |
440 | + } | |
441 | + })); | |
442 | + } | |
443 | + } | |
444 | + for (Future<?> future : futures) { | |
445 | + future.get(); | |
436 | 446 | } |
447 | + } finally { | |
448 | + tmpInitExecutor.shutdownNow(); | |
437 | 449 | } |
450 | + } finally { | |
451 | + updateLock.unlock(); | |
438 | 452 | } |
439 | 453 | log.info("Initialized tenant states."); |
440 | 454 | } catch (Exception e) { | ... | ... |
... | ... | @@ -437,6 +437,7 @@ public class SqlDatabaseUpgradeService implements DatabaseEntitiesUpgradeService |
437 | 437 | case "3.2.1": |
438 | 438 | try (Connection conn = DriverManager.getConnection(dbUrl, dbUserName, dbPassword)) { |
439 | 439 | log.info("Updating schema ..."); |
440 | + conn.createStatement().execute("CREATE INDEX IF NOT EXISTS idx_audit_log_tenant_id_and_created_time ON audit_log(tenant_id, created_time);"); | |
440 | 441 | schemaUpdateFile = Paths.get(installScripts.getDataDir(), "upgrade", "3.2.1", SCHEMA_UPDATE_SQL); |
441 | 442 | loadSql(schemaUpdateFile, conn); |
442 | 443 | conn.createStatement().execute("UPDATE tb_schema_settings SET schema_version = 3002002;"); | ... | ... |
... | ... | @@ -178,7 +178,11 @@ public class TimescaleTsDatabaseUpgradeService extends AbstractSqlTsDatabaseUpgr |
178 | 178 | } |
179 | 179 | break; |
180 | 180 | case "3.1.1": |
181 | + break; | |
181 | 182 | case "3.2.1": |
183 | + try (Connection conn = DriverManager.getConnection(dbUrl, dbUserName, dbPassword)) { | |
184 | + loadSql(conn, LOAD_TTL_FUNCTIONS_SQL); | |
185 | + } | |
182 | 186 | break; |
183 | 187 | default: |
184 | 188 | throw new RuntimeException("Unable to upgrade SQL database, unsupported fromVersion: " + fromVersion); | ... | ... |
... | ... | @@ -283,10 +283,12 @@ public class DefaultTbClusterService implements TbClusterService { |
283 | 283 | byte[] msgBytes = encodingService.encode(msg); |
284 | 284 | TbQueueProducer<TbProtoQueueMsg<ToRuleEngineNotificationMsg>> toRuleEngineProducer = producerProvider.getRuleEngineNotificationsMsgProducer(); |
285 | 285 | Set<String> tbRuleEngineServices = new HashSet<>(partitionService.getAllServiceIds(ServiceType.TB_RULE_ENGINE)); |
286 | - if (msg.getEntityId().getEntityType().equals(EntityType.TENANT) | |
287 | - || msg.getEntityId().getEntityType().equals(EntityType.TENANT_PROFILE) | |
288 | - || msg.getEntityId().getEntityType().equals(EntityType.DEVICE_PROFILE) | |
289 | - || msg.getEntityId().getEntityType().equals(EntityType.API_USAGE_STATE)) { | |
286 | + EntityType entityType = msg.getEntityId().getEntityType(); | |
287 | + if (entityType.equals(EntityType.TENANT) | |
288 | + || entityType.equals(EntityType.TENANT_PROFILE) | |
289 | + || entityType.equals(EntityType.DEVICE_PROFILE) | |
290 | + || entityType.equals(EntityType.API_USAGE_STATE) | |
291 | + || (entityType.equals(EntityType.DEVICE) && msg.getEvent() == ComponentLifecycleEvent.UPDATED)) { | |
290 | 292 | TbQueueProducer<TbProtoQueueMsg<ToCoreNotificationMsg>> toCoreNfProducer = producerProvider.getTbCoreNotificationsMsgProducer(); |
291 | 293 | Set<String> tbCoreServices = partitionService.getAllServiceIds(ServiceType.TB_CORE); |
292 | 294 | for (String serviceId : tbCoreServices) { | ... | ... |
... | ... | @@ -108,13 +108,18 @@ public class RuleNodeJsScriptEngine implements org.thingsboard.rule.engine.api.S |
108 | 108 | } |
109 | 109 | |
110 | 110 | @Override |
111 | - public TbMsg executeUpdate(TbMsg msg) throws ScriptException { | |
111 | + public List<TbMsg> executeUpdate(TbMsg msg) throws ScriptException { | |
112 | 112 | JsonNode result = executeScript(msg); |
113 | - if (!result.isObject()) { | |
113 | + if (result.isObject()) { | |
114 | + return Collections.singletonList(unbindMsg(result, msg)); | |
115 | + } else if (result.isArray()){ | |
116 | + List<TbMsg> res = new ArrayList<>(result.size()); | |
117 | + result.forEach(jsonObject -> res.add(unbindMsg(jsonObject, msg))); | |
118 | + return res; | |
119 | + } else { | |
114 | 120 | log.warn("Wrong result type: {}", result.getNodeType()); |
115 | 121 | throw new ScriptException("Wrong result type: " + result.getNodeType()); |
116 | 122 | } |
117 | - return unbindMsg(result, msg); | |
118 | 123 | } |
119 | 124 | |
120 | 125 | @Override | ... | ... |
... | ... | @@ -44,7 +44,8 @@ public class CustomerUserPermissions extends AbstractPermissions { |
44 | 44 | |
45 | 45 | private static final PermissionChecker customerEntityPermissionChecker = |
46 | 46 | new PermissionChecker.GenericPermissionChecker(Operation.READ, Operation.READ_CREDENTIALS, |
47 | - Operation.READ_ATTRIBUTES, Operation.READ_TELEMETRY, Operation.RPC_CALL, Operation.CLAIM_DEVICES) { | |
47 | + Operation.READ_ATTRIBUTES, Operation.READ_TELEMETRY, Operation.RPC_CALL, Operation.CLAIM_DEVICES, | |
48 | + Operation.WRITE, Operation.WRITE_ATTRIBUTES, Operation.WRITE_TELEMETRY) { | |
48 | 49 | |
49 | 50 | @Override |
50 | 51 | @SuppressWarnings("unchecked") | ... | ... |
... | ... | @@ -213,7 +213,7 @@ public class DefaultSubscriptionManagerService extends TbApplicationEventListene |
213 | 213 | }, s -> true, s -> { |
214 | 214 | List<TsKvEntry> subscriptionUpdate = null; |
215 | 215 | for (TsKvEntry kv : ts) { |
216 | - if (isInTimeRange(s, kv.getTs()) && (s.isAllKeys() || s.getKeyStates().containsKey((kv.getKey())))) { | |
216 | + if ((s.isAllKeys() || s.getKeyStates().containsKey((kv.getKey())))) { | |
217 | 217 | if (subscriptionUpdate == null) { |
218 | 218 | subscriptionUpdate = new ArrayList<>(); |
219 | 219 | } |
... | ... | @@ -375,11 +375,6 @@ public class DefaultSubscriptionManagerService extends TbApplicationEventListene |
375 | 375 | } |
376 | 376 | } |
377 | 377 | |
378 | - private boolean isInTimeRange(TbTimeseriesSubscription subscription, long kvTime) { | |
379 | - return (subscription.getStartTime() == 0 || subscription.getStartTime() <= kvTime) | |
380 | - && (subscription.getEndTime() == 0 || subscription.getEndTime() >= kvTime); | |
381 | - } | |
382 | - | |
383 | 378 | private void removeSubscriptionFromEntityMap(TbSubscription sub) { |
384 | 379 | Set<TbSubscription> entitySubSet = subscriptionsByEntityId.get(sub.getEntityId()); |
385 | 380 | if (entitySubSet != null) { |
... | ... | @@ -429,16 +424,9 @@ public class DefaultSubscriptionManagerService extends TbApplicationEventListene |
429 | 424 | serviceId, subscription.getSessionId(), subscription.getSubscriptionId(), subscription.getEntityId()); |
430 | 425 | |
431 | 426 | long curTs = System.currentTimeMillis(); |
432 | - List<ReadTsKvQuery> queries = new ArrayList<>(); | |
433 | - subscription.getKeyStates().forEach((key, value) -> { | |
434 | - if (curTs > value) { | |
435 | - long startTs = subscription.getStartTime() > 0 ? Math.max(subscription.getStartTime(), value + 1L) : (value + 1L); | |
436 | - long endTs = subscription.getEndTime() > 0 ? Math.min(subscription.getEndTime(), curTs) : curTs; | |
437 | - queries.add(new BaseReadTsKvQuery(key, startTs, endTs, 0, 1000, Aggregation.NONE)); | |
438 | - } | |
439 | - }); | |
440 | - if (!queries.isEmpty()) { | |
441 | - DonAsynchron.withCallback(tsService.findAll(subscription.getTenantId(), subscription.getEntityId(), queries), | |
427 | + | |
428 | + if (subscription.isLatestValues()) { | |
429 | + DonAsynchron.withCallback(tsService.findLatest(subscription.getTenantId(), subscription.getEntityId(), subscription.getKeyStates().keySet()), | |
442 | 430 | missedUpdates -> { |
443 | 431 | if (missedUpdates != null && !missedUpdates.isEmpty()) { |
444 | 432 | TopicPartitionInfo tpi = partitionService.getNotificationsTopic(ServiceType.TB_CORE, subscription.getServiceId()); |
... | ... | @@ -447,6 +435,26 @@ public class DefaultSubscriptionManagerService extends TbApplicationEventListene |
447 | 435 | }, |
448 | 436 | e -> log.error("Failed to fetch missed updates.", e), |
449 | 437 | tsCallBackExecutor); |
438 | + } else { | |
439 | + List<ReadTsKvQuery> queries = new ArrayList<>(); | |
440 | + subscription.getKeyStates().forEach((key, value) -> { | |
441 | + if (curTs > value) { | |
442 | + long startTs = subscription.getStartTime() > 0 ? Math.max(subscription.getStartTime(), value + 1L) : (value + 1L); | |
443 | + long endTs = subscription.getEndTime() > 0 ? Math.min(subscription.getEndTime(), curTs) : curTs; | |
444 | + queries.add(new BaseReadTsKvQuery(key, startTs, endTs, 0, 1000, Aggregation.NONE)); | |
445 | + } | |
446 | + }); | |
447 | + if (!queries.isEmpty()) { | |
448 | + DonAsynchron.withCallback(tsService.findAll(subscription.getTenantId(), subscription.getEntityId(), queries), | |
449 | + missedUpdates -> { | |
450 | + if (missedUpdates != null && !missedUpdates.isEmpty()) { | |
451 | + TopicPartitionInfo tpi = partitionService.getNotificationsTopic(ServiceType.TB_CORE, subscription.getServiceId()); | |
452 | + toCoreNotificationsProducer.send(tpi, toProto(subscription, missedUpdates), null); | |
453 | + } | |
454 | + }, | |
455 | + e -> log.error("Failed to fetch missed updates.", e), | |
456 | + tsCallBackExecutor); | |
457 | + } | |
450 | 458 | } |
451 | 459 | } |
452 | 460 | |
... | ... | @@ -468,12 +476,19 @@ public class DefaultSubscriptionManagerService extends TbApplicationEventListene |
468 | 476 | data.forEach((key, value) -> { |
469 | 477 | TbSubscriptionUpdateValueListProto.Builder dataBuilder = TbSubscriptionUpdateValueListProto.newBuilder(); |
470 | 478 | dataBuilder.setKey(key); |
471 | - value.forEach(v -> { | |
479 | + boolean hasData = false; | |
480 | + for (Object v : value) { | |
472 | 481 | Object[] array = (Object[]) v; |
473 | 482 | dataBuilder.addTs((long) array[0]); |
474 | - dataBuilder.addValue((String) array[1]); | |
475 | - }); | |
476 | - builder.addData(dataBuilder.build()); | |
483 | + String strVal = (String) array[1]; | |
484 | + if (strVal != null) { | |
485 | + hasData = true; | |
486 | + dataBuilder.addValue(strVal); | |
487 | + } | |
488 | + } | |
489 | + if (hasData) { | |
490 | + builder.addData(dataBuilder.build()); | |
491 | + } | |
477 | 492 | }); |
478 | 493 | |
479 | 494 | ToCoreNotificationMsg toCoreMsg = ToCoreNotificationMsg.newBuilder().setToLocalSubscriptionServiceMsg( | ... | ... |
... | ... | @@ -215,7 +215,7 @@ public class DefaultTbEntityDataSubscriptionService implements TbEntityDataSubsc |
215 | 215 | } else { |
216 | 216 | historyFuture = Futures.immediateFuture(ctx); |
217 | 217 | } |
218 | - Futures.addCallback(historyFuture, new FutureCallback<TbEntityDataSubCtx>() { | |
218 | + Futures.addCallback(historyFuture, new FutureCallback<>() { | |
219 | 219 | @Override |
220 | 220 | public void onSuccess(@Nullable TbEntityDataSubCtx theCtx) { |
221 | 221 | if (cmd.getLatestCmd() != null) { |
... | ... | @@ -278,7 +278,7 @@ public class DefaultTbEntityDataSubscriptionService implements TbEntityDataSubsc |
278 | 278 | wsService.sendWsMsg(ctx.getSessionId(), update); |
279 | 279 | } else { |
280 | 280 | ctx.fetchAlarms(); |
281 | - ctx.createSubscriptions(cmd.getQuery().getLatestValues(), true); | |
281 | + ctx.createLatestValuesSubscriptions(cmd.getQuery().getLatestValues()); | |
282 | 282 | if (adq.getPageLink().getTimeWindow() > 0) { |
283 | 283 | TbAlarmDataSubCtx finalCtx = ctx; |
284 | 284 | ScheduledFuture<?> task = scheduler.scheduleWithFixedDelay( |
... | ... | @@ -419,7 +419,7 @@ public class DefaultTbEntityDataSubscriptionService implements TbEntityDataSubsc |
419 | 419 | } |
420 | 420 | wsService.sendWsMsg(ctx.getSessionId(), update); |
421 | 421 | if (subscribe) { |
422 | - ctx.createSubscriptions(keys.stream().map(key -> new EntityKey(EntityKeyType.TIME_SERIES, key)).collect(Collectors.toList()), false); | |
422 | + ctx.createTimeseriesSubscriptions(keys.stream().map(key -> new EntityKey(EntityKeyType.TIME_SERIES, key)).collect(Collectors.toList()), cmd.getStartTs(), cmd.getEndTs()); | |
423 | 423 | } |
424 | 424 | ctx.getData().getData().forEach(ed -> ed.getTimeseries().clear()); |
425 | 425 | return ctx; |
... | ... | @@ -468,7 +468,7 @@ public class DefaultTbEntityDataSubscriptionService implements TbEntityDataSubsc |
468 | 468 | update = new EntityDataUpdate(ctx.getCmdId(), null, ctx.getData().getData(), ctx.getMaxEntitiesPerDataSubscription()); |
469 | 469 | } |
470 | 470 | wsService.sendWsMsg(ctx.getSessionId(), update); |
471 | - ctx.createSubscriptions(latestCmd.getKeys(), true); | |
471 | + ctx.createLatestValuesSubscriptions(latestCmd.getKeys()); | |
472 | 472 | } |
473 | 473 | |
474 | 474 | @Override |
... | ... | @@ -484,7 +484,7 @@ public class DefaultTbEntityDataSubscriptionService implements TbEntityDataSubsc |
484 | 484 | wsService.sendWsMsg(ctx.getSessionId(), update); |
485 | 485 | ctx.setInitialDataSent(true); |
486 | 486 | } |
487 | - ctx.createSubscriptions(latestCmd.getKeys(), true); | |
487 | + ctx.createLatestValuesSubscriptions(latestCmd.getKeys()); | |
488 | 488 | } |
489 | 489 | } |
490 | 490 | ... | ... |
... | ... | @@ -115,10 +115,18 @@ public abstract class TbAbstractDataSubCtx<T extends AbstractDataQuery<? extends |
115 | 115 | } |
116 | 116 | } |
117 | 117 | |
118 | - public void createSubscriptions(List<EntityKey> keys, boolean resultToLatestValues) { | |
118 | + public void createLatestValuesSubscriptions(List<EntityKey> keys) { | |
119 | + createSubscriptions(keys, true, 0, 0); | |
120 | + } | |
121 | + | |
122 | + public void createTimeseriesSubscriptions(List<EntityKey> keys, long startTs, long endTs) { | |
123 | + createSubscriptions(keys, false, startTs, endTs); | |
124 | + } | |
125 | + | |
126 | + private void createSubscriptions(List<EntityKey> keys, boolean latestValues, long startTs, long endTs) { | |
119 | 127 | Map<EntityKeyType, List<EntityKey>> keysByType = getEntityKeyByTypeMap(keys); |
120 | 128 | for (EntityData entityData : data.getData()) { |
121 | - List<TbSubscription> entitySubscriptions = addSubscriptions(entityData, keysByType, resultToLatestValues); | |
129 | + List<TbSubscription> entitySubscriptions = addSubscriptions(entityData, keysByType, latestValues, startTs, endTs); | |
122 | 130 | entitySubscriptions.forEach(localSubscriptionService::addSubscription); |
123 | 131 | } |
124 | 132 | } |
... | ... | @@ -129,14 +137,14 @@ public abstract class TbAbstractDataSubCtx<T extends AbstractDataQuery<? extends |
129 | 137 | return keysByType; |
130 | 138 | } |
131 | 139 | |
132 | - protected List<TbSubscription> addSubscriptions(EntityData entityData, Map<EntityKeyType, List<EntityKey>> keysByType, boolean resultToLatestValues) { | |
140 | + protected List<TbSubscription> addSubscriptions(EntityData entityData, Map<EntityKeyType, List<EntityKey>> keysByType, boolean latestValues, long startTs, long endTs) { | |
133 | 141 | List<TbSubscription> subscriptionList = new ArrayList<>(); |
134 | 142 | keysByType.forEach((keysType, keysList) -> { |
135 | 143 | int subIdx = sessionRef.getSessionSubIdSeq().incrementAndGet(); |
136 | 144 | subToEntityIdMap.put(subIdx, entityData.getEntityId()); |
137 | 145 | switch (keysType) { |
138 | 146 | case TIME_SERIES: |
139 | - subscriptionList.add(createTsSub(entityData, subIdx, keysList, resultToLatestValues)); | |
147 | + subscriptionList.add(createTsSub(entityData, subIdx, keysList, latestValues, startTs, endTs)); | |
140 | 148 | break; |
141 | 149 | case CLIENT_ATTRIBUTE: |
142 | 150 | subscriptionList.add(createAttrSub(entityData, subIdx, keysType, TbAttributeSubscriptionScope.CLIENT_SCOPE, keysList)); |
... | ... | @@ -171,9 +179,9 @@ public abstract class TbAbstractDataSubCtx<T extends AbstractDataQuery<? extends |
171 | 179 | .build(); |
172 | 180 | } |
173 | 181 | |
174 | - private TbSubscription createTsSub(EntityData entityData, int subIdx, List<EntityKey> subKeys, boolean resultToLatestValues) { | |
182 | + private TbSubscription createTsSub(EntityData entityData, int subIdx, List<EntityKey> subKeys, boolean latestValues, long startTs, long endTs) { | |
175 | 183 | Map<String, Long> keyStates = buildKeyStats(entityData, EntityKeyType.TIME_SERIES, subKeys); |
176 | - if (entityData.getTimeseries() != null) { | |
184 | + if (!latestValues && entityData.getTimeseries() != null) { | |
177 | 185 | entityData.getTimeseries().forEach((k, v) -> { |
178 | 186 | long ts = Arrays.stream(v).map(TsValue::getTs).max(Long::compareTo).orElse(0L); |
179 | 187 | log.trace("[{}][{}] Updating key: {} with ts: {}", serviceId, cmdId, k, ts); |
... | ... | @@ -187,9 +195,12 @@ public abstract class TbAbstractDataSubCtx<T extends AbstractDataQuery<? extends |
187 | 195 | .subscriptionId(subIdx) |
188 | 196 | .tenantId(sessionRef.getSecurityCtx().getTenantId()) |
189 | 197 | .entityId(entityData.getEntityId()) |
190 | - .updateConsumer((sessionId, subscriptionUpdate) -> sendWsMsg(sessionId, subscriptionUpdate, EntityKeyType.TIME_SERIES, resultToLatestValues)) | |
198 | + .updateConsumer((sessionId, subscriptionUpdate) -> sendWsMsg(sessionId, subscriptionUpdate, EntityKeyType.TIME_SERIES, latestValues)) | |
191 | 199 | .allKeys(false) |
192 | 200 | .keyStates(keyStates) |
201 | + .latestValues(latestValues) | |
202 | + .startTime(startTs) | |
203 | + .endTime(endTs) | |
193 | 204 | .build(); |
194 | 205 | } |
195 | 206 | ... | ... |
... | ... | @@ -132,8 +132,8 @@ public class TbAlarmDataSubCtx extends TbAbstractDataSubCtx<AlarmDataQuery> { |
132 | 132 | } |
133 | 133 | |
134 | 134 | @Override |
135 | - public void createSubscriptions(List<EntityKey> keys, boolean resultToLatestValues) { | |
136 | - super.createSubscriptions(keys, resultToLatestValues); | |
135 | + public void createLatestValuesSubscriptions(List<EntityKey> keys) { | |
136 | + super.createLatestValuesSubscriptions(keys); | |
137 | 137 | createAlarmSubscriptions(); |
138 | 138 | } |
139 | 139 | |
... | ... | @@ -282,7 +282,7 @@ public class TbAlarmDataSubCtx extends TbAbstractDataSubCtx<AlarmDataQuery> { |
282 | 282 | newSubsList.forEach( |
283 | 283 | entity -> { |
284 | 284 | log.trace("[{}][{}] Found new subscription for entity: {}", sessionRef.getSessionId(), cmdId, entity.getEntityId()); |
285 | - subsToAdd.addAll(addSubscriptions(entity, keysByType, true)); | |
285 | + subsToAdd.addAll(addSubscriptions(entity, keysByType, true, 0, 0)); | |
286 | 286 | } |
287 | 287 | ); |
288 | 288 | } | ... | ... |
... | ... | @@ -50,9 +50,6 @@ public class TbEntityDataSubCtx extends TbAbstractDataSubCtx<EntityDataQuery> { |
50 | 50 | |
51 | 51 | @Getter |
52 | 52 | @Setter |
53 | - private TimeSeriesCmd tsCmd; | |
54 | - @Getter | |
55 | - @Setter | |
56 | 53 | private boolean initialDataSent; |
57 | 54 | private TimeSeriesCmd curTsCmd; |
58 | 55 | private LatestValueCmd latestValueCmd; |
... | ... | @@ -137,7 +134,8 @@ public class TbEntityDataSubCtx extends TbAbstractDataSubCtx<EntityDataQuery> { |
137 | 134 | for (TsValue update : new ArrayList<>(updateList)) { |
138 | 135 | if (update.getTs() < v.getTs()) { |
139 | 136 | log.trace("[{}][{}][{}] Removed stale update for key: {} and ts: {}", sessionId, cmdId, subscriptionUpdate.getSubscriptionId(), k, update.getTs()); |
140 | - updateList.remove(update); | |
137 | + // Looks like this is redundant feature and our UI is ready to merge the updates. | |
138 | + //updateList.remove(update); | |
141 | 139 | } else if ((update.getTs() == v.getTs() && update.getValue().equals(v.getValue()))) { |
142 | 140 | log.trace("[{}][{}][{}] Removed duplicate update for key: {} and ts: {}", sessionId, cmdId, subscriptionUpdate.getSubscriptionId(), k, update.getTs()); |
143 | 141 | updateList.remove(update); |
... | ... | @@ -182,25 +180,18 @@ public class TbEntityDataSubCtx extends TbAbstractDataSubCtx<EntityDataQuery> { |
182 | 180 | subIdsToCancel.forEach(subToEntityIdMap::remove); |
183 | 181 | List<EntityData> newSubsList = newDataMap.entrySet().stream().filter(entry -> !currentSubs.contains(entry.getKey())).map(Map.Entry::getValue).collect(Collectors.toList()); |
184 | 182 | if (!newSubsList.isEmpty()) { |
185 | - boolean resultToLatestValues; | |
186 | - List<EntityKey> keys = null; | |
187 | - if (curTsCmd != null) { | |
188 | - resultToLatestValues = false; | |
189 | - keys = curTsCmd.getKeys().stream().map(key -> new EntityKey(EntityKeyType.TIME_SERIES, key)).collect(Collectors.toList()); | |
190 | - } else if (latestValueCmd != null) { | |
191 | - resultToLatestValues = true; | |
192 | - keys = latestValueCmd.getKeys(); | |
193 | - } else { | |
194 | - resultToLatestValues = true; | |
195 | - } | |
196 | - if (keys != null && !keys.isEmpty()) { | |
197 | - Map<EntityKeyType, List<EntityKey>> keysByType = getEntityKeyByTypeMap(keys); | |
198 | - newSubsList.forEach( | |
199 | - entity -> { | |
200 | - log.trace("[{}][{}] Found new subscription for entity: {}", sessionRef.getSessionId(), cmdId, entity.getEntityId()); | |
201 | - subsToAdd.addAll(addSubscriptions(entity, keysByType, resultToLatestValues)); | |
202 | - } | |
203 | - ); | |
183 | + // NOTE: We ignore the TS subscriptions for new entities here, because widgets will re-init it's content and will create new subscriptions. | |
184 | + if (curTsCmd == null && latestValueCmd != null) { | |
185 | + List<EntityKey> keys = latestValueCmd.getKeys(); | |
186 | + if (keys != null && !keys.isEmpty()) { | |
187 | + Map<EntityKeyType, List<EntityKey>> keysByType = getEntityKeyByTypeMap(keys); | |
188 | + newSubsList.forEach( | |
189 | + entity -> { | |
190 | + log.trace("[{}][{}] Found new subscription for entity: {}", sessionRef.getSessionId(), cmdId, entity.getEntityId()); | |
191 | + subsToAdd.addAll(addSubscriptions(entity, keysByType, true, 0, 0)); | |
192 | + } | |
193 | + ); | |
194 | + } | |
204 | 195 | } |
205 | 196 | } |
206 | 197 | wsService.sendWsMsg(sessionRef.getSessionId(), new EntityDataUpdate(cmdId, data, null, maxEntitiesPerDataSubscription)); | ... | ... |
... | ... | @@ -83,6 +83,7 @@ public class TbSubscriptionUtils { |
83 | 83 | TbSubscriptionKetStateProto.newBuilder().setKey(key).setTs(value).build())); |
84 | 84 | tSubProto.setStartTime(tSub.getStartTime()); |
85 | 85 | tSubProto.setEndTime(tSub.getEndTime()); |
86 | + tSubProto.setLatestValues(tSub.isLatestValues()); | |
86 | 87 | msgBuilder.setTelemetrySub(tSubProto.build()); |
87 | 88 | break; |
88 | 89 | case ATTRIBUTES: |
... | ... | @@ -146,6 +147,7 @@ public class TbSubscriptionUtils { |
146 | 147 | telemetrySub.getKeyStatesList().forEach(ksProto -> keyStates.put(ksProto.getKey(), ksProto.getTs())); |
147 | 148 | builder.startTime(telemetrySub.getStartTime()); |
148 | 149 | builder.endTime(telemetrySub.getEndTime()); |
150 | + builder.latestValues(telemetrySub.getLatestValues()); | |
149 | 151 | builder.keyStates(keyStates); |
150 | 152 | return builder.build(); |
151 | 153 | } | ... | ... |
... | ... | @@ -34,16 +34,19 @@ public class TbTimeseriesSubscription extends TbSubscription<TelemetrySubscripti |
34 | 34 | private final long startTime; |
35 | 35 | @Getter |
36 | 36 | private final long endTime; |
37 | + @Getter | |
38 | + private final boolean latestValues; | |
37 | 39 | |
38 | 40 | @Builder |
39 | 41 | public TbTimeseriesSubscription(String serviceId, String sessionId, int subscriptionId, TenantId tenantId, EntityId entityId, |
40 | 42 | BiConsumer<String, TelemetrySubscriptionUpdate> updateConsumer, |
41 | - boolean allKeys, Map<String, Long> keyStates, long startTime, long endTime) { | |
43 | + boolean allKeys, Map<String, Long> keyStates, long startTime, long endTime, boolean latestValues) { | |
42 | 44 | super(serviceId, sessionId, subscriptionId, tenantId, entityId, TbSubscriptionType.TIMESERIES, updateConsumer); |
43 | 45 | this.allKeys = allKeys; |
44 | 46 | this.keyStates = keyStates; |
45 | 47 | this.startTime = startTime; |
46 | 48 | this.endTime = endTime; |
49 | + this.latestValues = latestValues; | |
47 | 50 | } |
48 | 51 | |
49 | 52 | @Override | ... | ... |
... | ... | @@ -22,9 +22,7 @@ import com.google.common.util.concurrent.ListenableFuture; |
22 | 22 | import lombok.extern.slf4j.Slf4j; |
23 | 23 | import org.checkerframework.checker.nullness.qual.Nullable; |
24 | 24 | import org.springframework.beans.factory.annotation.Autowired; |
25 | -import org.springframework.context.event.EventListener; | |
26 | 25 | import org.springframework.stereotype.Service; |
27 | -import org.thingsboard.common.util.ThingsBoardThreadFactory; | |
28 | 26 | import org.thingsboard.server.common.data.alarm.Alarm; |
29 | 27 | import org.thingsboard.server.common.data.alarm.AlarmInfo; |
30 | 28 | import org.thingsboard.server.common.data.alarm.AlarmQuery; |
... | ... | @@ -35,43 +33,22 @@ import org.thingsboard.server.common.data.id.AlarmId; |
35 | 33 | import org.thingsboard.server.common.data.id.CustomerId; |
36 | 34 | import org.thingsboard.server.common.data.id.EntityId; |
37 | 35 | import org.thingsboard.server.common.data.id.TenantId; |
38 | -import org.thingsboard.server.common.data.kv.AttributeKvEntry; | |
39 | -import org.thingsboard.server.common.data.kv.BaseAttributeKvEntry; | |
40 | -import org.thingsboard.server.common.data.kv.BooleanDataEntry; | |
41 | -import org.thingsboard.server.common.data.kv.DoubleDataEntry; | |
42 | -import org.thingsboard.server.common.data.kv.LongDataEntry; | |
43 | -import org.thingsboard.server.common.data.kv.StringDataEntry; | |
44 | -import org.thingsboard.server.common.data.kv.TsKvEntry; | |
45 | 36 | import org.thingsboard.server.common.data.page.PageData; |
46 | 37 | import org.thingsboard.server.common.data.query.AlarmData; |
47 | -import org.thingsboard.server.common.data.query.AlarmDataPageLink; | |
48 | 38 | import org.thingsboard.server.common.data.query.AlarmDataQuery; |
49 | 39 | import org.thingsboard.server.common.msg.queue.ServiceType; |
50 | 40 | import org.thingsboard.server.common.msg.queue.TbCallback; |
51 | 41 | import org.thingsboard.server.common.msg.queue.TopicPartitionInfo; |
52 | 42 | import org.thingsboard.server.dao.alarm.AlarmOperationResult; |
53 | 43 | import org.thingsboard.server.dao.alarm.AlarmService; |
54 | -import org.thingsboard.server.dao.attributes.AttributesService; | |
55 | -import org.thingsboard.server.dao.timeseries.TimeseriesService; | |
56 | 44 | import org.thingsboard.server.gen.transport.TransportProtos; |
57 | -import org.thingsboard.server.queue.discovery.PartitionChangeEvent; | |
58 | 45 | import org.thingsboard.server.queue.discovery.PartitionService; |
59 | 46 | import org.thingsboard.server.service.queue.TbClusterService; |
60 | 47 | import org.thingsboard.server.service.subscription.SubscriptionManagerService; |
61 | 48 | import org.thingsboard.server.service.subscription.TbSubscriptionUtils; |
62 | -import org.thingsboard.server.service.telemetry.sub.AlarmSubscriptionUpdate; | |
63 | 49 | |
64 | -import javax.annotation.PostConstruct; | |
65 | -import javax.annotation.PreDestroy; | |
66 | 50 | import java.util.Collection; |
67 | -import java.util.Collections; | |
68 | -import java.util.List; | |
69 | 51 | import java.util.Optional; |
70 | -import java.util.Set; | |
71 | -import java.util.concurrent.ConcurrentHashMap; | |
72 | -import java.util.concurrent.ExecutorService; | |
73 | -import java.util.concurrent.Executors; | |
74 | -import java.util.function.Consumer; | |
75 | 52 | |
76 | 53 | /** |
77 | 54 | * Created by ashvayka on 27.03.18. |
... | ... | @@ -124,9 +101,15 @@ public class DefaultAlarmSubscriptionService extends AbstractSubscriptionService |
124 | 101 | |
125 | 102 | @Override |
126 | 103 | public ListenableFuture<Boolean> clearAlarm(TenantId tenantId, AlarmId alarmId, JsonNode details, long clearTs) { |
104 | + ListenableFuture<AlarmOperationResult> result = clearAlarmForResult(tenantId, alarmId, details, clearTs); | |
105 | + return Futures.transform(result, AlarmOperationResult::isSuccessful, wsCallBackExecutor); | |
106 | + } | |
107 | + | |
108 | + @Override | |
109 | + public ListenableFuture<AlarmOperationResult> clearAlarmForResult(TenantId tenantId, AlarmId alarmId, JsonNode details, long clearTs) { | |
127 | 110 | ListenableFuture<AlarmOperationResult> result = alarmService.clearAlarm(tenantId, alarmId, details, clearTs); |
128 | 111 | Futures.addCallback(result, new AlarmUpdateCallback(), wsCallBackExecutor); |
129 | - return Futures.transform(result, AlarmOperationResult::isSuccessful, wsCallBackExecutor); | |
112 | + return result; | |
130 | 113 | } |
131 | 114 | |
132 | 115 | @Override | ... | ... |
... | ... | @@ -110,7 +110,7 @@ security: |
110 | 110 | # Enable/disable claiming devices, if false -> the device's [claimingAllowed] SERVER_SCOPE attribute must be set to [true] to allow claiming specific device |
111 | 111 | allowClaimingByDefault: "${SECURITY_CLAIM_ALLOW_CLAIMING_BY_DEFAULT:true}" |
112 | 112 | # Time allowed to claim the device in milliseconds |
113 | - duration: "${SECURITY_CLAIM_DURATION:60000}" # 1 minute, note this value must equal claimDevices.timeToLiveInMinutes value | |
113 | + duration: "${SECURITY_CLAIM_DURATION:86400000}" # 1 minute, note this value must equal claimDevices.timeToLiveInMinutes value | |
114 | 114 | basic: |
115 | 115 | enabled: "${SECURITY_BASIC_ENABLED:false}" |
116 | 116 | oauth2: |
... | ... | @@ -348,8 +348,8 @@ caffeine: |
348 | 348 | timeToLiveInMinutes: 1440 |
349 | 349 | maxSize: 0 |
350 | 350 | claimDevices: |
351 | - timeToLiveInMinutes: 1 | |
352 | - maxSize: 0 | |
351 | + timeToLiveInMinutes: 1440 | |
352 | + maxSize: 1000 | |
353 | 353 | securitySettings: |
354 | 354 | timeToLiveInMinutes: 1440 |
355 | 355 | maxSize: 0 |
... | ... | @@ -625,11 +625,11 @@ queue: |
625 | 625 | security.protocol: "${TB_QUEUE_KAFKA_CONFLUENT_SECURITY_PROTOCOL:SASL_SSL}" |
626 | 626 | other: |
627 | 627 | topic-properties: |
628 | - rule-engine: "${TB_QUEUE_KAFKA_RE_TOPIC_PROPERTIES:retention.ms:604800000;segment.bytes:26214400;retention.bytes:1048576000;partitions:1}" | |
629 | - core: "${TB_QUEUE_KAFKA_CORE_TOPIC_PROPERTIES:retention.ms:604800000;segment.bytes:26214400;retention.bytes:1048576000;partitions:1}" | |
630 | - transport-api: "${TB_QUEUE_KAFKA_TA_TOPIC_PROPERTIES:retention.ms:604800000;segment.bytes:26214400;retention.bytes:1048576000;partitions:1}" | |
631 | - notifications: "${TB_QUEUE_KAFKA_NOTIFICATIONS_TOPIC_PROPERTIES:retention.ms:604800000;segment.bytes:26214400;retention.bytes:1048576000;partitions:1}" | |
632 | - js-executor: "${TB_QUEUE_KAFKA_JE_TOPIC_PROPERTIES:retention.ms:604800000;segment.bytes:26214400;retention.bytes:104857600;partitions:100}" | |
628 | + rule-engine: "${TB_QUEUE_KAFKA_RE_TOPIC_PROPERTIES:retention.ms:604800000;segment.bytes:26214400;retention.bytes:1048576000;partitions:1;min.insync.replicas:1}" | |
629 | + core: "${TB_QUEUE_KAFKA_CORE_TOPIC_PROPERTIES:retention.ms:604800000;segment.bytes:26214400;retention.bytes:1048576000;partitions:1;min.insync.replicas:1}" | |
630 | + transport-api: "${TB_QUEUE_KAFKA_TA_TOPIC_PROPERTIES:retention.ms:604800000;segment.bytes:26214400;retention.bytes:1048576000;partitions:1;min.insync.replicas:1}" | |
631 | + notifications: "${TB_QUEUE_KAFKA_NOTIFICATIONS_TOPIC_PROPERTIES:retention.ms:604800000;segment.bytes:26214400;retention.bytes:1048576000;partitions:1;min.insync.replicas:1}" | |
632 | + js-executor: "${TB_QUEUE_KAFKA_JE_TOPIC_PROPERTIES:retention.ms:604800000;segment.bytes:26214400;retention.bytes:104857600;partitions:100;min.insync.replicas:1}" | |
633 | 633 | consumer-stats: |
634 | 634 | enabled: "${TB_QUEUE_KAFKA_CONSUMER_STATS_ENABLED:true}" |
635 | 635 | print-interval-ms: "${TB_QUEUE_KAFKA_CONSUMER_STATS_MIN_PRINT_INTERVAL_MS:60000}" | ... | ... |
application/src/test/java/org/thingsboard/server/transport/AbstractTransportIntegrationTest.java
renamed from
application/src/test/java/org/thingsboard/server/mqtt/AbstractMqttIntegrationTest.java
... | ... | @@ -13,7 +13,7 @@ |
13 | 13 | * See the License for the specific language governing permissions and |
14 | 14 | * limitations under the License. |
15 | 15 | */ |
16 | -package org.thingsboard.server.mqtt; | |
16 | +package org.thingsboard.server.transport; | |
17 | 17 | |
18 | 18 | import com.fasterxml.jackson.databind.node.ObjectNode; |
19 | 19 | import lombok.extern.slf4j.Slf4j; |
... | ... | @@ -56,13 +56,14 @@ import static org.junit.Assert.assertNotNull; |
56 | 56 | import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; |
57 | 57 | |
58 | 58 | @Slf4j |
59 | -public abstract class AbstractMqttIntegrationTest extends AbstractControllerTest { | |
59 | +public abstract class AbstractTransportIntegrationTest extends AbstractControllerTest { | |
60 | 60 | |
61 | 61 | protected static final String MQTT_URL = "tcp://localhost:1883"; |
62 | + protected static final String COAP_BASE_URL = "coap://localhost:5683/api/v1/"; | |
62 | 63 | |
63 | - private static final AtomicInteger atomicInteger = new AtomicInteger(2); | |
64 | + protected static final AtomicInteger atomicInteger = new AtomicInteger(2); | |
64 | 65 | |
65 | - public static final String DEVICE_TELEMETRY_PROTO_SCHEMA = "syntax =\"proto3\";\n" + | |
66 | + protected static final String DEVICE_TELEMETRY_PROTO_SCHEMA = "syntax =\"proto3\";\n" + | |
66 | 67 | "\n" + |
67 | 68 | "package test;\n" + |
68 | 69 | "\n" + |
... | ... | @@ -83,7 +84,7 @@ public abstract class AbstractMqttIntegrationTest extends AbstractControllerTest |
83 | 84 | " }\n" + |
84 | 85 | "}"; |
85 | 86 | |
86 | - public static final String DEVICE_ATTRIBUTES_PROTO_SCHEMA = "syntax =\"proto3\";\n" + | |
87 | + protected static final String DEVICE_ATTRIBUTES_PROTO_SCHEMA = "syntax =\"proto3\";\n" + | |
87 | 88 | "\n" + |
88 | 89 | "package test;\n" + |
89 | 90 | "\n" + |
... | ... | @@ -110,81 +111,8 @@ public abstract class AbstractMqttIntegrationTest extends AbstractControllerTest |
110 | 111 | protected Device savedDevice; |
111 | 112 | protected String accessToken; |
112 | 113 | |
113 | - protected Device savedGateway; | |
114 | - protected String gatewayAccessToken; | |
115 | - | |
116 | 114 | protected DeviceProfile deviceProfile; |
117 | 115 | |
118 | - protected void processBeforeTest (String deviceName, String gatewayName, TransportPayloadType payloadType, String telemetryTopic, String attributesTopic) throws Exception { | |
119 | - this.processBeforeTest(deviceName, gatewayName, payloadType, telemetryTopic, attributesTopic, null, null, DeviceProfileProvisionType.DISABLED, null, null); | |
120 | - } | |
121 | - | |
122 | - protected void processBeforeTest(String deviceName, | |
123 | - String gatewayName, | |
124 | - TransportPayloadType payloadType, | |
125 | - String telemetryTopic, | |
126 | - String attributesTopic, | |
127 | - String telemetryProtoSchema, | |
128 | - String attributesProtoSchema, | |
129 | - DeviceProfileProvisionType provisionType, | |
130 | - String provisionKey, String provisionSecret | |
131 | - ) throws Exception { | |
132 | - loginSysAdmin(); | |
133 | - | |
134 | - Tenant tenant = new Tenant(); | |
135 | - tenant.setTitle("My tenant"); | |
136 | - savedTenant = doPost("/api/tenant", tenant, Tenant.class); | |
137 | - Assert.assertNotNull(savedTenant); | |
138 | - | |
139 | - tenantAdmin = new User(); | |
140 | - tenantAdmin.setAuthority(Authority.TENANT_ADMIN); | |
141 | - tenantAdmin.setTenantId(savedTenant.getId()); | |
142 | - tenantAdmin.setEmail("tenant" + atomicInteger.getAndIncrement() + "@thingsboard.org"); | |
143 | - tenantAdmin.setFirstName("Joe"); | |
144 | - tenantAdmin.setLastName("Downs"); | |
145 | - | |
146 | - tenantAdmin = createUserAndLogin(tenantAdmin, "testPassword1"); | |
147 | - | |
148 | - Device device = new Device(); | |
149 | - device.setName(deviceName); | |
150 | - device.setType("default"); | |
151 | - | |
152 | - Device gateway = new Device(); | |
153 | - gateway.setName(gatewayName); | |
154 | - gateway.setType("default"); | |
155 | - ObjectNode additionalInfo = mapper.createObjectNode(); | |
156 | - additionalInfo.put("gateway", true); | |
157 | - gateway.setAdditionalInfo(additionalInfo); | |
158 | - | |
159 | - if (payloadType != null) { | |
160 | - DeviceProfile mqttDeviceProfile = createMqttDeviceProfile(payloadType, telemetryTopic, attributesTopic, telemetryProtoSchema, attributesProtoSchema, provisionType, provisionKey, provisionSecret); | |
161 | - deviceProfile = doPost("/api/deviceProfile", mqttDeviceProfile, DeviceProfile.class); | |
162 | - device.setType(deviceProfile.getName()); | |
163 | - device.setDeviceProfileId(deviceProfile.getId()); | |
164 | - gateway.setType(deviceProfile.getName()); | |
165 | - gateway.setDeviceProfileId(deviceProfile.getId()); | |
166 | - } | |
167 | - | |
168 | - savedDevice = doPost("/api/device", device, Device.class); | |
169 | - | |
170 | - DeviceCredentials deviceCredentials = | |
171 | - doGet("/api/device/" + savedDevice.getId().getId().toString() + "/credentials", DeviceCredentials.class); | |
172 | - | |
173 | - savedGateway = doPost("/api/device", gateway, Device.class); | |
174 | - | |
175 | - DeviceCredentials gatewayCredentials = | |
176 | - doGet("/api/device/" + savedGateway.getId().getId().toString() + "/credentials", DeviceCredentials.class); | |
177 | - | |
178 | - assertEquals(savedDevice.getId(), deviceCredentials.getDeviceId()); | |
179 | - accessToken = deviceCredentials.getCredentialsId(); | |
180 | - assertNotNull(accessToken); | |
181 | - | |
182 | - assertEquals(savedGateway.getId(), gatewayCredentials.getDeviceId()); | |
183 | - gatewayAccessToken = gatewayCredentials.getCredentialsId(); | |
184 | - assertNotNull(gatewayAccessToken); | |
185 | - | |
186 | - } | |
187 | - | |
188 | 116 | protected void processAfterTest() throws Exception { |
189 | 117 | loginSysAdmin(); |
190 | 118 | if (savedTenant != null) { |
... | ... | @@ -192,22 +120,6 @@ public abstract class AbstractMqttIntegrationTest extends AbstractControllerTest |
192 | 120 | } |
193 | 121 | } |
194 | 122 | |
195 | - protected MqttAsyncClient getMqttAsyncClient(String accessToken) throws MqttException { | |
196 | - String clientId = MqttAsyncClient.generateClientId(); | |
197 | - MqttAsyncClient client = new MqttAsyncClient(MQTT_URL, clientId, new MemoryPersistence()); | |
198 | - | |
199 | - MqttConnectOptions options = new MqttConnectOptions(); | |
200 | - options.setUserName(accessToken); | |
201 | - client.connect(options).waitForCompletion(); | |
202 | - return client; | |
203 | - } | |
204 | - | |
205 | - protected void publishMqttMsg(MqttAsyncClient client, byte[] payload, String topic) throws MqttException { | |
206 | - MqttMessage message = new MqttMessage(); | |
207 | - message.setPayload(payload); | |
208 | - client.publish(topic, message); | |
209 | - } | |
210 | - | |
211 | 123 | protected List<TransportProtos.KeyValueProto> getKvProtos(List<String> expectedKeys) { |
212 | 124 | List<TransportProtos.KeyValueProto> keyValueProtos = new ArrayList<>(); |
213 | 125 | TransportProtos.KeyValueProto strKeyValueProto = getKeyValueProto(expectedKeys.get(0), "value1", TransportProtos.KeyValueType.STRING_V); |
... | ... | @@ -247,72 +159,6 @@ public abstract class AbstractMqttIntegrationTest extends AbstractControllerTest |
247 | 159 | return keyValueProtoBuilder.build(); |
248 | 160 | } |
249 | 161 | |
250 | - protected DeviceProfile createMqttDeviceProfile(TransportPayloadType transportPayloadType, | |
251 | - String telemetryTopic, String attributesTopic, | |
252 | - String telemetryProtoSchema, String attributesProtoSchema, | |
253 | - DeviceProfileProvisionType provisionType, | |
254 | - String provisionKey, String provisionSecret) { | |
255 | - DeviceProfile deviceProfile = new DeviceProfile(); | |
256 | - deviceProfile.setName(transportPayloadType.name()); | |
257 | - deviceProfile.setType(DeviceProfileType.DEFAULT); | |
258 | - deviceProfile.setTransportType(DeviceTransportType.MQTT); | |
259 | - deviceProfile.setProvisionType(provisionType); | |
260 | - deviceProfile.setProvisionDeviceKey(provisionKey); | |
261 | - deviceProfile.setDescription(transportPayloadType.name() + " Test"); | |
262 | - DeviceProfileData deviceProfileData = new DeviceProfileData(); | |
263 | - DefaultDeviceProfileConfiguration configuration = new DefaultDeviceProfileConfiguration(); | |
264 | - MqttDeviceProfileTransportConfiguration mqttDeviceProfileTransportConfiguration = new MqttDeviceProfileTransportConfiguration(); | |
265 | - if (!StringUtils.isEmpty(telemetryTopic)) { | |
266 | - mqttDeviceProfileTransportConfiguration.setDeviceTelemetryTopic(telemetryTopic); | |
267 | - } | |
268 | - if (!StringUtils.isEmpty(attributesTopic)) { | |
269 | - mqttDeviceProfileTransportConfiguration.setDeviceAttributesTopic(attributesTopic); | |
270 | - } | |
271 | - TransportPayloadTypeConfiguration transportPayloadTypeConfiguration; | |
272 | - if (TransportPayloadType.JSON.equals(transportPayloadType)) { | |
273 | - transportPayloadTypeConfiguration = new JsonTransportPayloadConfiguration(); | |
274 | - } else { | |
275 | - ProtoTransportPayloadConfiguration protoTransportPayloadConfiguration = new ProtoTransportPayloadConfiguration(); | |
276 | - if (StringUtils.isEmpty(telemetryProtoSchema)) { | |
277 | - telemetryProtoSchema = DEVICE_TELEMETRY_PROTO_SCHEMA; | |
278 | - } | |
279 | - if (StringUtils.isEmpty(attributesProtoSchema)) { | |
280 | - attributesProtoSchema = DEVICE_ATTRIBUTES_PROTO_SCHEMA; | |
281 | - } | |
282 | - protoTransportPayloadConfiguration.setDeviceTelemetryProtoSchema(telemetryProtoSchema); | |
283 | - protoTransportPayloadConfiguration.setDeviceAttributesProtoSchema(attributesProtoSchema); | |
284 | - transportPayloadTypeConfiguration = protoTransportPayloadConfiguration; | |
285 | - } | |
286 | - mqttDeviceProfileTransportConfiguration.setTransportPayloadTypeConfiguration(transportPayloadTypeConfiguration); | |
287 | - deviceProfileData.setTransportConfiguration(mqttDeviceProfileTransportConfiguration); | |
288 | - DeviceProfileProvisionConfiguration provisionConfiguration; | |
289 | - switch (provisionType) { | |
290 | - case ALLOW_CREATE_NEW_DEVICES: | |
291 | - provisionConfiguration = new AllowCreateNewDevicesDeviceProfileProvisionConfiguration(provisionSecret); | |
292 | - break; | |
293 | - case CHECK_PRE_PROVISIONED_DEVICES: | |
294 | - provisionConfiguration = new CheckPreProvisionedDevicesDeviceProfileProvisionConfiguration(provisionSecret); | |
295 | - break; | |
296 | - case DISABLED: | |
297 | - default: | |
298 | - provisionConfiguration = new DisabledDeviceProfileProvisionConfiguration(provisionSecret); | |
299 | - break; | |
300 | - } | |
301 | - deviceProfileData.setProvisionConfiguration(provisionConfiguration); | |
302 | - deviceProfileData.setConfiguration(configuration); | |
303 | - deviceProfile.setProfileData(deviceProfileData); | |
304 | - deviceProfile.setDefault(false); | |
305 | - deviceProfile.setDefaultRuleChainId(null); | |
306 | - return deviceProfile; | |
307 | - } | |
308 | - | |
309 | - protected TransportProtos.PostAttributeMsg getPostAttributeMsg(List<String> expectedKeys) { | |
310 | - List<TransportProtos.KeyValueProto> kvProtos = getKvProtos(expectedKeys); | |
311 | - TransportProtos.PostAttributeMsg.Builder builder = TransportProtos.PostAttributeMsg.newBuilder(); | |
312 | - builder.addAllKv(kvProtos); | |
313 | - return builder.build(); | |
314 | - } | |
315 | - | |
316 | 162 | protected <T> T doExecuteWithRetriesAndInterval(SupplierWithThrowable<T> supplier, int retries, int intervalMs) throws Exception { |
317 | 163 | int count = 0; |
318 | 164 | T result = null; | ... | ... |
application/src/test/java/org/thingsboard/server/transport/TransportNoSqlTestSuite.java
renamed from
application/src/test/java/org/thingsboard/server/mqtt/MqttNoSqlTestSuite.java
... | ... | @@ -13,7 +13,7 @@ |
13 | 13 | * See the License for the specific language governing permissions and |
14 | 14 | * limitations under the License. |
15 | 15 | */ |
16 | -package org.thingsboard.server.mqtt; | |
16 | +package org.thingsboard.server.transport; | |
17 | 17 | |
18 | 18 | import org.cassandraunit.dataset.cql.ClassPathCQLDataSet; |
19 | 19 | import org.junit.BeforeClass; |
... | ... | @@ -28,8 +28,8 @@ import java.util.Arrays; |
28 | 28 | |
29 | 29 | @RunWith(ClasspathSuite.class) |
30 | 30 | @ClasspathSuite.ClassnameFilters({ |
31 | - "org.thingsboard.server.mqtt.*.nosql.*Test"}) | |
32 | -public class MqttNoSqlTestSuite { | |
31 | + "org.thingsboard.server.transport.*.telemetry.timeseries.nosql.*Test"}) | |
32 | +public class TransportNoSqlTestSuite { | |
33 | 33 | |
34 | 34 | @ClassRule |
35 | 35 | public static CustomSqlUnit sqlUnit = new CustomSqlUnit( | ... | ... |
application/src/test/java/org/thingsboard/server/transport/TransportSqlTestSuite.java
renamed from
application/src/test/java/org/thingsboard/server/mqtt/MqttSqlTestSuite.java
... | ... | @@ -13,7 +13,7 @@ |
13 | 13 | * See the License for the specific language governing permissions and |
14 | 14 | * limitations under the License. |
15 | 15 | */ |
16 | -package org.thingsboard.server.mqtt; | |
16 | +package org.thingsboard.server.transport; | |
17 | 17 | |
18 | 18 | import org.junit.BeforeClass; |
19 | 19 | import org.junit.ClassRule; |
... | ... | @@ -26,15 +26,15 @@ import java.util.Arrays; |
26 | 26 | |
27 | 27 | @RunWith(ClasspathSuite.class) |
28 | 28 | @ClasspathSuite.ClassnameFilters({ |
29 | - "org.thingsboard.server.mqtt.rpc.sql.*Test", | |
30 | - "org.thingsboard.server.mqtt.telemetry.timeseries.sql.*Test", | |
31 | - "org.thingsboard.server.mqtt.telemetry.attributes.sql.*Test", | |
32 | - "org.thingsboard.server.mqtt.attributes.updates.sql.*Test", | |
33 | - "org.thingsboard.server.mqtt.attributes.request.sql.*Test", | |
34 | - "org.thingsboard.server.mqtt.claim.sql.*Test", | |
35 | - "org.thingsboard.server.mqtt.provision.sql.*Test" | |
29 | + "org.thingsboard.server.transport.*.rpc.sql.*Test", | |
30 | + "org.thingsboard.server.transport.*.telemetry.timeseries.sql.*Test", | |
31 | + "org.thingsboard.server.transport.*.telemetry.attributes.sql.*Test", | |
32 | + "org.thingsboard.server.transport.*.attributes.updates.sql.*Test", | |
33 | + "org.thingsboard.server.transport.*.attributes.request.sql.*Test", | |
34 | + "org.thingsboard.server.transport.*.claim.sql.*Test", | |
35 | + "org.thingsboard.server.transport.*.provision.sql.*Test" | |
36 | 36 | }) |
37 | -public class MqttSqlTestSuite { | |
37 | +public class TransportSqlTestSuite { | |
38 | 38 | |
39 | 39 | @ClassRule |
40 | 40 | public static CustomSqlUnit sqlUnit = new CustomSqlUnit( | ... | ... |
application/src/test/java/org/thingsboard/server/transport/coap/AbstractCoapIntegrationTest.java
0 → 100644
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.eclipse.californium.core.CoapClient; | |
20 | +import org.junit.Assert; | |
21 | +import org.springframework.util.StringUtils; | |
22 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
23 | +import org.thingsboard.server.common.data.Device; | |
24 | +import org.thingsboard.server.common.data.DeviceProfile; | |
25 | +import org.thingsboard.server.common.data.DeviceProfileProvisionType; | |
26 | +import org.thingsboard.server.common.data.DeviceProfileType; | |
27 | +import org.thingsboard.server.common.data.DeviceTransportType; | |
28 | +import org.thingsboard.server.common.data.Tenant; | |
29 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
30 | +import org.thingsboard.server.common.data.User; | |
31 | +import org.thingsboard.server.common.data.device.profile.AllowCreateNewDevicesDeviceProfileProvisionConfiguration; | |
32 | +import org.thingsboard.server.common.data.device.profile.CheckPreProvisionedDevicesDeviceProfileProvisionConfiguration; | |
33 | +import org.thingsboard.server.common.data.device.profile.CoapDeviceProfileTransportConfiguration; | |
34 | +import org.thingsboard.server.common.data.device.profile.CoapDeviceTypeConfiguration; | |
35 | +import org.thingsboard.server.common.data.device.profile.DefaultCoapDeviceTypeConfiguration; | |
36 | +import org.thingsboard.server.common.data.device.profile.DefaultDeviceProfileConfiguration; | |
37 | +import org.thingsboard.server.common.data.device.profile.DeviceProfileData; | |
38 | +import org.thingsboard.server.common.data.device.profile.DeviceProfileProvisionConfiguration; | |
39 | +import org.thingsboard.server.common.data.device.profile.DisabledDeviceProfileProvisionConfiguration; | |
40 | +import org.thingsboard.server.common.data.device.profile.EfentoCoapDeviceTypeConfiguration; | |
41 | +import org.thingsboard.server.common.data.device.profile.JsonTransportPayloadConfiguration; | |
42 | +import org.thingsboard.server.common.data.device.profile.ProtoTransportPayloadConfiguration; | |
43 | +import org.thingsboard.server.common.data.device.profile.TransportPayloadTypeConfiguration; | |
44 | +import org.thingsboard.server.common.data.security.Authority; | |
45 | +import org.thingsboard.server.common.data.security.DeviceCredentials; | |
46 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
47 | +import org.thingsboard.server.transport.AbstractTransportIntegrationTest; | |
48 | + | |
49 | +import static org.junit.Assert.assertEquals; | |
50 | +import static org.junit.Assert.assertNotNull; | |
51 | + | |
52 | +@Slf4j | |
53 | +public abstract class AbstractCoapIntegrationTest extends AbstractTransportIntegrationTest { | |
54 | + | |
55 | + protected void processBeforeTest(String deviceName, CoapDeviceType coapDeviceType, TransportPayloadType payloadType) throws Exception { | |
56 | + this.processBeforeTest(deviceName, coapDeviceType, payloadType, null, null, DeviceProfileProvisionType.DISABLED, null, null); | |
57 | + } | |
58 | + | |
59 | + protected void processBeforeTest(String deviceName, | |
60 | + CoapDeviceType coapDeviceType, | |
61 | + TransportPayloadType payloadType, | |
62 | + String telemetryProtoSchema, | |
63 | + String attributesProtoSchema, | |
64 | + DeviceProfileProvisionType provisionType, | |
65 | + String provisionKey, String provisionSecret | |
66 | + ) throws Exception { | |
67 | + loginSysAdmin(); | |
68 | + | |
69 | + Tenant tenant = new Tenant(); | |
70 | + tenant.setTitle("My tenant"); | |
71 | + savedTenant = doPost("/api/tenant", tenant, Tenant.class); | |
72 | + Assert.assertNotNull(savedTenant); | |
73 | + | |
74 | + tenantAdmin = new User(); | |
75 | + tenantAdmin.setAuthority(Authority.TENANT_ADMIN); | |
76 | + tenantAdmin.setTenantId(savedTenant.getId()); | |
77 | + tenantAdmin.setEmail("tenant" + atomicInteger.getAndIncrement() + "@thingsboard.org"); | |
78 | + tenantAdmin.setFirstName("Joe"); | |
79 | + tenantAdmin.setLastName("Downs"); | |
80 | + | |
81 | + tenantAdmin = createUserAndLogin(tenantAdmin, "testPassword1"); | |
82 | + | |
83 | + Device device = new Device(); | |
84 | + device.setName(deviceName); | |
85 | + device.setType("default"); | |
86 | + | |
87 | + if (coapDeviceType != null) { | |
88 | + DeviceProfile coapDeviceProfile = createCoapDeviceProfile(payloadType, coapDeviceType, attributesProtoSchema, provisionType, provisionKey, provisionSecret, telemetryProtoSchema); | |
89 | + deviceProfile = doPost("/api/deviceProfile", coapDeviceProfile, DeviceProfile.class); | |
90 | + device.setType(deviceProfile.getName()); | |
91 | + device.setDeviceProfileId(deviceProfile.getId()); | |
92 | + } | |
93 | + | |
94 | + savedDevice = doPost("/api/device", device, Device.class); | |
95 | + | |
96 | + DeviceCredentials deviceCredentials = | |
97 | + doGet("/api/device/" + savedDevice.getId().getId().toString() + "/credentials", DeviceCredentials.class); | |
98 | + | |
99 | + assertEquals(savedDevice.getId(), deviceCredentials.getDeviceId()); | |
100 | + accessToken = deviceCredentials.getCredentialsId(); | |
101 | + assertNotNull(accessToken); | |
102 | + | |
103 | + } | |
104 | + | |
105 | + protected DeviceProfile createCoapDeviceProfile(TransportPayloadType transportPayloadType, CoapDeviceType coapDeviceType, | |
106 | + String attributesProtoSchema, DeviceProfileProvisionType provisionType, | |
107 | + String provisionKey, String provisionSecret, String telemetryProtoSchema) { | |
108 | + DeviceProfile deviceProfile = new DeviceProfile(); | |
109 | + deviceProfile.setName(transportPayloadType.name()); | |
110 | + deviceProfile.setType(DeviceProfileType.DEFAULT); | |
111 | + deviceProfile.setProvisionType(provisionType); | |
112 | + deviceProfile.setProvisionDeviceKey(provisionKey); | |
113 | + deviceProfile.setDescription(transportPayloadType.name() + " Test"); | |
114 | + DeviceProfileData deviceProfileData = new DeviceProfileData(); | |
115 | + DefaultDeviceProfileConfiguration configuration = new DefaultDeviceProfileConfiguration(); | |
116 | + deviceProfile.setTransportType(DeviceTransportType.COAP); | |
117 | + CoapDeviceProfileTransportConfiguration coapDeviceProfileTransportConfiguration = new CoapDeviceProfileTransportConfiguration(); | |
118 | + CoapDeviceTypeConfiguration coapDeviceTypeConfiguration; | |
119 | + if (CoapDeviceType.DEFAULT.equals(coapDeviceType)) { | |
120 | + DefaultCoapDeviceTypeConfiguration defaultCoapDeviceTypeConfiguration = new DefaultCoapDeviceTypeConfiguration(); | |
121 | + TransportPayloadTypeConfiguration transportPayloadTypeConfiguration; | |
122 | + if (TransportPayloadType.PROTOBUF.equals(transportPayloadType)) { | |
123 | + ProtoTransportPayloadConfiguration protoTransportPayloadConfiguration = new ProtoTransportPayloadConfiguration(); | |
124 | + if (StringUtils.isEmpty(telemetryProtoSchema)) { | |
125 | + telemetryProtoSchema = DEVICE_TELEMETRY_PROTO_SCHEMA; | |
126 | + } | |
127 | + if (StringUtils.isEmpty(attributesProtoSchema)) { | |
128 | + attributesProtoSchema = DEVICE_ATTRIBUTES_PROTO_SCHEMA; | |
129 | + } | |
130 | + protoTransportPayloadConfiguration.setDeviceTelemetryProtoSchema(telemetryProtoSchema); | |
131 | + protoTransportPayloadConfiguration.setDeviceAttributesProtoSchema(attributesProtoSchema); | |
132 | + transportPayloadTypeConfiguration = protoTransportPayloadConfiguration; | |
133 | + } else { | |
134 | + transportPayloadTypeConfiguration = new JsonTransportPayloadConfiguration(); | |
135 | + } | |
136 | + defaultCoapDeviceTypeConfiguration.setTransportPayloadTypeConfiguration(transportPayloadTypeConfiguration); | |
137 | + coapDeviceTypeConfiguration = defaultCoapDeviceTypeConfiguration; | |
138 | + } else { | |
139 | + coapDeviceTypeConfiguration = new EfentoCoapDeviceTypeConfiguration(); | |
140 | + } | |
141 | + coapDeviceProfileTransportConfiguration.setCoapDeviceTypeConfiguration(coapDeviceTypeConfiguration); | |
142 | + deviceProfileData.setTransportConfiguration(coapDeviceProfileTransportConfiguration); | |
143 | + DeviceProfileProvisionConfiguration provisionConfiguration; | |
144 | + switch (provisionType) { | |
145 | + case ALLOW_CREATE_NEW_DEVICES: | |
146 | + provisionConfiguration = new AllowCreateNewDevicesDeviceProfileProvisionConfiguration(provisionSecret); | |
147 | + break; | |
148 | + case CHECK_PRE_PROVISIONED_DEVICES: | |
149 | + provisionConfiguration = new CheckPreProvisionedDevicesDeviceProfileProvisionConfiguration(provisionSecret); | |
150 | + break; | |
151 | + case DISABLED: | |
152 | + default: | |
153 | + provisionConfiguration = new DisabledDeviceProfileProvisionConfiguration(provisionSecret); | |
154 | + break; | |
155 | + } | |
156 | + deviceProfileData.setProvisionConfiguration(provisionConfiguration); | |
157 | + deviceProfileData.setConfiguration(configuration); | |
158 | + deviceProfile.setProfileData(deviceProfileData); | |
159 | + deviceProfile.setDefault(false); | |
160 | + deviceProfile.setDefaultRuleChainId(null); | |
161 | + return deviceProfile; | |
162 | + } | |
163 | + | |
164 | + protected CoapClient getCoapClient(FeatureType featureType) { | |
165 | + return new CoapClient(getFeatureTokenUrl(accessToken, featureType)); | |
166 | + } | |
167 | + | |
168 | + protected CoapClient getCoapClient(String featureTokenUrl) { | |
169 | + return new CoapClient(featureTokenUrl); | |
170 | + } | |
171 | + | |
172 | + protected String getFeatureTokenUrl(String token, FeatureType featureType) { | |
173 | + return COAP_BASE_URL + token + "/" + featureType.name().toLowerCase(); | |
174 | + } | |
175 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.thingsboard.server.transport.coap.AbstractCoapIntegrationTest; | |
20 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
21 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
22 | +import org.thingsboard.server.gen.transport.TransportProtos; | |
23 | + | |
24 | +import java.util.ArrayList; | |
25 | +import java.util.List; | |
26 | + | |
27 | +@Slf4j | |
28 | +public abstract class AbstractCoapAttributesIntegrationTest extends AbstractCoapIntegrationTest { | |
29 | + | |
30 | + protected static final String POST_ATTRIBUTES_PAYLOAD = "{\"attribute1\":\"value1\",\"attribute2\":true,\"attribute3\":42.0,\"attribute4\":73," + | |
31 | + "\"attribute5\":{\"someNumber\":42,\"someArray\":[1,2,3],\"someNestedObject\":{\"key\":\"value\"}}}"; | |
32 | + | |
33 | + protected void processBeforeTest(String deviceName, CoapDeviceType coapDeviceType, TransportPayloadType payloadType) throws Exception { | |
34 | + super.processBeforeTest(deviceName, coapDeviceType, payloadType); | |
35 | + } | |
36 | + | |
37 | + protected void processAfterTest() throws Exception { | |
38 | + super.processAfterTest(); | |
39 | + } | |
40 | + | |
41 | + protected List<TransportProtos.TsKvProto> getTsKvProtoList() { | |
42 | + TransportProtos.TsKvProto tsKvProtoAttribute1 = getTsKvProto("attribute1", "value1", TransportProtos.KeyValueType.STRING_V); | |
43 | + TransportProtos.TsKvProto tsKvProtoAttribute2 = getTsKvProto("attribute2", "true", TransportProtos.KeyValueType.BOOLEAN_V); | |
44 | + TransportProtos.TsKvProto tsKvProtoAttribute3 = getTsKvProto("attribute3", "42.0", TransportProtos.KeyValueType.DOUBLE_V); | |
45 | + TransportProtos.TsKvProto tsKvProtoAttribute4 = getTsKvProto("attribute4", "73", TransportProtos.KeyValueType.LONG_V); | |
46 | + TransportProtos.TsKvProto tsKvProtoAttribute5 = getTsKvProto("attribute5", "{\"someNumber\":42,\"someArray\":[1,2,3],\"someNestedObject\":{\"key\":\"value\"}}", TransportProtos.KeyValueType.JSON_V); | |
47 | + List<TransportProtos.TsKvProto> tsKvProtoList = new ArrayList<>(); | |
48 | + tsKvProtoList.add(tsKvProtoAttribute1); | |
49 | + tsKvProtoList.add(tsKvProtoAttribute2); | |
50 | + tsKvProtoList.add(tsKvProtoAttribute3); | |
51 | + tsKvProtoList.add(tsKvProtoAttribute4); | |
52 | + tsKvProtoList.add(tsKvProtoAttribute5); | |
53 | + return tsKvProtoList; | |
54 | + } | |
55 | + | |
56 | + protected TransportProtos.TsKvProto getTsKvProto(String key, String value, TransportProtos.KeyValueType keyValueType) { | |
57 | + TransportProtos.TsKvProto.Builder tsKvProtoBuilder = TransportProtos.TsKvProto.newBuilder(); | |
58 | + TransportProtos.KeyValueProto keyValueProto = getKeyValueProto(key, value, keyValueType); | |
59 | + tsKvProtoBuilder.setKv(keyValueProto); | |
60 | + return tsKvProtoBuilder.build(); | |
61 | + } | |
62 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.request; | |
17 | + | |
18 | +import com.fasterxml.jackson.core.type.TypeReference; | |
19 | +import com.google.protobuf.InvalidProtocolBufferException; | |
20 | +import lombok.extern.slf4j.Slf4j; | |
21 | +import org.eclipse.californium.core.CoapClient; | |
22 | +import org.eclipse.californium.core.CoapResponse; | |
23 | +import org.eclipse.californium.core.coap.CoAP; | |
24 | +import org.eclipse.californium.core.coap.MediaTypeRegistry; | |
25 | +import org.junit.After; | |
26 | +import org.junit.Before; | |
27 | +import org.junit.Test; | |
28 | +import org.thingsboard.server.transport.coap.attributes.AbstractCoapAttributesIntegrationTest; | |
29 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
30 | +import org.thingsboard.server.dao.util.mapping.JacksonUtil; | |
31 | + | |
32 | +import java.nio.charset.StandardCharsets; | |
33 | +import java.util.List; | |
34 | + | |
35 | +import static org.junit.Assert.assertEquals; | |
36 | +import static org.junit.Assert.assertNotNull; | |
37 | +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; | |
38 | + | |
39 | +@Slf4j | |
40 | +public abstract class AbstractCoapAttributesRequestIntegrationTest extends AbstractCoapAttributesIntegrationTest { | |
41 | + | |
42 | + protected static final long CLIENT_REQUEST_TIMEOUT = 60000L; | |
43 | + | |
44 | + @Before | |
45 | + public void beforeTest() throws Exception { | |
46 | + processBeforeTest("Test Request attribute values from the server", null, null); | |
47 | + } | |
48 | + | |
49 | + @After | |
50 | + public void afterTest() throws Exception { | |
51 | + processAfterTest(); | |
52 | + } | |
53 | + | |
54 | + @Test | |
55 | + public void testRequestAttributesValuesFromTheServer() throws Exception { | |
56 | + processTestRequestAttributesValuesFromTheServer(); | |
57 | + } | |
58 | + | |
59 | + protected void processTestRequestAttributesValuesFromTheServer() throws Exception { | |
60 | + postAttributes(); | |
61 | + | |
62 | + long start = System.currentTimeMillis(); | |
63 | + long end = System.currentTimeMillis() + 5000; | |
64 | + | |
65 | + List<String> savedAttributeKeys = null; | |
66 | + while (start <= end) { | |
67 | + savedAttributeKeys = doGetAsyncTyped("/api/plugins/telemetry/DEVICE/" + savedDevice.getId().getId() + "/keys/attributes/CLIENT_SCOPE", new TypeReference<>() {}); | |
68 | + if (savedAttributeKeys.size() == 5) { | |
69 | + break; | |
70 | + } | |
71 | + Thread.sleep(100); | |
72 | + start += 100; | |
73 | + } | |
74 | + assertNotNull(savedAttributeKeys); | |
75 | + | |
76 | + String keys = "attribute1,attribute2,attribute3,attribute4,attribute5"; | |
77 | + String featureTokenUrl = getFeatureTokenUrl(accessToken, FeatureType.ATTRIBUTES) + "?clientKeys=" + keys + "&sharedKeys=" + keys; | |
78 | + CoapClient client = getCoapClient(featureTokenUrl); | |
79 | + | |
80 | + CoapResponse getAttributesResponse = client.setTimeout(CLIENT_REQUEST_TIMEOUT).get(); | |
81 | + validateResponse(getAttributesResponse); | |
82 | + } | |
83 | + | |
84 | + protected void postAttributes() throws Exception { | |
85 | + doPostAsync("/api/plugins/telemetry/DEVICE/" + savedDevice.getId().getId() + "/attributes/SHARED_SCOPE", POST_ATTRIBUTES_PAYLOAD, String.class, status().isOk()); | |
86 | + CoapClient client = getCoapClient(FeatureType.ATTRIBUTES); | |
87 | + CoapResponse coapResponse = client.setTimeout(CLIENT_REQUEST_TIMEOUT).post(POST_ATTRIBUTES_PAYLOAD.getBytes(), MediaTypeRegistry.APPLICATION_JSON); | |
88 | + assertEquals(CoAP.ResponseCode.CREATED, coapResponse.getCode()); | |
89 | + } | |
90 | + | |
91 | + protected void validateResponse(CoapResponse getAttributesResponse) throws InvalidProtocolBufferException { | |
92 | + assertEquals(CoAP.ResponseCode.CONTENT, getAttributesResponse.getCode()); | |
93 | + String expectedRequestPayload = "{\"client\":{\"attribute1\":\"value1\",\"attribute2\":true,\"attribute3\":42.0,\"attribute4\":73,\"attribute5\":{\"someNumber\":42,\"someArray\":[1,2,3],\"someNestedObject\":{\"key\":\"value\"}}},\"shared\":{\"attribute1\":\"value1\",\"attribute2\":true,\"attribute3\":42.0,\"attribute4\":73,\"attribute5\":{\"someNumber\":42,\"someArray\":[1,2,3],\"someNestedObject\":{\"key\":\"value\"}}}}"; | |
94 | + assertEquals(JacksonUtil.toJsonNode(expectedRequestPayload), JacksonUtil.toJsonNode(new String(getAttributesResponse.getPayload(), StandardCharsets.UTF_8))); | |
95 | + } | |
96 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.request; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.junit.After; | |
20 | +import org.junit.Before; | |
21 | +import org.junit.Test; | |
22 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
23 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
24 | + | |
25 | +@Slf4j | |
26 | +public abstract class AbstractCoapAttributesRequestJsonIntegrationTest extends AbstractCoapAttributesRequestIntegrationTest { | |
27 | + | |
28 | + @Before | |
29 | + public void beforeTest() throws Exception { | |
30 | + processBeforeTest("Test Request attribute values from the server json", CoapDeviceType.DEFAULT, TransportPayloadType.JSON); | |
31 | + } | |
32 | + | |
33 | + @After | |
34 | + public void afterTest() throws Exception { | |
35 | + processAfterTest(); | |
36 | + } | |
37 | + | |
38 | + @Test | |
39 | + public void testRequestAttributesValuesFromTheServer() throws Exception { | |
40 | + super.testRequestAttributesValuesFromTheServer(); | |
41 | + } | |
42 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.request; | |
17 | + | |
18 | +import com.github.os72.protobuf.dynamic.DynamicSchema; | |
19 | +import com.google.protobuf.Descriptors; | |
20 | +import com.google.protobuf.DynamicMessage; | |
21 | +import com.google.protobuf.InvalidProtocolBufferException; | |
22 | +import com.squareup.wire.schema.internal.parser.ProtoFileElement; | |
23 | +import lombok.extern.slf4j.Slf4j; | |
24 | +import org.eclipse.californium.core.CoapClient; | |
25 | +import org.eclipse.californium.core.CoapResponse; | |
26 | +import org.eclipse.californium.core.coap.CoAP; | |
27 | +import org.eclipse.californium.core.coap.MediaTypeRegistry; | |
28 | +import org.junit.After; | |
29 | +import org.junit.Test; | |
30 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
31 | +import org.thingsboard.server.common.data.DeviceProfileProvisionType; | |
32 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
33 | +import org.thingsboard.server.common.data.device.profile.CoapDeviceProfileTransportConfiguration; | |
34 | +import org.thingsboard.server.common.data.device.profile.CoapDeviceTypeConfiguration; | |
35 | +import org.thingsboard.server.common.data.device.profile.DefaultCoapDeviceTypeConfiguration; | |
36 | +import org.thingsboard.server.common.data.device.profile.DeviceProfileTransportConfiguration; | |
37 | +import org.thingsboard.server.common.data.device.profile.ProtoTransportPayloadConfiguration; | |
38 | +import org.thingsboard.server.common.data.device.profile.TransportPayloadTypeConfiguration; | |
39 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
40 | +import org.thingsboard.server.gen.transport.TransportProtos; | |
41 | + | |
42 | +import java.util.List; | |
43 | +import java.util.stream.Collectors; | |
44 | + | |
45 | +import static org.junit.Assert.assertEquals; | |
46 | +import static org.junit.Assert.assertNotNull; | |
47 | +import static org.junit.Assert.assertTrue; | |
48 | +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; | |
49 | + | |
50 | +@Slf4j | |
51 | +public abstract class AbstractCoapAttributesRequestProtoIntegrationTest extends AbstractCoapAttributesRequestIntegrationTest { | |
52 | + | |
53 | + public static final String ATTRIBUTES_SCHEMA_STR = "syntax =\"proto3\";\n" + | |
54 | + "\n" + | |
55 | + "package test;\n" + | |
56 | + "\n" + | |
57 | + "message PostAttributes {\n" + | |
58 | + " string attribute1 = 1;\n" + | |
59 | + " bool attribute2 = 2;\n" + | |
60 | + " double attribute3 = 3;\n" + | |
61 | + " int32 attribute4 = 4;\n" + | |
62 | + " JsonObject attribute5 = 5;\n" + | |
63 | + "\n" + | |
64 | + " message JsonObject {\n" + | |
65 | + " int32 someNumber = 6;\n" + | |
66 | + " repeated int32 someArray = 7;\n" + | |
67 | + " NestedJsonObject someNestedObject = 8;\n" + | |
68 | + " message NestedJsonObject {\n" + | |
69 | + " string key = 9;\n" + | |
70 | + " }\n" + | |
71 | + " }\n" + | |
72 | + "}"; | |
73 | + | |
74 | + @After | |
75 | + public void afterTest() throws Exception { | |
76 | + processAfterTest(); | |
77 | + } | |
78 | + | |
79 | + @Test | |
80 | + public void testRequestAttributesValuesFromTheServer() throws Exception { | |
81 | + super.processBeforeTest("Test Request attribute values from the server proto", CoapDeviceType.DEFAULT, | |
82 | + TransportPayloadType.PROTOBUF, null, ATTRIBUTES_SCHEMA_STR, DeviceProfileProvisionType.DISABLED, null, null); | |
83 | + processTestRequestAttributesValuesFromTheServer(); | |
84 | + } | |
85 | + | |
86 | + protected void postAttributes() throws Exception { | |
87 | + doPostAsync("/api/plugins/telemetry/DEVICE/" + savedDevice.getId().getId() + "/attributes/SHARED_SCOPE", POST_ATTRIBUTES_PAYLOAD, String.class, status().isOk()); | |
88 | + DeviceProfileTransportConfiguration transportConfiguration = deviceProfile.getProfileData().getTransportConfiguration(); | |
89 | + assertTrue(transportConfiguration instanceof CoapDeviceProfileTransportConfiguration); | |
90 | + CoapDeviceProfileTransportConfiguration coapTransportConfiguration = (CoapDeviceProfileTransportConfiguration) transportConfiguration; | |
91 | + CoapDeviceTypeConfiguration coapDeviceTypeConfiguration = coapTransportConfiguration.getCoapDeviceTypeConfiguration(); | |
92 | + assertTrue(coapDeviceTypeConfiguration instanceof DefaultCoapDeviceTypeConfiguration); | |
93 | + DefaultCoapDeviceTypeConfiguration defaultCoapDeviceTypeConfiguration = (DefaultCoapDeviceTypeConfiguration) coapDeviceTypeConfiguration; | |
94 | + TransportPayloadTypeConfiguration transportPayloadTypeConfiguration = defaultCoapDeviceTypeConfiguration.getTransportPayloadTypeConfiguration(); | |
95 | + assertTrue(transportPayloadTypeConfiguration instanceof ProtoTransportPayloadConfiguration); | |
96 | + ProtoTransportPayloadConfiguration protoTransportPayloadConfiguration = (ProtoTransportPayloadConfiguration) transportPayloadTypeConfiguration; | |
97 | + ProtoFileElement transportProtoSchema = protoTransportPayloadConfiguration.getTransportProtoSchema(ATTRIBUTES_SCHEMA_STR); | |
98 | + DynamicSchema attributesSchema = protoTransportPayloadConfiguration.getDynamicSchema(transportProtoSchema, ProtoTransportPayloadConfiguration.ATTRIBUTES_PROTO_SCHEMA); | |
99 | + | |
100 | + DynamicMessage.Builder nestedJsonObjectBuilder = attributesSchema.newMessageBuilder("PostAttributes.JsonObject.NestedJsonObject"); | |
101 | + Descriptors.Descriptor nestedJsonObjectBuilderDescriptor = nestedJsonObjectBuilder.getDescriptorForType(); | |
102 | + assertNotNull(nestedJsonObjectBuilderDescriptor); | |
103 | + DynamicMessage nestedJsonObject = nestedJsonObjectBuilder.setField(nestedJsonObjectBuilderDescriptor.findFieldByName("key"), "value").build(); | |
104 | + | |
105 | + DynamicMessage.Builder jsonObjectBuilder = attributesSchema.newMessageBuilder("PostAttributes.JsonObject"); | |
106 | + Descriptors.Descriptor jsonObjectBuilderDescriptor = jsonObjectBuilder.getDescriptorForType(); | |
107 | + assertNotNull(jsonObjectBuilderDescriptor); | |
108 | + DynamicMessage jsonObject = jsonObjectBuilder | |
109 | + .setField(jsonObjectBuilderDescriptor.findFieldByName("someNumber"), 42) | |
110 | + .addRepeatedField(jsonObjectBuilderDescriptor.findFieldByName("someArray"), 1) | |
111 | + .addRepeatedField(jsonObjectBuilderDescriptor.findFieldByName("someArray"), 2) | |
112 | + .addRepeatedField(jsonObjectBuilderDescriptor.findFieldByName("someArray"), 3) | |
113 | + .setField(jsonObjectBuilderDescriptor.findFieldByName("someNestedObject"), nestedJsonObject) | |
114 | + .build(); | |
115 | + | |
116 | + DynamicMessage.Builder postAttributesBuilder = attributesSchema.newMessageBuilder("PostAttributes"); | |
117 | + Descriptors.Descriptor postAttributesMsgDescriptor = postAttributesBuilder.getDescriptorForType(); | |
118 | + assertNotNull(postAttributesMsgDescriptor); | |
119 | + DynamicMessage postAttributesMsg = postAttributesBuilder | |
120 | + .setField(postAttributesMsgDescriptor.findFieldByName("attribute1"), "value1") | |
121 | + .setField(postAttributesMsgDescriptor.findFieldByName("attribute2"), true) | |
122 | + .setField(postAttributesMsgDescriptor.findFieldByName("attribute3"), 42.0) | |
123 | + .setField(postAttributesMsgDescriptor.findFieldByName("attribute4"), 73) | |
124 | + .setField(postAttributesMsgDescriptor.findFieldByName("attribute5"), jsonObject) | |
125 | + .build(); | |
126 | + byte[] payload = postAttributesMsg.toByteArray(); | |
127 | + CoapClient client = getCoapClient(FeatureType.ATTRIBUTES); | |
128 | + CoapResponse coapResponse = client.setTimeout(CLIENT_REQUEST_TIMEOUT).post(payload, MediaTypeRegistry.APPLICATION_JSON); | |
129 | + assertEquals(CoAP.ResponseCode.CREATED, coapResponse.getCode()); | |
130 | + } | |
131 | + | |
132 | + protected void validateResponse(CoapResponse getAttributesResponse) throws InvalidProtocolBufferException { | |
133 | + TransportProtos.GetAttributeResponseMsg expectedAttributesResponse = getExpectedAttributeResponseMsg(); | |
134 | + TransportProtos.GetAttributeResponseMsg actualAttributesResponse = TransportProtos.GetAttributeResponseMsg.parseFrom(getAttributesResponse.getPayload()); | |
135 | + assertEquals(expectedAttributesResponse.getRequestId(), actualAttributesResponse.getRequestId()); | |
136 | + List<TransportProtos.KeyValueProto> expectedClientKeyValueProtos = expectedAttributesResponse.getClientAttributeListList().stream().map(TransportProtos.TsKvProto::getKv).collect(Collectors.toList()); | |
137 | + List<TransportProtos.KeyValueProto> expectedSharedKeyValueProtos = expectedAttributesResponse.getSharedAttributeListList().stream().map(TransportProtos.TsKvProto::getKv).collect(Collectors.toList()); | |
138 | + List<TransportProtos.KeyValueProto> actualClientKeyValueProtos = actualAttributesResponse.getClientAttributeListList().stream().map(TransportProtos.TsKvProto::getKv).collect(Collectors.toList()); | |
139 | + List<TransportProtos.KeyValueProto> actualSharedKeyValueProtos = actualAttributesResponse.getSharedAttributeListList().stream().map(TransportProtos.TsKvProto::getKv).collect(Collectors.toList()); | |
140 | + assertTrue(actualClientKeyValueProtos.containsAll(expectedClientKeyValueProtos)); | |
141 | + assertTrue(actualSharedKeyValueProtos.containsAll(expectedSharedKeyValueProtos)); | |
142 | + } | |
143 | + | |
144 | + private TransportProtos.GetAttributeResponseMsg getExpectedAttributeResponseMsg() { | |
145 | + TransportProtos.GetAttributeResponseMsg.Builder result = TransportProtos.GetAttributeResponseMsg.newBuilder(); | |
146 | + List<TransportProtos.TsKvProto> tsKvProtoList = getTsKvProtoList(); | |
147 | + result.addAllClientAttributeList(tsKvProtoList); | |
148 | + result.addAllSharedAttributeList(tsKvProtoList); | |
149 | + result.setRequestId(0); | |
150 | + return result.build(); | |
151 | + } | |
152 | + | |
153 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.request.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.attributes.request.AbstractCoapAttributesRequestJsonIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapAttributesRequestJsonSqlIntegrationTest extends AbstractCoapAttributesRequestJsonIntegrationTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.request.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.attributes.request.AbstractCoapAttributesRequestProtoIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapAttributesRequestProtoSqlIntegrationTest extends AbstractCoapAttributesRequestProtoIntegrationTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.request.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.attributes.request.AbstractCoapAttributesRequestIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapAttributesRequestSqlIntegrationTest extends AbstractCoapAttributesRequestIntegrationTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.updates; | |
17 | + | |
18 | +import com.google.protobuf.InvalidProtocolBufferException; | |
19 | +import lombok.extern.slf4j.Slf4j; | |
20 | +import org.eclipse.californium.core.CoapClient; | |
21 | +import org.eclipse.californium.core.CoapHandler; | |
22 | +import org.eclipse.californium.core.CoapObserveRelation; | |
23 | +import org.eclipse.californium.core.CoapResponse; | |
24 | +import org.eclipse.californium.core.coap.CoAP; | |
25 | +import org.eclipse.californium.core.coap.Request; | |
26 | +import org.junit.After; | |
27 | +import org.junit.Before; | |
28 | +import org.junit.Test; | |
29 | +import org.thingsboard.server.transport.coap.attributes.AbstractCoapAttributesIntegrationTest; | |
30 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
31 | +import org.thingsboard.server.dao.util.mapping.JacksonUtil; | |
32 | + | |
33 | +import java.nio.charset.StandardCharsets; | |
34 | +import java.util.concurrent.CountDownLatch; | |
35 | +import java.util.concurrent.TimeUnit; | |
36 | + | |
37 | +import static org.junit.Assert.assertEquals; | |
38 | +import static org.junit.Assert.assertNotNull; | |
39 | +import static org.junit.Assert.assertTrue; | |
40 | +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; | |
41 | + | |
42 | +@Slf4j | |
43 | +public abstract class AbstractCoapAttributesUpdatesIntegrationTest extends AbstractCoapAttributesIntegrationTest { | |
44 | + | |
45 | + private static final String RESPONSE_ATTRIBUTES_PAYLOAD_DELETED = "{\"deleted\":[\"attribute5\"]}"; | |
46 | + | |
47 | + @Before | |
48 | + public void beforeTest() throws Exception { | |
49 | + processBeforeTest("Test Subscribe to attribute updates", null, null); | |
50 | + } | |
51 | + | |
52 | + @After | |
53 | + public void afterTest() throws Exception { | |
54 | + processAfterTest(); | |
55 | + } | |
56 | + | |
57 | + @Test | |
58 | + public void testSubscribeToAttributesUpdatesFromTheServer() throws Exception { | |
59 | + processTestSubscribeToAttributesUpdates(); | |
60 | + } | |
61 | + | |
62 | + protected void processTestSubscribeToAttributesUpdates() throws Exception { | |
63 | + | |
64 | + CoapClient client = getCoapClient(FeatureType.ATTRIBUTES); | |
65 | + | |
66 | + CountDownLatch latch = new CountDownLatch(1); | |
67 | + TestCoapCallback testCoapCallback = new TestCoapCallback(latch); | |
68 | + | |
69 | + Request request = Request.newGet().setObserve(); | |
70 | + request.setType(CoAP.Type.CON); | |
71 | + CoapObserveRelation observeRelation = client.observe(request, testCoapCallback); | |
72 | + | |
73 | + Thread.sleep(1000); | |
74 | + | |
75 | + doPostAsync("/api/plugins/telemetry/DEVICE/" + savedDevice.getId().getId() + "/attributes/SHARED_SCOPE", POST_ATTRIBUTES_PAYLOAD, String.class, status().isOk()); | |
76 | + latch.await(3, TimeUnit.SECONDS); | |
77 | + | |
78 | + validateUpdateAttributesResponse(testCoapCallback); | |
79 | + | |
80 | + latch = new CountDownLatch(1); | |
81 | + | |
82 | + doDelete("/api/plugins/telemetry/DEVICE/" + savedDevice.getId().getId() + "/SHARED_SCOPE?keys=attribute5", String.class); | |
83 | + latch.await(3, TimeUnit.SECONDS); | |
84 | + | |
85 | + validateDeleteAttributesResponse(testCoapCallback); | |
86 | + | |
87 | + observeRelation.proactiveCancel(); | |
88 | + assertTrue(observeRelation.isCanceled()); | |
89 | + } | |
90 | + | |
91 | + protected void validateUpdateAttributesResponse(TestCoapCallback callback) throws InvalidProtocolBufferException { | |
92 | + assertNotNull(callback.getPayloadBytes()); | |
93 | + assertNotNull(callback.getObserve()); | |
94 | + assertEquals(0, callback.getObserve().intValue()); | |
95 | + String response = new String(callback.getPayloadBytes(), StandardCharsets.UTF_8); | |
96 | + assertEquals(JacksonUtil.toJsonNode(POST_ATTRIBUTES_PAYLOAD), JacksonUtil.toJsonNode(response)); | |
97 | + } | |
98 | + | |
99 | + protected void validateDeleteAttributesResponse(TestCoapCallback callback) throws InvalidProtocolBufferException { | |
100 | + assertNotNull(callback.getPayloadBytes()); | |
101 | + assertNotNull(callback.getObserve()); | |
102 | + assertEquals(1, callback.getObserve().intValue()); | |
103 | + String response = new String(callback.getPayloadBytes(), StandardCharsets.UTF_8); | |
104 | + assertEquals(JacksonUtil.toJsonNode(RESPONSE_ATTRIBUTES_PAYLOAD_DELETED), JacksonUtil.toJsonNode(response)); | |
105 | + } | |
106 | + | |
107 | + protected static class TestCoapCallback implements CoapHandler { | |
108 | + | |
109 | + private final CountDownLatch latch; | |
110 | + | |
111 | + private Integer observe; | |
112 | + private byte[] payloadBytes; | |
113 | + | |
114 | + public byte[] getPayloadBytes() { | |
115 | + return payloadBytes; | |
116 | + } | |
117 | + | |
118 | + public Integer getObserve() { | |
119 | + return observe; | |
120 | + } | |
121 | + | |
122 | + private TestCoapCallback(CountDownLatch latch) { | |
123 | + this.latch = latch; | |
124 | + } | |
125 | + | |
126 | + @Override | |
127 | + public void onLoad(CoapResponse response) { | |
128 | + assertNotNull(response.getPayload()); | |
129 | + assertEquals(response.getCode(), CoAP.ResponseCode.CONTENT); | |
130 | + observe = response.getOptions().getObserve(); | |
131 | + payloadBytes = response.getPayload(); | |
132 | + latch.countDown(); | |
133 | + } | |
134 | + | |
135 | + @Override | |
136 | + public void onError() { | |
137 | + log.warn("Command Response Ack Error, No connect"); | |
138 | + } | |
139 | + | |
140 | + } | |
141 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.updates; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.junit.After; | |
20 | +import org.junit.Before; | |
21 | +import org.junit.Test; | |
22 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
23 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
24 | + | |
25 | +@Slf4j | |
26 | +public abstract class AbstractCoapAttributesUpdatesJsonIntegrationTest extends AbstractCoapAttributesUpdatesIntegrationTest { | |
27 | + | |
28 | + @Before | |
29 | + public void beforeTest() throws Exception { | |
30 | + processBeforeTest("Test Subscribe to attribute updates", CoapDeviceType.DEFAULT, TransportPayloadType.JSON); | |
31 | + } | |
32 | + | |
33 | + @After | |
34 | + public void afterTest() throws Exception { | |
35 | + processAfterTest(); | |
36 | + } | |
37 | + | |
38 | + @Test | |
39 | + public void testSubscribeToAttributesUpdatesFromTheServer() throws Exception { | |
40 | + super.testSubscribeToAttributesUpdatesFromTheServer(); | |
41 | + } | |
42 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.updates; | |
17 | + | |
18 | +import com.google.protobuf.InvalidProtocolBufferException; | |
19 | +import lombok.extern.slf4j.Slf4j; | |
20 | +import org.junit.After; | |
21 | +import org.junit.Before; | |
22 | +import org.junit.Test; | |
23 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
24 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
25 | +import org.thingsboard.server.gen.transport.TransportProtos; | |
26 | + | |
27 | +import java.util.List; | |
28 | +import java.util.stream.Collectors; | |
29 | + | |
30 | +import static org.junit.Assert.assertEquals; | |
31 | +import static org.junit.Assert.assertNotNull; | |
32 | +import static org.junit.Assert.assertTrue; | |
33 | + | |
34 | +@Slf4j | |
35 | +public abstract class AbstractCoapAttributesUpdatesProtoIntegrationTest extends AbstractCoapAttributesUpdatesIntegrationTest { | |
36 | + | |
37 | + @Before | |
38 | + public void beforeTest() throws Exception { | |
39 | + processBeforeTest("Test Subscribe to attribute updates", CoapDeviceType.DEFAULT, TransportPayloadType.PROTOBUF); | |
40 | + } | |
41 | + | |
42 | + @After | |
43 | + public void afterTest() throws Exception { | |
44 | + processAfterTest(); | |
45 | + } | |
46 | + | |
47 | + @Test | |
48 | + public void testSubscribeToAttributesUpdatesFromTheServer() throws Exception { | |
49 | + processTestSubscribeToAttributesUpdates(); | |
50 | + } | |
51 | + | |
52 | + protected void validateUpdateAttributesResponse(TestCoapCallback callback) throws InvalidProtocolBufferException { | |
53 | + assertNotNull(callback.getPayloadBytes()); | |
54 | + TransportProtos.AttributeUpdateNotificationMsg.Builder attributeUpdateNotificationMsgBuilder = TransportProtos.AttributeUpdateNotificationMsg.newBuilder(); | |
55 | + List<TransportProtos.TsKvProto> tsKvProtoList = getTsKvProtoList(); | |
56 | + attributeUpdateNotificationMsgBuilder.addAllSharedUpdated(tsKvProtoList); | |
57 | + | |
58 | + TransportProtos.AttributeUpdateNotificationMsg expectedAttributeUpdateNotificationMsg = attributeUpdateNotificationMsgBuilder.build(); | |
59 | + TransportProtos.AttributeUpdateNotificationMsg actualAttributeUpdateNotificationMsg = TransportProtos.AttributeUpdateNotificationMsg.parseFrom(callback.getPayloadBytes()); | |
60 | + | |
61 | + List<TransportProtos.KeyValueProto> actualSharedUpdatedList = actualAttributeUpdateNotificationMsg.getSharedUpdatedList().stream().map(TransportProtos.TsKvProto::getKv).collect(Collectors.toList()); | |
62 | + List<TransportProtos.KeyValueProto> expectedSharedUpdatedList = expectedAttributeUpdateNotificationMsg.getSharedUpdatedList().stream().map(TransportProtos.TsKvProto::getKv).collect(Collectors.toList()); | |
63 | + | |
64 | + assertEquals(expectedSharedUpdatedList.size(), actualSharedUpdatedList.size()); | |
65 | + assertTrue(actualSharedUpdatedList.containsAll(expectedSharedUpdatedList)); | |
66 | + | |
67 | + } | |
68 | + | |
69 | + protected void validateDeleteAttributesResponse(TestCoapCallback callback) throws InvalidProtocolBufferException { | |
70 | + assertNotNull(callback.getPayloadBytes()); | |
71 | + TransportProtos.AttributeUpdateNotificationMsg.Builder attributeUpdateNotificationMsgBuilder = TransportProtos.AttributeUpdateNotificationMsg.newBuilder(); | |
72 | + attributeUpdateNotificationMsgBuilder.addSharedDeleted("attribute5"); | |
73 | + | |
74 | + TransportProtos.AttributeUpdateNotificationMsg expectedAttributeUpdateNotificationMsg = attributeUpdateNotificationMsgBuilder.build(); | |
75 | + TransportProtos.AttributeUpdateNotificationMsg actualAttributeUpdateNotificationMsg = TransportProtos.AttributeUpdateNotificationMsg.parseFrom(callback.getPayloadBytes()); | |
76 | + | |
77 | + assertEquals(expectedAttributeUpdateNotificationMsg.getSharedDeletedList().size(), actualAttributeUpdateNotificationMsg.getSharedDeletedList().size()); | |
78 | + assertEquals("attribute5", actualAttributeUpdateNotificationMsg.getSharedDeletedList().get(0)); | |
79 | + | |
80 | + } | |
81 | + | |
82 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.updates.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.attributes.updates.AbstractCoapAttributesUpdatesIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapAttributesUpdatesSqlIntegrationTest extends AbstractCoapAttributesUpdatesIntegrationTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.updates.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.attributes.updates.AbstractCoapAttributesUpdatesJsonIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapAttributesUpdatesSqlJsonIntegrationTest extends AbstractCoapAttributesUpdatesJsonIntegrationTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.attributes.updates.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.attributes.updates.AbstractCoapAttributesUpdatesProtoIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapAttributesUpdatesSqlProtoIntegrationTest extends AbstractCoapAttributesUpdatesProtoIntegrationTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.claim; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.eclipse.californium.core.CoapClient; | |
20 | +import org.eclipse.californium.core.CoapResponse; | |
21 | +import org.eclipse.californium.core.coap.CoAP; | |
22 | +import org.eclipse.californium.core.coap.MediaTypeRegistry; | |
23 | +import org.eclipse.californium.elements.exception.ConnectorException; | |
24 | +import org.junit.After; | |
25 | +import org.junit.Before; | |
26 | +import org.junit.Test; | |
27 | +import org.thingsboard.server.transport.coap.AbstractCoapIntegrationTest; | |
28 | +import org.thingsboard.server.common.data.ClaimRequest; | |
29 | +import org.thingsboard.server.common.data.Customer; | |
30 | +import org.thingsboard.server.common.data.Device; | |
31 | +import org.thingsboard.server.common.data.User; | |
32 | +import org.thingsboard.server.common.data.security.Authority; | |
33 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
34 | +import org.thingsboard.server.dao.device.claim.ClaimResponse; | |
35 | +import org.thingsboard.server.dao.device.claim.ClaimResult; | |
36 | + | |
37 | +import java.io.IOException; | |
38 | + | |
39 | +import static org.junit.Assert.assertEquals; | |
40 | +import static org.junit.Assert.assertNotNull; | |
41 | +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; | |
42 | + | |
43 | +@Slf4j | |
44 | +public abstract class AbstractCoapClaimDeviceTest extends AbstractCoapIntegrationTest { | |
45 | + | |
46 | + protected static final String CUSTOMER_USER_PASSWORD = "customerUser123!"; | |
47 | + | |
48 | + protected User customerAdmin; | |
49 | + protected Customer savedCustomer; | |
50 | + | |
51 | + @Before | |
52 | + public void beforeTest() throws Exception { | |
53 | + super.processBeforeTest("Test Claim device", null, null); | |
54 | + createCustomerAndUser(); | |
55 | + } | |
56 | + | |
57 | + protected void createCustomerAndUser() throws Exception { | |
58 | + Customer customer = new Customer(); | |
59 | + customer.setTenantId(savedTenant.getId()); | |
60 | + customer.setTitle("Test Claiming Customer"); | |
61 | + savedCustomer = doPost("/api/customer", customer, Customer.class); | |
62 | + assertNotNull(savedCustomer); | |
63 | + assertEquals(savedTenant.getId(), savedCustomer.getTenantId()); | |
64 | + | |
65 | + User user = new User(); | |
66 | + user.setAuthority(Authority.CUSTOMER_USER); | |
67 | + user.setTenantId(savedTenant.getId()); | |
68 | + user.setCustomerId(savedCustomer.getId()); | |
69 | + user.setEmail("customer@thingsboard.org"); | |
70 | + | |
71 | + customerAdmin = createUser(user, CUSTOMER_USER_PASSWORD); | |
72 | + assertNotNull(customerAdmin); | |
73 | + assertEquals(customerAdmin.getCustomerId(), savedCustomer.getId()); | |
74 | + } | |
75 | + | |
76 | + @After | |
77 | + public void afterTest() throws Exception { | |
78 | + super.processAfterTest(); | |
79 | + } | |
80 | + | |
81 | + @Test | |
82 | + public void testClaimingDevice() throws Exception { | |
83 | + processTestClaimingDevice(false); | |
84 | + } | |
85 | + | |
86 | + @Test | |
87 | + public void testClaimingDeviceWithoutSecretAndDuration() throws Exception { | |
88 | + processTestClaimingDevice(true); | |
89 | + } | |
90 | + | |
91 | + protected void processTestClaimingDevice(boolean emptyPayload) throws Exception { | |
92 | + log.warn("[testClaimingDevice] Device: {}, Transport type: {}", savedDevice.getName(), savedDevice.getType()); | |
93 | + CoapClient client = getCoapClient(FeatureType.CLAIM); | |
94 | + byte[] payloadBytes; | |
95 | + byte[] failurePayloadBytes; | |
96 | + if (emptyPayload) { | |
97 | + payloadBytes = "{}".getBytes(); | |
98 | + failurePayloadBytes = "{\"durationMs\":1}".getBytes(); | |
99 | + } else { | |
100 | + payloadBytes = "{\"secretKey\":\"value\", \"durationMs\":60000}".getBytes(); | |
101 | + failurePayloadBytes = "{\"secretKey\":\"value\", \"durationMs\":1}".getBytes(); | |
102 | + } | |
103 | + validateClaimResponse(emptyPayload, client, payloadBytes, failurePayloadBytes); | |
104 | + } | |
105 | + | |
106 | + protected void validateClaimResponse(boolean emptyPayload, CoapClient client, byte[] payloadBytes, byte[] failurePayloadBytes) throws Exception { | |
107 | + postClaimRequest(client, failurePayloadBytes); | |
108 | + | |
109 | + loginUser(customerAdmin.getName(), CUSTOMER_USER_PASSWORD); | |
110 | + ClaimRequest claimRequest; | |
111 | + if (!emptyPayload) { | |
112 | + claimRequest = new ClaimRequest("value"); | |
113 | + } else { | |
114 | + claimRequest = new ClaimRequest(null); | |
115 | + } | |
116 | + | |
117 | + ClaimResponse claimResponse = doExecuteWithRetriesAndInterval( | |
118 | + () -> doPostClaimAsync("/api/customer/device/" + savedDevice.getName() + "/claim", claimRequest, ClaimResponse.class, status().isBadRequest()), | |
119 | + 20, | |
120 | + 100 | |
121 | + ); | |
122 | + | |
123 | + assertEquals(claimResponse, ClaimResponse.FAILURE); | |
124 | + | |
125 | + postClaimRequest(client, payloadBytes); | |
126 | + | |
127 | + ClaimResult claimResult = doExecuteWithRetriesAndInterval( | |
128 | + () -> doPostClaimAsync("/api/customer/device/" + savedDevice.getName() + "/claim", claimRequest, ClaimResult.class, status().isOk()), | |
129 | + 20, | |
130 | + 100 | |
131 | + ); | |
132 | + assertEquals(claimResult.getResponse(), ClaimResponse.SUCCESS); | |
133 | + Device claimedDevice = claimResult.getDevice(); | |
134 | + assertNotNull(claimedDevice); | |
135 | + assertNotNull(claimedDevice.getCustomerId()); | |
136 | + assertEquals(customerAdmin.getCustomerId(), claimedDevice.getCustomerId()); | |
137 | + | |
138 | + claimResponse = doPostClaimAsync("/api/customer/device/" + savedDevice.getName() + "/claim", claimRequest, ClaimResponse.class, status().isBadRequest()); | |
139 | + assertEquals(claimResponse, ClaimResponse.CLAIMED); | |
140 | + } | |
141 | + | |
142 | + private void postClaimRequest(CoapClient client, byte[] payload) throws IOException, ConnectorException { | |
143 | + CoapResponse coapResponse = client.setTimeout((long) 60000).post(payload, MediaTypeRegistry.APPLICATION_JSON); | |
144 | + assertEquals(CoAP.ResponseCode.CREATED, coapResponse.getCode()); | |
145 | + } | |
146 | + | |
147 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.claim; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.junit.After; | |
20 | +import org.junit.Before; | |
21 | +import org.junit.Test; | |
22 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
23 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
24 | + | |
25 | +@Slf4j | |
26 | +public abstract class AbstractCoapClaimJsonDeviceTest extends AbstractCoapClaimDeviceTest { | |
27 | + | |
28 | + @Before | |
29 | + public void beforeTest() throws Exception { | |
30 | + super.processBeforeTest("Test Claim device Json", CoapDeviceType.DEFAULT, TransportPayloadType.JSON); | |
31 | + createCustomerAndUser(); | |
32 | + } | |
33 | + | |
34 | + @After | |
35 | + public void afterTest() throws Exception { | |
36 | + super.afterTest(); | |
37 | + } | |
38 | + | |
39 | + @Test | |
40 | + public void testClaimingDevice() throws Exception { | |
41 | + super.testClaimingDevice(); | |
42 | + } | |
43 | + | |
44 | + @Test | |
45 | + public void testClaimingDeviceWithoutSecretAndDuration() throws Exception { | |
46 | + super.testClaimingDeviceWithoutSecretAndDuration(); | |
47 | + } | |
48 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.claim; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.eclipse.californium.core.CoapClient; | |
20 | +import org.junit.After; | |
21 | +import org.junit.Before; | |
22 | +import org.junit.Test; | |
23 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
24 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
25 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
26 | +import org.thingsboard.server.gen.transport.TransportApiProtos; | |
27 | + | |
28 | +@Slf4j | |
29 | +public abstract class AbstractCoapClaimProtoDeviceTest extends AbstractCoapClaimDeviceTest { | |
30 | + | |
31 | + @Before | |
32 | + public void beforeTest() throws Exception { | |
33 | + processBeforeTest("Test Claim device Proto", CoapDeviceType.DEFAULT, TransportPayloadType.PROTOBUF); | |
34 | + createCustomerAndUser(); | |
35 | + } | |
36 | + | |
37 | + @After | |
38 | + public void afterTest() throws Exception { super.afterTest(); } | |
39 | + | |
40 | + @Test | |
41 | + public void testClaimingDevice() throws Exception { | |
42 | + processTestClaimingDevice(false); | |
43 | + } | |
44 | + | |
45 | + @Test | |
46 | + public void testClaimingDeviceWithoutSecretAndDuration() throws Exception { | |
47 | + processTestClaimingDevice(true); | |
48 | + } | |
49 | + | |
50 | + @Override | |
51 | + protected void processTestClaimingDevice(boolean emptyPayload) throws Exception { | |
52 | + CoapClient client = getCoapClient(FeatureType.CLAIM); | |
53 | + byte[] payloadBytes; | |
54 | + if (emptyPayload) { | |
55 | + TransportApiProtos.ClaimDevice claimDevice = getClaimDevice(0, emptyPayload); | |
56 | + payloadBytes = claimDevice.toByteArray(); | |
57 | + } else { | |
58 | + TransportApiProtos.ClaimDevice claimDevice = getClaimDevice(60000, emptyPayload); | |
59 | + payloadBytes = claimDevice.toByteArray(); | |
60 | + } | |
61 | + TransportApiProtos.ClaimDevice claimDevice = getClaimDevice(1, emptyPayload); | |
62 | + byte[] failurePayloadBytes = claimDevice.toByteArray(); | |
63 | + validateClaimResponse(emptyPayload, client, payloadBytes, failurePayloadBytes); | |
64 | + } | |
65 | + | |
66 | + private TransportApiProtos.ClaimDevice getClaimDevice(long duration, boolean emptyPayload) { | |
67 | + TransportApiProtos.ClaimDevice.Builder claimDeviceBuilder = TransportApiProtos.ClaimDevice.newBuilder(); | |
68 | + if (!emptyPayload) { | |
69 | + claimDeviceBuilder.setSecretKey("value"); | |
70 | + } | |
71 | + if (duration > 0) { | |
72 | + claimDeviceBuilder.setSecretKey("value"); | |
73 | + claimDeviceBuilder.setDurationMs(duration); | |
74 | + } else { | |
75 | + claimDeviceBuilder.setDurationMs(0); | |
76 | + } | |
77 | + return claimDeviceBuilder.build(); | |
78 | + } | |
79 | + | |
80 | + | |
81 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.claim.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.claim.AbstractCoapClaimJsonDeviceTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapClaimDeviceJsonSqlTest extends AbstractCoapClaimJsonDeviceTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.claim.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.claim.AbstractCoapClaimProtoDeviceTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapClaimDeviceProtoSqlTest extends AbstractCoapClaimProtoDeviceTest { | |
23 | +} | ... | ... |
application/src/test/java/org/thingsboard/server/transport/coap/claim/sql/CoapClaimDeviceSqlTest.java
renamed from
application/src/test/java/org/thingsboard/server/mqtt/telemetry/attributes/nosql/MqttAttributesNoSqlProtoIntegrationTest.java
... | ... | @@ -13,11 +13,11 @@ |
13 | 13 | * See the License for the specific language governing permissions and |
14 | 14 | * limitations under the License. |
15 | 15 | */ |
16 | -package org.thingsboard.server.mqtt.telemetry.attributes.nosql; | |
16 | +package org.thingsboard.server.transport.coap.claim.sql; | |
17 | 17 | |
18 | -import org.thingsboard.server.dao.service.DaoNoSqlTest; | |
19 | -import org.thingsboard.server.mqtt.telemetry.attributes.AbstractMqttAttributesProtoIntegrationTest; | |
18 | +import org.thingsboard.server.transport.coap.claim.AbstractCoapClaimDeviceTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | 20 | |
21 | -@DaoNoSqlTest | |
22 | -public class MqttAttributesNoSqlProtoIntegrationTest extends AbstractMqttAttributesProtoIntegrationTest { | |
21 | +@DaoSqlTest | |
22 | +public class CoapClaimDeviceSqlTest extends AbstractCoapClaimDeviceTest { | |
23 | 23 | } | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.provision; | |
17 | + | |
18 | +import com.google.gson.JsonObject; | |
19 | +import lombok.extern.slf4j.Slf4j; | |
20 | +import org.eclipse.californium.core.CoapClient; | |
21 | +import org.eclipse.californium.core.CoapResponse; | |
22 | +import org.eclipse.californium.core.coap.MediaTypeRegistry; | |
23 | +import org.eclipse.californium.elements.exception.ConnectorException; | |
24 | +import org.junit.After; | |
25 | +import org.junit.Assert; | |
26 | +import org.junit.Test; | |
27 | +import org.springframework.beans.factory.annotation.Autowired; | |
28 | +import org.thingsboard.server.transport.coap.AbstractCoapIntegrationTest; | |
29 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
30 | +import org.thingsboard.server.common.data.Device; | |
31 | +import org.thingsboard.server.common.data.DeviceProfileProvisionType; | |
32 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
33 | +import org.thingsboard.server.common.data.security.DeviceCredentials; | |
34 | +import org.thingsboard.server.common.msg.EncryptionUtil; | |
35 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
36 | +import org.thingsboard.server.common.transport.util.JsonUtils; | |
37 | +import org.thingsboard.server.dao.device.DeviceCredentialsService; | |
38 | +import org.thingsboard.server.dao.device.DeviceService; | |
39 | +import org.thingsboard.server.dao.device.provision.ProvisionResponseStatus; | |
40 | + | |
41 | +import java.io.IOException; | |
42 | + | |
43 | +import static org.junit.Assert.assertEquals; | |
44 | + | |
45 | +@Slf4j | |
46 | +public abstract class AbstractCoapProvisionJsonDeviceTest extends AbstractCoapIntegrationTest { | |
47 | + | |
48 | + @Autowired | |
49 | + DeviceCredentialsService deviceCredentialsService; | |
50 | + | |
51 | + @Autowired | |
52 | + DeviceService deviceService; | |
53 | + | |
54 | + @After | |
55 | + public void afterTest() throws Exception { | |
56 | + super.processAfterTest(); | |
57 | + } | |
58 | + | |
59 | + @Test | |
60 | + public void testProvisioningDisabledDevice() throws Exception { | |
61 | + processTestProvisioningDisabledDevice(); | |
62 | + } | |
63 | + | |
64 | + @Test | |
65 | + public void testProvisioningCheckPreProvisionedDevice() throws Exception { | |
66 | + processTestProvisioningCheckPreProvisionedDevice(); | |
67 | + } | |
68 | + | |
69 | + @Test | |
70 | + public void testProvisioningCreateNewDeviceWithoutCredentials() throws Exception { | |
71 | + processTestProvisioningCreateNewDeviceWithoutCredentials(); | |
72 | + } | |
73 | + | |
74 | + @Test | |
75 | + public void testProvisioningCreateNewDeviceWithAccessToken() throws Exception { | |
76 | + processTestProvisioningCreateNewDeviceWithAccessToken(); | |
77 | + } | |
78 | + | |
79 | + @Test | |
80 | + public void testProvisioningCreateNewDeviceWithCert() throws Exception { | |
81 | + processTestProvisioningCreateNewDeviceWithCert(); | |
82 | + } | |
83 | + | |
84 | + @Test | |
85 | + public void testProvisioningWithBadKeyDevice() throws Exception { | |
86 | + processTestProvisioningWithBadKeyDevice(); | |
87 | + } | |
88 | + | |
89 | + | |
90 | + private void processTestProvisioningDisabledDevice() throws Exception { | |
91 | + super.processBeforeTest("Test Provision device", CoapDeviceType.DEFAULT, TransportPayloadType.JSON, null, null, DeviceProfileProvisionType.DISABLED, null, null); | |
92 | + byte[] result = createCoapClientAndPublish().getPayload(); | |
93 | + JsonObject response = JsonUtils.parse(new String(result)).getAsJsonObject(); | |
94 | + Assert.assertEquals("Provision data was not found!", response.get("errorMsg").getAsString()); | |
95 | + Assert.assertEquals(ProvisionResponseStatus.NOT_FOUND.name(), response.get("status").getAsString()); | |
96 | + } | |
97 | + | |
98 | + | |
99 | + private void processTestProvisioningCreateNewDeviceWithoutCredentials() throws Exception { | |
100 | + super.processBeforeTest("Test Provision device3", CoapDeviceType.DEFAULT, TransportPayloadType.JSON, null, null, DeviceProfileProvisionType.ALLOW_CREATE_NEW_DEVICES, "testProvisionKey", "testProvisionSecret"); | |
101 | + byte[] result = createCoapClientAndPublish().getPayload(); | |
102 | + JsonObject response = JsonUtils.parse(new String(result)).getAsJsonObject(); | |
103 | + | |
104 | + Device createdDevice = deviceService.findDeviceByTenantIdAndName(savedTenant.getTenantId(), "Test Provision device"); | |
105 | + | |
106 | + Assert.assertNotNull(createdDevice); | |
107 | + | |
108 | + DeviceCredentials deviceCredentials = deviceCredentialsService.findDeviceCredentialsByDeviceId(savedTenant.getTenantId(), createdDevice.getId()); | |
109 | + | |
110 | + Assert.assertEquals(deviceCredentials.getCredentialsType().name(), response.get("credentialsType").getAsString()); | |
111 | + Assert.assertEquals(ProvisionResponseStatus.SUCCESS.name(), response.get("status").getAsString()); | |
112 | + } | |
113 | + | |
114 | + | |
115 | + private void processTestProvisioningCreateNewDeviceWithAccessToken() throws Exception { | |
116 | + super.processBeforeTest("Test Provision device3", CoapDeviceType.DEFAULT, TransportPayloadType.JSON, null, null, DeviceProfileProvisionType.ALLOW_CREATE_NEW_DEVICES, "testProvisionKey", "testProvisionSecret"); | |
117 | + String requestCredentials = ",\"credentialsType\": \"ACCESS_TOKEN\",\"token\": \"test_token\""; | |
118 | + byte[] result = createCoapClientAndPublish(requestCredentials).getPayload(); | |
119 | + JsonObject response = JsonUtils.parse(new String(result)).getAsJsonObject(); | |
120 | + | |
121 | + Device createdDevice = deviceService.findDeviceByTenantIdAndName(savedTenant.getTenantId(), "Test Provision device"); | |
122 | + | |
123 | + Assert.assertNotNull(createdDevice); | |
124 | + | |
125 | + DeviceCredentials deviceCredentials = deviceCredentialsService.findDeviceCredentialsByDeviceId(savedTenant.getTenantId(), createdDevice.getId()); | |
126 | + | |
127 | + Assert.assertEquals(deviceCredentials.getCredentialsType().name(), response.get("credentialsType").getAsString()); | |
128 | + Assert.assertEquals(deviceCredentials.getCredentialsType().name(), "ACCESS_TOKEN"); | |
129 | + Assert.assertEquals(deviceCredentials.getCredentialsId(), "test_token"); | |
130 | + Assert.assertEquals(ProvisionResponseStatus.SUCCESS.name(), response.get("status").getAsString()); | |
131 | + } | |
132 | + | |
133 | + | |
134 | + private void processTestProvisioningCreateNewDeviceWithCert() throws Exception { | |
135 | + super.processBeforeTest("Test Provision device3", CoapDeviceType.DEFAULT, TransportPayloadType.JSON, null, null, DeviceProfileProvisionType.ALLOW_CREATE_NEW_DEVICES, "testProvisionKey", "testProvisionSecret"); | |
136 | + String requestCredentials = ",\"credentialsType\": \"X509_CERTIFICATE\",\"hash\": \"testHash\""; | |
137 | + byte[] result = createCoapClientAndPublish(requestCredentials).getPayload(); | |
138 | + JsonObject response = JsonUtils.parse(new String(result)).getAsJsonObject(); | |
139 | + | |
140 | + Device createdDevice = deviceService.findDeviceByTenantIdAndName(savedTenant.getTenantId(), "Test Provision device"); | |
141 | + | |
142 | + Assert.assertNotNull(createdDevice); | |
143 | + | |
144 | + DeviceCredentials deviceCredentials = deviceCredentialsService.findDeviceCredentialsByDeviceId(savedTenant.getTenantId(), createdDevice.getId()); | |
145 | + | |
146 | + Assert.assertEquals(deviceCredentials.getCredentialsType().name(), response.get("credentialsType").getAsString()); | |
147 | + Assert.assertEquals(deviceCredentials.getCredentialsType().name(), "X509_CERTIFICATE"); | |
148 | + | |
149 | + String cert = EncryptionUtil.trimNewLines(deviceCredentials.getCredentialsValue()); | |
150 | + String sha3Hash = EncryptionUtil.getSha3Hash(cert); | |
151 | + | |
152 | + Assert.assertEquals(deviceCredentials.getCredentialsId(), sha3Hash); | |
153 | + | |
154 | + Assert.assertEquals(deviceCredentials.getCredentialsValue(), "testHash"); | |
155 | + Assert.assertEquals(ProvisionResponseStatus.SUCCESS.name(), response.get("status").getAsString()); | |
156 | + } | |
157 | + | |
158 | + private void processTestProvisioningCheckPreProvisionedDevice() throws Exception { | |
159 | + super.processBeforeTest("Test Provision device", CoapDeviceType.DEFAULT, TransportPayloadType.JSON, null, null, DeviceProfileProvisionType.CHECK_PRE_PROVISIONED_DEVICES, "testProvisionKey", "testProvisionSecret"); | |
160 | + byte[] result = createCoapClientAndPublish().getPayload(); | |
161 | + JsonObject response = JsonUtils.parse(new String(result)).getAsJsonObject(); | |
162 | + | |
163 | + DeviceCredentials deviceCredentials = deviceCredentialsService.findDeviceCredentialsByDeviceId(savedTenant.getTenantId(), savedDevice.getId()); | |
164 | + | |
165 | + Assert.assertEquals(deviceCredentials.getCredentialsType().name(), response.get("credentialsType").getAsString()); | |
166 | + Assert.assertEquals(ProvisionResponseStatus.SUCCESS.name(), response.get("status").getAsString()); | |
167 | + } | |
168 | + | |
169 | + private void processTestProvisioningWithBadKeyDevice() throws Exception { | |
170 | + super.processBeforeTest("Test Provision device", CoapDeviceType.DEFAULT, TransportPayloadType.JSON, null, null, DeviceProfileProvisionType.CHECK_PRE_PROVISIONED_DEVICES, "testProvisionKeyOrig", "testProvisionSecret"); | |
171 | + byte[] result = createCoapClientAndPublish().getPayload(); | |
172 | + JsonObject response = JsonUtils.parse(new String(result)).getAsJsonObject(); | |
173 | + Assert.assertEquals("Provision data was not found!", response.get("errorMsg").getAsString()); | |
174 | + Assert.assertEquals(ProvisionResponseStatus.NOT_FOUND.name(), response.get("status").getAsString()); | |
175 | + } | |
176 | + | |
177 | + private CoapResponse createCoapClientAndPublish() throws Exception { | |
178 | + return createCoapClientAndPublish(""); | |
179 | + } | |
180 | + | |
181 | + private CoapResponse createCoapClientAndPublish(String deviceCredentials) throws Exception { | |
182 | + String provisionRequestMsg = createTestProvisionMessage(deviceCredentials); | |
183 | + CoapClient client = getCoapClient(FeatureType.PROVISION); | |
184 | + return postProvision(client, provisionRequestMsg.getBytes()); | |
185 | + } | |
186 | + | |
187 | + | |
188 | + private CoapResponse postProvision(CoapClient client, byte[] payload) throws IOException, ConnectorException { | |
189 | + return client.setTimeout((long) 60000).post(payload, MediaTypeRegistry.APPLICATION_JSON); | |
190 | + } | |
191 | + | |
192 | + private String createTestProvisionMessage(String deviceCredentials) { | |
193 | + return "{\"deviceName\":\"Test Provision device\",\"provisionDeviceKey\":\"testProvisionKey\", \"provisionDeviceSecret\":\"testProvisionSecret\"" + deviceCredentials + "}"; | |
194 | + } | |
195 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.provision; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.eclipse.californium.core.CoapClient; | |
20 | +import org.eclipse.californium.core.CoapResponse; | |
21 | +import org.eclipse.californium.core.coap.MediaTypeRegistry; | |
22 | +import org.eclipse.californium.elements.exception.ConnectorException; | |
23 | +import org.junit.After; | |
24 | +import org.junit.Assert; | |
25 | +import org.junit.Test; | |
26 | +import org.springframework.beans.factory.annotation.Autowired; | |
27 | +import org.thingsboard.server.transport.coap.AbstractCoapIntegrationTest; | |
28 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
29 | +import org.thingsboard.server.common.data.Device; | |
30 | +import org.thingsboard.server.common.data.DeviceProfileProvisionType; | |
31 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
32 | +import org.thingsboard.server.common.data.security.DeviceCredentials; | |
33 | +import org.thingsboard.server.common.data.security.DeviceCredentialsType; | |
34 | +import org.thingsboard.server.common.msg.EncryptionUtil; | |
35 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
36 | +import org.thingsboard.server.dao.device.DeviceCredentialsService; | |
37 | +import org.thingsboard.server.dao.device.DeviceService; | |
38 | +import org.thingsboard.server.dao.device.provision.ProvisionResponseStatus; | |
39 | +import org.thingsboard.server.gen.transport.TransportProtos.CredentialsDataProto; | |
40 | +import org.thingsboard.server.gen.transport.TransportProtos.CredentialsType; | |
41 | +import org.thingsboard.server.gen.transport.TransportProtos.ProvisionDeviceCredentialsMsg; | |
42 | +import org.thingsboard.server.gen.transport.TransportProtos.ProvisionDeviceRequestMsg; | |
43 | +import org.thingsboard.server.gen.transport.TransportProtos.ProvisionDeviceResponseMsg; | |
44 | +import org.thingsboard.server.gen.transport.TransportProtos.ValidateDeviceTokenRequestMsg; | |
45 | +import org.thingsboard.server.gen.transport.TransportProtos.ValidateDeviceX509CertRequestMsg; | |
46 | + | |
47 | +import java.io.IOException; | |
48 | + | |
49 | +@Slf4j | |
50 | +public abstract class AbstractCoapProvisionProtoDeviceTest extends AbstractCoapIntegrationTest { | |
51 | + | |
52 | + @Autowired | |
53 | + DeviceCredentialsService deviceCredentialsService; | |
54 | + | |
55 | + @Autowired | |
56 | + DeviceService deviceService; | |
57 | + | |
58 | + @After | |
59 | + public void afterTest() throws Exception { | |
60 | + super.processAfterTest(); | |
61 | + } | |
62 | + | |
63 | + @Test | |
64 | + public void testProvisioningDisabledDevice() throws Exception { | |
65 | + processTestProvisioningDisabledDevice(); | |
66 | + } | |
67 | + | |
68 | + @Test | |
69 | + public void testProvisioningCheckPreProvisionedDevice() throws Exception { | |
70 | + processTestProvisioningCheckPreProvisionedDevice(); | |
71 | + } | |
72 | + | |
73 | + @Test | |
74 | + public void testProvisioningCreateNewDeviceWithoutCredentials() throws Exception { | |
75 | + processTestProvisioningCreateNewDeviceWithoutCredentials(); | |
76 | + } | |
77 | + | |
78 | + @Test | |
79 | + public void testProvisioningCreateNewDeviceWithAccessToken() throws Exception { | |
80 | + processTestProvisioningCreateNewDeviceWithAccessToken(); | |
81 | + } | |
82 | + | |
83 | + @Test | |
84 | + public void testProvisioningCreateNewDeviceWithCert() throws Exception { | |
85 | + processTestProvisioningCreateNewDeviceWithCert(); | |
86 | + } | |
87 | + | |
88 | + @Test | |
89 | + public void testProvisioningWithBadKeyDevice() throws Exception { | |
90 | + processTestProvisioningWithBadKeyDevice(); | |
91 | + } | |
92 | + | |
93 | + | |
94 | + private void processTestProvisioningDisabledDevice() throws Exception { | |
95 | + super.processBeforeTest("Test Provision device", CoapDeviceType.DEFAULT, TransportPayloadType.PROTOBUF, null, null, DeviceProfileProvisionType.DISABLED, null, null); | |
96 | + ProvisionDeviceResponseMsg result = ProvisionDeviceResponseMsg.parseFrom(createCoapClientAndPublish().getPayload()); | |
97 | + Assert.assertNotNull(result); | |
98 | + Assert.assertEquals(ProvisionResponseStatus.NOT_FOUND.name(), result.getStatus().toString()); | |
99 | + } | |
100 | + | |
101 | + private void processTestProvisioningCreateNewDeviceWithoutCredentials() throws Exception { | |
102 | + super.processBeforeTest("Test Provision device3", CoapDeviceType.DEFAULT, TransportPayloadType.PROTOBUF, null, null, DeviceProfileProvisionType.ALLOW_CREATE_NEW_DEVICES, "testProvisionKey", "testProvisionSecret"); | |
103 | + ProvisionDeviceResponseMsg response = ProvisionDeviceResponseMsg.parseFrom(createCoapClientAndPublish().getPayload()); | |
104 | + | |
105 | + Device createdDevice = deviceService.findDeviceByTenantIdAndName(savedTenant.getTenantId(), "Test Provision device"); | |
106 | + | |
107 | + Assert.assertNotNull(createdDevice); | |
108 | + | |
109 | + DeviceCredentials deviceCredentials = deviceCredentialsService.findDeviceCredentialsByDeviceId(savedTenant.getTenantId(), createdDevice.getId()); | |
110 | + | |
111 | + Assert.assertEquals(deviceCredentials.getCredentialsType().name(), response.getCredentialsType().toString()); | |
112 | + Assert.assertEquals(ProvisionResponseStatus.SUCCESS.name(), response.getStatus().toString()); | |
113 | + } | |
114 | + | |
115 | + private void processTestProvisioningCreateNewDeviceWithAccessToken() throws Exception { | |
116 | + super.processBeforeTest("Test Provision device3", CoapDeviceType.DEFAULT, TransportPayloadType.PROTOBUF, null, null, DeviceProfileProvisionType.ALLOW_CREATE_NEW_DEVICES, "testProvisionKey", "testProvisionSecret"); | |
117 | + CredentialsDataProto requestCredentials = CredentialsDataProto.newBuilder().setValidateDeviceTokenRequestMsg(ValidateDeviceTokenRequestMsg.newBuilder().setToken("test_token").build()).build(); | |
118 | + | |
119 | + ProvisionDeviceResponseMsg response = ProvisionDeviceResponseMsg.parseFrom(createCoapClientAndPublish(createTestsProvisionMessage(CredentialsType.ACCESS_TOKEN, requestCredentials)).getPayload()); | |
120 | + | |
121 | + Device createdDevice = deviceService.findDeviceByTenantIdAndName(savedTenant.getTenantId(), "Test Provision device"); | |
122 | + | |
123 | + Assert.assertNotNull(createdDevice); | |
124 | + | |
125 | + DeviceCredentials deviceCredentials = deviceCredentialsService.findDeviceCredentialsByDeviceId(savedTenant.getTenantId(), createdDevice.getId()); | |
126 | + | |
127 | + Assert.assertEquals(deviceCredentials.getCredentialsType().name(), response.getCredentialsType().toString()); | |
128 | + Assert.assertEquals(deviceCredentials.getCredentialsType(), DeviceCredentialsType.ACCESS_TOKEN); | |
129 | + Assert.assertEquals(deviceCredentials.getCredentialsId(), "test_token"); | |
130 | + Assert.assertEquals(ProvisionResponseStatus.SUCCESS.name(), response.getStatus().toString()); | |
131 | + } | |
132 | + | |
133 | + private void processTestProvisioningCreateNewDeviceWithCert() throws Exception { | |
134 | + super.processBeforeTest("Test Provision device3", CoapDeviceType.DEFAULT, TransportPayloadType.PROTOBUF, null, null, DeviceProfileProvisionType.ALLOW_CREATE_NEW_DEVICES, "testProvisionKey", "testProvisionSecret"); | |
135 | + CredentialsDataProto requestCredentials = CredentialsDataProto.newBuilder().setValidateDeviceX509CertRequestMsg(ValidateDeviceX509CertRequestMsg.newBuilder().setHash("testHash").build()).build(); | |
136 | + | |
137 | + ProvisionDeviceResponseMsg response = ProvisionDeviceResponseMsg.parseFrom(createCoapClientAndPublish(createTestsProvisionMessage(CredentialsType.X509_CERTIFICATE, requestCredentials)).getPayload()); | |
138 | + | |
139 | + Device createdDevice = deviceService.findDeviceByTenantIdAndName(savedTenant.getTenantId(), "Test Provision device"); | |
140 | + | |
141 | + Assert.assertNotNull(createdDevice); | |
142 | + | |
143 | + DeviceCredentials deviceCredentials = deviceCredentialsService.findDeviceCredentialsByDeviceId(savedTenant.getTenantId(), createdDevice.getId()); | |
144 | + | |
145 | + Assert.assertEquals(deviceCredentials.getCredentialsType().name(), response.getCredentialsType().toString()); | |
146 | + Assert.assertEquals(deviceCredentials.getCredentialsType(), DeviceCredentialsType.X509_CERTIFICATE); | |
147 | + | |
148 | + String cert = EncryptionUtil.trimNewLines(deviceCredentials.getCredentialsValue()); | |
149 | + String sha3Hash = EncryptionUtil.getSha3Hash(cert); | |
150 | + | |
151 | + Assert.assertEquals(deviceCredentials.getCredentialsId(), sha3Hash); | |
152 | + | |
153 | + Assert.assertEquals(deviceCredentials.getCredentialsValue(), "testHash"); | |
154 | + Assert.assertEquals(ProvisionResponseStatus.SUCCESS.name(), response.getStatus().toString()); | |
155 | + } | |
156 | + | |
157 | + private void processTestProvisioningCheckPreProvisionedDevice() throws Exception { | |
158 | + super.processBeforeTest("Test Provision device", CoapDeviceType.DEFAULT, TransportPayloadType.PROTOBUF, null, null, DeviceProfileProvisionType.CHECK_PRE_PROVISIONED_DEVICES, "testProvisionKey", "testProvisionSecret"); | |
159 | + ProvisionDeviceResponseMsg response = ProvisionDeviceResponseMsg.parseFrom(createCoapClientAndPublish().getPayload()); | |
160 | + | |
161 | + DeviceCredentials deviceCredentials = deviceCredentialsService.findDeviceCredentialsByDeviceId(savedTenant.getTenantId(), savedDevice.getId()); | |
162 | + | |
163 | + Assert.assertEquals(deviceCredentials.getCredentialsType().name(), response.getCredentialsType().toString()); | |
164 | + Assert.assertEquals(ProvisionResponseStatus.SUCCESS.name(), response.getStatus().toString()); | |
165 | + } | |
166 | + | |
167 | + private void processTestProvisioningWithBadKeyDevice() throws Exception { | |
168 | + super.processBeforeTest("Test Provision device", CoapDeviceType.DEFAULT, TransportPayloadType.PROTOBUF, null, null, DeviceProfileProvisionType.CHECK_PRE_PROVISIONED_DEVICES, "testProvisionKeyOrig", "testProvisionSecret"); | |
169 | + ProvisionDeviceResponseMsg response = ProvisionDeviceResponseMsg.parseFrom(createCoapClientAndPublish().getPayload()); | |
170 | + Assert.assertEquals(ProvisionResponseStatus.NOT_FOUND.name(), response.getStatus().toString()); | |
171 | + } | |
172 | + | |
173 | + private CoapResponse createCoapClientAndPublish() throws Exception { | |
174 | + byte[] provisionRequestMsg = createTestProvisionMessage(); | |
175 | + return createCoapClientAndPublish(provisionRequestMsg); | |
176 | + } | |
177 | + | |
178 | + private CoapResponse createCoapClientAndPublish(byte[] provisionRequestMsg) throws Exception { | |
179 | + CoapClient client = getCoapClient(FeatureType.PROVISION); | |
180 | + return postProvision(client, provisionRequestMsg); | |
181 | + } | |
182 | + | |
183 | + private CoapResponse postProvision(CoapClient client, byte[] payload) throws IOException, ConnectorException { | |
184 | + return client.setTimeout((long) 60000).post(payload, MediaTypeRegistry.APPLICATION_JSON); | |
185 | + } | |
186 | + | |
187 | + private byte[] createTestsProvisionMessage(CredentialsType credentialsType, CredentialsDataProto credentialsData) throws Exception { | |
188 | + return ProvisionDeviceRequestMsg.newBuilder() | |
189 | + .setDeviceName("Test Provision device") | |
190 | + .setCredentialsType(credentialsType != null ? credentialsType : CredentialsType.ACCESS_TOKEN) | |
191 | + .setCredentialsDataProto(credentialsData != null ? credentialsData: CredentialsDataProto.newBuilder().build()) | |
192 | + .setProvisionDeviceCredentialsMsg( | |
193 | + ProvisionDeviceCredentialsMsg.newBuilder() | |
194 | + .setProvisionDeviceKey("testProvisionKey") | |
195 | + .setProvisionDeviceSecret("testProvisionSecret") | |
196 | + ).build() | |
197 | + .toByteArray(); | |
198 | + } | |
199 | + | |
200 | + | |
201 | + private byte[] createTestProvisionMessage() throws Exception { | |
202 | + return createTestsProvisionMessage(null, null); | |
203 | + } | |
204 | + | |
205 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.provision.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.provision.AbstractCoapProvisionJsonDeviceTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapProvisionDeviceJsonSqlTest extends AbstractCoapProvisionJsonDeviceTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.provision.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.provision.AbstractCoapProvisionProtoDeviceTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapProvisionDeviceProtoSqlTest extends AbstractCoapProvisionProtoDeviceTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.rpc; | |
17 | + | |
18 | +import com.datastax.oss.driver.api.core.uuid.Uuids; | |
19 | +import lombok.extern.slf4j.Slf4j; | |
20 | +import org.junit.After; | |
21 | +import org.junit.Assert; | |
22 | +import org.junit.Before; | |
23 | +import org.junit.Test; | |
24 | +import org.thingsboard.server.service.security.AccessValidator; | |
25 | + | |
26 | +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; | |
27 | + | |
28 | +@Slf4j | |
29 | +public abstract class AbstractCoapServerSideRpcDefaultIntegrationTest extends AbstractCoapServerSideRpcIntegrationTest { | |
30 | + | |
31 | + @Before | |
32 | + public void beforeTest() throws Exception { | |
33 | + processBeforeTest("RPC test device", null, null); | |
34 | + } | |
35 | + | |
36 | + @After | |
37 | + public void afterTest() throws Exception { | |
38 | + super.processAfterTest(); | |
39 | + } | |
40 | + | |
41 | + @Test | |
42 | + public void testServerCoapOneWayRpcDeviceOffline() throws Exception { | |
43 | + String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"24\",\"value\": 1},\"timeout\": 6000}"; | |
44 | + String deviceId = savedDevice.getId().getId().toString(); | |
45 | + | |
46 | + doPostAsync("/api/plugins/rpc/oneway/" + deviceId, setGpioRequest, String.class, status().is(409), | |
47 | + asyncContextTimeoutToUseRpcPlugin); | |
48 | + } | |
49 | + | |
50 | + @Test | |
51 | + public void testServerCoapOneWayRpcDeviceDoesNotExist() throws Exception { | |
52 | + String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"25\",\"value\": 1}}"; | |
53 | + String nonExistentDeviceId = Uuids.timeBased().toString(); | |
54 | + | |
55 | + String result = doPostAsync("/api/plugins/rpc/oneway/" + nonExistentDeviceId, setGpioRequest, String.class, | |
56 | + status().isNotFound()); | |
57 | + Assert.assertEquals(AccessValidator.DEVICE_WITH_REQUESTED_ID_NOT_FOUND, result); | |
58 | + } | |
59 | + | |
60 | + @Test | |
61 | + public void testServerCoapTwoWayRpcDeviceOffline() throws Exception { | |
62 | + String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"27\",\"value\": 1},\"timeout\": 6000}"; | |
63 | + String deviceId = savedDevice.getId().getId().toString(); | |
64 | + | |
65 | + doPostAsync("/api/plugins/rpc/twoway/" + deviceId, setGpioRequest, String.class, status().is(409), | |
66 | + asyncContextTimeoutToUseRpcPlugin); | |
67 | + } | |
68 | + | |
69 | + @Test | |
70 | + public void testServerCoapTwoWayRpcDeviceDoesNotExist() throws Exception { | |
71 | + String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"28\",\"value\": 1}}"; | |
72 | + String nonExistentDeviceId = Uuids.timeBased().toString(); | |
73 | + | |
74 | + String result = doPostAsync("/api/plugins/rpc/twoway/" + nonExistentDeviceId, setGpioRequest, String.class, | |
75 | + status().isNotFound()); | |
76 | + Assert.assertEquals(AccessValidator.DEVICE_WITH_REQUESTED_ID_NOT_FOUND, result); | |
77 | + } | |
78 | + | |
79 | + @Test | |
80 | + public void testServerCoapOneWayRpc() throws Exception { | |
81 | + processOneWayRpcTest(); | |
82 | + } | |
83 | + | |
84 | + @Test | |
85 | + public void testServerCoapTwoWayRpc() throws Exception { | |
86 | + processTwoWayRpcTest(); | |
87 | + } | |
88 | + | |
89 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.rpc; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.apache.commons.lang3.StringUtils; | |
20 | +import org.eclipse.californium.core.CoapClient; | |
21 | +import org.eclipse.californium.core.CoapHandler; | |
22 | +import org.eclipse.californium.core.CoapObserveRelation; | |
23 | +import org.eclipse.californium.core.CoapResponse; | |
24 | +import org.eclipse.californium.core.coap.CoAP; | |
25 | +import org.eclipse.californium.core.coap.MediaTypeRegistry; | |
26 | +import org.eclipse.californium.core.coap.Request; | |
27 | +import org.junit.Assert; | |
28 | +import org.thingsboard.server.transport.coap.AbstractCoapIntegrationTest; | |
29 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
30 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
31 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
32 | + | |
33 | +import java.util.concurrent.CountDownLatch; | |
34 | +import java.util.concurrent.TimeUnit; | |
35 | + | |
36 | +import static org.junit.Assert.assertEquals; | |
37 | +import static org.junit.Assert.assertNotNull; | |
38 | +import static org.junit.Assert.assertTrue; | |
39 | +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; | |
40 | + | |
41 | +@Slf4j | |
42 | +public abstract class AbstractCoapServerSideRpcIntegrationTest extends AbstractCoapIntegrationTest { | |
43 | + | |
44 | + protected static final String DEVICE_RESPONSE = "{\"value1\":\"A\",\"value2\":\"B\"}"; | |
45 | + | |
46 | + protected Long asyncContextTimeoutToUseRpcPlugin; | |
47 | + | |
48 | + protected void processBeforeTest(String deviceName, CoapDeviceType coapDeviceType, TransportPayloadType payloadType) throws Exception { | |
49 | + super.processBeforeTest(deviceName, coapDeviceType, payloadType); | |
50 | + asyncContextTimeoutToUseRpcPlugin = 10000L; | |
51 | + } | |
52 | + | |
53 | + protected void processOneWayRpcTest() throws Exception { | |
54 | + CoapClient client = getCoapClient(FeatureType.RPC); | |
55 | + client.useCONs(); | |
56 | + | |
57 | + CountDownLatch latch = new CountDownLatch(1); | |
58 | + TestCoapCallback testCoapCallback = new TestCoapCallback(client, latch, true); | |
59 | + | |
60 | + Request request = Request.newGet().setObserve(); | |
61 | + CoapObserveRelation observeRelation = client.observe(request, testCoapCallback); | |
62 | + | |
63 | + String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"23\",\"value\": 1}}"; | |
64 | + String deviceId = savedDevice.getId().getId().toString(); | |
65 | + String result = doPostAsync("/api/plugins/rpc/oneway/" + deviceId, setGpioRequest, String.class, status().isOk()); | |
66 | + Assert.assertTrue(StringUtils.isEmpty(result)); | |
67 | + latch.await(3, TimeUnit.SECONDS); | |
68 | + assertEquals(0, testCoapCallback.getObserve().intValue()); | |
69 | + observeRelation.proactiveCancel(); | |
70 | + assertTrue(observeRelation.isCanceled()); | |
71 | + } | |
72 | + | |
73 | + protected void processTwoWayRpcTest() throws Exception { | |
74 | + CoapClient client = getCoapClient(FeatureType.RPC); | |
75 | + client.useCONs(); | |
76 | + | |
77 | + CountDownLatch latch = new CountDownLatch(1); | |
78 | + TestCoapCallback testCoapCallback = new TestCoapCallback(client, latch, false); | |
79 | + | |
80 | + Request request = Request.newGet().setObserve(); | |
81 | + request.setType(CoAP.Type.CON); | |
82 | + CoapObserveRelation observeRelation = client.observe(request, testCoapCallback); | |
83 | + | |
84 | + String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"26\",\"value\": 1}}"; | |
85 | + String deviceId = savedDevice.getId().getId().toString(); | |
86 | + | |
87 | + String expected = "{\"value1\":\"A\",\"value2\":\"B\"}"; | |
88 | + | |
89 | + String result = doPostAsync("/api/plugins/rpc/twoway/" + deviceId, setGpioRequest, String.class, status().isOk()); | |
90 | + latch.await(3, TimeUnit.SECONDS); | |
91 | + | |
92 | + assertEquals(expected, result); | |
93 | + assertEquals(0, testCoapCallback.getObserve().intValue()); | |
94 | + observeRelation.proactiveCancel(); | |
95 | + assertTrue(observeRelation.isCanceled()); | |
96 | + | |
97 | +// // TODO: 3/11/21 Fix test to validate next RPC | |
98 | +// latch = new CountDownLatch(1); | |
99 | +// | |
100 | +// result = doPostAsync("/api/plugins/rpc/twoway/" + deviceId, setGpioRequest, String.class, status().isOk()); | |
101 | +// latch.await(3, TimeUnit.SECONDS); | |
102 | +// | |
103 | +// assertEquals(expected, result); | |
104 | +// assertEquals(1, testCoapCallback.getObserve().intValue()); | |
105 | + } | |
106 | + | |
107 | + protected void processOnLoadResponse(CoapResponse response, CoapClient client, Integer observe, CountDownLatch latch) { | |
108 | + client.setURI(getRpcResponseFeatureTokenUrl(accessToken, observe)); | |
109 | + client.post(new CoapHandler() { | |
110 | + @Override | |
111 | + public void onLoad(CoapResponse response) { | |
112 | + log.warn("Command Response Ack: {}, {}", response.getCode(), response.getResponseText()); | |
113 | + latch.countDown(); | |
114 | + } | |
115 | + | |
116 | + @Override | |
117 | + public void onError() { | |
118 | + log.warn("Command Response Ack Error, No connect"); | |
119 | + } | |
120 | + }, DEVICE_RESPONSE, MediaTypeRegistry.APPLICATION_JSON); | |
121 | + } | |
122 | + | |
123 | + protected String getRpcResponseFeatureTokenUrl(String token, int requestId) { | |
124 | + return COAP_BASE_URL + token + "/" + FeatureType.RPC.name().toLowerCase() + "/" + requestId; | |
125 | + } | |
126 | + | |
127 | + private class TestCoapCallback implements CoapHandler { | |
128 | + | |
129 | + private final CoapClient client; | |
130 | + private final CountDownLatch latch; | |
131 | + private final boolean isOneWayRpc; | |
132 | + | |
133 | + public Integer getObserve() { | |
134 | + return observe; | |
135 | + } | |
136 | + | |
137 | + private Integer observe; | |
138 | + | |
139 | + private TestCoapCallback(CoapClient client, CountDownLatch latch, boolean isOneWayRpc) { | |
140 | + this.client = client; | |
141 | + this.latch = latch; | |
142 | + this.isOneWayRpc = isOneWayRpc; | |
143 | + } | |
144 | + | |
145 | + @Override | |
146 | + public void onLoad(CoapResponse response) { | |
147 | + log.warn("coap response: {}, {}", response, response.getCode()); | |
148 | + assertNotNull(response.getPayload()); | |
149 | + assertEquals(response.getCode(), CoAP.ResponseCode.CONTENT); | |
150 | + observe = response.getOptions().getObserve(); | |
151 | + if (!isOneWayRpc) { | |
152 | + processOnLoadResponse(response, client, observe, latch); | |
153 | + } else { | |
154 | + latch.countDown(); | |
155 | + } | |
156 | + } | |
157 | + | |
158 | + @Override | |
159 | + public void onError() { | |
160 | + log.warn("Command Response Ack Error, No connect"); | |
161 | + } | |
162 | + | |
163 | + } | |
164 | + | |
165 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.rpc; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.junit.After; | |
20 | +import org.junit.Before; | |
21 | +import org.junit.Test; | |
22 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
23 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
24 | + | |
25 | +@Slf4j | |
26 | +public abstract class AbstractCoapServerSideRpcJsonIntegrationTest extends AbstractCoapServerSideRpcIntegrationTest { | |
27 | + | |
28 | + @Before | |
29 | + public void beforeTest() throws Exception { | |
30 | + processBeforeTest("RPC test device", CoapDeviceType.DEFAULT, TransportPayloadType.JSON); | |
31 | + } | |
32 | + | |
33 | + @After | |
34 | + public void afterTest() throws Exception { | |
35 | + super.processAfterTest(); | |
36 | + } | |
37 | + | |
38 | + @Test | |
39 | + public void testServerMqttOneWayRpc() throws Exception { | |
40 | + processOneWayRpcTest(); | |
41 | + } | |
42 | + | |
43 | + @Test | |
44 | + public void testServerMqttTwoWayRpc() throws Exception { | |
45 | + processTwoWayRpcTest(); | |
46 | + } | |
47 | + | |
48 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.rpc; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.eclipse.californium.core.CoapClient; | |
20 | +import org.eclipse.californium.core.CoapHandler; | |
21 | +import org.eclipse.californium.core.CoapResponse; | |
22 | +import org.eclipse.californium.core.coap.MediaTypeRegistry; | |
23 | +import org.junit.After; | |
24 | +import org.junit.Before; | |
25 | +import org.junit.Test; | |
26 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
27 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
28 | +import org.thingsboard.server.gen.transport.TransportProtos; | |
29 | + | |
30 | +import java.util.concurrent.CountDownLatch; | |
31 | + | |
32 | +@Slf4j | |
33 | +public abstract class AbstractCoapServerSideRpcProtoIntegrationTest extends AbstractCoapServerSideRpcIntegrationTest { | |
34 | + | |
35 | + @Before | |
36 | + public void beforeTest() throws Exception { | |
37 | + processBeforeTest("RPC test device", CoapDeviceType.DEFAULT, TransportPayloadType.PROTOBUF); | |
38 | + } | |
39 | + | |
40 | + @After | |
41 | + public void afterTest() throws Exception { | |
42 | + super.processAfterTest(); | |
43 | + } | |
44 | + | |
45 | + @Test | |
46 | + public void testServerMqttOneWayRpc() throws Exception { | |
47 | + processOneWayRpcTest(); | |
48 | + } | |
49 | + | |
50 | + @Test | |
51 | + public void testServerMqttTwoWayRpc() throws Exception { | |
52 | + processTwoWayRpcTest(); | |
53 | + } | |
54 | + | |
55 | + @Override | |
56 | + protected void processOnLoadResponse(CoapResponse response, CoapClient client, Integer observe, CountDownLatch latch) { | |
57 | + client.setURI(getRpcResponseFeatureTokenUrl(accessToken, observe)); | |
58 | + TransportProtos.ToDeviceRpcResponseMsg toDeviceRpcResponseMsg = TransportProtos.ToDeviceRpcResponseMsg.newBuilder() | |
59 | + .setPayload(DEVICE_RESPONSE) | |
60 | + .setRequestId(observe) | |
61 | + .build(); | |
62 | + client.post(new CoapHandler() { | |
63 | + @Override | |
64 | + public void onLoad(CoapResponse response) { | |
65 | + log.warn("Command Response Ack: {}, {}", response.getCode(), response.getResponseText()); | |
66 | + latch.countDown(); | |
67 | + } | |
68 | + | |
69 | + @Override | |
70 | + public void onError() { | |
71 | + log.warn("Command Response Ack Error, No connect"); | |
72 | + } | |
73 | + }, toDeviceRpcResponseMsg.toByteArray(), MediaTypeRegistry.APPLICATION_JSON); | |
74 | + } | |
75 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.rpc.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.rpc.AbstractCoapServerSideRpcJsonIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapServerSideRpcJsonSqlIntegrationTest extends AbstractCoapServerSideRpcJsonIntegrationTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.rpc.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.rpc.AbstractCoapServerSideRpcProtoIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | + | |
22 | +@DaoSqlTest | |
23 | +public class CoapServerSideRpcProtoSqlIntegrationTest extends AbstractCoapServerSideRpcProtoIntegrationTest { | |
24 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.rpc.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.rpc.AbstractCoapServerSideRpcDefaultIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +@DaoSqlTest | |
22 | +public class CoapServerSideRpcSqlIntegrationTest extends AbstractCoapServerSideRpcDefaultIntegrationTest { | |
23 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.telemetry.attributes; | |
17 | + | |
18 | +import com.fasterxml.jackson.core.JsonProcessingException; | |
19 | +import com.fasterxml.jackson.core.type.TypeReference; | |
20 | +import lombok.extern.slf4j.Slf4j; | |
21 | +import org.eclipse.californium.core.CoapClient; | |
22 | +import org.eclipse.californium.core.CoapResponse; | |
23 | +import org.eclipse.californium.core.coap.CoAP; | |
24 | +import org.eclipse.californium.core.coap.MediaTypeRegistry; | |
25 | +import org.eclipse.californium.elements.exception.ConnectorException; | |
26 | +import org.junit.After; | |
27 | +import org.junit.Before; | |
28 | +import org.junit.Test; | |
29 | +import org.thingsboard.server.transport.coap.AbstractCoapIntegrationTest; | |
30 | +import org.thingsboard.server.common.data.id.DeviceId; | |
31 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
32 | + | |
33 | +import java.io.IOException; | |
34 | +import java.util.Arrays; | |
35 | +import java.util.HashSet; | |
36 | +import java.util.LinkedHashMap; | |
37 | +import java.util.List; | |
38 | +import java.util.Map; | |
39 | +import java.util.Set; | |
40 | + | |
41 | +import static org.junit.Assert.assertEquals; | |
42 | +import static org.junit.Assert.assertNotNull; | |
43 | +import static org.junit.Assert.assertTrue; | |
44 | + | |
45 | +@Slf4j | |
46 | +public abstract class AbstractCoapAttributesIntegrationTest extends AbstractCoapIntegrationTest { | |
47 | + | |
48 | + private static final String PAYLOAD_VALUES_STR = "{\"key1\":\"value1\", \"key2\":true, \"key3\": 3.0, \"key4\": 4," + | |
49 | + " \"key5\": {\"someNumber\": 42, \"someArray\": [1,2,3], \"someNestedObject\": {\"key\": \"value\"}}}"; | |
50 | + | |
51 | + @Before | |
52 | + public void beforeTest() throws Exception { | |
53 | + processBeforeTest("Test Post Attributes device", null, null); | |
54 | + } | |
55 | + | |
56 | + @After | |
57 | + public void afterTest() throws Exception { | |
58 | + processAfterTest(); | |
59 | + } | |
60 | + | |
61 | + @Test | |
62 | + public void testPushAttributes() throws Exception { | |
63 | + List<String> expectedKeys = Arrays.asList("key1", "key2", "key3", "key4", "key5"); | |
64 | + processAttributesTest(expectedKeys, PAYLOAD_VALUES_STR.getBytes()); | |
65 | + } | |
66 | + | |
67 | + protected void processAttributesTest(List<String> expectedKeys, byte[] payload) throws Exception { | |
68 | + log.warn("[testPushAttributes] Device: {}, Transport type: {}", savedDevice.getName(), savedDevice.getType()); | |
69 | + CoapClient client = getCoapClient(FeatureType.ATTRIBUTES); | |
70 | + | |
71 | + postAttributes(client, payload); | |
72 | + | |
73 | + DeviceId deviceId = savedDevice.getId(); | |
74 | + | |
75 | + long start = System.currentTimeMillis(); | |
76 | + long end = System.currentTimeMillis() + 5000; | |
77 | + | |
78 | + List<String> actualKeys = null; | |
79 | + while (start <= end) { | |
80 | + actualKeys = doGetAsyncTyped("/api/plugins/telemetry/DEVICE/" + deviceId + "/keys/attributes/CLIENT_SCOPE", new TypeReference<>() {}); | |
81 | + if (actualKeys.size() == expectedKeys.size()) { | |
82 | + break; | |
83 | + } | |
84 | + Thread.sleep(100); | |
85 | + start += 100; | |
86 | + } | |
87 | + assertNotNull(actualKeys); | |
88 | + | |
89 | + Set<String> actualKeySet = new HashSet<>(actualKeys); | |
90 | + | |
91 | + Set<String> expectedKeySet = new HashSet<>(expectedKeys); | |
92 | + | |
93 | + assertEquals(expectedKeySet, actualKeySet); | |
94 | + | |
95 | + String getAttributesValuesUrl = getAttributesValuesUrl(deviceId, actualKeySet); | |
96 | + List<Map<String, Object>> values = doGetAsyncTyped(getAttributesValuesUrl, new TypeReference<>() {}); | |
97 | + assertAttributesValues(values, expectedKeySet); | |
98 | + String deleteAttributesUrl = "/api/plugins/telemetry/DEVICE/" + deviceId + "/CLIENT_SCOPE?keys=" + String.join(",", actualKeySet); | |
99 | + doDelete(deleteAttributesUrl); | |
100 | + } | |
101 | + | |
102 | + private void postAttributes(CoapClient client, byte[] payload) throws IOException, ConnectorException { | |
103 | + if (payload == null) { | |
104 | + payload = PAYLOAD_VALUES_STR.getBytes(); | |
105 | + } | |
106 | + CoapResponse coapResponse = client.setTimeout((long) 60000).post(payload, MediaTypeRegistry.APPLICATION_JSON); | |
107 | + assertEquals(CoAP.ResponseCode.CREATED, coapResponse.getCode()); | |
108 | + } | |
109 | + | |
110 | + @SuppressWarnings("unchecked") | |
111 | + protected void assertAttributesValues(List<Map<String, Object>> deviceValues, Set<String> expectedKeySet) throws JsonProcessingException { | |
112 | + for (Map<String, Object> map : deviceValues) { | |
113 | + String key = (String) map.get("key"); | |
114 | + Object value = map.get("value"); | |
115 | + assertTrue(expectedKeySet.contains(key)); | |
116 | + switch (key) { | |
117 | + case "key1": | |
118 | + assertEquals("value1", value); | |
119 | + break; | |
120 | + case "key2": | |
121 | + assertEquals(true, value); | |
122 | + break; | |
123 | + case "key3": | |
124 | + assertEquals(3.0, value); | |
125 | + break; | |
126 | + case "key4": | |
127 | + assertEquals(4, value); | |
128 | + break; | |
129 | + case "key5": | |
130 | + assertNotNull(value); | |
131 | + assertEquals(3, ((LinkedHashMap) value).size()); | |
132 | + assertEquals(42, ((LinkedHashMap) value).get("someNumber")); | |
133 | + assertEquals(Arrays.asList(1, 2, 3), ((LinkedHashMap) value).get("someArray")); | |
134 | + LinkedHashMap<String, String> someNestedObject = (LinkedHashMap) ((LinkedHashMap) value).get("someNestedObject"); | |
135 | + assertEquals("value", someNestedObject.get("key")); | |
136 | + break; | |
137 | + } | |
138 | + } | |
139 | + } | |
140 | + | |
141 | + private String getAttributesValuesUrl(DeviceId deviceId, Set<String> actualKeySet) { | |
142 | + return "/api/plugins/telemetry/DEVICE/" + deviceId + "/values/attributes/CLIENT_SCOPE?keys=" + String.join(",", actualKeySet); | |
143 | + } | |
144 | + | |
145 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.telemetry.attributes; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.junit.After; | |
20 | +import org.junit.Before; | |
21 | +import org.junit.Test; | |
22 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
23 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
24 | + | |
25 | +@Slf4j | |
26 | +public abstract class AbstractCoapAttributesJsonIntegrationTest extends AbstractCoapAttributesIntegrationTest { | |
27 | + | |
28 | + @Before | |
29 | + public void beforeTest() throws Exception { | |
30 | + processBeforeTest("Test Post Attributes device Json", CoapDeviceType.DEFAULT, TransportPayloadType.JSON); | |
31 | + } | |
32 | + | |
33 | + @After | |
34 | + public void afterTest() throws Exception { | |
35 | + processAfterTest(); | |
36 | + } | |
37 | + | |
38 | + @Test | |
39 | + public void testPushAttributes() throws Exception { | |
40 | + super.testPushAttributes(); | |
41 | + } | |
42 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.telemetry.attributes; | |
17 | + | |
18 | +import com.github.os72.protobuf.dynamic.DynamicSchema; | |
19 | +import com.google.protobuf.Descriptors; | |
20 | +import com.google.protobuf.DynamicMessage; | |
21 | +import com.squareup.wire.schema.internal.parser.ProtoFileElement; | |
22 | +import lombok.extern.slf4j.Slf4j; | |
23 | +import org.junit.After; | |
24 | +import org.junit.Test; | |
25 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
26 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
27 | +import org.thingsboard.server.common.data.device.profile.CoapDeviceProfileTransportConfiguration; | |
28 | +import org.thingsboard.server.common.data.device.profile.CoapDeviceTypeConfiguration; | |
29 | +import org.thingsboard.server.common.data.device.profile.DefaultCoapDeviceTypeConfiguration; | |
30 | +import org.thingsboard.server.common.data.device.profile.DeviceProfileTransportConfiguration; | |
31 | +import org.thingsboard.server.common.data.device.profile.ProtoTransportPayloadConfiguration; | |
32 | +import org.thingsboard.server.common.data.device.profile.TransportPayloadTypeConfiguration; | |
33 | + | |
34 | +import java.util.Arrays; | |
35 | +import java.util.List; | |
36 | + | |
37 | +import static org.junit.Assert.assertNotNull; | |
38 | +import static org.junit.Assert.assertTrue; | |
39 | + | |
40 | +@Slf4j | |
41 | +public abstract class AbstractCoapAttributesProtoIntegrationTest extends AbstractCoapAttributesIntegrationTest { | |
42 | + | |
43 | + @After | |
44 | + public void afterTest() throws Exception { | |
45 | + processAfterTest(); | |
46 | + } | |
47 | + | |
48 | + @Test | |
49 | + public void testPushAttributes() throws Exception { | |
50 | + super.processBeforeTest("Test Post Attributes device Proto", CoapDeviceType.DEFAULT, TransportPayloadType.PROTOBUF); | |
51 | + List<String> expectedKeys = Arrays.asList("key1", "key2", "key3", "key4", "key5"); | |
52 | + DeviceProfileTransportConfiguration transportConfiguration = deviceProfile.getProfileData().getTransportConfiguration(); | |
53 | + assertTrue(transportConfiguration instanceof CoapDeviceProfileTransportConfiguration); | |
54 | + CoapDeviceProfileTransportConfiguration coapTransportConfiguration = (CoapDeviceProfileTransportConfiguration) transportConfiguration; | |
55 | + CoapDeviceTypeConfiguration coapDeviceTypeConfiguration = coapTransportConfiguration.getCoapDeviceTypeConfiguration(); | |
56 | + assertTrue(coapDeviceTypeConfiguration instanceof DefaultCoapDeviceTypeConfiguration); | |
57 | + DefaultCoapDeviceTypeConfiguration defaultCoapDeviceTypeConfiguration = (DefaultCoapDeviceTypeConfiguration) coapDeviceTypeConfiguration; | |
58 | + TransportPayloadTypeConfiguration transportPayloadTypeConfiguration = defaultCoapDeviceTypeConfiguration.getTransportPayloadTypeConfiguration(); | |
59 | + assertTrue(transportPayloadTypeConfiguration instanceof ProtoTransportPayloadConfiguration); | |
60 | + ProtoTransportPayloadConfiguration protoTransportPayloadConfiguration = (ProtoTransportPayloadConfiguration) transportPayloadTypeConfiguration; | |
61 | + ProtoFileElement transportProtoSchemaFile = protoTransportPayloadConfiguration.getTransportProtoSchema(DEVICE_ATTRIBUTES_PROTO_SCHEMA); | |
62 | + DynamicSchema attributesSchema = protoTransportPayloadConfiguration.getDynamicSchema(transportProtoSchemaFile, ProtoTransportPayloadConfiguration.ATTRIBUTES_PROTO_SCHEMA); | |
63 | + | |
64 | + DynamicMessage.Builder nestedJsonObjectBuilder = attributesSchema.newMessageBuilder("PostAttributes.JsonObject.NestedJsonObject"); | |
65 | + Descriptors.Descriptor nestedJsonObjectBuilderDescriptor = nestedJsonObjectBuilder.getDescriptorForType(); | |
66 | + assertNotNull(nestedJsonObjectBuilderDescriptor); | |
67 | + DynamicMessage nestedJsonObject = nestedJsonObjectBuilder.setField(nestedJsonObjectBuilderDescriptor.findFieldByName("key"), "value").build(); | |
68 | + | |
69 | + DynamicMessage.Builder jsonObjectBuilder = attributesSchema.newMessageBuilder("PostAttributes.JsonObject"); | |
70 | + Descriptors.Descriptor jsonObjectBuilderDescriptor = jsonObjectBuilder.getDescriptorForType(); | |
71 | + assertNotNull(jsonObjectBuilderDescriptor); | |
72 | + DynamicMessage jsonObject = jsonObjectBuilder | |
73 | + .setField(jsonObjectBuilderDescriptor.findFieldByName("someNumber"), 42) | |
74 | + .addRepeatedField(jsonObjectBuilderDescriptor.findFieldByName("someArray"), 1) | |
75 | + .addRepeatedField(jsonObjectBuilderDescriptor.findFieldByName("someArray"), 2) | |
76 | + .addRepeatedField(jsonObjectBuilderDescriptor.findFieldByName("someArray"), 3) | |
77 | + .setField(jsonObjectBuilderDescriptor.findFieldByName("someNestedObject"), nestedJsonObject) | |
78 | + .build(); | |
79 | + | |
80 | + DynamicMessage.Builder postAttributesBuilder = attributesSchema.newMessageBuilder("PostAttributes"); | |
81 | + Descriptors.Descriptor postAttributesMsgDescriptor = postAttributesBuilder.getDescriptorForType(); | |
82 | + assertNotNull(postAttributesMsgDescriptor); | |
83 | + DynamicMessage postAttributesMsg = postAttributesBuilder | |
84 | + .setField(postAttributesMsgDescriptor.findFieldByName("key1"), "value1") | |
85 | + .setField(postAttributesMsgDescriptor.findFieldByName("key2"), true) | |
86 | + .setField(postAttributesMsgDescriptor.findFieldByName("key3"), 3.0) | |
87 | + .setField(postAttributesMsgDescriptor.findFieldByName("key4"), 4) | |
88 | + .setField(postAttributesMsgDescriptor.findFieldByName("key5"), jsonObject) | |
89 | + .build(); | |
90 | + processAttributesTest(expectedKeys, postAttributesMsg.toByteArray()); | |
91 | + } | |
92 | + | |
93 | +} | ... | ... |
application/src/test/java/org/thingsboard/server/transport/coap/telemetry/attributes/sql/CoapAttributesSqlIntegrationTest.java
renamed from
application/src/test/java/org/thingsboard/server/mqtt/rpc/nosql/MqttServerSideRpcNoSqlIntegrationTest.java
... | ... | @@ -13,14 +13,14 @@ |
13 | 13 | * See the License for the specific language governing permissions and |
14 | 14 | * limitations under the License. |
15 | 15 | */ |
16 | -package org.thingsboard.server.mqtt.rpc.nosql; | |
16 | +package org.thingsboard.server.transport.coap.telemetry.attributes.sql; | |
17 | 17 | |
18 | -import org.thingsboard.server.dao.service.DaoNoSqlTest; | |
19 | -import org.thingsboard.server.mqtt.rpc.AbstractMqttServerSideRpcDefaultIntegrationTest; | |
18 | +import org.thingsboard.server.transport.coap.telemetry.attributes.AbstractCoapAttributesIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | 20 | |
21 | 21 | /** |
22 | 22 | * Created by Valerii Sosliuk on 8/22/2017. |
23 | 23 | */ |
24 | -@DaoNoSqlTest | |
25 | -public class MqttServerSideRpcNoSqlIntegrationTest extends AbstractMqttServerSideRpcDefaultIntegrationTest { | |
24 | +@DaoSqlTest | |
25 | +public class CoapAttributesSqlIntegrationTest extends AbstractCoapAttributesIntegrationTest { | |
26 | 26 | } | ... | ... |
application/src/test/java/org/thingsboard/server/transport/coap/telemetry/attributes/sql/CoapAttributesSqlJsonIntegrationTest.java
renamed from
application/src/test/java/org/thingsboard/server/mqtt/DbConfigurationTestRule.java
... | ... | @@ -13,19 +13,14 @@ |
13 | 13 | * See the License for the specific language governing permissions and |
14 | 14 | * limitations under the License. |
15 | 15 | */ |
16 | -package org.thingsboard.server.mqtt; | |
16 | +package org.thingsboard.server.transport.coap.telemetry.attributes.sql; | |
17 | 17 | |
18 | -import org.junit.rules.TestRule; | |
19 | -import org.junit.runner.Description; | |
20 | -import org.junit.runners.model.Statement; | |
18 | +import org.thingsboard.server.transport.coap.telemetry.attributes.AbstractCoapAttributesJsonIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
21 | 20 | |
22 | 21 | /** |
23 | - * Created by ashvayka on 11.05.18. | |
22 | + * Created by Valerii Sosliuk on 8/22/2017. | |
24 | 23 | */ |
25 | -public class DbConfigurationTestRule implements TestRule { | |
26 | - | |
27 | - @Override | |
28 | - public Statement apply(Statement base, Description description) { | |
29 | - return null; | |
30 | - } | |
24 | +@DaoSqlTest | |
25 | +public class CoapAttributesSqlJsonIntegrationTest extends AbstractCoapAttributesJsonIntegrationTest { | |
31 | 26 | } | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.telemetry.attributes.sql; | |
17 | + | |
18 | +import org.thingsboard.server.transport.coap.telemetry.attributes.AbstractCoapAttributesProtoIntegrationTest; | |
19 | +import org.thingsboard.server.dao.service.DaoSqlTest; | |
20 | + | |
21 | +/** | |
22 | + * Created by Valerii Sosliuk on 8/22/2017. | |
23 | + */ | |
24 | +@DaoSqlTest | |
25 | +public class CoapAttributesSqlProtoIntegrationTest extends AbstractCoapAttributesProtoIntegrationTest { | |
26 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.telemetry.timeseries; | |
17 | + | |
18 | +import com.fasterxml.jackson.core.type.TypeReference; | |
19 | +import lombok.extern.slf4j.Slf4j; | |
20 | +import org.eclipse.californium.core.CoapClient; | |
21 | +import org.eclipse.californium.core.CoapResponse; | |
22 | +import org.eclipse.californium.core.coap.CoAP; | |
23 | +import org.eclipse.californium.core.coap.MediaTypeRegistry; | |
24 | +import org.eclipse.californium.elements.exception.ConnectorException; | |
25 | +import org.junit.After; | |
26 | +import org.junit.Before; | |
27 | +import org.junit.Test; | |
28 | +import org.thingsboard.server.transport.coap.AbstractCoapIntegrationTest; | |
29 | +import org.thingsboard.server.common.msg.session.FeatureType; | |
30 | + | |
31 | +import java.io.IOException; | |
32 | +import java.util.Arrays; | |
33 | +import java.util.HashSet; | |
34 | +import java.util.List; | |
35 | +import java.util.Map; | |
36 | +import java.util.Set; | |
37 | + | |
38 | +import static org.junit.Assert.assertEquals; | |
39 | +import static org.junit.Assert.assertNotNull; | |
40 | + | |
41 | +@Slf4j | |
42 | +public abstract class AbstractCoapTimeseriesIntegrationTest extends AbstractCoapIntegrationTest { | |
43 | + | |
44 | + private static final String PAYLOAD_VALUES_STR = "{\"key1\":\"value1\", \"key2\":true, \"key3\": 3.0, \"key4\": 4," + | |
45 | + " \"key5\": {\"someNumber\": 42, \"someArray\": [1,2,3], \"someNestedObject\": {\"key\": \"value\"}}}"; | |
46 | + | |
47 | + @Before | |
48 | + public void beforeTest() throws Exception { | |
49 | + processBeforeTest("Test Post Telemetry device", null, null); | |
50 | + } | |
51 | + | |
52 | + @After | |
53 | + public void afterTest() throws Exception { | |
54 | + processAfterTest(); | |
55 | + } | |
56 | + | |
57 | + @Test | |
58 | + public void testPushTelemetry() throws Exception { | |
59 | + processTestPostTelemetry(null, false); | |
60 | + } | |
61 | + | |
62 | + @Test | |
63 | + public void testPushTelemetryWithTs() throws Exception { | |
64 | + String payloadStr = "{\"ts\": 10000, \"values\": " + PAYLOAD_VALUES_STR + "}"; | |
65 | + processTestPostTelemetry(payloadStr.getBytes(), true); | |
66 | + } | |
67 | + | |
68 | + protected void processTestPostTelemetry(byte[] payloadBytes, boolean withTs) throws Exception { | |
69 | + log.warn("[testPushTelemetry] Device: {}, Transport type: {}", savedDevice.getName(), savedDevice.getType()); | |
70 | + List<String> expectedKeys = Arrays.asList("key1", "key2", "key3", "key4", "key5"); | |
71 | + CoapClient coapClient = getCoapClient(FeatureType.TELEMETRY); | |
72 | + postTelemetry(coapClient, payloadBytes); | |
73 | + | |
74 | + String deviceId = savedDevice.getId().getId().toString(); | |
75 | + | |
76 | + long start = System.currentTimeMillis(); | |
77 | + long end = System.currentTimeMillis() + 5000; | |
78 | + | |
79 | + List<String> actualKeys = null; | |
80 | + while (start <= end) { | |
81 | + actualKeys = doGetAsyncTyped("/api/plugins/telemetry/DEVICE/" + deviceId + "/keys/timeseries", new TypeReference<>() {}); | |
82 | + if (actualKeys.size() == expectedKeys.size()) { | |
83 | + break; | |
84 | + } | |
85 | + Thread.sleep(100); | |
86 | + start += 100; | |
87 | + } | |
88 | + assertNotNull(actualKeys); | |
89 | + | |
90 | + Set<String> actualKeySet = new HashSet<>(actualKeys); | |
91 | + Set<String> expectedKeySet = new HashSet<>(expectedKeys); | |
92 | + | |
93 | + assertEquals(expectedKeySet, actualKeySet); | |
94 | + | |
95 | + String getTelemetryValuesUrl; | |
96 | + if (withTs) { | |
97 | + getTelemetryValuesUrl = "/api/plugins/telemetry/DEVICE/" + deviceId + "/values/timeseries?startTs=0&endTs=15000&keys=" + String.join(",", actualKeySet); | |
98 | + } else { | |
99 | + getTelemetryValuesUrl = "/api/plugins/telemetry/DEVICE/" + deviceId + "/values/timeseries?keys=" + String.join(",", actualKeySet); | |
100 | + } | |
101 | + start = System.currentTimeMillis(); | |
102 | + end = System.currentTimeMillis() + 5000; | |
103 | + Map<String, List<Map<String, Object>>> values = null; | |
104 | + while (start <= end) { | |
105 | + values = doGetAsyncTyped(getTelemetryValuesUrl, new TypeReference<>() {}); | |
106 | + boolean valid = values.size() == expectedKeys.size(); | |
107 | + if (valid) { | |
108 | + for (String key : expectedKeys) { | |
109 | + List<Map<String, Object>> tsValues = values.get(key); | |
110 | + if (tsValues != null && tsValues.size() > 0) { | |
111 | + Object ts = tsValues.get(0).get("ts"); | |
112 | + if (ts == null) { | |
113 | + valid = false; | |
114 | + break; | |
115 | + } | |
116 | + } else { | |
117 | + valid = false; | |
118 | + break; | |
119 | + } | |
120 | + } | |
121 | + } | |
122 | + if (valid) { | |
123 | + break; | |
124 | + } | |
125 | + Thread.sleep(100); | |
126 | + start += 100; | |
127 | + } | |
128 | + assertNotNull(values); | |
129 | + | |
130 | + if (withTs) { | |
131 | + assertTs(values, expectedKeys, 10000, 0); | |
132 | + } | |
133 | + assertValues(values, 0); | |
134 | + } | |
135 | + | |
136 | + private void postTelemetry(CoapClient client, byte[] payload) throws IOException, ConnectorException { | |
137 | + if (payload == null) { | |
138 | + payload = PAYLOAD_VALUES_STR.getBytes(); | |
139 | + } | |
140 | + CoapResponse coapResponse = client.setTimeout((long) 60000).post(payload, MediaTypeRegistry.APPLICATION_JSON); | |
141 | + assertEquals(CoAP.ResponseCode.CREATED, coapResponse.getCode()); | |
142 | + } | |
143 | + | |
144 | + private void assertTs(Map<String, List<Map<String, Object>>> deviceValues, List<String> expectedKeys, int ts, int arrayIndex) { | |
145 | + assertEquals(ts, deviceValues.get(expectedKeys.get(0)).get(arrayIndex).get("ts")); | |
146 | + assertEquals(ts, deviceValues.get(expectedKeys.get(1)).get(arrayIndex).get("ts")); | |
147 | + assertEquals(ts, deviceValues.get(expectedKeys.get(2)).get(arrayIndex).get("ts")); | |
148 | + assertEquals(ts, deviceValues.get(expectedKeys.get(3)).get(arrayIndex).get("ts")); | |
149 | + assertEquals(ts, deviceValues.get(expectedKeys.get(4)).get(arrayIndex).get("ts")); | |
150 | + } | |
151 | + | |
152 | + private void assertValues(Map<String, List<Map<String, Object>>> deviceValues, int arrayIndex) { | |
153 | + for (Map.Entry<String, List<Map<String, Object>>> entry : deviceValues.entrySet()) { | |
154 | + String key = entry.getKey(); | |
155 | + List<Map<String, Object>> tsKv = entry.getValue(); | |
156 | + String value = (String) tsKv.get(arrayIndex).get("value"); | |
157 | + switch (key) { | |
158 | + case "key1": | |
159 | + assertEquals("value1", value); | |
160 | + break; | |
161 | + case "key2": | |
162 | + assertEquals("true", value); | |
163 | + break; | |
164 | + case "key3": | |
165 | + assertEquals("3.0", value); | |
166 | + break; | |
167 | + case "key4": | |
168 | + assertEquals("4", value); | |
169 | + break; | |
170 | + case "key5": | |
171 | + assertEquals("{\"someNumber\":42,\"someArray\":[1,2,3],\"someNestedObject\":{\"key\":\"value\"}}", value); | |
172 | + break; | |
173 | + } | |
174 | + } | |
175 | + } | |
176 | + | |
177 | + | |
178 | +} | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 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.transport.coap.telemetry.timeseries; | |
17 | + | |
18 | +import lombok.extern.slf4j.Slf4j; | |
19 | +import org.junit.After; | |
20 | +import org.junit.Before; | |
21 | +import org.junit.Test; | |
22 | +import org.thingsboard.server.common.data.CoapDeviceType; | |
23 | +import org.thingsboard.server.common.data.TransportPayloadType; | |
24 | + | |
25 | +@Slf4j | |
26 | +public abstract class AbstractCoapTimeseriesJsonIntegrationTest extends AbstractCoapTimeseriesIntegrationTest { | |
27 | + | |
28 | + @Before | |
29 | + public void beforeTest() throws Exception { | |
30 | + processBeforeTest("Test Post Telemetry device json payload", CoapDeviceType.DEFAULT, TransportPayloadType.JSON); | |
31 | + } | |
32 | + | |
33 | + @After | |
34 | + public void afterTest() throws Exception { | |
35 | + processAfterTest(); | |
36 | + } | |
37 | + | |
38 | + | |
39 | + @Test | |
40 | + public void testPushTelemetry() throws Exception { | |
41 | + super.testPushTelemetry(); | |
42 | + } | |
43 | + | |
44 | + @Test | |
45 | + public void testPushTelemetryWithTs() throws Exception { | |
46 | + super.testPushTelemetryWithTs(); | |
47 | + } | |
48 | + | |
49 | + | |
50 | +} | ... | ... |