Commit 8c9a6c3e4742dff95e751b91bb17247738bdcf3f

Authored by Andrew Shvayka
2 parents 2590f04a 0f14a36c

Merge with master

Showing 73 changed files with 2793 additions and 463 deletions

Too many changes to show.

To preserve performance only 73 of 82 files are displayed.

  1 +--
  2 +-- Copyright © 2016-2018 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 +DROP MATERIALIZED VIEW IF EXISTS thingsboard.entity_view_by_tenant_and_name;
  18 +DROP MATERIALIZED VIEW IF EXISTS thingsboard.entity_view_by_tenant_and_search_text;
  19 +DROP MATERIALIZED VIEW IF EXISTS thingsboard.entity_view_by_tenant_and_customer;
  20 +DROP MATERIALIZED VIEW IF EXISTS thingsboard.entity_view_by_tenant_and_entity_id;
  21 +
  22 +DROP TABLE IF EXISTS thingsboard.entity_views;
  23 +
  24 +CREATE TABLE IF NOT EXISTS thingsboard.entity_view (
  25 + id timeuuid,
  26 + entity_id timeuuid,
  27 + entity_type text,
  28 + tenant_id timeuuid,
  29 + customer_id timeuuid,
  30 + name text,
  31 + type text,
  32 + keys text,
  33 + start_ts bigint,
  34 + end_ts bigint,
  35 + search_text text,
  36 + additional_info text,
  37 + PRIMARY KEY (id, entity_id, tenant_id, customer_id, type)
  38 +);
  39 +
  40 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_name AS
  41 + SELECT *
  42 + from thingsboard.entity_view
  43 + WHERE tenant_id IS NOT NULL
  44 + AND entity_id IS NOT NULL
  45 + AND customer_id IS NOT NULL
  46 + AND type IS NOT NULL
  47 + AND name IS NOT NULL
  48 + AND id IS NOT NULL
  49 + PRIMARY KEY (tenant_id, name, id, customer_id, entity_id, type)
  50 + WITH CLUSTERING ORDER BY (name ASC, id DESC, customer_id DESC);
  51 +
  52 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_search_text AS
  53 + SELECT *
  54 + from thingsboard.entity_view
  55 + WHERE tenant_id IS NOT NULL
  56 + AND entity_id IS NOT NULL
  57 + AND customer_id IS NOT NULL
  58 + AND type IS NOT NULL
  59 + AND search_text IS NOT NULL
  60 + AND id IS NOT NULL
  61 + PRIMARY KEY (tenant_id, search_text, id, customer_id, entity_id, type)
  62 + WITH CLUSTERING ORDER BY (search_text ASC, id DESC, customer_id DESC);
  63 +
  64 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_by_type_and_search_text AS
  65 + SELECT *
  66 + from thingsboard.entity_view
  67 + WHERE tenant_id IS NOT NULL
  68 + AND entity_id IS NOT NULL
  69 + AND customer_id IS NOT NULL
  70 + AND type IS NOT NULL
  71 + AND search_text IS NOT NULL
  72 + AND id IS NOT NULL
  73 + PRIMARY KEY (tenant_id, type, search_text, id, customer_id, entity_id)
  74 + WITH CLUSTERING ORDER BY (type ASC, search_text ASC, id DESC, customer_id DESC);
  75 +
  76 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_customer AS
  77 + SELECT *
  78 + from thingsboard.entity_view
  79 + WHERE tenant_id IS NOT NULL
  80 + AND customer_id IS NOT NULL
  81 + AND entity_id IS NOT NULL
  82 + AND type IS NOT NULL
  83 + AND search_text IS NOT NULL
  84 + AND id IS NOT NULL
  85 + PRIMARY KEY (tenant_id, customer_id, search_text, id, entity_id, type)
  86 + WITH CLUSTERING ORDER BY (customer_id DESC, search_text ASC, id DESC);
  87 +
  88 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_customer_and_type AS
  89 + SELECT *
  90 + from thingsboard.entity_view
  91 + WHERE tenant_id IS NOT NULL
  92 + AND customer_id IS NOT NULL
  93 + AND entity_id IS NOT NULL
  94 + AND type IS NOT NULL
  95 + AND search_text IS NOT NULL
  96 + AND id IS NOT NULL
  97 + PRIMARY KEY (tenant_id, type, customer_id, search_text, id, entity_id)
  98 + WITH CLUSTERING ORDER BY (type ASC, customer_id DESC, search_text ASC, id DESC);
  99 +
  100 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_entity_id AS
  101 + SELECT *
  102 + from thingsboard.entity_view
  103 + WHERE tenant_id IS NOT NULL
  104 + AND customer_id IS NOT NULL
  105 + AND entity_id IS NOT NULL
  106 + AND type IS NOT NULL
  107 + AND search_text IS NOT NULL
  108 + AND id IS NOT NULL
  109 + PRIMARY KEY (tenant_id, entity_id, customer_id, search_text, id, type)
  110 + WITH CLUSTERING ORDER BY (entity_id DESC, customer_id DESC, search_text ASC, id DESC);
  1 +--
  2 +-- Copyright © 2016-2018 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 +DROP TABLE IF EXISTS entity_views;
  18 +
  19 +CREATE TABLE IF NOT EXISTS entity_view (
  20 + id varchar(31) NOT NULL CONSTRAINT entity_view_pkey PRIMARY KEY,
  21 + entity_id varchar(31),
  22 + entity_type varchar(255),
  23 + tenant_id varchar(31),
  24 + customer_id varchar(31),
  25 + type varchar(255),
  26 + name varchar(255),
  27 + keys varchar(255),
  28 + start_ts bigint,
  29 + end_ts bigint,
  30 + search_text varchar(255),
  31 + additional_info varchar
  32 +);
@@ -15,6 +15,8 @@ @@ -15,6 +15,8 @@
15 */ 15 */
16 package org.thingsboard.server.controller; 16 package org.thingsboard.server.controller;
17 17
  18 +import lombok.Getter;
  19 +import org.springframework.beans.factory.annotation.Value;
18 import org.springframework.http.HttpStatus; 20 import org.springframework.http.HttpStatus;
19 import org.springframework.security.access.prepost.PreAuthorize; 21 import org.springframework.security.access.prepost.PreAuthorize;
20 import org.springframework.web.bind.annotation.PathVariable; 22 import org.springframework.web.bind.annotation.PathVariable;
@@ -49,6 +51,11 @@ public class DashboardController extends BaseController { @@ -49,6 +51,11 @@ public class DashboardController extends BaseController {
49 51
50 public static final String DASHBOARD_ID = "dashboardId"; 52 public static final String DASHBOARD_ID = "dashboardId";
51 53
  54 + @Value("${dashboard.max_datapoints_limit}")
  55 + @Getter
  56 + private long maxDatapointsLimit;
  57 +
  58 +
52 @PreAuthorize("hasAnyAuthority('SYS_ADMIN', 'TENANT_ADMIN', 'CUSTOMER_USER')") 59 @PreAuthorize("hasAnyAuthority('SYS_ADMIN', 'TENANT_ADMIN', 'CUSTOMER_USER')")
53 @RequestMapping(value = "/dashboard/serverTime", method = RequestMethod.GET) 60 @RequestMapping(value = "/dashboard/serverTime", method = RequestMethod.GET)
54 @ResponseBody 61 @ResponseBody
@@ -57,6 +64,13 @@ public class DashboardController extends BaseController { @@ -57,6 +64,13 @@ public class DashboardController extends BaseController {
57 } 64 }
58 65
59 @PreAuthorize("hasAnyAuthority('SYS_ADMIN', 'TENANT_ADMIN', 'CUSTOMER_USER')") 66 @PreAuthorize("hasAnyAuthority('SYS_ADMIN', 'TENANT_ADMIN', 'CUSTOMER_USER')")
  67 + @RequestMapping(value = "/dashboard/maxDatapointsLimit", method = RequestMethod.GET)
  68 + @ResponseBody
  69 + public long getMaxDatapointsLimit() throws ThingsboardException {
  70 + return maxDatapointsLimit;
  71 + }
  72 +
  73 + @PreAuthorize("hasAnyAuthority('SYS_ADMIN', 'TENANT_ADMIN', 'CUSTOMER_USER')")
60 @RequestMapping(value = "/dashboard/info/{dashboardId}", method = RequestMethod.GET) 74 @RequestMapping(value = "/dashboard/info/{dashboardId}", method = RequestMethod.GET)
61 @ResponseBody 75 @ResponseBody
62 public DashboardInfo getDashboardInfoById(@PathVariable(DASHBOARD_ID) String strDashboardId) throws ThingsboardException { 76 public DashboardInfo getDashboardInfoById(@PathVariable(DASHBOARD_ID) String strDashboardId) throws ThingsboardException {
@@ -15,6 +15,7 @@ @@ -15,6 +15,7 @@
15 */ 15 */
16 package org.thingsboard.server.controller; 16 package org.thingsboard.server.controller;
17 17
  18 +import com.google.common.util.concurrent.ListenableFuture;
18 import org.springframework.http.HttpStatus; 19 import org.springframework.http.HttpStatus;
19 import org.springframework.security.access.prepost.PreAuthorize; 20 import org.springframework.security.access.prepost.PreAuthorize;
20 import org.springframework.web.bind.annotation.PathVariable; 21 import org.springframework.web.bind.annotation.PathVariable;
@@ -26,6 +27,7 @@ import org.springframework.web.bind.annotation.ResponseBody; @@ -26,6 +27,7 @@ import org.springframework.web.bind.annotation.ResponseBody;
26 import org.springframework.web.bind.annotation.ResponseStatus; 27 import org.springframework.web.bind.annotation.ResponseStatus;
27 import org.springframework.web.bind.annotation.RestController; 28 import org.springframework.web.bind.annotation.RestController;
28 import org.thingsboard.server.common.data.Customer; 29 import org.thingsboard.server.common.data.Customer;
  30 +import org.thingsboard.server.common.data.EntitySubtype;
29 import org.thingsboard.server.common.data.EntityType; 31 import org.thingsboard.server.common.data.EntityType;
30 import org.thingsboard.server.common.data.EntityView; 32 import org.thingsboard.server.common.data.EntityView;
31 import org.thingsboard.server.common.data.audit.ActionType; 33 import org.thingsboard.server.common.data.audit.ActionType;
@@ -38,6 +40,7 @@ import org.thingsboard.server.common.data.page.TextPageData; @@ -38,6 +40,7 @@ import org.thingsboard.server.common.data.page.TextPageData;
38 import org.thingsboard.server.common.data.page.TextPageLink; 40 import org.thingsboard.server.common.data.page.TextPageLink;
39 import org.thingsboard.server.dao.exception.IncorrectParameterException; 41 import org.thingsboard.server.dao.exception.IncorrectParameterException;
40 import org.thingsboard.server.dao.model.ModelConstants; 42 import org.thingsboard.server.dao.model.ModelConstants;
  43 +import org.thingsboard.server.service.security.model.SecurityUser;
41 44
42 import java.util.List; 45 import java.util.List;
43 import java.util.stream.Collectors; 46 import java.util.stream.Collectors;
@@ -161,6 +164,7 @@ public class EntityViewController extends BaseController { @@ -161,6 +164,7 @@ public class EntityViewController extends BaseController {
161 public TextPageData<EntityView> getCustomerEntityViews( 164 public TextPageData<EntityView> getCustomerEntityViews(
162 @PathVariable("customerId") String strCustomerId, 165 @PathVariable("customerId") String strCustomerId,
163 @RequestParam int limit, 166 @RequestParam int limit,
  167 + @RequestParam(required = false) String type,
164 @RequestParam(required = false) String textSearch, 168 @RequestParam(required = false) String textSearch,
165 @RequestParam(required = false) String idOffset, 169 @RequestParam(required = false) String idOffset,
166 @RequestParam(required = false) String textOffset) throws ThingsboardException { 170 @RequestParam(required = false) String textOffset) throws ThingsboardException {
@@ -170,7 +174,11 @@ public class EntityViewController extends BaseController { @@ -170,7 +174,11 @@ public class EntityViewController extends BaseController {
170 CustomerId customerId = new CustomerId(toUUID(strCustomerId)); 174 CustomerId customerId = new CustomerId(toUUID(strCustomerId));
171 checkCustomerId(customerId); 175 checkCustomerId(customerId);
172 TextPageLink pageLink = createPageLink(limit, textSearch, idOffset, textOffset); 176 TextPageLink pageLink = createPageLink(limit, textSearch, idOffset, textOffset);
173 - return checkNotNull(entityViewService.findEntityViewsByTenantIdAndCustomerId(tenantId, customerId, pageLink)); 177 + if (type != null && type.trim().length() > 0) {
  178 + return checkNotNull(entityViewService.findEntityViewsByTenantIdAndCustomerIdAndType(tenantId, customerId, pageLink, type));
  179 + } else {
  180 + return checkNotNull(entityViewService.findEntityViewsByTenantIdAndCustomerId(tenantId, customerId, pageLink));
  181 + }
174 } catch (Exception e) { 182 } catch (Exception e) {
175 throw handleException(e); 183 throw handleException(e);
176 } 184 }
@@ -181,13 +189,19 @@ public class EntityViewController extends BaseController { @@ -181,13 +189,19 @@ public class EntityViewController extends BaseController {
181 @ResponseBody 189 @ResponseBody
182 public TextPageData<EntityView> getTenantEntityViews( 190 public TextPageData<EntityView> getTenantEntityViews(
183 @RequestParam int limit, 191 @RequestParam int limit,
  192 + @RequestParam(required = false) String type,
184 @RequestParam(required = false) String textSearch, 193 @RequestParam(required = false) String textSearch,
185 @RequestParam(required = false) String idOffset, 194 @RequestParam(required = false) String idOffset,
186 @RequestParam(required = false) String textOffset) throws ThingsboardException { 195 @RequestParam(required = false) String textOffset) throws ThingsboardException {
187 try { 196 try {
188 TenantId tenantId = getCurrentUser().getTenantId(); 197 TenantId tenantId = getCurrentUser().getTenantId();
189 TextPageLink pageLink = createPageLink(limit, textSearch, idOffset, textOffset); 198 TextPageLink pageLink = createPageLink(limit, textSearch, idOffset, textOffset);
190 - return checkNotNull(entityViewService.findEntityViewByTenantId(tenantId, pageLink)); 199 +
  200 + if (type != null && type.trim().length() > 0) {
  201 + return checkNotNull(entityViewService.findEntityViewByTenantIdAndType(tenantId, pageLink, type));
  202 + } else {
  203 + return checkNotNull(entityViewService.findEntityViewByTenantId(tenantId, pageLink));
  204 + }
191 } catch (Exception e) { 205 } catch (Exception e) {
192 throw handleException(e); 206 throw handleException(e);
193 } 207 }
@@ -199,6 +213,7 @@ public class EntityViewController extends BaseController { @@ -199,6 +213,7 @@ public class EntityViewController extends BaseController {
199 public List<EntityView> findByQuery(@RequestBody EntityViewSearchQuery query) throws ThingsboardException { 213 public List<EntityView> findByQuery(@RequestBody EntityViewSearchQuery query) throws ThingsboardException {
200 checkNotNull(query); 214 checkNotNull(query);
201 checkNotNull(query.getParameters()); 215 checkNotNull(query.getParameters());
  216 + checkNotNull(query.getEntityViewTypes());
202 checkEntityId(query.getParameters().getEntityId()); 217 checkEntityId(query.getParameters().getEntityId());
203 try { 218 try {
204 List<EntityView> entityViews = checkNotNull(entityViewService.findEntityViewsByQuery(query).get()); 219 List<EntityView> entityViews = checkNotNull(entityViewService.findEntityViewsByQuery(query).get());
@@ -215,4 +230,18 @@ public class EntityViewController extends BaseController { @@ -215,4 +230,18 @@ public class EntityViewController extends BaseController {
215 throw handleException(e); 230 throw handleException(e);
216 } 231 }
217 } 232 }
  233 +
  234 + @PreAuthorize("hasAnyAuthority('TENANT_ADMIN', 'CUSTOMER_USER')")
  235 + @RequestMapping(value = "/entityView/types", method = RequestMethod.GET)
  236 + @ResponseBody
  237 + public List<EntitySubtype> getEntityViewTypes() throws ThingsboardException {
  238 + try {
  239 + SecurityUser user = getCurrentUser();
  240 + TenantId tenantId = user.getTenantId();
  241 + ListenableFuture<List<EntitySubtype>> entityViewTypes = entityViewService.findEntityViewTypesByTenantId(tenantId);
  242 + return checkNotNull(entityViewTypes.get());
  243 + } catch (Exception e) {
  244 + throw handleException(e);
  245 + }
  246 + }
218 } 247 }
@@ -97,6 +97,11 @@ public class ThingsboardInstallService { @@ -97,6 +97,11 @@ public class ThingsboardInstallService {
97 97
98 databaseUpgradeService.upgradeDatabase("2.0.0"); 98 databaseUpgradeService.upgradeDatabase("2.0.0");
99 99
  100 + case "2.1.1":
  101 + log.info("Upgrading ThingsBoard from version 2.1.1 to 2.1.2 ...");
  102 +
  103 + databaseUpgradeService.upgradeDatabase("2.1.1");
  104 +
100 log.info("Updating system data..."); 105 log.info("Updating system data...");
101 106
102 systemDataLoaderService.deleteSystemWidgetBundle("charts"); 107 systemDataLoaderService.deleteSystemWidgetBundle("charts");
@@ -39,10 +39,19 @@ import static org.thingsboard.server.service.install.DatabaseHelper.CONFIGURATIO @@ -39,10 +39,19 @@ import static org.thingsboard.server.service.install.DatabaseHelper.CONFIGURATIO
39 import static org.thingsboard.server.service.install.DatabaseHelper.CUSTOMER_ID; 39 import static org.thingsboard.server.service.install.DatabaseHelper.CUSTOMER_ID;
40 import static org.thingsboard.server.service.install.DatabaseHelper.DASHBOARD; 40 import static org.thingsboard.server.service.install.DatabaseHelper.DASHBOARD;
41 import static org.thingsboard.server.service.install.DatabaseHelper.DEVICE; 41 import static org.thingsboard.server.service.install.DatabaseHelper.DEVICE;
  42 +import static org.thingsboard.server.service.install.DatabaseHelper.END_TS;
  43 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_ID;
  44 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_TYPE;
  45 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_VIEW;
  46 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_VIEWS;
42 import static org.thingsboard.server.service.install.DatabaseHelper.ID; 47 import static org.thingsboard.server.service.install.DatabaseHelper.ID;
  48 +import static org.thingsboard.server.service.install.DatabaseHelper.KEYS;
  49 +import static org.thingsboard.server.service.install.DatabaseHelper.NAME;
43 import static org.thingsboard.server.service.install.DatabaseHelper.SEARCH_TEXT; 50 import static org.thingsboard.server.service.install.DatabaseHelper.SEARCH_TEXT;
  51 +import static org.thingsboard.server.service.install.DatabaseHelper.START_TS;
44 import static org.thingsboard.server.service.install.DatabaseHelper.TENANT_ID; 52 import static org.thingsboard.server.service.install.DatabaseHelper.TENANT_ID;
45 import static org.thingsboard.server.service.install.DatabaseHelper.TITLE; 53 import static org.thingsboard.server.service.install.DatabaseHelper.TITLE;
  54 +import static org.thingsboard.server.service.install.DatabaseHelper.TYPE;
46 55
47 @Service 56 @Service
48 @NoSqlDao 57 @NoSqlDao
@@ -203,6 +212,46 @@ public class CassandraDatabaseUpgradeService implements DatabaseUpgradeService { @@ -203,6 +212,46 @@ public class CassandraDatabaseUpgradeService implements DatabaseUpgradeService {
203 log.info("Schema updated."); 212 log.info("Schema updated.");
204 213
205 break; 214 break;
  215 +
  216 + case "2.0.0":
  217 +
  218 + log.info("Updating schema ...");
  219 + schemaUpdateFile = Paths.get(installScripts.getDataDir(), "upgrade", "2.1.1", SCHEMA_UPDATE_CQL);
  220 + loadCql(schemaUpdateFile);
  221 + log.info("Schema updated.");
  222 +
  223 + break;
  224 +
  225 + case "2.1.1":
  226 +
  227 + log.info("Upgrading Cassandra DataBase from version {} to 2.1.2 ...", fromVersion);
  228 +
  229 + cluster.getSession();
  230 +
  231 + ks = cluster.getCluster().getMetadata().getKeyspace(cluster.getKeyspaceName());
  232 +
  233 + log.info("Dumping entity views ...");
  234 + Path entityViewsDump = CassandraDbHelper.dumpCfIfExists(ks, cluster.getSession(), ENTITY_VIEWS,
  235 + new String[]{ID, ENTITY_ID, ENTITY_TYPE, TENANT_ID, CUSTOMER_ID, NAME, TYPE, KEYS, START_TS, END_TS, SEARCH_TEXT, ADDITIONAL_INFO},
  236 + new String[]{"", "", "", "", "", "", "default", "", "0", "0", "", ""},
  237 + "tb-entity-views");
  238 + log.info("Entity views dumped.");
  239 +
  240 + log.info("Updating schema ...");
  241 + schemaUpdateFile = Paths.get(installScripts.getDataDir(), "upgrade", "2.1.2", SCHEMA_UPDATE_CQL);
  242 + loadCql(schemaUpdateFile);
  243 + log.info("Schema updated.");
  244 +
  245 + log.info("Restoring entity views ...");
  246 + if (entityViewsDump != null) {
  247 + CassandraDbHelper.loadCf(ks, cluster.getSession(), ENTITY_VIEW,
  248 + new String[]{ID, ENTITY_ID, ENTITY_TYPE, TENANT_ID, CUSTOMER_ID, NAME, TYPE, KEYS, START_TS, END_TS, SEARCH_TEXT, ADDITIONAL_INFO}, entityViewsDump);
  249 + Files.deleteIfExists(entityViewsDump);
  250 + }
  251 + log.info("Entity views restored.");
  252 +
  253 + break;
  254 +
206 default: 255 default:
207 throw new RuntimeException("Unable to upgrade Cassandra database, unsupported fromVersion: " + fromVersion); 256 throw new RuntimeException("Unable to upgrade Cassandra database, unsupported fromVersion: " + fromVersion);
208 } 257 }
@@ -45,14 +45,23 @@ public class DatabaseHelper { @@ -45,14 +45,23 @@ public class DatabaseHelper {
45 public static final CSVFormat CSV_DUMP_FORMAT = CSVFormat.DEFAULT.withNullString("\\N"); 45 public static final CSVFormat CSV_DUMP_FORMAT = CSVFormat.DEFAULT.withNullString("\\N");
46 46
47 public static final String DEVICE = "device"; 47 public static final String DEVICE = "device";
  48 + public static final String ENTITY_ID = "entity_id";
48 public static final String TENANT_ID = "tenant_id"; 49 public static final String TENANT_ID = "tenant_id";
  50 + public static final String ENTITY_TYPE = "entity_type";
49 public static final String CUSTOMER_ID = "customer_id"; 51 public static final String CUSTOMER_ID = "customer_id";
50 public static final String SEARCH_TEXT = "search_text"; 52 public static final String SEARCH_TEXT = "search_text";
51 public static final String ADDITIONAL_INFO = "additional_info"; 53 public static final String ADDITIONAL_INFO = "additional_info";
52 public static final String ASSET = "asset"; 54 public static final String ASSET = "asset";
53 public static final String DASHBOARD = "dashboard"; 55 public static final String DASHBOARD = "dashboard";
  56 + public static final String ENTITY_VIEWS = "entity_views";
  57 + public static final String ENTITY_VIEW = "entity_view";
54 public static final String ID = "id"; 58 public static final String ID = "id";
55 public static final String TITLE = "title"; 59 public static final String TITLE = "title";
  60 + public static final String TYPE = "type";
  61 + public static final String NAME = "name";
  62 + public static final String KEYS = "keys";
  63 + public static final String START_TS = "start_ts";
  64 + public static final String END_TS = "end_ts";
56 public static final String ASSIGNED_CUSTOMERS = "assigned_customers"; 65 public static final String ASSIGNED_CUSTOMERS = "assigned_customers";
57 public static final String CONFIGURATION = "configuration"; 66 public static final String CONFIGURATION = "configuration";
58 67
@@ -31,14 +31,24 @@ import java.nio.file.Paths; @@ -31,14 +31,24 @@ import java.nio.file.Paths;
31 import java.sql.Connection; 31 import java.sql.Connection;
32 import java.sql.DriverManager; 32 import java.sql.DriverManager;
33 33
  34 +import static org.thingsboard.server.service.install.DatabaseHelper.ADDITIONAL_INFO;
34 import static org.thingsboard.server.service.install.DatabaseHelper.ASSIGNED_CUSTOMERS; 35 import static org.thingsboard.server.service.install.DatabaseHelper.ASSIGNED_CUSTOMERS;
35 import static org.thingsboard.server.service.install.DatabaseHelper.CONFIGURATION; 36 import static org.thingsboard.server.service.install.DatabaseHelper.CONFIGURATION;
36 import static org.thingsboard.server.service.install.DatabaseHelper.CUSTOMER_ID; 37 import static org.thingsboard.server.service.install.DatabaseHelper.CUSTOMER_ID;
37 import static org.thingsboard.server.service.install.DatabaseHelper.DASHBOARD; 38 import static org.thingsboard.server.service.install.DatabaseHelper.DASHBOARD;
  39 +import static org.thingsboard.server.service.install.DatabaseHelper.END_TS;
  40 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_ID;
  41 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_TYPE;
  42 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_VIEW;
  43 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_VIEWS;
38 import static org.thingsboard.server.service.install.DatabaseHelper.ID; 44 import static org.thingsboard.server.service.install.DatabaseHelper.ID;
  45 +import static org.thingsboard.server.service.install.DatabaseHelper.KEYS;
  46 +import static org.thingsboard.server.service.install.DatabaseHelper.NAME;
39 import static org.thingsboard.server.service.install.DatabaseHelper.SEARCH_TEXT; 47 import static org.thingsboard.server.service.install.DatabaseHelper.SEARCH_TEXT;
  48 +import static org.thingsboard.server.service.install.DatabaseHelper.START_TS;
40 import static org.thingsboard.server.service.install.DatabaseHelper.TENANT_ID; 49 import static org.thingsboard.server.service.install.DatabaseHelper.TENANT_ID;
41 import static org.thingsboard.server.service.install.DatabaseHelper.TITLE; 50 import static org.thingsboard.server.service.install.DatabaseHelper.TITLE;
  51 +import static org.thingsboard.server.service.install.DatabaseHelper.TYPE;
42 52
43 @Service 53 @Service
44 @Profile("install") 54 @Profile("install")
@@ -115,6 +125,30 @@ public class SqlDatabaseUpgradeService implements DatabaseUpgradeService { @@ -115,6 +125,30 @@ public class SqlDatabaseUpgradeService implements DatabaseUpgradeService {
115 log.info("Schema updated."); 125 log.info("Schema updated.");
116 } 126 }
117 break; 127 break;
  128 + case "2.1.1":
  129 + try (Connection conn = DriverManager.getConnection(dbUrl, dbUserName, dbPassword)) {
  130 +
  131 + log.info("Dumping entity views ...");
  132 + Path entityViewsDump = SqlDbHelper.dumpTableIfExists(conn, ENTITY_VIEWS,
  133 + new String[]{ID, ENTITY_ID, ENTITY_TYPE, TENANT_ID, CUSTOMER_ID, TYPE, NAME, KEYS, START_TS, END_TS, SEARCH_TEXT, ADDITIONAL_INFO},
  134 + new String[]{"", "", "", "", "", "default", "", "", "0", "0", "", ""},
  135 + "tb-entity-views", true);
  136 + log.info("Entity views dumped.");
  137 +
  138 + log.info("Updating schema ...");
  139 + schemaUpdateFile = Paths.get(installScripts.getDataDir(), "upgrade", "2.1.2", SCHEMA_UPDATE_SQL);
  140 + loadSql(schemaUpdateFile, conn);
  141 + log.info("Schema updated.");
  142 +
  143 + log.info("Restoring entity views ...");
  144 + if (entityViewsDump != null) {
  145 + SqlDbHelper.loadTable(conn, ENTITY_VIEW,
  146 + new String[]{ID, ENTITY_ID, ENTITY_TYPE, TENANT_ID, CUSTOMER_ID, TYPE, NAME, KEYS, START_TS, END_TS, SEARCH_TEXT, ADDITIONAL_INFO}, entityViewsDump, true);
  147 + Files.deleteIfExists(entityViewsDump);
  148 + }
  149 + log.info("Entity views restored.");
  150 + }
  151 + break;
118 152
119 default: 153 default:
120 throw new RuntimeException("Unable to upgrade SQL database, unsupported fromVersion: " + fromVersion); 154 throw new RuntimeException("Unable to upgrade SQL database, unsupported fromVersion: " + fromVersion);
@@ -147,6 +147,8 @@ public class CassandraDbHelper { @@ -147,6 +147,8 @@ public class CassandraDbHelper {
147 str = new Double(row.getDouble(index)).toString(); 147 str = new Double(row.getDouble(index)).toString();
148 } else if (type == DataType.cint()) { 148 } else if (type == DataType.cint()) {
149 str = new Integer(row.getInt(index)).toString(); 149 str = new Integer(row.getInt(index)).toString();
  150 + } else if (type == DataType.bigint()) {
  151 + str = new Long(row.getLong(index)).toString();
150 } else if (type == DataType.uuid()) { 152 } else if (type == DataType.uuid()) {
151 str = row.getUUID(index).toString(); 153 str = row.getUUID(index).toString();
152 } else if (type == DataType.timeuuid()) { 154 } else if (type == DataType.timeuuid()) {
@@ -193,6 +195,8 @@ public class CassandraDbHelper { @@ -193,6 +195,8 @@ public class CassandraDbHelper {
193 boundStatement.setDouble(column, Double.valueOf(value)); 195 boundStatement.setDouble(column, Double.valueOf(value));
194 } else if (type == DataType.cint()) { 196 } else if (type == DataType.cint()) {
195 boundStatement.setInt(column, Integer.valueOf(value)); 197 boundStatement.setInt(column, Integer.valueOf(value));
  198 + } else if (type == DataType.bigint()) {
  199 + boundStatement.setLong(column, Long.valueOf(value));
196 } else if (type == DataType.uuid()) { 200 } else if (type == DataType.uuid()) {
197 boundStatement.setUUID(column, UUID.fromString(value)); 201 boundStatement.setUUID(column, UUID.fromString(value));
198 } else if (type == DataType.timeuuid()) { 202 } else if (type == DataType.timeuuid()) {
@@ -77,46 +77,10 @@ security: @@ -77,46 +77,10 @@ security:
77 # Enable/disable access to Tenant Administrators JWT token by System Administrator or Customer Users JWT token by Tenant Administrator 77 # Enable/disable access to Tenant Administrators JWT token by System Administrator or Customer Users JWT token by Tenant Administrator
78 user_token_access_enabled: "${SECURITY_USER_TOKEN_ACCESS_ENABLED:true}" 78 user_token_access_enabled: "${SECURITY_USER_TOKEN_ACCESS_ENABLED:true}"
79 79
80 -# Device communication protocol parameters  
81 -http:  
82 - request_timeout: "${HTTP_REQUEST_TIMEOUT:60000}"  
83 -  
84 -# MQTT server parameters  
85 -mqtt:  
86 - # Enable/disable mqtt transport protocol.  
87 - enabled: "${MQTT_ENABLED:false}"  
88 - bind_address: "${MQTT_BIND_ADDRESS:0.0.0.0}"  
89 - bind_port: "${MQTT_BIND_PORT:1883}"  
90 - adaptor: "${MQTT_ADAPTOR_NAME:JsonMqttAdaptor}"  
91 - timeout: "${MQTT_TIMEOUT:10000}"  
92 - netty:  
93 - leak_detector_level: "${NETTY_LEASK_DETECTOR_LVL:DISABLED}"  
94 - boss_group_thread_count: "${NETTY_BOSS_GROUP_THREADS:1}"  
95 - worker_group_thread_count: "${NETTY_WORKER_GROUP_THREADS:12}"  
96 - max_payload_size: "${NETTY_MAX_PAYLOAD_SIZE:65536}"  
97 - # MQTT SSL configuration  
98 - ssl:  
99 - # Enable/disable SSL support  
100 - enabled: "${MQTT_SSL_ENABLED:false}"  
101 - # SSL protocol: See http://docs.oracle.com/javase/8/docs/technotes/guides/security/StandardNames.html#SSLContext  
102 - protocol: "${MQTT_SSL_PROTOCOL:TLSv1.2}"  
103 - # Path to the key store that holds the SSL certificate  
104 - key_store: "${MQTT_SSL_KEY_STORE:mqttserver.jks}"  
105 - # Password used to access the key store  
106 - key_store_password: "${MQTT_SSL_KEY_STORE_PASSWORD:server_ks_password}"  
107 - # Password used to access the key  
108 - key_password: "${MQTT_SSL_KEY_PASSWORD:server_key_password}"  
109 - # Type of the key store  
110 - key_store_type: "${MQTT_SSL_KEY_STORE_TYPE:JKS}"  
111 -  
112 -# CoAP server parameters  
113 -coap:  
114 - # Enable/disable coap transport protocol.  
115 - enabled: "${COAP_ENABLED:false}"  
116 - bind_address: "${COAP_BIND_ADDRESS:0.0.0.0}"  
117 - bind_port: "${COAP_BIND_PORT:5683}"  
118 - adaptor: "${COAP_ADAPTOR_NAME:JsonCoapAdaptor}"  
119 - timeout: "${COAP_TIMEOUT:10000}" 80 +# Dashboard parameters
  81 +dashboard:
  82 + # Maximum allowed datapoints fetched by widgets
  83 + max_datapoints_limit: "${DASHBOARD_MAX_DATAPOINTS_LIMIT:50000}"
120 84
121 #Quota parameters 85 #Quota parameters
122 quota: 86 quota:
@@ -136,8 +100,8 @@ quota: @@ -136,8 +100,8 @@ quota:
136 # Array of blacklist hosts 100 # Array of blacklist hosts
137 blacklist: "${QUOTA_HOST_BLACKLIST:}" 101 blacklist: "${QUOTA_HOST_BLACKLIST:}"
138 log: 102 log:
139 - topSize: 10  
140 - intervalMin: 2 103 + topSize: "${QUOTA_HOST_LOG_TOP_SIZE:10}"
  104 + intervalMin: "${QUOTA_HOST_LOG_INTERVAL_MIN:2}"
141 rule: 105 rule:
142 tenant: 106 tenant:
143 # Max allowed number of API requests in interval for single tenant 107 # Max allowed number of API requests in interval for single tenant
@@ -153,10 +117,10 @@ quota: @@ -153,10 +117,10 @@ quota:
153 # Array of whitelist tenants 117 # Array of whitelist tenants
154 whitelist: "${QUOTA_TENANT_WHITELIST:}" 118 whitelist: "${QUOTA_TENANT_WHITELIST:}"
155 # Array of blacklist tenants 119 # Array of blacklist tenants
156 - blacklist: "${QUOTA_HOST_BLACKLIST:}" 120 + blacklist: "${QUOTA_HOST_TENANT_BLACKLIST:}"
157 log: 121 log:
158 - topSize: 10  
159 - intervalMin: 2 122 + topSize: "${QUOTA_TENANT_LOG_TOP_SIZE:10}"
  123 + intervalMin: "${QUOTA_TENANT_LOG_INTERVAL_MIN:2}"
160 124
161 database: 125 database:
162 entities: 126 entities:
@@ -185,7 +149,8 @@ cassandra: @@ -185,7 +149,8 @@ cassandra:
185 init_timeout_ms: "${CASSANDRA_CLUSTER_INIT_TIMEOUT_MS:300000}" 149 init_timeout_ms: "${CASSANDRA_CLUSTER_INIT_TIMEOUT_MS:300000}"
186 # Specify cassandra claster initialization retry interval (if no hosts available during startup) 150 # Specify cassandra claster initialization retry interval (if no hosts available during startup)
187 init_retry_interval_ms: "${CASSANDRA_CLUSTER_INIT_RETRY_INTERVAL_MS:3000}" 151 init_retry_interval_ms: "${CASSANDRA_CLUSTER_INIT_RETRY_INTERVAL_MS:3000}"
188 - 152 + max_requests_per_connection_local: "${CASSANDRA_MAX_REQUESTS_PER_CONNECTION_LOCAL:32768}"
  153 + max_requests_per_connection_remote: "${CASSANDRA_MAX_REQUESTS_PER_CONNECTION_REMOTE:32768}"
189 # Credential parameters # 154 # Credential parameters #
190 credentials: "${CASSANDRA_USE_CREDENTIALS:false}" 155 credentials: "${CASSANDRA_USE_CREDENTIALS:false}"
191 # Specify your username 156 # Specify your username
@@ -466,4 +431,42 @@ transport: @@ -466,4 +431,42 @@ transport:
466 poll_interval: "${TB_RULE_ENGINE_POLL_INTERVAL_MS:25}" 431 poll_interval: "${TB_RULE_ENGINE_POLL_INTERVAL_MS:25}"
467 auto_commit_interval: "${TB_RULE_ENGINE_AUTO_COMMIT_INTERVAL_MS:100}" 432 auto_commit_interval: "${TB_RULE_ENGINE_AUTO_COMMIT_INTERVAL_MS:100}"
468 notifications: 433 notifications:
469 - topic: "${TB_TRANSPORT_NOTIFICATIONS_TOPIC:tb.transport.notifications}"  
  434 + topic: "${TB_TRANSPORT_NOTIFICATIONS_TOPIC:tb.transport.notifications}"
  435 + # Local HTTP transport parameters
  436 + http:
  437 + request_timeout: "${HTTP_REQUEST_TIMEOUT:60000}"
  438 + # Local MQTT transport parameters
  439 + mqtt:
  440 + # Enable/disable mqtt transport protocol.
  441 + enabled: "${MQTT_ENABLED:false}"
  442 + bind_address: "${MQTT_BIND_ADDRESS:0.0.0.0}"
  443 + bind_port: "${MQTT_BIND_PORT:1883}"
  444 + adaptor: "${MQTT_ADAPTOR_NAME:JsonMqttAdaptor}"
  445 + timeout: "${MQTT_TIMEOUT:10000}"
  446 + netty:
  447 + leak_detector_level: "${NETTY_LEASK_DETECTOR_LVL:DISABLED}"
  448 + boss_group_thread_count: "${NETTY_BOSS_GROUP_THREADS:1}"
  449 + worker_group_thread_count: "${NETTY_WORKER_GROUP_THREADS:12}"
  450 + max_payload_size: "${NETTY_MAX_PAYLOAD_SIZE:65536}"
  451 + # MQTT SSL configuration
  452 + ssl:
  453 + # Enable/disable SSL support
  454 + enabled: "${MQTT_SSL_ENABLED:false}"
  455 + # SSL protocol: See http://docs.oracle.com/javase/8/docs/technotes/guides/security/StandardNames.html#SSLContext
  456 + protocol: "${MQTT_SSL_PROTOCOL:TLSv1.2}"
  457 + # Path to the key store that holds the SSL certificate
  458 + key_store: "${MQTT_SSL_KEY_STORE:mqttserver.jks}"
  459 + # Password used to access the key store
  460 + key_store_password: "${MQTT_SSL_KEY_STORE_PASSWORD:server_ks_password}"
  461 + # Password used to access the key
  462 + key_password: "${MQTT_SSL_KEY_PASSWORD:server_key_password}"
  463 + # Type of the key store
  464 + key_store_type: "${MQTT_SSL_KEY_STORE_TYPE:JKS}"
  465 + # Local CoAP transport parameters
  466 + coap:
  467 + # Enable/disable coap transport protocol.
  468 + enabled: "${COAP_ENABLED:false}"
  469 + bind_address: "${COAP_BIND_ADDRESS:0.0.0.0}"
  470 + bind_port: "${COAP_BIND_PORT:5683}"
  471 + adaptor: "${COAP_ADAPTOR_NAME:JsonCoapAdaptor}"
  472 + timeout: "${COAP_TIMEOUT:10000}"
@@ -85,11 +85,11 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes @@ -85,11 +85,11 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
85 testDevice = doPost("/api/device", device, Device.class); 85 testDevice = doPost("/api/device", device, Device.class);
86 86
87 telemetry = new TelemetryEntityView( 87 telemetry = new TelemetryEntityView(
88 - Arrays.asList("109", "209", "309"), 88 + Arrays.asList("tsKey1", "tsKey2", "tsKey3"),
89 new AttributesEntityView( 89 new AttributesEntityView(
90 - Arrays.asList("caValue1", "caValue2", "caValue3", "caValue4"),  
91 - Arrays.asList("saValue1", "saValue2", "saValue3", "saValue4"),  
92 - Arrays.asList("shValue1", "shValue2", "shValue3", "shValue4"))); 90 + Arrays.asList("caKey1", "caKey2", "caKey3", "caKey4"),
  91 + Arrays.asList("saKey1", "saKey2", "saKey3", "saKey4"),
  92 + Arrays.asList("shKey1", "shKey2", "shKey3", "shKey4")));
93 } 93 }
94 94
95 @After 95 @After
@@ -144,7 +144,9 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes @@ -144,7 +144,9 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
144 144
145 @Test 145 @Test
146 public void testSaveEntityViewWithEmptyName() throws Exception { 146 public void testSaveEntityViewWithEmptyName() throws Exception {
147 - doPost("/api/entityView", new EntityView()) 147 + EntityView entityView = new EntityView();
  148 + entityView.setType("default");
  149 + doPost("/api/entityView", entityView)
148 .andExpect(status().isBadRequest()) 150 .andExpect(status().isBadRequest())
149 .andExpect(statusReason(containsString("Entity view name should be specified!"))); 151 .andExpect(statusReason(containsString("Entity view name should be specified!")));
150 } 152 }
@@ -322,10 +324,10 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes @@ -322,10 +324,10 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
322 @Test 324 @Test
323 public void testTheCopyOfAttrsIntoTSForTheView() throws Exception { 325 public void testTheCopyOfAttrsIntoTSForTheView() throws Exception {
324 Set<String> actualAttributesSet = 326 Set<String> actualAttributesSet =
325 - getAttributesByKeys("{\"caValue1\":\"value1\", \"caValue2\":true, \"caValue3\":42.0, \"caValue4\":73}"); 327 + getAttributesByKeys("{\"caKey1\":\"value1\", \"caKey2\":true, \"caKey3\":42.0, \"caKey4\":73}");
326 328
327 Set<String> expectedActualAttributesSet = 329 Set<String> expectedActualAttributesSet =
328 - new HashSet<>(Arrays.asList("caValue1", "caValue2", "caValue3", "caValue4")); 330 + new HashSet<>(Arrays.asList("caKey1", "caKey2", "caKey3", "caKey4"));
329 assertTrue(actualAttributesSet.containsAll(expectedActualAttributesSet)); 331 assertTrue(actualAttributesSet.containsAll(expectedActualAttributesSet));
330 Thread.sleep(1000); 332 Thread.sleep(1000);
331 333
@@ -333,18 +335,18 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes @@ -333,18 +335,18 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
333 List<Map<String, Object>> values = doGetAsync("/api/plugins/telemetry/ENTITY_VIEW/" + savedView.getId().getId().toString() + 335 List<Map<String, Object>> values = doGetAsync("/api/plugins/telemetry/ENTITY_VIEW/" + savedView.getId().getId().toString() +
334 "/values/attributes?keys=" + String.join(",", actualAttributesSet), List.class); 336 "/values/attributes?keys=" + String.join(",", actualAttributesSet), List.class);
335 337
336 - assertEquals("value1", getValue(values, "caValue1"));  
337 - assertEquals(true, getValue(values, "caValue2"));  
338 - assertEquals(42.0, getValue(values, "caValue3"));  
339 - assertEquals(73, getValue(values, "caValue4")); 338 + assertEquals("value1", getValue(values, "caKey1"));
  339 + assertEquals(true, getValue(values, "caKey2"));
  340 + assertEquals(42.0, getValue(values, "caKey3"));
  341 + assertEquals(73, getValue(values, "caKey4"));
340 } 342 }
341 343
342 @Test 344 @Test
343 public void testTheCopyOfAttrsOutOfTSForTheView() throws Exception { 345 public void testTheCopyOfAttrsOutOfTSForTheView() throws Exception {
344 Set<String> actualAttributesSet = 346 Set<String> actualAttributesSet =
345 - getAttributesByKeys("{\"caValue1\":\"value1\", \"caValue2\":true, \"caValue3\":42.0, \"caValue4\":73}"); 347 + getAttributesByKeys("{\"caKey1\":\"value1\", \"caKey2\":true, \"caKey3\":42.0, \"caKey4\":73}");
346 348
347 - Set<String> expectedActualAttributesSet = new HashSet<>(Arrays.asList("caValue1", "caValue2", "caValue3", "caValue4")); 349 + Set<String> expectedActualAttributesSet = new HashSet<>(Arrays.asList("caKey1", "caKey2", "caKey3", "caKey4"));
348 assertTrue(actualAttributesSet.containsAll(expectedActualAttributesSet)); 350 assertTrue(actualAttributesSet.containsAll(expectedActualAttributesSet));
349 Thread.sleep(1000); 351 Thread.sleep(1000);
350 352
@@ -355,6 +357,7 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes @@ -355,6 +357,7 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
355 view.setEntityId(testDevice.getId()); 357 view.setEntityId(testDevice.getId());
356 view.setTenantId(savedTenant.getId()); 358 view.setTenantId(savedTenant.getId());
357 view.setName("Test entity view"); 359 view.setName("Test entity view");
  360 + view.setType("default");
358 view.setKeys(telemetry); 361 view.setKeys(telemetry);
359 view.setStartTimeMs((long) getValue(valueTelemetryOfDevices, "lastActivityTime") * 10); 362 view.setStartTimeMs((long) getValue(valueTelemetryOfDevices, "lastActivityTime") * 10);
360 view.setEndTimeMs((long) getValue(valueTelemetryOfDevices, "lastActivityTime") / 10); 363 view.setEndTimeMs((long) getValue(valueTelemetryOfDevices, "lastActivityTime") / 10);
@@ -365,6 +368,69 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes @@ -365,6 +368,69 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
365 assertEquals(0, values.size()); 368 assertEquals(0, values.size());
366 } 369 }
367 370
  371 +
  372 + @Test
  373 + public void testGetTelemetryWhenEntityViewTimeRangeInsideTimestampRange() throws Exception {
  374 + uploadTelemetry("{\"tsKey1\":\"value1\", \"tsKey2\":true, \"tsKey3\":40.0}");
  375 + Thread.sleep(1000);
  376 + long startTimeMs = System.currentTimeMillis();
  377 + uploadTelemetry("{\"tsKey1\":\"value2\", \"tsKey2\":false, \"tsKey3\":80.0}");
  378 + Thread.sleep(1000);
  379 + uploadTelemetry("{\"tsKey1\":\"value3\", \"tsKey2\":false, \"tsKey3\":120.0}");
  380 + long endTimeMs = System.currentTimeMillis();
  381 + uploadTelemetry("{\"tsKey1\":\"value4\", \"tsKey2\":true, \"tsKey3\":160.0}");
  382 +
  383 + String deviceId = testDevice.getId().getId().toString();
  384 + Set<String> keys = getTelemetryKeys("DEVICE", deviceId);
  385 + Thread.sleep(1000);
  386 +
  387 + EntityView view = createEntityView("Test entity view", startTimeMs, endTimeMs);
  388 + EntityView savedView = doPost("/api/entityView", view, EntityView.class);
  389 + String entityViewId = savedView.getId().getId().toString();
  390 +
  391 + Map<String, List<Map<String, String>>> expectedValues = getTelemetryValues("DEVICE", deviceId, keys, 0L, (startTimeMs + endTimeMs) / 2);
  392 + Assert.assertEquals(2, expectedValues.get("tsKey1").size());
  393 + Assert.assertEquals(2, expectedValues.get("tsKey2").size());
  394 + Assert.assertEquals(2, expectedValues.get("tsKey3").size());
  395 +
  396 + Map<String, List<Map<String, String>>> actualValues = getTelemetryValues("ENTITY_VIEW", entityViewId, keys, 0L, (startTimeMs + endTimeMs) / 2);
  397 + Assert.assertEquals(1, actualValues.get("tsKey1").size());
  398 + Assert.assertEquals(1, actualValues.get("tsKey2").size());
  399 + Assert.assertEquals(1, actualValues.get("tsKey3").size());
  400 + }
  401 +
  402 + private void uploadTelemetry(String strKvs) throws Exception {
  403 + String viewDeviceId = testDevice.getId().getId().toString();
  404 + DeviceCredentials deviceCredentials =
  405 + doGet("/api/device/" + viewDeviceId + "/credentials", DeviceCredentials.class);
  406 + assertEquals(testDevice.getId(), deviceCredentials.getDeviceId());
  407 +
  408 + String accessToken = deviceCredentials.getCredentialsId();
  409 + assertNotNull(accessToken);
  410 +
  411 + String clientId = MqttAsyncClient.generateClientId();
  412 + MqttAsyncClient client = new MqttAsyncClient("tcp://localhost:1883", clientId);
  413 +
  414 + MqttConnectOptions options = new MqttConnectOptions();
  415 + options.setUserName(accessToken);
  416 + client.connect(options);
  417 + Thread.sleep(3000);
  418 +
  419 + MqttMessage message = new MqttMessage();
  420 + message.setPayload(strKvs.getBytes());
  421 + client.publish("v1/devices/me/telemetry", message);
  422 + Thread.sleep(1000);
  423 + }
  424 +
  425 + private Set<String> getTelemetryKeys(String type, String id) throws Exception {
  426 + return new HashSet<>(doGetAsync("/api/plugins/telemetry/" + type + "/" + id + "/keys/timeseries", List.class));
  427 + }
  428 +
  429 + private Map<String, List<Map<String, String>>> getTelemetryValues(String type, String id, Set<String> keys, Long startTs, Long endTs) throws Exception {
  430 + return doGetAsync("/api/plugins/telemetry/" + type + "/" + id +
  431 + "/values/timeseries?keys=" + String.join(",", keys) + "&startTs=" + startTs + "&endTs=" + endTs, Map.class);
  432 + }
  433 +
368 private Set<String> getAttributesByKeys(String stringKV) throws Exception { 434 private Set<String> getAttributesByKeys(String stringKV) throws Exception {
369 String viewDeviceId = testDevice.getId().getId().toString(); 435 String viewDeviceId = testDevice.getId().getId().toString();
370 DeviceCredentials deviceCredentials = 436 DeviceCredentials deviceCredentials =
@@ -387,7 +453,7 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes @@ -387,7 +453,7 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
387 client.publish("v1/devices/me/attributes", message); 453 client.publish("v1/devices/me/attributes", message);
388 Thread.sleep(1000); 454 Thread.sleep(1000);
389 455
390 - return new HashSet<>(doGetAsync("/api/plugins/telemetry/DEVICE/" + viewDeviceId + "/keys/attributes", List.class)); 456 + return new HashSet<>(doGetAsync("/api/plugins/telemetry/DEVICE/" + viewDeviceId + "/keys/attributes", List.class));
391 } 457 }
392 458
393 private Object getValue(List<Map<String, Object>> values, String stringValue) { 459 private Object getValue(List<Map<String, Object>> values, String stringValue) {
@@ -398,12 +464,20 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes @@ -398,12 +464,20 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
398 } 464 }
399 465
400 private EntityView getNewSavedEntityView(String name) throws Exception { 466 private EntityView getNewSavedEntityView(String name) throws Exception {
  467 + EntityView view = createEntityView(name, 0, 0);
  468 + return doPost("/api/entityView", view, EntityView.class);
  469 + }
  470 +
  471 + private EntityView createEntityView(String name, long startTimeMs, long endTimeMs) {
401 EntityView view = new EntityView(); 472 EntityView view = new EntityView();
402 view.setEntityId(testDevice.getId()); 473 view.setEntityId(testDevice.getId());
403 view.setTenantId(savedTenant.getId()); 474 view.setTenantId(savedTenant.getId());
404 view.setName(name); 475 view.setName(name);
  476 + view.setType("default");
405 view.setKeys(telemetry); 477 view.setKeys(telemetry);
406 - return doPost("/api/entityView", view, EntityView.class); 478 + view.setStartTimeMs(startTimeMs);
  479 + view.setEndTimeMs(endTimeMs);
  480 + return view;
407 } 481 }
408 482
409 private Customer getNewCustomer(String title) { 483 private Customer getNewCustomer(String title) {
@@ -16,6 +16,7 @@ @@ -16,6 +16,7 @@
16 package org.thingsboard.server.mqtt.rpc; 16 package org.thingsboard.server.mqtt.rpc;
17 17
18 import com.datastax.driver.core.utils.UUIDs; 18 import com.datastax.driver.core.utils.UUIDs;
  19 +import io.netty.handler.codec.mqtt.MqttQoS;
19 import lombok.extern.slf4j.Slf4j; 20 import lombok.extern.slf4j.Slf4j;
20 import org.apache.commons.lang3.StringUtils; 21 import org.apache.commons.lang3.StringUtils;
21 import org.eclipse.paho.client.mqttv3.IMqttDeliveryToken; 22 import org.eclipse.paho.client.mqttv3.IMqttDeliveryToken;
@@ -23,19 +24,19 @@ import org.eclipse.paho.client.mqttv3.MqttAsyncClient; @@ -23,19 +24,19 @@ import org.eclipse.paho.client.mqttv3.MqttAsyncClient;
23 import org.eclipse.paho.client.mqttv3.MqttCallback; 24 import org.eclipse.paho.client.mqttv3.MqttCallback;
24 import org.eclipse.paho.client.mqttv3.MqttConnectOptions; 25 import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
25 import org.eclipse.paho.client.mqttv3.MqttMessage; 26 import org.eclipse.paho.client.mqttv3.MqttMessage;
26 -import org.junit.After;  
27 -import org.junit.Assert;  
28 -import org.junit.Before;  
29 -import org.junit.Test; 27 +import org.junit.*;
30 import org.thingsboard.server.common.data.Device; 28 import org.thingsboard.server.common.data.Device;
31 import org.thingsboard.server.common.data.Tenant; 29 import org.thingsboard.server.common.data.Tenant;
32 import org.thingsboard.server.common.data.User; 30 import org.thingsboard.server.common.data.User;
33 import org.thingsboard.server.common.data.security.Authority; 31 import org.thingsboard.server.common.data.security.Authority;
34 import org.thingsboard.server.common.data.security.DeviceCredentials; 32 import org.thingsboard.server.common.data.security.DeviceCredentials;
35 import org.thingsboard.server.controller.AbstractControllerTest; 33 import org.thingsboard.server.controller.AbstractControllerTest;
  34 +import org.thingsboard.server.mqtt.telemetry.AbstractMqttTelemetryIntegrationTest;
36 import org.thingsboard.server.service.security.AccessValidator; 35 import org.thingsboard.server.service.security.AccessValidator;
37 36
38 import java.util.Arrays; 37 import java.util.Arrays;
  38 +import java.util.concurrent.CountDownLatch;
  39 +import java.util.concurrent.TimeUnit;
39 40
40 import static org.junit.Assert.assertEquals; 41 import static org.junit.Assert.assertEquals;
41 import static org.junit.Assert.assertNotNull; 42 import static org.junit.Assert.assertNotNull;
@@ -101,13 +102,19 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC @@ -101,13 +102,19 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC
101 MqttConnectOptions options = new MqttConnectOptions(); 102 MqttConnectOptions options = new MqttConnectOptions();
102 options.setUserName(accessToken); 103 options.setUserName(accessToken);
103 client.connect(options).waitForCompletion(); 104 client.connect(options).waitForCompletion();
104 - client.subscribe("v1/devices/me/rpc/request/+", 1);  
105 - client.setCallback(new TestMqttCallback(client)); 105 +
  106 + CountDownLatch latch = new CountDownLatch(1);
  107 + TestMqttCallback callback = new TestMqttCallback(client, latch);
  108 + client.setCallback(callback);
  109 +
  110 + client.subscribe("v1/devices/me/rpc/request/+", MqttQoS.AT_MOST_ONCE.value());
106 111
107 String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"23\",\"value\": 1}}"; 112 String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"23\",\"value\": 1}}";
108 String deviceId = savedDevice.getId().getId().toString(); 113 String deviceId = savedDevice.getId().getId().toString();
109 String result = doPostAsync("/api/plugins/rpc/oneway/" + deviceId, setGpioRequest, String.class, status().isOk()); 114 String result = doPostAsync("/api/plugins/rpc/oneway/" + deviceId, setGpioRequest, String.class, status().isOk());
110 Assert.assertTrue(StringUtils.isEmpty(result)); 115 Assert.assertTrue(StringUtils.isEmpty(result));
  116 + latch.await(3, TimeUnit.SECONDS);
  117 + assertEquals(MqttQoS.AT_MOST_ONCE.value(), callback.getQoS());
111 } 118 }
112 119
113 @Test 120 @Test
@@ -156,7 +163,7 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC @@ -156,7 +163,7 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC
156 options.setUserName(accessToken); 163 options.setUserName(accessToken);
157 client.connect(options).waitForCompletion(); 164 client.connect(options).waitForCompletion();
158 client.subscribe("v1/devices/me/rpc/request/+", 1); 165 client.subscribe("v1/devices/me/rpc/request/+", 1);
159 - client.setCallback(new TestMqttCallback(client)); 166 + client.setCallback(new TestMqttCallback(client, new CountDownLatch(1)));
160 167
161 String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"23\",\"value\": 1}}"; 168 String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"23\",\"value\": 1}}";
162 String deviceId = savedDevice.getId().getId().toString(); 169 String deviceId = savedDevice.getId().getId().toString();
@@ -204,9 +211,16 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC @@ -204,9 +211,16 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC
204 private static class TestMqttCallback implements MqttCallback { 211 private static class TestMqttCallback implements MqttCallback {
205 212
206 private final MqttAsyncClient client; 213 private final MqttAsyncClient client;
  214 + private final CountDownLatch latch;
  215 + private Integer qoS;
207 216
208 - TestMqttCallback(MqttAsyncClient client) { 217 + TestMqttCallback(MqttAsyncClient client, CountDownLatch latch) {
209 this.client = client; 218 this.client = client;
  219 + this.latch = latch;
  220 + }
  221 +
  222 + int getQoS() {
  223 + return qoS;
210 } 224 }
211 225
212 @Override 226 @Override
@@ -219,7 +233,9 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC @@ -219,7 +233,9 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC
219 MqttMessage message = new MqttMessage(); 233 MqttMessage message = new MqttMessage();
220 String responseTopic = requestTopic.replace("request", "response"); 234 String responseTopic = requestTopic.replace("request", "response");
221 message.setPayload("{\"value1\":\"A\", \"value2\":\"B\"}".getBytes("UTF-8")); 235 message.setPayload("{\"value1\":\"A\", \"value2\":\"B\"}".getBytes("UTF-8"));
  236 + qoS = mqttMessage.getQos();
222 client.publish(responseTopic, message); 237 client.publish(responseTopic, message);
  238 + latch.countDown();
223 } 239 }
224 240
225 @Override 241 @Override
@@ -15,10 +15,9 @@ @@ -15,10 +15,9 @@
15 */ 15 */
16 package org.thingsboard.server.mqtt.telemetry; 16 package org.thingsboard.server.mqtt.telemetry;
17 17
  18 +import io.netty.handler.codec.mqtt.MqttQoS;
18 import lombok.extern.slf4j.Slf4j; 19 import lombok.extern.slf4j.Slf4j;
19 -import org.eclipse.paho.client.mqttv3.MqttAsyncClient;  
20 -import org.eclipse.paho.client.mqttv3.MqttConnectOptions;  
21 -import org.eclipse.paho.client.mqttv3.MqttMessage; 20 +import org.eclipse.paho.client.mqttv3.*;
22 import org.junit.Before; 21 import org.junit.Before;
23 import org.junit.Ignore; 22 import org.junit.Ignore;
24 import org.junit.Test; 23 import org.junit.Test;
@@ -30,9 +29,12 @@ import org.thingsboard.server.dao.service.DaoNoSqlTest; @@ -30,9 +29,12 @@ import org.thingsboard.server.dao.service.DaoNoSqlTest;
30 29
31 import java.net.URI; 30 import java.net.URI;
32 import java.util.*; 31 import java.util.*;
  32 +import java.util.concurrent.CountDownLatch;
  33 +import java.util.concurrent.TimeUnit;
33 34
34 import static org.junit.Assert.assertEquals; 35 import static org.junit.Assert.assertEquals;
35 import static org.junit.Assert.assertNotNull; 36 import static org.junit.Assert.assertNotNull;
  37 +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;
36 38
37 /** 39 /**
38 * @author Valerii Sosliuk 40 * @author Valerii Sosliuk
@@ -94,4 +96,62 @@ public abstract class AbstractMqttTelemetryIntegrationTest extends AbstractContr @@ -94,4 +96,62 @@ public abstract class AbstractMqttTelemetryIntegrationTest extends AbstractContr
94 assertEquals("3.0", values.get("key3").get(0).get("value")); 96 assertEquals("3.0", values.get("key3").get(0).get("value"));
95 assertEquals("4", values.get("key4").get(0).get("value")); 97 assertEquals("4", values.get("key4").get(0).get("value"));
96 } 98 }
  99 +
  100 + @Test
  101 + public void testMqttQoSLevel() throws Exception {
  102 + String clientId = MqttAsyncClient.generateClientId();
  103 + MqttAsyncClient client = new MqttAsyncClient(MQTT_URL, clientId);
  104 +
  105 + MqttConnectOptions options = new MqttConnectOptions();
  106 + options.setUserName(accessToken);
  107 + client.connect(options).waitForCompletion(3000);
  108 + CountDownLatch latch = new CountDownLatch(1);
  109 + TestMqttCallback callback = new TestMqttCallback(client, latch);
  110 + client.setCallback(callback);
  111 + client.subscribe("v1/devices/me/attributes", MqttQoS.AT_MOST_ONCE.value());
  112 + String payload = "{\"key\":\"value\"}";
  113 + String result = doPostAsync("/api/plugins/telemetry/" + savedDevice.getId() + "/SHARED_SCOPE", payload, String.class, status().isOk());
  114 + latch.await(3, TimeUnit.SECONDS);
  115 + assertEquals(payload, callback.getPayload());
  116 + assertEquals(MqttQoS.AT_MOST_ONCE.value(), callback.getQoS());
  117 + }
  118 +
  119 + private static class TestMqttCallback implements MqttCallback {
  120 +
  121 + private final MqttAsyncClient client;
  122 + private final CountDownLatch latch;
  123 + private Integer qoS;
  124 + private String payload;
  125 +
  126 + String getPayload() {
  127 + return payload;
  128 + }
  129 +
  130 + TestMqttCallback(MqttAsyncClient client, CountDownLatch latch) {
  131 + this.client = client;
  132 + this.latch = latch;
  133 + }
  134 +
  135 + int getQoS() {
  136 + return qoS;
  137 + }
  138 +
  139 + @Override
  140 + public void connectionLost(Throwable throwable) {
  141 + }
  142 +
  143 + @Override
  144 + public void messageArrived(String requestTopic, MqttMessage mqttMessage) {
  145 + payload = new String(mqttMessage.getPayload());
  146 + qoS = mqttMessage.getQos();
  147 + latch.countDown();
  148 + }
  149 +
  150 + @Override
  151 + public void deliveryComplete(IMqttDeliveryToken iMqttDeliveryToken) {
  152 +
  153 + }
  154 + }
  155 +
  156 +
97 } 157 }
@@ -40,6 +40,7 @@ public class EntityView extends SearchTextBasedWithAdditionalInfo<EntityViewId> @@ -40,6 +40,7 @@ public class EntityView extends SearchTextBasedWithAdditionalInfo<EntityViewId>
40 private TenantId tenantId; 40 private TenantId tenantId;
41 private CustomerId customerId; 41 private CustomerId customerId;
42 private String name; 42 private String name;
  43 + private String type;
43 private TelemetryEntityView keys; 44 private TelemetryEntityView keys;
44 private long startTimeMs; 45 private long startTimeMs;
45 private long endTimeMs; 46 private long endTimeMs;
@@ -30,6 +30,7 @@ public class EntityViewSearchQuery { @@ -30,6 +30,7 @@ public class EntityViewSearchQuery {
30 30
31 private RelationsSearchParameters parameters; 31 private RelationsSearchParameters parameters;
32 private String relationType; 32 private String relationType;
  33 + private List<String> entityViewTypes;
33 34
34 public EntityRelationsQuery toEntitySearchQuery() { 35 public EntityRelationsQuery toEntitySearchQuery() {
35 EntityRelationsQuery query = new EntityRelationsQuery(); 36 EntityRelationsQuery query = new EntityRelationsQuery();
@@ -15,8 +15,13 @@ @@ -15,8 +15,13 @@
15 */ 15 */
16 package org.thingsboard.server.dao.entityview; 16 package org.thingsboard.server.dao.entityview;
17 17
  18 +import com.datastax.driver.core.ResultSet;
  19 +import com.datastax.driver.core.ResultSetFuture;
18 import com.datastax.driver.core.Statement; 20 import com.datastax.driver.core.Statement;
19 import com.datastax.driver.core.querybuilder.Select; 21 import com.datastax.driver.core.querybuilder.Select;
  22 +import com.datastax.driver.mapping.Result;
  23 +import com.google.common.base.Function;
  24 +import com.google.common.util.concurrent.Futures;
20 import com.google.common.util.concurrent.ListenableFuture; 25 import com.google.common.util.concurrent.ListenableFuture;
21 import lombok.extern.slf4j.Slf4j; 26 import lombok.extern.slf4j.Slf4j;
22 import org.springframework.stereotype.Component; 27 import org.springframework.stereotype.Component;
@@ -30,6 +35,8 @@ import org.thingsboard.server.dao.model.nosql.EntityViewEntity; @@ -30,6 +35,8 @@ import org.thingsboard.server.dao.model.nosql.EntityViewEntity;
30 import org.thingsboard.server.dao.nosql.CassandraAbstractSearchTextDao; 35 import org.thingsboard.server.dao.nosql.CassandraAbstractSearchTextDao;
31 import org.thingsboard.server.dao.util.NoSqlDao; 36 import org.thingsboard.server.dao.util.NoSqlDao;
32 37
  38 +import javax.annotation.Nullable;
  39 +import java.util.ArrayList;
33 import java.util.Arrays; 40 import java.util.Arrays;
34 import java.util.Collections; 41 import java.util.Collections;
35 import java.util.List; 42 import java.util.List;
@@ -39,14 +46,21 @@ import java.util.UUID; @@ -39,14 +46,21 @@ import java.util.UUID;
39 import static com.datastax.driver.core.querybuilder.QueryBuilder.eq; 46 import static com.datastax.driver.core.querybuilder.QueryBuilder.eq;
40 import static com.datastax.driver.core.querybuilder.QueryBuilder.select; 47 import static com.datastax.driver.core.querybuilder.QueryBuilder.select;
41 import static org.thingsboard.server.dao.model.ModelConstants.CUSTOMER_ID_PROPERTY; 48 import static org.thingsboard.server.dao.model.ModelConstants.CUSTOMER_ID_PROPERTY;
  49 +import static org.thingsboard.server.dao.model.ModelConstants.DEVICE_TYPE_PROPERTY;
42 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_ID_COLUMN; 50 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_ID_COLUMN;
  51 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_SUBTYPE_COLUMN_FAMILY_NAME;
  52 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_SUBTYPE_ENTITY_TYPE_PROPERTY;
  53 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_SUBTYPE_TENANT_ID_PROPERTY;
  54 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_AND_TYPE_CF;
43 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_CF; 55 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_CF;
44 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_ENTITY_ID_CF; 56 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_ENTITY_ID_CF;
45 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_NAME; 57 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_NAME;
46 -import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_COLUMN_FAMILY_NAME; 58 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_CF;
  59 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_BY_TYPE_AND_SEARCH_TEXT_CF;
47 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_NAME_PROPERTY; 60 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_NAME_PROPERTY;
48 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TABLE_FAMILY_NAME; 61 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TABLE_FAMILY_NAME;
49 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TENANT_ID_PROPERTY; 62 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TENANT_ID_PROPERTY;
  63 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TYPE_PROPERTY;
50 import static org.thingsboard.server.dao.model.ModelConstants.TENANT_ID_PROPERTY; 64 import static org.thingsboard.server.dao.model.ModelConstants.TENANT_ID_PROPERTY;
51 65
52 /** 66 /**
@@ -82,7 +96,7 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit @@ -82,7 +96,7 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit
82 public List<EntityView> findEntityViewsByTenantId(UUID tenantId, TextPageLink pageLink) { 96 public List<EntityView> findEntityViewsByTenantId(UUID tenantId, TextPageLink pageLink) {
83 log.debug("Try to find entity views by tenantId [{}] and pageLink [{}]", tenantId, pageLink); 97 log.debug("Try to find entity views by tenantId [{}] and pageLink [{}]", tenantId, pageLink);
84 List<EntityViewEntity> entityViewEntities = 98 List<EntityViewEntity> entityViewEntities =
85 - findPageWithTextSearch(ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_COLUMN_FAMILY_NAME, 99 + findPageWithTextSearch(ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_CF,
86 Collections.singletonList(eq(TENANT_ID_PROPERTY, tenantId)), pageLink); 100 Collections.singletonList(eq(TENANT_ID_PROPERTY, tenantId)), pageLink);
87 log.trace("Found entity views [{}] by tenantId [{}] and pageLink [{}]", 101 log.trace("Found entity views [{}] by tenantId [{}] and pageLink [{}]",
88 entityViewEntities, tenantId, pageLink); 102 entityViewEntities, tenantId, pageLink);
@@ -90,6 +104,18 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit @@ -90,6 +104,18 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit
90 } 104 }
91 105
92 @Override 106 @Override
  107 + public List<EntityView> findEntityViewsByTenantIdAndType(UUID tenantId, String type, TextPageLink pageLink) {
  108 + log.debug("Try to find entity views by tenantId [{}], type [{}] and pageLink [{}]", tenantId, type, pageLink);
  109 + List<EntityViewEntity> entityViewEntities =
  110 + findPageWithTextSearch(ENTITY_VIEW_BY_TENANT_BY_TYPE_AND_SEARCH_TEXT_CF,
  111 + Arrays.asList(eq(ENTITY_VIEW_TYPE_PROPERTY, type),
  112 + eq(TENANT_ID_PROPERTY, tenantId)), pageLink);
  113 + log.trace("Found entity views [{}] by tenantId [{}], type [{}] and pageLink [{}]",
  114 + entityViewEntities, tenantId, type, pageLink);
  115 + return DaoUtil.convertDataList(entityViewEntities);
  116 + }
  117 +
  118 + @Override
93 public Optional<EntityView> findEntityViewByTenantIdAndName(UUID tenantId, String name) { 119 public Optional<EntityView> findEntityViewByTenantIdAndName(UUID tenantId, String name) {
94 Select.Where query = select().from(ENTITY_VIEW_BY_TENANT_AND_NAME).where(); 120 Select.Where query = select().from(ENTITY_VIEW_BY_TENANT_AND_NAME).where();
95 query.and(eq(ENTITY_VIEW_TENANT_ID_PROPERTY, tenantId)); 121 query.and(eq(ENTITY_VIEW_TENANT_ID_PROPERTY, tenantId));
@@ -111,6 +137,19 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit @@ -111,6 +137,19 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit
111 } 137 }
112 138
113 @Override 139 @Override
  140 + public List<EntityView> findEntityViewsByTenantIdAndCustomerIdAndType(UUID tenantId, UUID customerId, String type, TextPageLink pageLink) {
  141 + log.debug("Try to find entity views by tenantId [{}], customerId[{}], type [{}] and pageLink [{}]",
  142 + tenantId, customerId, type, pageLink);
  143 + List<EntityViewEntity> entityViewEntities = findPageWithTextSearch(
  144 + ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_AND_TYPE_CF,
  145 + Arrays.asList(eq(DEVICE_TYPE_PROPERTY, type), eq(CUSTOMER_ID_PROPERTY, customerId), eq(TENANT_ID_PROPERTY, tenantId)),
  146 + pageLink);
  147 + log.trace("Found find entity views [{}] by tenantId [{}], customerId [{}], type [{}] and pageLink [{}]",
  148 + entityViewEntities, tenantId, customerId, type, pageLink);
  149 + return DaoUtil.convertDataList(entityViewEntities);
  150 + }
  151 +
  152 + @Override
114 public ListenableFuture<List<EntityView>> findEntityViewsByTenantIdAndEntityIdAsync(UUID tenantId, UUID entityId) { 153 public ListenableFuture<List<EntityView>> findEntityViewsByTenantIdAndEntityIdAsync(UUID tenantId, UUID entityId) {
115 log.debug("Try to find entity views by tenantId [{}] and entityId [{}]", tenantId, entityId); 154 log.debug("Try to find entity views by tenantId [{}] and entityId [{}]", tenantId, entityId);
116 Select.Where query = select().from(ENTITY_VIEW_BY_TENANT_AND_ENTITY_ID_CF).where(); 155 Select.Where query = select().from(ENTITY_VIEW_BY_TENANT_AND_ENTITY_ID_CF).where();
@@ -118,4 +157,30 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit @@ -118,4 +157,30 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit
118 query.and(eq(ENTITY_ID_COLUMN, entityId)); 157 query.and(eq(ENTITY_ID_COLUMN, entityId));
119 return findListByStatementAsync(query); 158 return findListByStatementAsync(query);
120 } 159 }
  160 +
  161 + @Override
  162 + public ListenableFuture<List<EntitySubtype>> findTenantEntityViewTypesAsync(UUID tenantId) {
  163 + Select select = select().from(ENTITY_SUBTYPE_COLUMN_FAMILY_NAME);
  164 + Select.Where query = select.where();
  165 + query.and(eq(ENTITY_SUBTYPE_TENANT_ID_PROPERTY, tenantId));
  166 + query.and(eq(ENTITY_SUBTYPE_ENTITY_TYPE_PROPERTY, EntityType.ENTITY_VIEW));
  167 + query.setConsistencyLevel(cluster.getDefaultReadConsistencyLevel());
  168 + ResultSetFuture resultSetFuture = executeAsyncRead(query);
  169 + return Futures.transform(resultSetFuture, new Function<ResultSet, List<EntitySubtype>>() {
  170 + @Nullable
  171 + @Override
  172 + public List<EntitySubtype> apply(@Nullable ResultSet resultSet) {
  173 + Result<EntitySubtypeEntity> result = cluster.getMapper(EntitySubtypeEntity.class).map(resultSet);
  174 + if (result != null) {
  175 + List<EntitySubtype> entitySubtypes = new ArrayList<>();
  176 + result.all().forEach((entitySubtypeEntity) ->
  177 + entitySubtypes.add(entitySubtypeEntity.toEntitySubtype())
  178 + );
  179 + return entitySubtypes;
  180 + } else {
  181 + return Collections.emptyList();
  182 + }
  183 + }
  184 + });
  185 + }
121 } 186 }
@@ -17,6 +17,7 @@ package org.thingsboard.server.dao.entityview; @@ -17,6 +17,7 @@ package org.thingsboard.server.dao.entityview;
17 17
18 import com.google.common.util.concurrent.ListenableFuture; 18 import com.google.common.util.concurrent.ListenableFuture;
19 import org.thingsboard.server.common.data.Device; 19 import org.thingsboard.server.common.data.Device;
  20 +import org.thingsboard.server.common.data.EntitySubtype;
20 import org.thingsboard.server.common.data.EntityView; 21 import org.thingsboard.server.common.data.EntityView;
21 import org.thingsboard.server.common.data.page.TextPageLink; 22 import org.thingsboard.server.common.data.page.TextPageLink;
22 import org.thingsboard.server.dao.Dao; 23 import org.thingsboard.server.dao.Dao;
@@ -48,6 +49,16 @@ public interface EntityViewDao extends Dao<EntityView> { @@ -48,6 +49,16 @@ public interface EntityViewDao extends Dao<EntityView> {
48 List<EntityView> findEntityViewsByTenantId(UUID tenantId, TextPageLink pageLink); 49 List<EntityView> findEntityViewsByTenantId(UUID tenantId, TextPageLink pageLink);
49 50
50 /** 51 /**
  52 + * Find entity views by tenantId, type and page link.
  53 + *
  54 + * @param tenantId the tenantId
  55 + * @param type the type
  56 + * @param pageLink the page link
  57 + * @return the list of entity view objects
  58 + */
  59 + List<EntityView> findEntityViewsByTenantIdAndType(UUID tenantId, String type, TextPageLink pageLink);
  60 +
  61 + /**
51 * Find entity views by tenantId and entity view name. 62 * Find entity views by tenantId and entity view name.
52 * 63 *
53 * @param tenantId the tenantId 64 * @param tenantId the tenantId
@@ -68,6 +79,27 @@ public interface EntityViewDao extends Dao<EntityView> { @@ -68,6 +79,27 @@ public interface EntityViewDao extends Dao<EntityView> {
68 UUID customerId, 79 UUID customerId,
69 TextPageLink pageLink); 80 TextPageLink pageLink);
70 81
  82 + /**
  83 + * Find entity views by tenantId, customerId, type and page link.
  84 + *
  85 + * @param tenantId the tenantId
  86 + * @param customerId the customerId
  87 + * @param type the type
  88 + * @param pageLink the page link
  89 + * @return the list of entity view objects
  90 + */
  91 + List<EntityView> findEntityViewsByTenantIdAndCustomerIdAndType(UUID tenantId,
  92 + UUID customerId,
  93 + String type,
  94 + TextPageLink pageLink);
71 95
72 ListenableFuture<List<EntityView>> findEntityViewsByTenantIdAndEntityIdAsync(UUID tenantId, UUID entityId); 96 ListenableFuture<List<EntityView>> findEntityViewsByTenantIdAndEntityIdAsync(UUID tenantId, UUID entityId);
  97 +
  98 + /**
  99 + * Find tenants entity view types.
  100 + *
  101 + * @return the list of tenant entity view type objects
  102 + */
  103 + ListenableFuture<List<EntitySubtype>> findTenantEntityViewTypesAsync(UUID tenantId);
  104 +
73 } 105 }
@@ -16,6 +16,7 @@ @@ -16,6 +16,7 @@
16 package org.thingsboard.server.dao.entityview; 16 package org.thingsboard.server.dao.entityview;
17 17
18 import com.google.common.util.concurrent.ListenableFuture; 18 import com.google.common.util.concurrent.ListenableFuture;
  19 +import org.thingsboard.server.common.data.EntitySubtype;
19 import org.thingsboard.server.common.data.EntityView; 20 import org.thingsboard.server.common.data.EntityView;
20 import org.thingsboard.server.common.data.entityview.EntityViewSearchQuery; 21 import org.thingsboard.server.common.data.entityview.EntityViewSearchQuery;
21 import org.thingsboard.server.common.data.id.CustomerId; 22 import org.thingsboard.server.common.data.id.CustomerId;
@@ -44,8 +45,12 @@ public interface EntityViewService { @@ -44,8 +45,12 @@ public interface EntityViewService {
44 45
45 TextPageData<EntityView> findEntityViewByTenantId(TenantId tenantId, TextPageLink pageLink); 46 TextPageData<EntityView> findEntityViewByTenantId(TenantId tenantId, TextPageLink pageLink);
46 47
  48 + TextPageData<EntityView> findEntityViewByTenantIdAndType(TenantId tenantId, TextPageLink pageLink, String type);
  49 +
47 TextPageData<EntityView> findEntityViewsByTenantIdAndCustomerId(TenantId tenantId, CustomerId customerId, TextPageLink pageLink); 50 TextPageData<EntityView> findEntityViewsByTenantIdAndCustomerId(TenantId tenantId, CustomerId customerId, TextPageLink pageLink);
48 51
  52 + TextPageData<EntityView> findEntityViewsByTenantIdAndCustomerIdAndType(TenantId tenantId, CustomerId customerId, TextPageLink pageLink, String type);
  53 +
49 ListenableFuture<List<EntityView>> findEntityViewsByQuery(EntityViewSearchQuery query); 54 ListenableFuture<List<EntityView>> findEntityViewsByQuery(EntityViewSearchQuery query);
50 55
51 ListenableFuture<EntityView> findEntityViewByIdAsync(EntityViewId entityViewId); 56 ListenableFuture<EntityView> findEntityViewByIdAsync(EntityViewId entityViewId);
@@ -55,4 +60,6 @@ public interface EntityViewService { @@ -55,4 +60,6 @@ public interface EntityViewService {
55 void deleteEntityView(EntityViewId entityViewId); 60 void deleteEntityView(EntityViewId entityViewId);
56 61
57 void deleteEntityViewsByTenantId(TenantId tenantId); 62 void deleteEntityViewsByTenantId(TenantId tenantId);
  63 +
  64 + ListenableFuture<List<EntitySubtype>> findEntityViewTypesByTenantId(TenantId tenantId);
58 } 65 }
@@ -15,6 +15,7 @@ @@ -15,6 +15,7 @@
15 */ 15 */
16 package org.thingsboard.server.dao.entityview; 16 package org.thingsboard.server.dao.entityview;
17 17
  18 +import com.google.common.base.Function;
18 import com.google.common.util.concurrent.FutureCallback; 19 import com.google.common.util.concurrent.FutureCallback;
19 import com.google.common.util.concurrent.Futures; 20 import com.google.common.util.concurrent.Futures;
20 import com.google.common.util.concurrent.ListenableFuture; 21 import com.google.common.util.concurrent.ListenableFuture;
@@ -29,6 +30,8 @@ import org.springframework.cache.annotation.Caching; @@ -29,6 +30,8 @@ import org.springframework.cache.annotation.Caching;
29 import org.springframework.stereotype.Service; 30 import org.springframework.stereotype.Service;
30 import org.thingsboard.server.common.data.Customer; 31 import org.thingsboard.server.common.data.Customer;
31 import org.thingsboard.server.common.data.DataConstants; 32 import org.thingsboard.server.common.data.DataConstants;
  33 +import org.thingsboard.server.common.data.Device;
  34 +import org.thingsboard.server.common.data.EntitySubtype;
32 import org.thingsboard.server.common.data.EntityType; 35 import org.thingsboard.server.common.data.EntityType;
33 import org.thingsboard.server.common.data.EntityView; 36 import org.thingsboard.server.common.data.EntityView;
34 import org.thingsboard.server.common.data.Tenant; 37 import org.thingsboard.server.common.data.Tenant;
@@ -54,6 +57,8 @@ import javax.annotation.Nullable; @@ -54,6 +57,8 @@ import javax.annotation.Nullable;
54 import java.util.ArrayList; 57 import java.util.ArrayList;
55 import java.util.Arrays; 58 import java.util.Arrays;
56 import java.util.Collection; 59 import java.util.Collection;
  60 +import java.util.Collections;
  61 +import java.util.Comparator;
57 import java.util.List; 62 import java.util.List;
58 import java.util.concurrent.ExecutionException; 63 import java.util.concurrent.ExecutionException;
59 import java.util.stream.Collectors; 64 import java.util.stream.Collectors;
@@ -63,6 +68,7 @@ import static org.thingsboard.server.common.data.CacheConstants.RELATIONS_CACHE; @@ -63,6 +68,7 @@ import static org.thingsboard.server.common.data.CacheConstants.RELATIONS_CACHE;
63 import static org.thingsboard.server.dao.model.ModelConstants.NULL_UUID; 68 import static org.thingsboard.server.dao.model.ModelConstants.NULL_UUID;
64 import static org.thingsboard.server.dao.service.Validator.validateId; 69 import static org.thingsboard.server.dao.service.Validator.validateId;
65 import static org.thingsboard.server.dao.service.Validator.validatePageLink; 70 import static org.thingsboard.server.dao.service.Validator.validatePageLink;
  71 +import static org.thingsboard.server.dao.service.Validator.validateString;
66 72
67 /** 73 /**
68 * Created by Victor Basanets on 8/28/2017. 74 * Created by Victor Basanets on 8/28/2017.
@@ -158,6 +164,16 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti @@ -158,6 +164,16 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
158 } 164 }
159 165
160 @Override 166 @Override
  167 + public TextPageData<EntityView> findEntityViewByTenantIdAndType(TenantId tenantId, TextPageLink pageLink, String type) {
  168 + log.trace("Executing findEntityViewByTenantIdAndType, tenantId [{}], pageLink [{}], type [{}]", tenantId, pageLink, type);
  169 + validateId(tenantId, INCORRECT_TENANT_ID + tenantId);
  170 + validatePageLink(pageLink, INCORRECT_PAGE_LINK + pageLink);
  171 + validateString(type, "Incorrect type " + type);
  172 + List<EntityView> entityViews = entityViewDao.findEntityViewsByTenantIdAndType(tenantId.getId(), type, pageLink);
  173 + return new TextPageData<>(entityViews, pageLink);
  174 + }
  175 +
  176 + @Override
161 public TextPageData<EntityView> findEntityViewsByTenantIdAndCustomerId(TenantId tenantId, CustomerId customerId, 177 public TextPageData<EntityView> findEntityViewsByTenantIdAndCustomerId(TenantId tenantId, CustomerId customerId,
162 TextPageLink pageLink) { 178 TextPageLink pageLink) {
163 log.trace("Executing findEntityViewByTenantIdAndCustomerId, tenantId [{}], customerId [{}]," + 179 log.trace("Executing findEntityViewByTenantIdAndCustomerId, tenantId [{}], customerId [{}]," +
@@ -171,6 +187,19 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti @@ -171,6 +187,19 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
171 } 187 }
172 188
173 @Override 189 @Override
  190 + public TextPageData<EntityView> findEntityViewsByTenantIdAndCustomerIdAndType(TenantId tenantId, CustomerId customerId, TextPageLink pageLink, String type) {
  191 + log.trace("Executing findEntityViewsByTenantIdAndCustomerIdAndType, tenantId [{}], customerId [{}]," +
  192 + " pageLink [{}], type [{}]", tenantId, customerId, pageLink, type);
  193 + validateId(tenantId, INCORRECT_TENANT_ID + tenantId);
  194 + validateId(customerId, INCORRECT_CUSTOMER_ID + customerId);
  195 + validatePageLink(pageLink, INCORRECT_PAGE_LINK + pageLink);
  196 + validateString(type, "Incorrect type " + type);
  197 + List<EntityView> entityViews = entityViewDao.findEntityViewsByTenantIdAndCustomerIdAndType(tenantId.getId(),
  198 + customerId.getId(), type, pageLink);
  199 + return new TextPageData<>(entityViews, pageLink);
  200 + }
  201 +
  202 + @Override
174 public ListenableFuture<List<EntityView>> findEntityViewsByQuery(EntityViewSearchQuery query) { 203 public ListenableFuture<List<EntityView>> findEntityViewsByQuery(EntityViewSearchQuery query) {
175 ListenableFuture<List<EntityRelation>> relations = relationService.findByQuery(query.toEntitySearchQuery()); 204 ListenableFuture<List<EntityRelation>> relations = relationService.findByQuery(query.toEntitySearchQuery());
176 ListenableFuture<List<EntityView>> entityViews = Futures.transformAsync(relations, r -> { 205 ListenableFuture<List<EntityView>> entityViews = Futures.transformAsync(relations, r -> {
@@ -184,6 +213,15 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti @@ -184,6 +213,15 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
184 } 213 }
185 return Futures.successfulAsList(futures); 214 return Futures.successfulAsList(futures);
186 }); 215 });
  216 +
  217 + entityViews = Futures.transform(entityViews, new Function<List<EntityView>, List<EntityView>>() {
  218 + @Nullable
  219 + @Override
  220 + public List<EntityView> apply(@Nullable List<EntityView> entityViewList) {
  221 + return entityViewList == null ? Collections.emptyList() : entityViewList.stream().filter(entityView -> query.getEntityViewTypes().contains(entityView.getType())).collect(Collectors.toList());
  222 + }
  223 + });
  224 +
187 return entityViews; 225 return entityViews;
188 } 226 }
189 227
@@ -216,6 +254,7 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti @@ -216,6 +254,7 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
216 public void onSuccess(@Nullable List<EntityView> result) { 254 public void onSuccess(@Nullable List<EntityView> result) {
217 cache.putIfAbsent(tenantIdAndEntityId, result); 255 cache.putIfAbsent(tenantIdAndEntityId, result);
218 } 256 }
  257 +
219 @Override 258 @Override
220 public void onFailure(Throwable t) { 259 public void onFailure(Throwable t) {
221 log.error("Error while finding entity views by tenantId and entityId", t); 260 log.error("Error while finding entity views by tenantId and entityId", t);
@@ -243,6 +282,18 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti @@ -243,6 +282,18 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
243 tenantEntityViewRemover.removeEntities(tenantId); 282 tenantEntityViewRemover.removeEntities(tenantId);
244 } 283 }
245 284
  285 + @Override
  286 + public ListenableFuture<List<EntitySubtype>> findEntityViewTypesByTenantId(TenantId tenantId) {
  287 + log.trace("Executing findEntityViewTypesByTenantId, tenantId [{}]", tenantId);
  288 + validateId(tenantId, INCORRECT_TENANT_ID + tenantId);
  289 + ListenableFuture<List<EntitySubtype>> tenantEntityViewTypes = entityViewDao.findTenantEntityViewTypesAsync(tenantId.getId());
  290 + return Futures.transform(tenantEntityViewTypes,
  291 + entityViewTypes -> {
  292 + entityViewTypes.sort(Comparator.comparing(EntitySubtype::getType));
  293 + return entityViewTypes;
  294 + });
  295 + }
  296 +
246 private ListenableFuture<List<Void>> copyAttributesFromEntityToEntityView(EntityView entityView, String scope, Collection<String> keys) { 297 private ListenableFuture<List<Void>> copyAttributesFromEntityToEntityView(EntityView entityView, String scope, Collection<String> keys) {
247 if (keys != null && !keys.isEmpty()) { 298 if (keys != null && !keys.isEmpty()) {
248 ListenableFuture<List<AttributeKvEntry>> getAttrFuture = attributesService.find(entityView.getEntityId(), scope, keys); 299 ListenableFuture<List<AttributeKvEntry>> getAttrFuture = attributesService.find(entityView.getEntityId(), scope, keys);
@@ -296,6 +347,9 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti @@ -296,6 +347,9 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
296 347
297 @Override 348 @Override
298 protected void validateDataImpl(EntityView entityView) { 349 protected void validateDataImpl(EntityView entityView) {
  350 + if (StringUtils.isEmpty(entityView.getType())) {
  351 + throw new DataValidationException("Entity View type should be specified!");
  352 + }
299 if (StringUtils.isEmpty(entityView.getName())) { 353 if (StringUtils.isEmpty(entityView.getName())) {
300 throw new DataValidationException("Entity view name should be specified!"); 354 throw new DataValidationException("Entity view name should be specified!");
301 } 355 }
@@ -145,18 +145,21 @@ public class ModelConstants { @@ -145,18 +145,21 @@ public class ModelConstants {
145 /** 145 /**
146 * Cassandra entityView constants. 146 * Cassandra entityView constants.
147 */ 147 */
148 - public static final String ENTITY_VIEW_TABLE_FAMILY_NAME = "entity_views"; 148 + public static final String ENTITY_VIEW_TABLE_FAMILY_NAME = "entity_view";
149 public static final String ENTITY_VIEW_ENTITY_ID_PROPERTY = ENTITY_ID_COLUMN; 149 public static final String ENTITY_VIEW_ENTITY_ID_PROPERTY = ENTITY_ID_COLUMN;
150 public static final String ENTITY_VIEW_TENANT_ID_PROPERTY = TENANT_ID_PROPERTY; 150 public static final String ENTITY_VIEW_TENANT_ID_PROPERTY = TENANT_ID_PROPERTY;
151 public static final String ENTITY_VIEW_CUSTOMER_ID_PROPERTY = CUSTOMER_ID_PROPERTY; 151 public static final String ENTITY_VIEW_CUSTOMER_ID_PROPERTY = CUSTOMER_ID_PROPERTY;
152 public static final String ENTITY_VIEW_NAME_PROPERTY = DEVICE_NAME_PROPERTY; 152 public static final String ENTITY_VIEW_NAME_PROPERTY = DEVICE_NAME_PROPERTY;
153 public static final String ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_CF = "entity_view_by_tenant_and_customer"; 153 public static final String ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_CF = "entity_view_by_tenant_and_customer";
  154 + public static final String ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_AND_TYPE_CF = "entity_view_by_tenant_and_customer_and_type";
154 public static final String ENTITY_VIEW_BY_TENANT_AND_ENTITY_ID_CF = "entity_view_by_tenant_and_entity_id"; 155 public static final String ENTITY_VIEW_BY_TENANT_AND_ENTITY_ID_CF = "entity_view_by_tenant_and_entity_id";
155 public static final String ENTITY_VIEW_KEYS_PROPERTY = "keys"; 156 public static final String ENTITY_VIEW_KEYS_PROPERTY = "keys";
  157 + public static final String ENTITY_VIEW_TYPE_PROPERTY = "type";
156 public static final String ENTITY_VIEW_START_TS_PROPERTY = "start_ts"; 158 public static final String ENTITY_VIEW_START_TS_PROPERTY = "start_ts";
157 public static final String ENTITY_VIEW_END_TS_PROPERTY = "end_ts"; 159 public static final String ENTITY_VIEW_END_TS_PROPERTY = "end_ts";
158 public static final String ENTITY_VIEW_ADDITIONAL_INFO_PROPERTY = ADDITIONAL_INFO_PROPERTY; 160 public static final String ENTITY_VIEW_ADDITIONAL_INFO_PROPERTY = ADDITIONAL_INFO_PROPERTY;
159 - public static final String ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_COLUMN_FAMILY_NAME = "entity_view_by_tenant_and_search_text"; 161 + public static final String ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_CF = "entity_view_by_tenant_and_search_text";
  162 + public static final String ENTITY_VIEW_BY_TENANT_BY_TYPE_AND_SEARCH_TEXT_CF = "entity_view_by_tenant_by_type_and_search_text";
160 public static final String ENTITY_VIEW_BY_TENANT_AND_NAME = "entity_view_by_tenant_and_name"; 163 public static final String ENTITY_VIEW_BY_TENANT_AND_NAME = "entity_view_by_tenant_and_name";
161 164
162 /** 165 /**
@@ -41,6 +41,7 @@ import javax.persistence.Enumerated; @@ -41,6 +41,7 @@ import javax.persistence.Enumerated;
41 import java.io.IOException; 41 import java.io.IOException;
42 import java.util.UUID; 42 import java.util.UUID;
43 43
  44 +import static org.thingsboard.server.dao.model.ModelConstants.DEVICE_TYPE_PROPERTY;
44 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_TYPE_PROPERTY; 45 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_TYPE_PROPERTY;
45 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TABLE_FAMILY_NAME; 46 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TABLE_FAMILY_NAME;
46 import static org.thingsboard.server.dao.model.ModelConstants.ID_PROPERTY; 47 import static org.thingsboard.server.dao.model.ModelConstants.ID_PROPERTY;
@@ -71,6 +72,10 @@ public class EntityViewEntity implements SearchTextEntity<EntityView> { @@ -71,6 +72,10 @@ public class EntityViewEntity implements SearchTextEntity<EntityView> {
71 @Column(name = ModelConstants.ENTITY_VIEW_CUSTOMER_ID_PROPERTY) 72 @Column(name = ModelConstants.ENTITY_VIEW_CUSTOMER_ID_PROPERTY)
72 private UUID customerId; 73 private UUID customerId;
73 74
  75 + @PartitionKey(value = 3)
  76 + @Column(name = DEVICE_TYPE_PROPERTY)
  77 + private String type;
  78 +
74 @Column(name = ModelConstants.ENTITY_VIEW_ENTITY_ID_PROPERTY) 79 @Column(name = ModelConstants.ENTITY_VIEW_ENTITY_ID_PROPERTY)
75 private UUID entityId; 80 private UUID entityId;
76 81
@@ -113,6 +118,7 @@ public class EntityViewEntity implements SearchTextEntity<EntityView> { @@ -113,6 +118,7 @@ public class EntityViewEntity implements SearchTextEntity<EntityView> {
113 if (entityView.getCustomerId() != null) { 118 if (entityView.getCustomerId() != null) {
114 this.customerId = entityView.getCustomerId().getId(); 119 this.customerId = entityView.getCustomerId().getId();
115 } 120 }
  121 + this.type = entityView.getType();
116 this.name = entityView.getName(); 122 this.name = entityView.getName();
117 try { 123 try {
118 this.keys = mapper.writeValueAsString(entityView.getKeys()); 124 this.keys = mapper.writeValueAsString(entityView.getKeys());
@@ -143,6 +149,7 @@ public class EntityViewEntity implements SearchTextEntity<EntityView> { @@ -143,6 +149,7 @@ public class EntityViewEntity implements SearchTextEntity<EntityView> {
143 if (customerId != null) { 149 if (customerId != null) {
144 entityView.setCustomerId(new CustomerId(customerId)); 150 entityView.setCustomerId(new CustomerId(customerId));
145 } 151 }
  152 + entityView.setType(type);
146 entityView.setName(name); 153 entityView.setName(name);
147 try { 154 try {
148 entityView.setKeys(mapper.readValue(keys, TelemetryEntityView.class)); 155 entityView.setKeys(mapper.readValue(keys, TelemetryEntityView.class));
@@ -69,6 +69,9 @@ public class EntityViewEntity extends BaseSqlEntity<EntityView> implements Searc @@ -69,6 +69,9 @@ public class EntityViewEntity extends BaseSqlEntity<EntityView> implements Searc
69 @Column(name = ModelConstants.ENTITY_VIEW_CUSTOMER_ID_PROPERTY) 69 @Column(name = ModelConstants.ENTITY_VIEW_CUSTOMER_ID_PROPERTY)
70 private String customerId; 70 private String customerId;
71 71
  72 + @Column(name = ModelConstants.DEVICE_TYPE_PROPERTY)
  73 + private String type;
  74 +
72 @Column(name = ModelConstants.ENTITY_VIEW_NAME_PROPERTY) 75 @Column(name = ModelConstants.ENTITY_VIEW_NAME_PROPERTY)
73 private String name; 76 private String name;
74 77
@@ -108,6 +111,7 @@ public class EntityViewEntity extends BaseSqlEntity<EntityView> implements Searc @@ -108,6 +111,7 @@ public class EntityViewEntity extends BaseSqlEntity<EntityView> implements Searc
108 if (entityView.getCustomerId() != null) { 111 if (entityView.getCustomerId() != null) {
109 this.customerId = toString(entityView.getCustomerId().getId()); 112 this.customerId = toString(entityView.getCustomerId().getId());
110 } 113 }
  114 + this.type = entityView.getType();
111 this.name = entityView.getName(); 115 this.name = entityView.getName();
112 try { 116 try {
113 this.keys = mapper.writeValueAsString(entityView.getKeys()); 117 this.keys = mapper.writeValueAsString(entityView.getKeys());
@@ -144,6 +148,7 @@ public class EntityViewEntity extends BaseSqlEntity<EntityView> implements Searc @@ -144,6 +148,7 @@ public class EntityViewEntity extends BaseSqlEntity<EntityView> implements Searc
144 if (customerId != null) { 148 if (customerId != null) {
145 entityView.setCustomerId(new CustomerId(toUUID(customerId))); 149 entityView.setCustomerId(new CustomerId(toUUID(customerId)));
146 } 150 }
  151 + entityView.setType(type);
147 entityView.setName(name); 152 entityView.setName(name);
148 try { 153 try {
149 entityView.setKeys(mapper.readValue(keys, TelemetryEntityView.class)); 154 entityView.setKeys(mapper.readValue(keys, TelemetryEntityView.class));
@@ -19,8 +19,6 @@ import org.springframework.data.domain.Pageable; @@ -19,8 +19,6 @@ import org.springframework.data.domain.Pageable;
19 import org.springframework.data.jpa.repository.Query; 19 import org.springframework.data.jpa.repository.Query;
20 import org.springframework.data.repository.CrudRepository; 20 import org.springframework.data.repository.CrudRepository;
21 import org.springframework.data.repository.query.Param; 21 import org.springframework.data.repository.query.Param;
22 -import org.thingsboard.server.common.data.EntityView;  
23 -import org.thingsboard.server.common.data.id.EntityId;  
24 import org.thingsboard.server.dao.model.sql.EntityViewEntity; 22 import org.thingsboard.server.dao.model.sql.EntityViewEntity;
25 import org.thingsboard.server.dao.util.SqlDao; 23 import org.thingsboard.server.dao.util.SqlDao;
26 24
@@ -36,21 +34,46 @@ public interface EntityViewRepository extends CrudRepository<EntityViewEntity, S @@ -36,21 +34,46 @@ public interface EntityViewRepository extends CrudRepository<EntityViewEntity, S
36 "AND LOWER(e.searchText) LIKE LOWER(CONCAT(:textSearch, '%')) " + 34 "AND LOWER(e.searchText) LIKE LOWER(CONCAT(:textSearch, '%')) " +
37 "AND e.id > :idOffset ORDER BY e.id") 35 "AND e.id > :idOffset ORDER BY e.id")
38 List<EntityViewEntity> findByTenantId(@Param("tenantId") String tenantId, 36 List<EntityViewEntity> findByTenantId(@Param("tenantId") String tenantId,
39 - @Param("textSearch") String textSearch,  
40 - @Param("idOffset") String idOffset,  
41 - Pageable pageable); 37 + @Param("textSearch") String textSearch,
  38 + @Param("idOffset") String idOffset,
  39 + Pageable pageable);
  40 +
  41 + @Query("SELECT e FROM EntityViewEntity e WHERE e.tenantId = :tenantId " +
  42 + "AND e.type = :type " +
  43 + "AND LOWER(e.searchText) LIKE LOWER(CONCAT(:textSearch, '%')) " +
  44 + "AND e.id > :idOffset ORDER BY e.id")
  45 + List<EntityViewEntity> findByTenantIdAndType(@Param("tenantId") String tenantId,
  46 + @Param("type") String type,
  47 + @Param("textSearch") String textSearch,
  48 + @Param("idOffset") String idOffset,
  49 + Pageable pageable);
42 50
43 @Query("SELECT e FROM EntityViewEntity e WHERE e.tenantId = :tenantId " + 51 @Query("SELECT e FROM EntityViewEntity e WHERE e.tenantId = :tenantId " +
44 "AND e.customerId = :customerId " + 52 "AND e.customerId = :customerId " +
45 "AND LOWER(e.searchText) LIKE LOWER(CONCAT(:searchText, '%')) " + 53 "AND LOWER(e.searchText) LIKE LOWER(CONCAT(:searchText, '%')) " +
46 "AND e.id > :idOffset ORDER BY e.id") 54 "AND e.id > :idOffset ORDER BY e.id")
47 List<EntityViewEntity> findByTenantIdAndCustomerId(@Param("tenantId") String tenantId, 55 List<EntityViewEntity> findByTenantIdAndCustomerId(@Param("tenantId") String tenantId,
48 - @Param("customerId") String customerId,  
49 - @Param("searchText") String searchText,  
50 - @Param("idOffset") String idOffset,  
51 - Pageable pageable); 56 + @Param("customerId") String customerId,
  57 + @Param("searchText") String searchText,
  58 + @Param("idOffset") String idOffset,
  59 + Pageable pageable);
  60 +
  61 + @Query("SELECT e FROM EntityViewEntity e WHERE e.tenantId = :tenantId " +
  62 + "AND e.customerId = :customerId " +
  63 + "AND e.type = :type " +
  64 + "AND LOWER(e.searchText) LIKE LOWER(CONCAT(:searchText, '%')) " +
  65 + "AND e.id > :idOffset ORDER BY e.id")
  66 + List<EntityViewEntity> findByTenantIdAndCustomerIdAndType(@Param("tenantId") String tenantId,
  67 + @Param("customerId") String customerId,
  68 + @Param("type") String type,
  69 + @Param("searchText") String searchText,
  70 + @Param("idOffset") String idOffset,
  71 + Pageable pageable);
52 72
53 EntityViewEntity findByTenantIdAndName(String tenantId, String name); 73 EntityViewEntity findByTenantIdAndName(String tenantId, String name);
54 74
55 List<EntityViewEntity> findAllByTenantIdAndEntityId(String tenantId, String entityId); 75 List<EntityViewEntity> findAllByTenantIdAndEntityId(String tenantId, String entityId);
  76 +
  77 + @Query("SELECT DISTINCT ev.type FROM EntityViewEntity ev WHERE ev.tenantId = :tenantId")
  78 + List<String> findTenantEntityViewTypes(@Param("tenantId") String tenantId);
56 } 79 }
@@ -24,7 +24,6 @@ import org.thingsboard.server.common.data.EntitySubtype; @@ -24,7 +24,6 @@ import org.thingsboard.server.common.data.EntitySubtype;
24 import org.thingsboard.server.common.data.EntityType; 24 import org.thingsboard.server.common.data.EntityType;
25 import org.thingsboard.server.common.data.EntityView; 25 import org.thingsboard.server.common.data.EntityView;
26 import org.thingsboard.server.common.data.UUIDConverter; 26 import org.thingsboard.server.common.data.UUIDConverter;
27 -import org.thingsboard.server.common.data.id.EntityId;  
28 import org.thingsboard.server.common.data.id.TenantId; 27 import org.thingsboard.server.common.data.id.TenantId;
29 import org.thingsboard.server.common.data.page.TextPageLink; 28 import org.thingsboard.server.common.data.page.TextPageLink;
30 import org.thingsboard.server.dao.DaoUtil; 29 import org.thingsboard.server.dao.DaoUtil;
@@ -41,7 +40,6 @@ import java.util.Optional; @@ -41,7 +40,6 @@ import java.util.Optional;
41 import java.util.UUID; 40 import java.util.UUID;
42 41
43 import static org.thingsboard.server.common.data.UUIDConverter.fromTimeUUID; 42 import static org.thingsboard.server.common.data.UUIDConverter.fromTimeUUID;
44 -import static org.thingsboard.server.common.data.UUIDConverter.fromTimeUUIDs;  
45 import static org.thingsboard.server.dao.model.ModelConstants.NULL_UUID_STR; 43 import static org.thingsboard.server.dao.model.ModelConstants.NULL_UUID_STR;
46 44
47 /** 45 /**
@@ -76,6 +74,17 @@ public class JpaEntityViewDao extends JpaAbstractSearchTextDao<EntityViewEntity, @@ -76,6 +74,17 @@ public class JpaEntityViewDao extends JpaAbstractSearchTextDao<EntityViewEntity,
76 } 74 }
77 75
78 @Override 76 @Override
  77 + public List<EntityView> findEntityViewsByTenantIdAndType(UUID tenantId, String type, TextPageLink pageLink) {
  78 + return DaoUtil.convertDataList(
  79 + entityViewRepository.findByTenantIdAndType(
  80 + fromTimeUUID(tenantId),
  81 + type,
  82 + Objects.toString(pageLink.getTextSearch(), ""),
  83 + pageLink.getIdOffset() == null ? NULL_UUID_STR : fromTimeUUID(pageLink.getIdOffset()),
  84 + new PageRequest(0, pageLink.getLimit())));
  85 + }
  86 +
  87 + @Override
79 public Optional<EntityView> findEntityViewByTenantIdAndName(UUID tenantId, String name) { 88 public Optional<EntityView> findEntityViewByTenantIdAndName(UUID tenantId, String name) {
80 return Optional.ofNullable( 89 return Optional.ofNullable(
81 DaoUtil.getData(entityViewRepository.findByTenantIdAndName(fromTimeUUID(tenantId), name))); 90 DaoUtil.getData(entityViewRepository.findByTenantIdAndName(fromTimeUUID(tenantId), name)));
@@ -96,8 +105,37 @@ public class JpaEntityViewDao extends JpaAbstractSearchTextDao<EntityViewEntity, @@ -96,8 +105,37 @@ public class JpaEntityViewDao extends JpaAbstractSearchTextDao<EntityViewEntity,
96 } 105 }
97 106
98 @Override 107 @Override
  108 + public List<EntityView> findEntityViewsByTenantIdAndCustomerIdAndType(UUID tenantId, UUID customerId, String type, TextPageLink pageLink) {
  109 + return DaoUtil.convertDataList(
  110 + entityViewRepository.findByTenantIdAndCustomerIdAndType(
  111 + fromTimeUUID(tenantId),
  112 + fromTimeUUID(customerId),
  113 + type,
  114 + Objects.toString(pageLink.getTextSearch(), ""),
  115 + pageLink.getIdOffset() == null ? NULL_UUID_STR : fromTimeUUID(pageLink.getIdOffset()),
  116 + new PageRequest(0, pageLink.getLimit())
  117 + ));
  118 + }
  119 +
  120 + @Override
99 public ListenableFuture<List<EntityView>> findEntityViewsByTenantIdAndEntityIdAsync(UUID tenantId, UUID entityId) { 121 public ListenableFuture<List<EntityView>> findEntityViewsByTenantIdAndEntityIdAsync(UUID tenantId, UUID entityId) {
100 return service.submit(() -> DaoUtil.convertDataList( 122 return service.submit(() -> DaoUtil.convertDataList(
101 entityViewRepository.findAllByTenantIdAndEntityId(UUIDConverter.fromTimeUUID(tenantId), UUIDConverter.fromTimeUUID(entityId)))); 123 entityViewRepository.findAllByTenantIdAndEntityId(UUIDConverter.fromTimeUUID(tenantId), UUIDConverter.fromTimeUUID(entityId))));
102 } 124 }
  125 +
  126 + @Override
  127 + public ListenableFuture<List<EntitySubtype>> findTenantEntityViewTypesAsync(UUID tenantId) {
  128 + return service.submit(() -> convertTenantEntityViewTypesToDto(tenantId, entityViewRepository.findTenantEntityViewTypes(fromTimeUUID(tenantId))));
  129 + }
  130 +
  131 + private List<EntitySubtype> convertTenantEntityViewTypesToDto(UUID tenantId, List<String> types) {
  132 + List<EntitySubtype> list = Collections.emptyList();
  133 + if (types != null && !types.isEmpty()) {
  134 + list = new ArrayList<>();
  135 + for (String type : types) {
  136 + list.add(new EntitySubtype(new TenantId(tenantId), EntityType.ENTITY_VIEW, type));
  137 + }
  138 + }
  139 + return list;
  140 + }
103 } 141 }
@@ -87,7 +87,11 @@ public class BaseTimeseriesService implements TimeseriesService { @@ -87,7 +87,11 @@ public class BaseTimeseriesService implements TimeseriesService {
87 .map(key -> new BaseReadTsKvQuery(key, entityView.getStartTimeMs(), entityView.getEndTimeMs(), 1, "ASC")) 87 .map(key -> new BaseReadTsKvQuery(key, entityView.getStartTimeMs(), entityView.getEndTimeMs(), 1, "ASC"))
88 .collect(Collectors.toList()); 88 .collect(Collectors.toList());
89 89
90 - return timeseriesDao.findAllAsync(entityView.getEntityId(), updateQueriesForEntityView(entityView, queries)); 90 + if (queries.size() > 0) {
  91 + return timeseriesDao.findAllAsync(entityView.getEntityId(), queries);
  92 + } else {
  93 + return Futures.immediateFuture(new ArrayList<>());
  94 + }
91 } 95 }
92 keys.forEach(key -> futures.add(timeseriesDao.findLatest(entityId, key))); 96 keys.forEach(key -> futures.add(timeseriesDao.findLatest(entityId, key)));
93 return Futures.allAsList(futures); 97 return Futures.allAsList(futures);
@@ -133,11 +137,20 @@ public class BaseTimeseriesService implements TimeseriesService { @@ -133,11 +137,20 @@ public class BaseTimeseriesService implements TimeseriesService {
133 137
134 private List<ReadTsKvQuery> updateQueriesForEntityView(EntityView entityView, List<ReadTsKvQuery> queries) { 138 private List<ReadTsKvQuery> updateQueriesForEntityView(EntityView entityView, List<ReadTsKvQuery> queries) {
135 return queries.stream().map(query -> { 139 return queries.stream().map(query -> {
136 - long startTs = entityView.getStartTimeMs() == 0 ? query.getStartTs() : entityView.getStartTimeMs();  
137 - long endTs = entityView.getEndTimeMs() == 0 ? query.getEndTs() : entityView.getEndTimeMs(); 140 + long startTs;
  141 + if (entityView.getStartTimeMs() != 0 && entityView.getStartTimeMs() > query.getStartTs()) {
  142 + startTs = entityView.getStartTimeMs();
  143 + } else {
  144 + startTs = query.getStartTs();
  145 + }
138 146
139 - return startTs <= query.getStartTs() && endTs >= query.getEndTs() ? query :  
140 - new BaseReadTsKvQuery(query.getKey(), startTs, endTs, query.getInterval(), query.getLimit(), query.getAggregation()); 147 + long endTs;
  148 + if (entityView.getEndTimeMs() != 0 && entityView.getEndTimeMs() < query.getEndTs()) {
  149 + endTs = entityView.getEndTimeMs();
  150 + } else {
  151 + endTs = query.getEndTs();
  152 + }
  153 + return new BaseReadTsKvQuery(query.getKey(), startTs, endTs, query.getInterval(), query.getLimit(), query.getAggregation());
141 }).collect(Collectors.toList()); 154 }).collect(Collectors.toList());
142 } 155 }
143 156
@@ -624,61 +624,90 @@ CREATE TABLE IF NOT EXISTS thingsboard.rule_node ( @@ -624,61 +624,90 @@ CREATE TABLE IF NOT EXISTS thingsboard.rule_node (
624 PRIMARY KEY (id) 624 PRIMARY KEY (id)
625 ); 625 );
626 626
627 -CREATE TABLE IF NOT EXISTS thingsboard.entity_views ( 627 +CREATE TABLE IF NOT EXISTS thingsboard.entity_view (
628 id timeuuid, 628 id timeuuid,
629 entity_id timeuuid, 629 entity_id timeuuid,
630 entity_type text, 630 entity_type text,
631 tenant_id timeuuid, 631 tenant_id timeuuid,
632 customer_id timeuuid, 632 customer_id timeuuid,
633 name text, 633 name text,
  634 + type text,
634 keys text, 635 keys text,
635 start_ts bigint, 636 start_ts bigint,
636 end_ts bigint, 637 end_ts bigint,
637 search_text text, 638 search_text text,
638 additional_info text, 639 additional_info text,
639 - PRIMARY KEY (id, entity_id, tenant_id, customer_id) 640 + PRIMARY KEY (id, entity_id, tenant_id, customer_id, type)
640 ); 641 );
641 642
642 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_name AS 643 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_name AS
643 SELECT * 644 SELECT *
644 - from thingsboard.entity_views 645 + from thingsboard.entity_view
645 WHERE tenant_id IS NOT NULL 646 WHERE tenant_id IS NOT NULL
646 AND entity_id IS NOT NULL 647 AND entity_id IS NOT NULL
647 AND customer_id IS NOT NULL 648 AND customer_id IS NOT NULL
  649 + AND type IS NOT NULL
648 AND name IS NOT NULL 650 AND name IS NOT NULL
649 AND id IS NOT NULL 651 AND id IS NOT NULL
650 - PRIMARY KEY (tenant_id, name, id, customer_id, entity_id) 652 + PRIMARY KEY (tenant_id, name, id, customer_id, entity_id, type)
651 WITH CLUSTERING ORDER BY (name ASC, id DESC, customer_id DESC); 653 WITH CLUSTERING ORDER BY (name ASC, id DESC, customer_id DESC);
652 654
653 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_search_text AS 655 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_search_text AS
654 SELECT * 656 SELECT *
655 - from thingsboard.entity_views 657 + from thingsboard.entity_view
656 WHERE tenant_id IS NOT NULL 658 WHERE tenant_id IS NOT NULL
657 AND entity_id IS NOT NULL 659 AND entity_id IS NOT NULL
658 AND customer_id IS NOT NULL 660 AND customer_id IS NOT NULL
  661 + AND type IS NOT NULL
659 AND search_text IS NOT NULL 662 AND search_text IS NOT NULL
660 AND id IS NOT NULL 663 AND id IS NOT NULL
661 - PRIMARY KEY (tenant_id, search_text, id, customer_id, entity_id) 664 + PRIMARY KEY (tenant_id, search_text, id, customer_id, entity_id, type)
662 WITH CLUSTERING ORDER BY (search_text ASC, id DESC, customer_id DESC); 665 WITH CLUSTERING ORDER BY (search_text ASC, id DESC, customer_id DESC);
663 666
  667 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_by_type_and_search_text AS
  668 + SELECT *
  669 + from thingsboard.entity_view
  670 + WHERE tenant_id IS NOT NULL
  671 + AND entity_id IS NOT NULL
  672 + AND customer_id IS NOT NULL
  673 + AND type IS NOT NULL
  674 + AND search_text IS NOT NULL
  675 + AND id IS NOT NULL
  676 + PRIMARY KEY (tenant_id, type, search_text, id, customer_id, entity_id)
  677 + WITH CLUSTERING ORDER BY (type ASC, search_text ASC, id DESC, customer_id DESC);
  678 +
664 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_customer AS 679 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_customer AS
665 SELECT * 680 SELECT *
666 - from thingsboard.entity_views 681 + from thingsboard.entity_view
667 WHERE tenant_id IS NOT NULL 682 WHERE tenant_id IS NOT NULL
668 AND customer_id IS NOT NULL 683 AND customer_id IS NOT NULL
669 AND entity_id IS NOT NULL 684 AND entity_id IS NOT NULL
  685 + AND type IS NOT NULL
670 AND search_text IS NOT NULL 686 AND search_text IS NOT NULL
671 AND id IS NOT NULL 687 AND id IS NOT NULL
672 - PRIMARY KEY (tenant_id, customer_id, search_text, id, entity_id) 688 + PRIMARY KEY (tenant_id, customer_id, search_text, id, entity_id, type)
673 WITH CLUSTERING ORDER BY (customer_id DESC, search_text ASC, id DESC); 689 WITH CLUSTERING ORDER BY (customer_id DESC, search_text ASC, id DESC);
674 690
  691 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_customer_and_type AS
  692 + SELECT *
  693 + from thingsboard.entity_view
  694 + WHERE tenant_id IS NOT NULL
  695 + AND customer_id IS NOT NULL
  696 + AND entity_id IS NOT NULL
  697 + AND type IS NOT NULL
  698 + AND search_text IS NOT NULL
  699 + AND id IS NOT NULL
  700 + PRIMARY KEY (tenant_id, type, customer_id, search_text, id, entity_id)
  701 + WITH CLUSTERING ORDER BY (type ASC, customer_id DESC, search_text ASC, id DESC);
  702 +
675 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_entity_id AS 703 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_entity_id AS
676 SELECT * 704 SELECT *
677 - from thingsboard.entity_views 705 + from thingsboard.entity_view
678 WHERE tenant_id IS NOT NULL 706 WHERE tenant_id IS NOT NULL
679 AND customer_id IS NOT NULL 707 AND customer_id IS NOT NULL
680 AND entity_id IS NOT NULL 708 AND entity_id IS NOT NULL
  709 + AND type IS NOT NULL
681 AND search_text IS NOT NULL 710 AND search_text IS NOT NULL
682 AND id IS NOT NULL 711 AND id IS NOT NULL
683 - PRIMARY KEY (tenant_id, entity_id, customer_id, search_text, id) 712 + PRIMARY KEY (tenant_id, entity_id, customer_id, search_text, id, type)
684 WITH CLUSTERING ORDER BY (entity_id DESC, customer_id DESC, search_text ASC, id DESC); 713 WITH CLUSTERING ORDER BY (entity_id DESC, customer_id DESC, search_text ASC, id DESC);
@@ -41,242 +41,4 @@ VALUES ( now ( ), 'mail', '{ @@ -41,242 +41,4 @@ VALUES ( now ( ), 'mail', '{
41 "enableTls": "false", 41 "enableTls": "false",
42 "username": "", 42 "username": "",
43 "password": "" 43 "password": ""
44 -}' );  
45 -  
46 -/** System widgets library **/  
47 -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )  
48 -VALUES ( now ( ), minTimeuuid ( 0 ), 'gpio_widgets', 'gpio widgets', 'GPIO widgets' );  
49 -  
50 -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )  
51 -VALUES ( now ( ), minTimeuuid ( 0 ), 'maps', 'maps', 'Maps' );  
52 -  
53 -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )  
54 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges', 'digital gauges', 'Digital gauges' );  
55 -  
56 -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )  
57 -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'charts', 'Charts' );  
58 -  
59 -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )  
60 -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'cards', 'Cards' );  
61 -  
62 -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )  
63 -VALUES ( now ( ), minTimeuuid ( 0 ), 'analogue_gauges', 'analogue gauges', 'Analogue gauges' );  
64 -  
65 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
66 -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'timeseries',  
67 -'{"type":"timeseries","sizeX":8,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.4.0/Chart.min.js"}],"templateHtml":"<canvas id=\"lineChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n \n var lineData = {\n labels: [],\n datasets: []\n };\n \n for (var i=0; i < self.ctx.data.length; i++) {\n var dataKey = self.ctx.data[i].dataKey;\n var keySettings = dataKey.settings;\n var backgroundColor = tinycolor(dataKey.color);\n backgroundColor.setAlpha(0.4);\n var dataset = {\n label: dataKey.label,\n data: [],\n borderColor: dataKey.color,\n borderWidth: 2,\n backgroundColor: backgroundColor.toRgbString(),\n pointRadius: keySettings.showPoints ? 1 : 0,\n fill: keySettings.fillLines || false,\n showLine: keySettings.showLines || true,\n spanGaps: false,\n lineTension: angular.isDefined(keySettings.tension) ? keySettings.tension : 0.2\n }\n lineData.datasets.push(dataset);\n }\n\n var ctx = $(''#lineChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''line'',\n data: lineData,\n options: {\n responsive: false,\n tooltips: {\n mode: ''index'',\n inersect: true\n },\n hover: {\n mode: ''index'',\n inersect: true\n },\n maintainAspectRatio: false,\n /*animation: {\n duration: 200,\n easing: ''linear''\n },*/\n elements: {\n line: {\n tension: 0.2\n } \n },\n scales: {\n xAxes: [{\n type: ''time'',\n ticks: {\n maxRotation: 20,\n autoSkip: true\n },\n time: {\n displayFormats: {\n second: ''hh:mm:ss'',\n minute: ''hh:mm:ss''\n }\n }\n }]\n },\n zoom: {\n onSelect: function(startTimeMs, endTimeMs) {\n self.ctx.timewindowFunctions.onUpdateTimewindow(startTimeMs, endTimeMs);\n },\n onResetSelect: function() {\n self.ctx.timewindowFunctions.onResetTimewindow();\n }\n }\n }\n });\n \n self.onResize();\n \n}\n\nself.onDataUpdated = function() {\n \n if (self.ctx.chart.zoom.isMouseInteraction) {\n return;\n }\n if (!self.ctx.tickUpdate) {\n for (var i = 0; i < self.ctx.data.length; i++) {\n var dataSetData = [];\n var dataKeyData = self.ctx.data[i].data;\n for (var i2 = 0; i2 < dataKeyData.length; i2 ++) {\n dataSetData.push({x: moment(dataKeyData[i2][0]), y: dataKeyData[i2][1]});\n \n }\n self.ctx.chart.data.datasets[i].data = dataSetData; \n }\n }\n \n self.ctx.chart.options.scales.xAxes[0].time.min = moment(self.ctx.timeWindow.minTime);\n self.ctx.chart.options.scales.xAxes[0].time.max = moment(self.ctx.timeWindow.maxTime);\n \n self.ctx.chart.update(0, true);\n\n}\n\nself.onResize = function() {\n self.ctx.chart.resize();\n self.ctx.chart.update(0, true);\n}\n\nself.onDestroy = function() {\n}\n\nfunction getYAxis(chartInstance) {\n var scales = chartInstance.scales;\n for (var scaleId in scales) {\n\t var scale = scales[scaleId];\n\n\t if (!scale.isHorizontal()) {\n\t\t return scale;\n\t }\n }\n}\n\nfunction getXAxis(chartInstance) {\n var scales = chartInstance.scales;\n for (var scaleId in scales) {\n\t var scale = scales[scaleId];\n\n\t if (scale.isHorizontal()) {\n\t\t return scale;\n\t }\n }\n}\n\nfunction eventPointer (e) {\n if (angular.isDefined(e.touches) && e.touches.length > 0) {\n return {\n x : e.touches[0].pageX,\n y : e.touches[0].pageY\n };\n } else if (angular.isDefined(e.changedTouches) && e.changedTouches.length > 0) {\n return {\n x : e.changedTouches[0].pageX,\n y : e.changedTouches[0].pageY\n };\n } else if (e.pageX || e.pageY) {\n return {\n x : e.pageX,\n y : e.pageY\n };\n } else if (e.clientX || e.clientY) {\n var\n d = document,\n b = d.body,\n de = d.documentElement;\n return {\n x: e.clientX + b.scrollLeft + de.scrollLeft,\n y: e.clientY + b.scrollTop + de.scrollTop\n };\n }\n}\n\nvar zoomPlugin = {\n beforeInit: function(chartInstance) {\n chartInstance.zoom = {};\n var node = chartInstance.zoom.node = chartInstance.chart.ctx.canvas;\n \n chartInstance.zoom.mouseDownHandler = function(event) {\n chartInstance.zoom.dragZoomStart = event;\n chartInstance.zoom.dragZoomStartPointer = eventPointer(event);\n chartInstance.zoom.isMouseInteraction = true;\n };\n\n node.addEventListener(''mousedown'', chartInstance.zoom.mouseDownHandler);\n \n chartInstance.zoom.mouseMoveHandler = function(event) {\n if (chartInstance.zoom.dragZoomStart) {\n chartInstance.update(0);\n chartInstance.zoom.dragZoomEnd = event;\n chartInstance.zoom.dragZoomEndPointer = eventPointer(event);\n }\n };\n \n node.addEventListener(''mousemove'', chartInstance.zoom.mouseMoveHandler);\n \n chartInstance.zoom.mouseUpHandler = function(event) {\n if (chartInstance.zoom.dragZoomStart) {\n \n var chartArea = chartInstance.chartArea;\n var yAxis = getYAxis(chartInstance);\n\t\t\t\t\tvar beginPoint = chartInstance.zoom.dragZoomStart;\n\t\t\t\t\tvar beginPointer = chartInstance.zoom.dragZoomStartPointer;\n\t\t\t\t\tvar upEventPointer = eventPointer(event);\n\t\t\t\t\tvar offsetX = beginPoint.target.getBoundingClientRect().left;\n\t\t\t\t\tvar startX = Math.min(beginPointer.x, upEventPointer.x) - offsetX;\n\t\t\t\t\tvar endX = Math.max(beginPointer.x, upEventPointer.x) - offsetX;\n\t\t\t\t\tvar dragDistance = endX - startX;\n\t\t\t\t\t\n\t\t\t\t\tif (dragDistance > 0) {\n \t\t\t\t\tvar xAxis = getXAxis(chartInstance);\n \t\t\t\t\tvar options = chartInstance.options;\n \t\t\t\t\tif (options.scales.xAxes[0].time) {\n \t\t\t\t\t startX = Math.max(startX, xAxis.left);\n \t\t\t\t\t endX = Math.min(endX, xAxis.right);\n \t\t\t\t\t if (endX - startX > 0) {\n \t\t\t\t\t startX = startX - xAxis.left;\n \t\t\t\t\t endX = endX - xAxis.left;\n \t\t\t\t\t var time = options.scales.xAxes[0].time;\n \t\t\t\t\t var min = time.min.valueOf();\n \t\t\t\t\t var max = time.max.valueOf();\n \t\t\t\t\t var axisDistance = xAxis.right - xAxis.left;\n \t\t\t\t\t var timeDistance = max - min;\n \t\t\t\t\t \n \t\t\t\t\t var zoomStartTime = min + startX / axisDistance * timeDistance;\n \t\t\t\t\t var zoomEndTime = min + endX / axisDistance * timeDistance;\n\n \t\t\t\t\t if (options.zoom && options.zoom.onSelect) {\n \t\t\t\t\t options.zoom.onSelect(zoomStartTime, zoomEndTime);\n \t\t\t\t\t }\n \t\t\t\t\t }\n \t\t\t\t\t}\n\t\t\t\t\t}\n \t\t\tchartInstance.zoom.dragZoomStart = null;\n \t\t\tchartInstance.zoom.dragZoomEnd = null; \n }\n chartInstance.zoom.isMouseInteraction = false;\n };\n \n node.addEventListener(''mouseup'', chartInstance.zoom.mouseUpHandler);\n \n chartInstance.zoom.mouseLeaveHandler = function(event) {\n if (chartInstance.zoom.dragZoomStart) {\n \t\t\tchartInstance.zoom.dragZoomStart = null;\n \t\t\tchartInstance.zoom.dragZoomEnd = null; \n }\n chartInstance.zoom.isMouseInteraction = false;\n };\n \n node.addEventListener(''mouseleave'', chartInstance.zoom.mouseLeaveHandler);\n \n chartInstance.zoom.dblClickHandler = function(event) {\n if (chartInstance.zoom.dragZoomStart) {\n \t\t\tchartInstance.zoom.dragZoomStart = null;\n \t\t\tchartInstance.zoom.dragZoomEnd = null; \n }\n var options = chartInstance.options;\n if (options.zoom && options.zoom.onResetSelect) {\n options.zoom.onResetSelect();\n }\n };\n \n node.addEventListener(''dblclick'', chartInstance.zoom.dblClickHandler);\n },\n beforeDatasetsDraw: function(chartInstance, easing) {\n \t\tvar ctx = chartInstance.chart.ctx;\n \t\tvar chartArea = chartInstance.chartArea;\n \t\tctx.save();\n \t\tctx.beginPath();\n \t\tif (chartInstance.zoom && chartInstance.zoom.dragZoomEnd) {\n \t\t\tvar yAxis = getYAxis(chartInstance);\n \t\t\tvar beginPoint = chartInstance.zoom.dragZoomStart;\n \t\t\tvar beginPointer = chartInstance.zoom.dragZoomStartPointer;\n \t\t\tvar endPoint = chartInstance.zoom.dragZoomEnd;\n \t\t\tvar endPointer = chartInstance.zoom.dragZoomEndPointer;\n \t\t\t\n \t\t\tvar offsetX = beginPoint.target.getBoundingClientRect().left;\n \t\t\tvar startX = Math.min(beginPointer.x, endPointer.x) - offsetX;\n \t\t\tvar endX = Math.max(beginPointer.x, endPointer.x) - offsetX;\n \t\t\tvar rectWidth = endX - startX;\n \t\t\tctx.fillStyle = ''rgba(157,203,255,0.1)'';\n \t\t\tctx.lineWidth = 1;\n \t\t\tctx.strokeRect(startX, yAxis.top, rectWidth, yAxis.bottom - yAxis.top);\n \t\t\tctx.fillRect(startX, yAxis.top, rectWidth, yAxis.bottom - yAxis.top);\n \t\t}\n \t\tif (chartArea) {\n \t\t ctx.rect(chartArea.left, chartArea.top, chartArea.right - chartArea.left, chartArea.bottom - chartArea.top);\n \t\t}\n\t\t ctx.clip(); \n },\n \n afterDatasetsDraw: function(chartInstance) {\n\t chartInstance.chart.ctx.restore();\n },\n \n destroy: function(chartInstance) {\n if (chartInstance.zoom) {\n var node = chartInstance.zoom.node;\n\t\t\t\tnode.removeEventListener(''mousedown'', chartInstance.zoom.mouseDownHandler);\n\t\t\t\tnode.removeEventListener(''mousemove'', chartInstance.zoom.mouseMoveHandler);\n\t\t\t\tnode.removeEventListener(''mouseup'', chartInstance.zoom.mouseUpHandler);\n\t\t\t\tnode.removeEventListener(''mouseleave'', chartInstance.zoom.mouseLeaveHandler);\t \n\t\t\t\tnode.removeEventListener(''dblclick'', chartInstance.zoom.dblClickHandler);\n\t\t\t\tdelete chartInstance.zoom;\n }\n }\n };\n\nChart.pluginService.register(zoomPlugin);\n","settingsSchema":"{}","dataKeySettingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"showLines\": {\n \"title\": \"Show lines\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"fillLines\": {\n \"title\": \"Fill lines\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"showPoints\": {\n \"title\": \"Show points\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"tension\": {\n \"title\": \"Line tension\",\n \"type\": \"number\",\n \"default\": 0.2\n }\n },\n \"required\": [\"showLines\", \"fillLines\", \"showPoints\"]\n },\n \"form\": [\n \"showLines\",\n \"fillLines\",\n \"showPoints\",\n \"tension\"\n ]\n}","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{\"showLines\":true,\"fillLines\":false,\"showPoints\":false},\"_hash\":0.5644745944820795,\"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;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#ffc107\",\"settings\":{\"showLines\":true,\"fillLines\":true,\"showPoints\":false},\"_hash\":0.18379294198604845,\"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\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Timeseries - Chart.js (Deprecated)\"}"}',  
68 -'Timeseries - Chart.js (Deprecated)' );  
69 -  
70 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
71 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',  
72 -'digital_vertical_bar',  
73 -'{"type":"latest","sizeX":2.5,"sizeY":4.5,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":60,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[\"#3d5afe\",\"#f44336\"],\"refreshAnimationType\":\"<>\",\"refreshAnimationTime\":700,\"startAnimationType\":\"<>\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":14},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":8,\"style\":\"normal\",\"weight\":\"normal\",\"color\":\"#cccccc\"},\"neonGlowBrightness\":20,\"decimals\":0,\"showUnitTitle\":true,\"gaugeColor\":\"#171a1c\",\"gaugeType\":\"verticalBar\",\"showTitle\":false,\"units\":\"°C\",\"minValue\":-60,\"dashThickness\":1.2},\"title\":\"Digital vertical bar\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
74 -'Digital vertical bar' );  
75 -  
76 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
77 -VALUES ( now ( ), minTimeuuid ( 0 ), 'gpio_widgets', 'gpio_panel',  
78 -'{"type":"latest","sizeX":5,"sizeY":2,"resources":[],"templateHtml":"<div class=\"gpio-panel\" style=\"height: 100%;\">\n <section layout=\"row\" ng-repeat=\"row in rows\">\n <section flex layout=\"row\" ng-repeat=\"cell in row\">\n <section layout=\"row\" flex ng-if=\"cell\" layout-align=\"{{$index===0 ? ''end center'' : ''start center''}}\">\n <span class=\"gpio-left-label\" ng-show=\"$index===0\">{{ cell.label }}</span>\n <section layout=\"row\" class=\"led-panel\" ng-class=\"$index===0 ? ''col-0'' : ''col-1''\"\n ng-style=\"{backgroundColor: ledPanelBackgroundColor }\">\n <span class=\"pin\" ng-show=\"$index===0\">{{cell.pin}}</span>\n <span class=\"led-container\">\n <tb-led-light size=\"prefferedRowHeight\"\n color-on=\"cell.colorOn\"\n color-off=\"cell.colorOff\"\n off-opacity=\"''0.9''\"\n tb-enabled=\"cell.enabled\">\n </tb-led-light>\n </span>\n <span class=\"pin\" ng-show=\"$index===1\">{{cell.pin}}</span>\n </section>\n <span class=\"gpio-right-label\" ng-show=\"$index===1\">{{ cell.label }}</span>\n </section>\n <section layout=\"row\" flex ng-if=\"!cell\">\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"led-panel\"\n ng-style=\"{backgroundColor: ledPanelBackgroundColor }\"></span>\n <span flex ng-show=\"$index===1\"></span>\n </section>\n </section>\n </section> \n</div>","templateCss":".error {\n font-size: 14px !important;\n color: maroon;/*rgb(250,250,250);*/\n background-color: transparent;\n padding: 6px;\n}\n\n.error span {\n margin: auto;\n}\n\n.gpio-panel {\n padding-top: 10px;\n white-space: nowrap;\n}\n\n.gpio-panel tb-led-light > div {\n margin: auto;\n}\n\n.led-panel {\n margin: 0;\n width: 66px;\n min-width: 66px;\n}\n\n.led-container {\n width: 48px;\n min-width: 48px;\n}\n\n.pin {\n margin-top: auto;\n margin-bottom: auto;\n color: white;\n font-size: 12px;\n width: 16px;\n min-width: 16px;\n}\n\n.led-panel.col-0 .pin {\n margin-left: auto;\n padding-left: 2px;\n text-align: right;\n}\n\n.led-panel.col-1 .pin {\n margin-right: auto;\n \n text-align: left;\n}\n\n.gpio-left-label {\n margin-right: 8px;\n}\n\n.gpio-right-label {\n margin-left: 8px;\n}","controllerScript":"self.onInit = function() {\n var i, gpio;\n \n var scope = self.ctx.$scope;\n var settings = self.ctx.settings;\n \n scope.gpioList = [];\n scope.gpioByPin = {};\n for (var g = 0; g < settings.gpioList.length; g++) {\n gpio = settings.gpioList[g];\n scope.gpioList.push(\n {\n row: gpio.row,\n col: gpio.col,\n pin: gpio.pin,\n label: gpio.label,\n enabled: false,\n colorOn: tinycolor(gpio.color).lighten(20).toHexString(),\n colorOff: tinycolor(gpio.color).darken().toHexString()\n }\n );\n scope.gpioByPin[gpio.pin] = scope.gpioList[scope.gpioList.length-1];\n }\n\n scope.ledPanelBackgroundColor = settings.ledPanelBackgroundColor || tinycolor(''green'').lighten(2).toRgbString();\n\n scope.gpioCells = {};\n var rowCount = 0;\n for (i = 0; i < scope.gpioList.length; i++) {\n gpio = scope.gpioList[i];\n scope.gpioCells[gpio.row+''_''+gpio.col] = gpio;\n rowCount = Math.max(rowCount, gpio.row+1);\n }\n \n scope.prefferedRowHeight = 32;\n scope.rows = [];\n for (i = 0; i < rowCount; i++) {\n var row = [];\n for (var c =0; c<2;c++) {\n if (scope.gpioCells[i+''_''+c]) {\n row[c] = scope.gpioCells[i+''_''+c];\n } else {\n row[c] = null;\n }\n }\n scope.rows.push(row);\n } \n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n var changed = false;\n for (var d = 0; d < self.ctx.data.length; d++) {\n var cellData = self.ctx.data[d];\n var dataKey = cellData.dataKey;\n var gpio = self.ctx.$scope.gpioByPin[dataKey.label];\n if (gpio) {\n var enabled = false;\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n enabled = (tvPair[1] === true || tvPair[1] === ''true'');\n }\n if (gpio.enabled != enabled) {\n changed = true;\n gpio.enabled = enabled;\n }\n }\n }\n if (changed) {\n self.ctx.$scope.$digest();\n } \n}\n\nself.onResize = function() {\n var rowCount = self.ctx.$scope.rows.length;\n var prefferedRowHeight = (self.ctx.height - 35)/rowCount;\n prefferedRowHeight = Math.min(32, prefferedRowHeight);\n prefferedRowHeight = Math.max(12, prefferedRowHeight);\n self.ctx.$scope.prefferedRowHeight = prefferedRowHeight;\n \n var ratio = prefferedRowHeight/32;\n \n var leftLabels = $(''.gpio-left-label'', self.ctx.$container);\n leftLabels.css(''font-size'', 16*ratio+''px'');\n var rightLabels = $(''.gpio-right-label'', self.ctx.$container);\n rightLabels.css(''font-size'', 16*ratio+''px'');\n var pins = $(''.pin'', self.ctx.$container);\n var pinsFontSize = Math.max(9, 12*ratio);\n pins.css(''font-size'', pinsFontSize+''px''); \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"gpioList\": {\n \"title\": \"Gpio leds\",\n \"type\": \"array\",\n \"minItems\" : 1,\n \"items\": {\n \"title\": \"Gpio led\",\n \"type\": \"object\",\n \"properties\": {\n \"pin\": {\n \"title\": \"Pin\",\n \"type\": \"number\"\n },\n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"row\": {\n \"title\": \"Row\",\n \"type\": \"number\"\n },\n \"col\": {\n \"title\": \"Column\",\n \"type\": \"number\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\",\n \"default\": \"red\"\n }\n },\n \"required\": [\"pin\", \"label\", \"row\", \"col\", \"color\"]\n }\n },\n \"ledPanelBackgroundColor\": {\n \"title\": \"LED panel background color\",\n \"type\": \"string\",\n \"default\": \"#008a00\"\n } \n },\n \"required\": [\"gpioList\", \n \"ledPanelBackgroundColor\"]\n },\n \"form\": [\n {\n \"key\": \"gpioList\",\n \"items\": [\n \"gpioList[].pin\",\n \"gpioList[].label\",\n \"gpioList[].row\",\n \"gpioList[].col\",\n {\n \"key\": \"gpioList[].color\",\n \"type\": \"color\"\n }\n ]\n },\n {\n \"key\": \"ledPanelBackgroundColor\",\n \"type\": \"color\"\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"gpioList\":[{\"pin\":1,\"label\":\"GPIO 1\",\"row\":0,\"col\":0,\"color\":\"#008000\",\"_uniqueKey\":0},{\"pin\":2,\"label\":\"GPIO 2\",\"row\":0,\"col\":1,\"color\":\"#ffff00\",\"_uniqueKey\":1},{\"pin\":3,\"label\":\"GPIO 3\",\"row\":1,\"col\":0,\"color\":\"#cf006f\",\"_uniqueKey\":2}],\"ledPanelBackgroundColor\":\"#b71c1c\"},\"title\":\"Basic GPIO Panel\",\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"1\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.22518255793320163,\"funcBody\":\"var period = time % 1500;\\nreturn period < 500;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"2\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.7008206860666621,\"funcBody\":\"var period = time % 1500;\\nreturn period >= 500 && period < 1000;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"3\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.42600325102193426,\"funcBody\":\"var period = time % 1500;\\nreturn period >= 1000;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}}}"}',  
79 -'Basic GPIO Panel' );  
80 -  
81 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
82 -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'timeseries_table',  
83 -'{"type":"timeseries","sizeX":8,"sizeY":6.5,"resources":[],"templateHtml":"<tb-timeseries-table-widget \n config=\"config\"\n table-id=\"tableId\"\n datasources=\"datasources\"\n data=\"data\">\n</tb-timeseries-table-widget>","templateCss":"","controllerScript":"self.onInit = function() {\n \n var scope = self.ctx.$scope;\n var id = self.ctx.$scope.$injector.get(''utils'').guid();\n\n scope.config = {\n settings: self.ctx.settings,\n widgetConfig: self.ctx.widgetConfig\n }\n\n scope.datasources = self.ctx.datasources;\n scope.data = self.ctx.data;\n scope.tableId = \"table-\"+id;\n \n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.data = self.ctx.data;\n self.ctx.$scope.$broadcast(''timeseries-table-data-updated'', self.ctx.$scope.tableId);\n}\n\nself.onDestroy = function() {\n}","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 },\n \"required\": []\n },\n \"form\": [\n \"showTimestamp\"\n ]\n}","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, filter)\",\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}","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},\"title\":\"Timeseries table\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":false,\"showLegend\":false}"}',  
84 -'Timeseries table' );  
85 -  
86 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
87 -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'polar_area_chart_js',  
88 -'{"type":"latest","sizeX":7,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.3.0/Chart.min.js"}],"templateHtml":"<canvas id=\"pieChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n var pieData = {\n labels: [],\n datasets: []\n };\n\n var dataset = {\n data: [],\n backgroundColor: [],\n borderColor: [],\n borderWidth: [],\n hoverBackgroundColor: []\n }\n \n pieData.datasets.push(dataset);\n \n for (var i = 0; i < self.ctx.data.length; i++) {\n var dataKey = self.ctx.data[i].dataKey;\n pieData.labels.push(dataKey.label);\n dataset.data.push(0);\n var hoverBackgroundColor = tinycolor(dataKey.color).lighten(15);\n var borderColor = tinycolor(dataKey.color).darken();\n dataset.backgroundColor.push(dataKey.color);\n dataset.borderColor.push(''#fff'');\n dataset.borderWidth.push(5);\n dataset.hoverBackgroundColor.push(hoverBackgroundColor.toRgbString());\n }\n\n var ctx = $(''#pieChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''polarArea'',\n data: pieData,\n options: {\n responsive: false,\n maintainAspectRatio: false\n }\n });\n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n for (var i = 0; i < self.ctx.data.length; i++) {\n var cellData = self.ctx.data[i];\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n var value = tvPair[1];\n self.ctx.chart.data.datasets[0].data[i] = parseFloat(value);\n }\n }\n \n self.ctx.chart.update();\n}\n\nself.onResize = function() {\n if (self.ctx.height >= 70) {\n try {\n self.ctx.chart.resize();\n } catch (e) {}\n }\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.545701115289893,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.2592906835158064,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.12880275585455747,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fifth\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.2074391823443591,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Polar Area - Chart.js\"}"}',  
89 -'Polar Area - Chart.js' );  
90 -  
91 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
92 -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'html_card',  
93 -'{"type":"static","sizeX":7.5,"sizeY":3,"resources":[],"templateHtml":"","templateCss":"","controllerScript":"self.onInit = function() {\n\n var cssParser = new cssjs();\n cssParser.testMode = false;\n var namespace = ''html-card-'' + hashCode(self.ctx.settings.cardCss);\n cssParser.cssPreviewNamespace = namespace;\n cssParser.createStyleElement(namespace, self.ctx.settings.cardCss);\n self.ctx.$container.addClass(namespace);\n cardHtml = self.ctx.settings.cardHtml;\n self.ctx.$container.html(cardHtml);\n \n function hashCode(str) {\n var hash = 0;\n var i, char;\n if (str.length === 0) return hash;\n for (i = 0; i < str.length; i++) {\n char = str.charCodeAt(i);\n hash = ((hash << 5) - hash) + char;\n hash = hash & hash;\n }\n return hash;\n }\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"required\": [\"cardHtml\"],\n \"properties\": {\n \"cardCss\": {\n \"title\": \"CSS\",\n \"type\": \"string\",\n \"default\": \".card {\\n font-weight: bold; \\n}\"\n },\n \"cardHtml\": {\n \"title\": \"HTML\",\n \"type\": \"string\",\n \"default\": \"<div class=''card''>HTML code here</div>\"\n }\n }\n },\n \"form\": [\n {\n \"key\": \"cardCss\",\n \"type\": \"css\"\n }, \n {\n \"key\": \"cardHtml\",\n \"type\": \"html\"\n } \n ]\n}","dataKeySettingsSchema":"{}\n","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\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"cardHtml\":\"<div class=''card''>HTML code here</div>\",\"cardCss\":\".card {\\n font-weight: bold;\\n font-size: 32px;\\n color: #999;\\n width: 100%;\\n height: 100%;\\n display: flex;\\n align-items: center;\\n justify-content: center;\\n}\"},\"title\":\"HTML Card\",\"dropShadow\":true}"}',  
94 -'HTML Card' );  
95 -  
96 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
97 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',  
98 -'simple_neon_gauge_justgage',  
99 -'{"type":"latest","sizeX":3,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#388e3c\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":1,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":40,\"dashThickness\":1.5,\"decimals\":0,\"gaugeType\":\"donut\"},\"title\":\"Simple neon gauge - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
100 -'Simple neon gauge - justGage' );  
101 -  
102 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
103 -VALUES ( now ( ), minTimeuuid ( 0 ), 'analogue_gauges',  
104 -'radial_gauge_canvas_gauges',  
105 -'{"type":"latest","sizeX":6,"sizeY":5,"resources":[],"templateHtml":"<canvas id=\"radialGauge\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbAnalogueRadialGauge(self.ctx, ''radialGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.getSettingsSchema = function() {\n return TbAnalogueRadialGauge.settingsSchema;\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 50 - 25;\\nif (value < -100) {\\n\\tvalue = -100;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"maxValue\":100,\"startAngle\":45,\"ticksAngle\":270,\"showBorder\":true,\"defaultColor\":\"#e65100\",\"needleCircleSize\":10,\"highlights\":[],\"showUnitTitle\":true,\"colorPlate\":\"#fff\",\"colorMajorTicks\":\"#444\",\"colorMinorTicks\":\"#666\",\"minorTicks\":10,\"valueInt\":3,\"valueDec\":0,\"highlightsWidth\":15,\"valueBox\":true,\"animation\":true,\"animationDuration\":500,\"animationRule\":\"cycle\",\"colorNeedleShadowUp\":\"rgba(2, 255, 255, 0)\",\"numbersFont\":{\"family\":\"Roboto\",\"size\":18,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#616161\"},\"titleFont\":{\"family\":\"Roboto\",\"size\":24,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#888\"},\"unitsFont\":{\"family\":\"Roboto\",\"size\":22,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#616161\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"size\":36,\"style\":\"normal\",\"weight\":\"normal\",\"shadowColor\":\"rgba(0, 0, 0, 0.49)\",\"color\":\"#444\"},\"minValue\":-100,\"colorNeedleShadowDown\":\"rgba(188,143,143,0.45)\",\"colorValueBoxRect\":\"#888\",\"colorValueBoxRectEnd\":\"#666\",\"colorValueBoxBackground\":\"#babab2\",\"colorValueBoxShadow\":\"rgba(0,0,0,1)\"},\"title\":\"Radial gauge - Canvas Gauges\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
106 -'Radial gauge - Canvas Gauges' );  
107 -  
108 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
109 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',  
110 -'digital_speedometer',  
111 -'{"type":"latest","sizeX":5,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < 45) {\\n\\tvalue = 45;\\n} else if (value > 130) {\\n\\tvalue = 130;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":180,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[\"#008000\",\"#fbc02d\",\"#f44336\"],\"refreshAnimationType\":\"linear\",\"refreshAnimationTime\":700,\"startAnimationType\":\"linear\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#ffffff\"},\"neonGlowBrightness\":40,\"dashThickness\":1.5,\"decimals\":0,\"unitTitle\":\"MPH\",\"showUnitTitle\":true,\"gaugeColor\":\"#171a1c\",\"gaugeType\":\"arc\"},\"title\":\"Digital speedometer\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
112 -'Digital speedometer' );  
113 -  
114 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
115 -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'doughnut_chart_js',  
116 -'{"type":"latest","sizeX":7,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.3.0/Chart.min.js"}],"templateHtml":"<canvas id=\"pieChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n var pieData = {\n labels: [],\n datasets: []\n };\n\n var dataset = {\n data: [],\n backgroundColor: [],\n borderColor: [],\n borderWidth: [],\n hoverBackgroundColor: []\n }\n \n var borderColor = self.ctx.settings.borderColor || ''#fff'';\n var borderWidth = angular.isDefined(self.ctx.settings.borderWidth) ? self.ctx.settings.borderWidth : 5;\n \n pieData.datasets.push(dataset);\n \n for (var i=0; i < self.ctx.data.length; i++) {\n var dataKey = self.ctx.data[i].dataKey;\n pieData.labels.push(dataKey.label);\n dataset.data.push(0);\n var hoverBackgroundColor = tinycolor(dataKey.color).lighten(15);\n dataset.backgroundColor.push(dataKey.color);\n dataset.borderColor.push(borderColor);\n dataset.borderWidth.push(borderWidth);\n dataset.hoverBackgroundColor.push(hoverBackgroundColor.toRgbString());\n }\n\n var options = {\n responsive: false,\n maintainAspectRatio: false,\n legend: {\n display: true,\n labels: {\n fontColor: ''#666''\n }\n },\n tooltips: {\n callbacks: {\n label: function(tooltipItem, data) {\n var label = data.labels[tooltipItem.index];\n var value = data.datasets[tooltipItem.datasetIndex].data[tooltipItem.index];\n var content = label + '': '' + value;\n var units = self.ctx.settings.units ? self.ctx.settings.units : self.ctx.units;\n if (units) {\n content += '' '' + units;\n } \n return content;\n }\n }\n }\n };\n\n if (self.ctx.settings.legend) {\n options.legend.display = self.ctx.settings.legend.display !== false;\n options.legend.labels.fontColor = self.ctx.settings.legend.labelsFontColor || ''#666'';\n }\n\n var ctx = $(''#pieChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''doughnut'',\n data: pieData,\n options: options\n });\n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n for (var i = 0; i < self.ctx.data.length; i++) {\n var cellData = self.ctx.data[i];\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n var value = tvPair[1];\n self.ctx.chart.data.datasets[0].data[i] = parseFloat(value);\n }\n }\n self.ctx.chart.update();\n}\n\nself.onResize = function() {\n self.ctx.chart.resize();\n}\n\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"borderWidth\": {\n \"title\": \"Border width\",\n \"type\": \"number\",\n \"default\": 5\n },\n \"borderColor\": {\n \"title\": \"Border color\",\n \"type\": \"string\",\n \"default\": \"#fff\"\n },\n \"legend\": {\n \"title\": \"Legend settings\",\n \"type\": \"object\",\n \"properties\": {\n \"display\": {\n \"title\": \"Display legend\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"labelsFontColor\": {\n \"title\": \"Labels font color\",\n \"type\": \"string\",\n \"default\": \"#666\"\n }\n }\n }\n },\n \"required\": []\n },\n \"form\": [\n \"borderWidth\", \n {\n \"key\": \"borderColor\",\n \"type\": \"color\"\n }, \n {\n \"key\": \"legend\",\n \"items\": [\n \"legend.display\",\n {\n \"key\": \"legend.labelsFontColor\",\n \"type\": \"color\"\n }\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#26a69a\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#f57c00\",\"settings\":{},\"_hash\":0.545701115289893,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#afb42b\",\"settings\":{},\"_hash\":0.2592906835158064,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#673ab7\",\"settings\":{},\"_hash\":0.12880275585455747,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"borderWidth\":5,\"borderColor\":\"#fff\",\"legend\":{\"display\":true,\"labelsFontColor\":\"#666666\"}},\"title\":\"Doughnut - Chart.js\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
117 -'Doughnut - Chart.js' );  
118 -  
119 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
120 -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'pie_chart_js',  
121 -'{"type":"latest","sizeX":8,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.3.0/Chart.min.js"}],"templateHtml":"<canvas id=\"pieChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n \n var pieData = {\n labels: [],\n datasets: []\n };\n\n var dataset = {\n data: [],\n backgroundColor: [],\n borderColor: [],\n borderWidth: [],\n hoverBackgroundColor: []\n }\n \n pieData.datasets.push(dataset);\n \n for (var i=0; i < self.ctx.data.length; i++) {\n var dataKey = self.ctx.data[i].dataKey;\n pieData.labels.push(dataKey.label);\n dataset.data.push(0);\n var hoverBackgroundColor = tinycolor(dataKey.color).lighten(15);\n var borderColor = tinycolor(dataKey.color).darken();\n dataset.backgroundColor.push(dataKey.color);\n dataset.borderColor.push(''#fff'');\n dataset.borderWidth.push(5);\n dataset.hoverBackgroundColor.push(hoverBackgroundColor.toRgbString());\n }\n\n var ctx = $(''#pieChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''pie'',\n data: pieData,\n options: {\n responsive: false,\n maintainAspectRatio: false\n }\n }); \n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n for (var i = 0; i < self.ctx.data.length; i++) {\n var cellData = self.ctx.data[i];\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n var value = tvPair[1];\n self.ctx.chart.data.datasets[0].data[i] = parseFloat(value);\n }\n }\n self.ctx.chart.update();\n}\n\nself.onResize = function() {\n self.ctx.chart.resize();\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.545701115289893,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.2592906835158064,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.12880275585455747,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Pie - Chart.js\"}"}',  
122 -'Pie - Chart.js' );  
123 -  
124 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
125 -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'bars',  
126 -'{"type":"latest","sizeX":7,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.3.0/Chart.min.js"}],"templateHtml":"<canvas id=\"barChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n var barData = {\n labels: [],\n datasets: []\n };\n \n for (var i = 0; i < self.ctx.datasources.length; i++) {\n var datasource = self.ctx.datasources[i];\n for (var d = 0; d < datasource.dataKeys.length; d++) {\n var dataset = {\n label: datasource.dataKeys[d].label,\n data: [0],\n backgroundColor: [datasource.dataKeys[d].color],\n borderColor: [datasource.dataKeys[d].color],\n borderWidth: 1\n }\n barData.datasets.push(dataset);\n }\n }\n\n var ctx = $(''#barChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''bar'',\n data: barData,\n options: {\n responsive: false,\n maintainAspectRatio: false,\n scales: {\n yAxes: [{\n ticks: {\n beginAtZero:true\n }\n }]\n }\n }\n });\n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n var c = 0;\n for (var i = 0; i < self.ctx.chart.data.datasets.length; i++) {\n var dataset = self.ctx.chart.data.datasets[i];\n var cellData = self.ctx.data[i]; \n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n var value = tvPair[1];\n dataset.data[0] = parseFloat(value);\n }\n }\n self.ctx.chart.update();\n}\n\nself.onResize = function() {\n self.ctx.chart.resize();\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.545701115289893,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.2592906835158064,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.12880275585455747,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Bars - Chart.js\"}"}',  
127 -'Bars - Chart.js' );  
128 -  
129 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
130 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges', 'digital_bar',  
131 -'{"type":"latest","sizeX":6,"sizeY":2.5,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < 80) {\\n\\tvalue = 80;\\n} else if (value > 160) {\\n\\tvalue = 160;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":180,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[\"#008000\",\"#fbc02d\",\"#f44336\"],\"refreshAnimationType\":\"linear\",\"refreshAnimationTime\":700,\"startAnimationType\":\"linear\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":18},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#ffffff\"},\"neonGlowBrightness\":40,\"dashThickness\":1.5,\"decimals\":0,\"unitTitle\":\"MPH\",\"showUnitTitle\":true,\"gaugeColor\":\"#171a1c\",\"gaugeType\":\"horizontalBar\",\"showTitle\":false},\"title\":\"Digital horizontal bar\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
132 -'Digital horizontal bar' );  
133 -  
134 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
135 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',  
136 -'mini_gauge_justgage',  
137 -'{"type":"latest","sizeX":2,"sizeY":2,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#7cb342\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ffffff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":0,\"dashThickness\":0,\"decimals\":0,\"roundedLineCap\":true,\"gaugeType\":\"donut\"},\"title\":\"Mini gauge - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
138 -'Mini gauge - justGage' );  
139 -  
140 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
141 -VALUES ( now ( ), minTimeuuid ( 0 ), 'maps', 'route_map_openstreetmap',  
142 -'{"type":"timeseries","sizeX":8.5,"sizeY":6,"resources":[],"templateHtml":"","templateCss":".leaflet-zoom-box {\n\tz-index: 9;\n}\n\n.leaflet-pane { z-index: 4; }\n\n.leaflet-tile-pane { z-index: 2; }\n.leaflet-overlay-pane { z-index: 4; }\n.leaflet-shadow-pane { z-index: 5; }\n.leaflet-marker-pane { z-index: 6; }\n.leaflet-tooltip-pane { z-index: 7; }\n.leaflet-popup-pane { z-index: 8; }\n\n.leaflet-map-pane canvas { z-index: 1; }\n.leaflet-map-pane svg { z-index: 2; }\n\n.leaflet-control {\n\tz-index: 9;\n}\n.leaflet-top,\n.leaflet-bottom {\n\tz-index: 11;\n}\n\n.tb-marker-label {\n border: none;\n background: none;\n box-shadow: none;\n}\n\n.tb-marker-label:before {\n border: none;\n background: none;\n}\n","controllerScript":"self.onInit = function() {\n self.ctx.map = new TbMapWidget(''openstreet-map'', true, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"title\": \"Route Map Configuration\",\n \"type\": \"object\",\n \"properties\": {\n \"defaultZoomLevel\": {\n \"title\": \"Default map zoom level (1 - 20)\",\n \"type\": \"number\"\n },\n \"fitMapBounds\": {\n \"title\": \"Fit map bounds to cover all markers\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"routesSettings\": {\n \"title\": \"Routes\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Route settings\",\n \"type\": \"object\",\n \"properties\": {\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"lat\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"lng\"\n },\n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n }, \n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"tooltipPattern\": {\n \"title\": \"Pattern ( for ex. ''Text ${keyName} units.'' or ''${#<key index>} units'' )\",\n \"type\": \"string\",\n \"default\": \"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\"\n },\n \"useColorFunction\": {\n \"title\": \"Use color function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"colorFunction\": {\n \"title\": \"Color function: f(data)\",\n \"type\": \"string\"\n },\n \"markerImage\": {\n \"title\": \"Custom marker image\",\n \"type\": \"string\"\n },\n \"markerImageSize\": {\n \"title\": \"Custom marker image size (px)\",\n \"type\": \"number\",\n \"default\": 34\n },\n \"useMarkerImageFunction\": {\n \"title\": \"Use marker image function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"markerImageFunction\": {\n \"title\": \"Marker image function: f(data, images)\",\n \"type\": \"string\"\n },\n \"markerImages\": {\n \"title\": \"Marker images\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker image\",\n \"type\": \"string\"\n }\n },\n \"strokeWeight\": {\n \"title\": \"Stroke weight\",\n \"type\": \"number\",\n \"default\": 2\n },\n \"strokeOpacity\": {\n \"title\": \"Stroke opacity\",\n \"type\": \"number\",\n \"default\": 1.0\n }\n }\n }\n }\n },\n \"required\": [\n ]\n },\n \"form\": [\n \"defaultZoomLevel\",\n \"fitMapBounds\",\n {\n \"key\": \"routesSettings\",\n \"items\": [\n \"routesSettings[].latKeyName\",\n \"routesSettings[].lngKeyName\",\n \"routesSettings[].showLabel\",\n \"routesSettings[].label\",\n \"routesSettings[].tooltipPattern\",\n {\n \"key\": \"routesSettings[].color\",\n \"type\": \"color\"\n },\n \"routesSettings[].useColorFunction\",\n {\n \"key\": \"routesSettings[].colorFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"routesSettings[].markerImage\",\n \"type\": \"image\"\n },\n \"routesSettings[].markerImageSize\",\n \"routesSettings[].useMarkerImageFunction\",\n {\n \"key\": \"routesSettings[].markerImageFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"routesSettings[].markerImages\",\n \"items\": [\n {\n \"key\": \"routesSettings[].markerImages[]\",\n \"type\": \"image\"\n }\n ]\n },\n \"routesSettings[].strokeWeight\",\n \"routesSettings[].strokeOpacity\"\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.8950926999078694,\"funcBody\":\"var lats = [37.7696499,\\n37.7699074,\\n37.7699536,\\n37.7697242,\\n37.7695189,\\n37.7696889,\\n37.7697153,\\n37.7701244,\\n37.7700604,\\n37.7705491,\\n37.7715705,\\n37.771752,\\n37.7707533,\\n37.769866];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lats[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.2757675428823283,\"funcBody\":\"var lons = [-122.4261215,\\n-122.4219157,\\n-122.4199623,\\n-122.4179074,\\n-122.4155876,\\n-122.4155521,\\n-122.4163203,\\n-122.4193876,\\n-122.4210496,\\n-122.422284,\\n-122.4232717,\\n-122.4235138,\\n-122.4247605,\\n-122.4258812];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lons[i];\"}],\"intervalSec\":60},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.14481354591724638,\"funcBody\":\"var value = prevValue;\\nif (time % 500 < 100) {\\n value = value + Math.random() * 40 - 20;\\n if (value < 45) {\\n \\tvalue = 45;\\n } else if (value > 130) {\\n \\tvalue = 130;\\n }\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":30000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"routesSettings\":[{\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"tooltipPattern\":\"<b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Speed:</b> ${Speed} MPH<br/><small>See advanced settings for details</small>\",\"strokeWeight\":4,\"label\":\"First route\",\"color\":\"#3d5afe\",\"strokeOpacity\":1,\"useColorFunction\":true,\"useMarkerImageFunction\":true,\"markerImages\":[\"\",\"\",\"\"],\"colorFunction\":\"var speed = data[''Speed''];\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n if (percent < 0.5) {\\n percent *=2*100; \\n return tinycolor.mix(''green'', ''yellow'', amount = percent).toHexString();\\n } else {\\n percent = (percent - 0.5)*2*100;\\n return tinycolor.mix(''yellow'', ''red'', amount = percent).toHexString();\\n }\\n}\\nreturn ''green'';\",\"markerImageFunction\":\"var speed = data[''Speed''];\\nvar res = {\\n url: images[0],\\n size: 55\\n};\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n var index = Math.floor(3 * percent);\\n res.url = images[index];\\n}\\nreturn res;\"}]},\"title\":\"Route Map - OpenStreetMap\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false}"}',  
143 -'Route Map - OpenStreetMap' );  
144 -  
145 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
146 -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'label_widget',  
147 -'{"type":"latest","sizeX":4.5,"sizeY":5,"resources":[],"templateHtml":"","templateCss":"#container {\n overflow: auto;\n}\n\n.tbDatasource-container {\n margin: 5px;\n padding: 8px;\n}\n\n.tbDatasource-title {\n font-size: 1.200rem;\n font-weight: 500;\n padding-bottom: 10px;\n}\n\n.tbDatasource-table {\n width: 100%;\n box-shadow: 0 0 10px #ccc;\n border-collapse: collapse;\n white-space: nowrap;\n font-size: 1.000rem;\n color: #757575;\n}\n\n.tbDatasource-table td {\n position: relative;\n border-top: 1px solid rgba(0, 0, 0, 0.12);\n border-bottom: 1px solid rgba(0, 0, 0, 0.12);\n padding: 0px 18px;\n box-sizing: border-box;\n}","controllerScript":"self.onInit = function() {\n self.ctx.varsRegex = /\\$\\{([^\\}]*)\\}/g;\n \n var imageUrl = self.ctx.settings.backgroundImageUrl ? self.ctx.settings.backgroundImageUrl :\n '''';\n\n self.ctx.$container.css(''background'', ''url(\"''+imageUrl+''\") no-repeat'');\n self.ctx.$container.css(''backgroundSize'', ''contain'');\n self.ctx.$container.css(''backgroundPosition'', ''50% 50%'');\n \n function processLabelPattern(pattern, data) {\n var match = self.ctx.varsRegex.exec(pattern);\n var replaceInfo = {};\n replaceInfo.variables = [];\n while (match !== null) {\n var variableInfo = {};\n variableInfo.dataKeyIndex = -1;\n var variable = match[0];\n var label = match[1];\n var valDec = 2;\n var splitVals = label.split('':'');\n if (splitVals.length > 1) {\n label = splitVals[0];\n valDec = parseFloat(splitVals[1]);\n }\n variableInfo.variable = variable;\n variableInfo.valDec = valDec;\n \n if (label.startsWith(''#'')) {\n var keyIndexStr = label.substring(1);\n var n = Math.floor(Number(keyIndexStr));\n if (String(n) === keyIndexStr && n >= 0) {\n variableInfo.dataKeyIndex = n;\n }\n }\n if (variableInfo.dataKeyIndex === -1) {\n for (var i = 0; i < data.length; i++) {\n var datasourceData = data[i];\n var dataKey = datasourceData.dataKey;\n if (dataKey.label === label) {\n variableInfo.dataKeyIndex = i;\n break;\n }\n }\n }\n replaceInfo.variables.push(variableInfo);\n match = self.ctx.varsRegex.exec(pattern);\n }\n return replaceInfo;\n }\n\n var configuredLabels = self.ctx.settings.labels;\n if (!configuredLabels) {\n configuredLabels = [];\n }\n \n self.ctx.labels = [];\n\n for (var l = 0; l < configuredLabels.length; l++) {\n var labelConfig = configuredLabels[l];\n var localConfig = {};\n localConfig.font = {};\n \n localConfig.pattern = labelConfig.pattern ? labelConfig.pattern : ''${#0}'';\n localConfig.x = labelConfig.x ? labelConfig.x : 0;\n localConfig.y = labelConfig.y ? labelConfig.y : 0;\n localConfig.backgroundColor = labelConfig.backgroundColor ? labelConfig.backgroundColor : ''rgba(0,0,0,0)'';\n \n var settingsFont = labelConfig.font;\n if (!settingsFont) {\n settingsFont = {};\n }\n \n localConfig.font.family = settingsFont.family || ''RobotoDraft'';\n localConfig.font.size = settingsFont.size ? settingsFont.size : 6;\n localConfig.font.style = settingsFont.style ? settingsFont.style : ''normal'';\n localConfig.font.weight = settingsFont.weight ? settingsFont.weight : ''500'';\n localConfig.font.color = settingsFont.color ? settingsFont.color : ''#fff'';\n \n localConfig.replaceInfo = processLabelPattern(localConfig.pattern, self.ctx.data);\n \n var label = {};\n var labelElement = $(''<div/>'');\n labelElement.css(''position'', ''absolute'');\n labelElement.css(''display'', ''none'');\n labelElement.css(''top'', ''0'');\n labelElement.css(''left'', ''0'');\n labelElement.css(''backgroundColor'', localConfig.backgroundColor);\n labelElement.css(''color'', localConfig.font.color);\n labelElement.css(''fontFamily'', localConfig.font.family);\n labelElement.css(''fontStyle'', localConfig.font.style);\n labelElement.css(''fontWeight'', localConfig.font.weight);\n \n labelElement.html(localConfig.pattern);\n self.ctx.$container.append(labelElement);\n label.element = labelElement;\n label.config = localConfig;\n label.htmlSet = false;\n label.visible = false;\n self.ctx.labels.push(label);\n }\n\n var bgImg = $(''<img />'');\n bgImg.hide();\n bgImg.bind(''load'', function()\n {\n self.ctx.bImageHeight = $(this).height();\n self.ctx.bImageWidth = $(this).width();\n self.onResize();\n });\n self.ctx.$container.append(bgImg);\n bgImg.attr(''src'', imageUrl);\n \n self.onDataUpdated();\n}\n\nself.onDataUpdated = function() {\n updateLabels();\n}\n\nself.onResize = function() {\n if (self.ctx.bImageHeight && self.ctx.bImageWidth) {\n var backgroundRect = {};\n var imageRatio = self.ctx.bImageWidth / self.ctx.bImageHeight;\n var componentRatio = self.ctx.width / self.ctx.height;\n if (componentRatio >= imageRatio) {\n backgroundRect.top = 0;\n backgroundRect.bottom = 1.0;\n backgroundRect.xRatio = imageRatio / componentRatio;\n backgroundRect.yRatio = 1;\n var offset = (1 - backgroundRect.xRatio) / 2;\n backgroundRect.left = offset;\n backgroundRect.right = 1 - offset;\n } else {\n backgroundRect.left = 0;\n backgroundRect.right = 1.0;\n backgroundRect.xRatio = 1;\n backgroundRect.yRatio = componentRatio / imageRatio;\n var offset = (1 - backgroundRect.yRatio) / 2;\n backgroundRect.top = offset;\n backgroundRect.bottom = 1 - offset;\n }\n for (var l = 0; l < self.ctx.labels.length; l++) {\n var label = self.ctx.labels[l];\n var labelLeft = backgroundRect.left*100 + (label.config.x*backgroundRect.xRatio);\n var labelTop = backgroundRect.top*100 + (label.config.y*backgroundRect.yRatio);\n var fontSize = self.ctx.height * backgroundRect.yRatio * label.config.font.size / 100;\n label.element.css(''top'', labelTop + ''%'');\n label.element.css(''left'', labelLeft + ''%'');\n label.element.css(''fontSize'', fontSize + ''px'');\n if (!label.visible) {\n label.element.css(''display'', ''block'');\n label.visible = true;\n }\n }\n } \n}\n\n\nfunction isNumber(n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n}\n\nfunction padValue(val, dec, int) {\n var i = 0;\n var s, strVal, n;\n\n val = parseFloat(val);\n n = (val < 0);\n val = Math.abs(val);\n\n if (dec > 0) {\n strVal = val.toFixed(dec).toString().split(''.'');\n s = int - strVal[0].length;\n\n for (; i < s; ++i) {\n strVal[0] = ''0'' + strVal[0];\n }\n\n strVal = (n ? ''-'' : '''') + strVal[0] + ''.'' + strVal[1];\n }\n\n else {\n strVal = Math.round(val).toString();\n s = int - strVal.length;\n\n for (; i < s; ++i) {\n strVal = ''0'' + strVal;\n }\n\n strVal = (n ? ''-'' : '''') + strVal;\n }\n\n return strVal;\n}\n\nfunction updateLabels() {\n for (var l = 0; l < self.ctx.labels.length; l++) {\n var label = self.ctx.labels[l];\n var text = label.config.pattern;\n var replaceInfo = label.config.replaceInfo;\n var updated = false;\n for (var v = 0; v < replaceInfo.variables.length; v++) {\n var variableInfo = replaceInfo.variables[v];\n var txtVal = '''';\n if (variableInfo.dataKeyIndex > -1) {\n var varData = self.ctx.data[variableInfo.dataKeyIndex].data;\n if (varData.length > 0) {\n var val = varData[varData.length-1][1];\n if (isNumber(val)) {\n txtVal = padValue(val, variableInfo.valDec, 0);\n updated = true;\n } else {\n txtVal = val;\n updated = true;\n }\n }\n }\n text = text.split(variableInfo.variable).join(txtVal);\n }\n if (updated || !label.htmlSet) {\n label.element.html(text);\n if (!label.htmlSet) {\n label.htmlSet = true;\n }\n }\n }\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"required\": [\"backgroundImageUrl\"],\n \"properties\": {\n \"backgroundImageUrl\": {\n \"title\": \"Background image\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"labels\": {\n \"title\": \"Labels\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Label\",\n \"type\": \"object\",\n \"required\": [\"pattern\"],\n \"properties\": {\n \"pattern\": {\n \"title\": \"Pattern ( for ex. ''Text ${keyName} units.'' or ''${#<key index>} units'' )\",\n \"type\": \"string\",\n \"default\": \"${#0}\"\n },\n \"x\": {\n \"title\": \"X (Percentage relative to background)\",\n \"type\": \"number\",\n \"default\": 50\n },\n \"y\": {\n \"title\": \"Y (Percentage relative to background)\",\n \"type\": \"number\",\n \"default\": 50\n },\n \"backgroundColor\": {\n \"title\": \"Backround color\",\n \"type\": \"string\",\n \"default\": \"rgba(0,0,0,0)\"\n },\n \"font\": {\n \"type\": \"object\",\n \"properties\": {\n \"family\": {\n \"title\": \"Font family\",\n \"type\": \"string\",\n \"default\": \"RobotoDraft\"\n },\n \"size\": {\n \"title\": \"Relative font size (percents)\",\n \"type\": \"number\",\n \"default\": 6\n },\n \"style\": {\n \"title\": \"Style\",\n \"type\": \"string\",\n \"default\": \"normal\"\n },\n \"weight\": {\n \"title\": \"Weight\",\n \"type\": \"string\",\n \"default\": \"500\"\n },\n \"color\": {\n \"title\": \"color\",\n \"type\": \"string\",\n \"default\": \"#fff\"\n }\n }\n }\n }\n }\n }\n }\n },\n \"form\": [\n {\n \"key\": \"backgroundImageUrl\",\n \"type\": \"image\"\n },\n {\n \"key\": \"labels\",\n \"items\": [\n \"labels[].pattern\",\n \"labels[].x\",\n \"labels[].y\",\n {\n \"key\": \"labels[].backgroundColor\",\n \"type\": \"color\"\n },\n \"labels[].font.family\",\n \"labels[].font.size\",\n {\n \"key\": \"labels[].font.style\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"normal\",\n \"label\": \"Normal\"\n },\n {\n \"value\": \"italic\",\n \"label\": \"Italic\"\n },\n {\n \"value\": \"oblique\",\n \"label\": \"Oblique\"\n }\n ]\n\n },\n {\n \"key\": \"labels[].font.weight\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"normal\",\n \"label\": \"Normal\"\n },\n {\n \"value\": \"bold\",\n \"label\": \"Bold\"\n },\n {\n \"value\": \"bolder\",\n \"label\": \"Bolder\"\n },\n {\n \"value\": \"lighter\",\n \"label\": \"Lighter\"\n },\n {\n \"value\": \"100\",\n \"label\": \"100\"\n },\n {\n \"value\": \"200\",\n \"label\": \"200\"\n },\n {\n \"value\": \"300\",\n \"label\": \"300\"\n },\n {\n \"value\": \"400\",\n \"label\": \"400\"\n },\n {\n \"value\": \"500\",\n \"label\": \"500\"\n },\n {\n \"value\": \"600\",\n \"label\": \"600\"\n },\n {\n \"value\": \"700\",\n \"label\": \"800\"\n },\n {\n \"value\": \"800\",\n \"label\": \"800\"\n },\n {\n \"value\": \"900\",\n \"label\": \"900\"\n }\n ]\n },\n {\n \"key\": \"labels[].font.color\",\n \"type\": \"color\"\n }\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"var\",\"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\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"backgroundImageUrl\":\"\",\"labels\":[{\"pattern\":\"Value: ${#0:2} units.\",\"x\":20,\"y\":47,\"font\":{\"color\":\"#515151\",\"family\":\"Roboto\",\"size\":6,\"style\":\"normal\",\"weight\":\"500\"}}]},\"title\":\"Label widget\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
148 -'Label widget' );  
149 -  
150 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
151 -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'simple_card',  
152 -'{"type":"latest","sizeX":5,"sizeY":3,"resources":[],"templateHtml":"","templateCss":"#container {\n overflow: auto;\n}\n\n.tbDatasource-container {\n width: 100%;\n height: 100%;\n overflow: hidden;\n}\n\n.tbDatasource-table {\n width: 100%;\n height: 100%;\n border-collapse: collapse;\n white-space: nowrap;\n font-weight: 100;\n text-align: right;\n}\n\n.tbDatasource-table td {\n padding: 12px;\n position: relative;\n box-sizing: border-box;\n}\n\n.tbDatasource-data-key {\n opacity: 0.7;\n font-weight: 400;\n font-size: 3.500rem;\n}\n\n.tbDatasource-value {\n font-size: 5.000rem;\n}","controllerScript":"self.onInit = function() {\n self.ctx.units = self.ctx.settings.units || self.ctx.units;\n self.ctx.valueDec = (typeof self.ctx.settings.valueDec !== ''undefined'' && self.ctx.settings.valueDec !== null)\n ? self.ctx.settings.valueDec : self.ctx.decimals;\n \n self.ctx.labelPosition = self.ctx.settings.labelPosition || ''left'';\n \n if (self.ctx.datasources.length > 0) {\n var tbDatasource = self.ctx.datasources[0];\n var datasourceId = ''tbDatasource'' + 0;\n self.ctx.$container.append(\n \"<div id=''\" + datasourceId +\n \"'' class=''tbDatasource-container''></div>\"\n );\n \n self.ctx.datasourceContainer = $(''#'' + datasourceId,\n self.ctx.$container);\n \n var tableId = ''table'' + 0;\n self.ctx.datasourceContainer.append(\n \"<table id=''\" + tableId +\n \"'' class=''tbDatasource-table''><col width=''30%''><col width=''70%''></table>\"\n );\n var table = $(''#'' + tableId, self.ctx.$container);\n if (self.ctx.labelPosition === ''top'') {\n table.css(''text-align'', ''left'');\n }\n \n if (tbDatasource.dataKeys.length > 0) {\n var dataKey = tbDatasource.dataKeys[0];\n var labelCellId = ''labelCell'' + 0;\n var cellId = ''cell'' + 0;\n if (self.ctx.labelPosition === ''left'') {\n table.append(\n \"<tr><td class=''tbDatasource-data-key'' id=''\" + labelCellId +\"''>\" +\n dataKey.label +\n \"</td><td class=''tbDatasource-value'' id=''\" +\n cellId +\n \"''></td></tr>\");\n } else {\n table.append(\n \"<tr style=''vertical-align: bottom;''><td class=''tbDatasource-data-key'' id=''\" + labelCellId +\"''>\" +\n dataKey.label +\n \"</td></tr><tr><td class=''tbDatasource-value'' id=''\" +\n cellId +\n \"''></td></tr>\");\n }\n self.ctx.labelCell = $(''#'' + labelCellId, table);\n self.ctx.valueCell = $(''#'' + cellId, table);\n self.ctx.valueCell.html(0 + '' '' + self.ctx.units);\n }\n }\n \n $.fn.textWidth = function(){\n var html_org = $(this).html();\n var html_calc = ''<span>'' + html_org + ''</span>'';\n $(this).html(html_calc);\n var width = $(this).find(''span:first'').width();\n $(this).html(html_org);\n return width;\n }; \n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n \n function isNumber(n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n }\n \n function padValue(val, dec, int) {\n var i = 0;\n var s, strVal, n;\n \n val = parseFloat(val);\n n = (val < 0);\n val = Math.abs(val);\n \n if (dec > 0) {\n strVal = val.toFixed(dec).toString().split(''.'');\n s = int - strVal[0].length;\n \n for (; i < s; ++i) {\n strVal[0] = ''0'' + strVal[0];\n }\n \n strVal = (n ? ''-'' : '''') + strVal[0] + ''.'' + strVal[1];\n }\n \n else {\n strVal = Math.round(val).toString();\n s = int - strVal.length;\n \n for (; i < s; ++i) {\n strVal = ''0'' + strVal;\n }\n \n strVal = (n ? ''-'' : '''') + strVal;\n }\n \n return strVal;\n }\n \n if (self.ctx.valueCell && self.ctx.data.length > 0) {\n var cellData = self.ctx.data[0];\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length -\n 1];\n var value = tvPair[1];\n var txtValue;\n if (isNumber(value)) {\n txtValue = padValue(value, self.ctx.alueDec, 0) + '' '' + self.ctx.units;\n } else {\n txtValue = value;\n }\n self.ctx.valueCell.html(txtValue);\n var targetWidth;\n var minDelta;\n if (self.ctx.labelPosition === ''left'') {\n targetWidth = self.ctx.datasourceContainer.width() - self.ctx.labelCell.width();\n minDelta = self.ctx.width/16 + self.ctx.padding;\n } else {\n targetWidth = self.ctx.datasourceContainer.width();\n minDelta = self.ctx.padding;\n }\n var delta = targetWidth - self.ctx.valueCell.textWidth();\n var fontSize = self.ctx.valueFontSize;\n if (targetWidth > minDelta) {\n while (delta < minDelta && fontSize > 6) {\n fontSize--;\n self.ctx.valueCell.css(''font-size'', fontSize+''px'');\n delta = targetWidth - self.ctx.valueCell.textWidth();\n }\n }\n }\n } \n \n}\n\nself.onResize = function() {\n var labelFontSize;\n if (self.ctx.labelPosition === ''top'') {\n self.ctx.padding = self.ctx.height/20;\n labelFontSize = self.ctx.height/4;\n self.ctx.valueFontSize = self.ctx.height/2;\n } else {\n self.ctx.padding = self.ctx.width/50;\n labelFontSize = self.ctx.height/2.5;\n self.ctx.valueFontSize = height/2;\n if (self.ctx.width/self.ctx.height <= 2.7) {\n labelFontSize = self.ctx.width/7;\n self.ctx.valueFontSize = self.ctx.width/6;\n }\n }\n self.ctx.padding = Math.min(12, self.ctx.padding);\n \n if (self.ctx.labelCell) {\n self.ctx.labelCell.css(''font-size'', labelFontSize+''px'');\n self.ctx.labelCell.css(''padding'', self.ctx.padding+''px'');\n }\n if (self.ctx.valueCell) {\n self.ctx.valueCell.css(''font-size'', self.ctx.valueFontSize+''px'');\n self.ctx.valueCell.css(''padding'', self.ctx.padding+''px'');\n } \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"labelPosition\": {\n \"title\": \"Label position\",\n \"type\": \"string\",\n \"default\": \"left\"\n }\n },\n \"required\": []\n },\n \"form\": [\n {\n \"key\": \"labelPosition\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"left\",\n \"label\": \"Left\"\n },\n {\n \"value\": \"top\",\n \"label\": \"Top\"\n }\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.2392660816082064,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ff5722\",\"color\":\"rgba(255, 255, 255, 0.87)\",\"padding\":\"16px\",\"settings\":{\"labelPosition\":\"top\"},\"title\":\"Simple card\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"units\":\"°C\",\"decimals\":0,\"useDashboardTimewindow\":true,\"showLegend\":false}"}',  
153 -'Simple card' );  
154 -  
155 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
156 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges', 'lcd_bar_gauge',  
157 -'{"type":"latest","sizeX":2,"sizeY":3.5,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Humidity\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#babab2\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\"linear\",\"refreshAnimationTime\":700,\"startAnimationType\":\"linear\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"400\",\"size\":16},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":0,\"dashThickness\":1.5,\"decimals\":0,\"showUnitTitle\":true,\"defaultColor\":\"#444444\",\"gaugeType\":\"verticalBar\",\"units\":\"%\"},\"title\":\"LCD bar gauge\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
158 -'LCD bar gauge' );  
159 -  
160 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
161 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',  
162 -'vertical_bar_justgage',  
163 -'{"type":"latest","sizeX":2,"sizeY":3.5,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#f57c00\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ffffff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#999999\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"500\",\"size\":12,\"color\":\"#666666\"},\"minMaxFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#666666\"},\"neonGlowBrightness\":0,\"decimals\":0,\"dashThickness\":1.5,\"gaugeColor\":\"#eeeeee\",\"showTitle\":false,\"gaugeType\":\"verticalBar\"},\"title\":\"Vertical bar - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
164 -'Vertical bar - justGage' );  
165 -  
166 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
167 -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'html_value_card',  
168 -'{"type":"latest","sizeX":7.5,"sizeY":3,"resources":[],"templateHtml":"","templateCss":"","controllerScript":"self.onInit = function() {\n self.ctx.varsRegex = /\\$\\{([^\\}]*)\\}/g;\n self.ctx.htmlSet = false;\n \n var cssParser = new cssjs();\n cssParser.testMode = false;\n var namespace = ''html-value-card-'' + hashCode(self.ctx.settings.cardCss);\n cssParser.cssPreviewNamespace = namespace;\n cssParser.createStyleElement(namespace, self.ctx.settings.cardCss);\n self.ctx.$container.addClass(namespace);\n self.ctx.html = self.ctx.settings.cardHtml;\n self.ctx.replaceInfo = processHtmlPattern(self.ctx.html, self.ctx.data);\n \n updateHtml();\n \n function hashCode(str) {\n var hash = 0;\n var i, char;\n if (str.length === 0) return hash;\n for (i = 0; i < str.length; i++) {\n char = str.charCodeAt(i);\n hash = ((hash << 5) - hash) + char;\n hash = hash & hash;\n }\n return hash;\n }\n \n function processHtmlPattern(pattern, data) {\n var match = self.ctx.varsRegex.exec(pattern);\n var replaceInfo = {};\n replaceInfo.variables = [];\n while (match !== null) {\n var variableInfo = {};\n variableInfo.dataKeyIndex = -1;\n var variable = match[0];\n var label = match[1];\n var valDec = 2;\n var splitVals = label.split('':'');\n if (splitVals.length > 1) {\n label = splitVals[0];\n valDec = parseFloat(splitVals[1]);\n }\n variableInfo.variable = variable;\n variableInfo.valDec = valDec;\n \n if (label.startsWith(''#'')) {\n var keyIndexStr = label.substring(1);\n var n = Math.floor(Number(keyIndexStr));\n if (String(n) === keyIndexStr && n >= 0) {\n variableInfo.dataKeyIndex = n;\n }\n }\n if (variableInfo.dataKeyIndex === -1) {\n for (var i = 0; i < data.length; i++) {\n var datasourceData = data[i];\n var dataKey = datasourceData.dataKey;\n if (dataKey.label === label) {\n variableInfo.dataKeyIndex = i;\n break;\n }\n }\n }\n replaceInfo.variables.push(variableInfo);\n match = self.ctx.varsRegex.exec(pattern);\n }\n return replaceInfo;\n } \n}\n\nself.onDataUpdated = function() {\n updateHtml();\n}\n\nself.onDestroy = function() {\n}\n\nfunction isNumber(n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n}\n\nfunction padValue(val, dec, int) {\n var i = 0;\n var s, strVal, n;\n\n val = parseFloat(val);\n n = (val < 0);\n val = Math.abs(val);\n\n if (dec > 0) {\n strVal = val.toFixed(dec).toString().split(''.'');\n s = int - strVal[0].length;\n\n for (; i < s; ++i) {\n strVal[0] = ''0'' + strVal[0];\n }\n\n strVal = (n ? ''-'' : '''') + strVal[0] + ''.'' + strVal[1];\n }\n\n else {\n strVal = Math.round(val).toString();\n s = int - strVal.length;\n\n for (; i < s; ++i) {\n strVal = ''0'' + strVal;\n }\n\n strVal = (n ? ''-'' : '''') + strVal;\n }\n\n return strVal;\n}\n\nfunction updateHtml() {\n var text = self.ctx.html;\n var updated = false;\n for (var v in self.ctx.replaceInfo.variables) {\n var variableInfo = self.ctx.replaceInfo.variables[v];\n var txtVal = '''';\n if (variableInfo.dataKeyIndex > -1) {\n var varData = self.ctx.data[variableInfo.dataKeyIndex].data;\n if (varData.length > 0) {\n var val = varData[varData.length-1][1];\n if (isNumber(val)) {\n txtVal = padValue(val, variableInfo.valDec, 0);\n } else {\n txtVal = val;\n }\n }\n }\n if (typeof variableInfo.lastVal === undefined ||\n variableInfo.lastVal !== txtVal) {\n updated = true;\n variableInfo.lastVal = txtVal;\n }\n text = text.split(variableInfo.variable).join(txtVal);\n }\n if (updated || !self.ctx.htmlSet) {\n self.ctx.$container.html(text);\n if (!self.ctx.htmlSet) {\n self.ctx.htmlSet = true;\n }\n }\n}\n\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"required\": [\"cardHtml\"],\n \"properties\": {\n \"cardCss\": {\n \"title\": \"CSS\",\n \"type\": \"string\",\n \"default\": \".card {\\n font-weight: bold; \\n}\"\n },\n \"cardHtml\": {\n \"title\": \"HTML\",\n \"type\": \"string\",\n \"default\": \"<div class=''card''>HTML code here</div>\"\n }\n }\n },\n \"form\": [\n {\n \"key\": \"cardCss\",\n \"type\": \"css\"\n }, \n {\n \"key\": \"cardHtml\",\n \"type\": \"html\"\n } \n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"My value\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"return Math.random() * 5.45;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"cardCss\":\".card {\\n width: 100%;\\n height: 100%;\\n border: 2px solid #ccc;\\n box-sizing: border-box;\\n}\\n\\n.card .content {\\n padding: 20px;\\n display: flex;\\n flex-direction: row;\\n align-items: center;\\n justify-content: space-around;\\n height: 100%;\\n box-sizing: border-box;\\n}\\n\\n.card .content .column {\\n display: flex;\\n flex-direction: column; \\n justify-content: space-around;\\n height: 100%;\\n}\\n\\n.card h1 {\\n text-transform: uppercase;\\n color: #999;\\n font-size: 20px;\\n font-weight: bold;\\n margin: 0;\\n padding-bottom: 10px;\\n line-height: 32px;\\n}\\n\\n.card .value {\\n font-size: 38px;\\n font-weight: 200;\\n}\\n\\n.card .description {\\n font-size: 20px;\\n color: #999;\\n}\\n\",\"cardHtml\":\"<div class=''card''>\\n <div class=''content''>\\n <div class=''column''>\\n <h1>Value title</h1>\\n <div class=''value''>\\n ${My value:2} units.\\n </div> \\n <div class=''description''>\\n Value description text\\n </div>\\n </div>\\n <img height=\\\"80px\\\" src=\\\"https://thingsboard.io/images/logo_small.png\\\" />\\n </div>\\n</div>\"},\"title\":\"HTML Value Card\",\"dropShadow\":false}"}',  
169 -'HTML Value Card' );  
170 -  
171 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
172 -VALUES ( now ( ), minTimeuuid ( 0 ), 'gpio_widgets',  
173 -'raspberry_pi_gpio_panel',  
174 -'{"type":"latest","sizeX":7,"sizeY":10.5,"resources":[],"templateHtml":"<div class=\"gpio-panel\" style=\"height: 100%;\">\n <section layout=\"row\" ng-repeat=\"row in rows\">\n <section flex layout=\"row\" ng-repeat=\"cell in row\">\n <section layout=\"row\" flex ng-if=\"cell\" layout-align=\"{{$index===0 ? ''end center'' : ''start center''}}\">\n <span class=\"gpio-left-label\" ng-show=\"$index===0\">{{ cell.label }}</span>\n <section layout=\"row\" class=\"led-panel\" ng-class=\"$index===0 ? ''col-0'' : ''col-1''\"\n ng-style=\"{backgroundColor: ledPanelBackgroundColor}\">\n <span class=\"pin\" ng-show=\"$index===0\">{{cell.pin}}</span>\n <span class=\"led-container\">\n <tb-led-light size=\"prefferedRowHeight\"\n color-on=\"cell.colorOn\"\n color-off=\"cell.colorOff\"\n off-opacity=\"''0.9''\"\n tb-enabled=\"cell.enabled\">\n </tb-led-light>\n </span>\n <span class=\"pin\" ng-show=\"$index===1\">{{cell.pin}}</span>\n </section>\n <span class=\"gpio-right-label\" ng-show=\"$index===1\">{{ cell.label }}</span>\n </section>\n <section layout=\"row\" flex ng-if=\"!cell\">\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"led-panel\"\n ng-style=\"{backgroundColor: ledPanelBackgroundColor}\"></span>\n <span flex ng-show=\"$index===1\"></span>\n </section>\n </section>\n </section> \n</div>","templateCss":".error {\n font-size: 14px !important;\n color: maroon;/*rgb(250,250,250);*/\n background-color: transparent;\n padding: 6px;\n}\n\n.error span {\n margin: auto;\n}\n\n.gpio-panel {\n padding-top: 10px;\n white-space: nowrap;\n}\n\n.gpio-panel tb-led-light > div {\n margin: auto;\n}\n\n.led-panel {\n margin: 0;\n width: 66px;\n min-width: 66px;\n}\n\n.led-container {\n width: 48px;\n min-width: 48px;\n}\n\n.pin {\n margin-top: auto;\n margin-bottom: auto;\n color: white;\n font-size: 12px;\n width: 16px;\n min-width: 16px;\n}\n\n.led-panel.col-0 .pin {\n margin-left: auto;\n padding-left: 2px;\n text-align: right;\n}\n\n.led-panel.col-1 .pin {\n margin-right: auto;\n \n text-align: left;\n}\n\n.gpio-left-label {\n margin-right: 8px;\n}\n\n.gpio-right-label {\n margin-left: 8px;\n}","controllerScript":"self.onInit = function() {\n var i, gpio;\n \n var scope = self.ctx.$scope;\n var settings = self.ctx.settings;\n \n scope.gpioList = [];\n scope.gpioByPin = {};\n for (var g = 0; g < settings.gpioList.length; g++) {\n gpio = settings.gpioList[g];\n scope.gpioList.push(\n {\n row: gpio.row,\n col: gpio.col,\n pin: gpio.pin,\n label: gpio.label,\n enabled: false,\n colorOn: tinycolor(gpio.color).lighten(20).toHexString(),\n colorOff: tinycolor(gpio.color).darken().toHexString()\n }\n );\n scope.gpioByPin[gpio.pin] = scope.gpioList[scope.gpioList.length-1];\n }\n\n scope.ledPanelBackgroundColor = settings.ledPanelBackgroundColor || tinycolor(''green'').lighten(2).toRgbString();\n\n scope.gpioCells = {};\n var rowCount = 0;\n for (i = 0; i < scope.gpioList.length; i++) {\n gpio = scope.gpioList[i];\n scope.gpioCells[gpio.row+''_''+gpio.col] = gpio;\n rowCount = Math.max(rowCount, gpio.row+1);\n }\n \n scope.prefferedRowHeight = 32;\n scope.rows = [];\n for (i = 0; i < rowCount; i++) {\n var row = [];\n for (var c =0; c<2;c++) {\n if (scope.gpioCells[i+''_''+c]) {\n row[c] = scope.gpioCells[i+''_''+c];\n } else {\n row[c] = null;\n }\n }\n scope.rows.push(row);\n } \n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n var changed = false;\n for (var d = 0; d < self.ctx.data.length; d++) {\n var cellData = self.ctx.data[d];\n var dataKey = cellData.dataKey;\n var gpio = self.ctx.$scope.gpioByPin[dataKey.label];\n if (gpio) {\n var enabled = false;\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n enabled = (tvPair[1] === true || tvPair[1] === ''true'');\n }\n if (gpio.enabled != enabled) {\n changed = true;\n gpio.enabled = enabled;\n }\n }\n }\n if (changed) {\n self.ctx.$scope.$digest();\n } \n}\n\nself.onResize = function() {\n var rowCount = self.ctx.$scope.rows.length;\n var prefferedRowHeight = (self.ctx.height - 35)/rowCount;\n prefferedRowHeight = Math.min(32, prefferedRowHeight);\n prefferedRowHeight = Math.max(12, prefferedRowHeight);\n self.ctx.$scope.prefferedRowHeight = prefferedRowHeight;\n \n var ratio = prefferedRowHeight/32;\n \n var leftLabels = $(''.gpio-left-label'', self.ctx.$container);\n leftLabels.css(''font-size'', 16*ratio+''px'');\n var rightLabels = $(''.gpio-right-label'', self.ctx.$container);\n rightLabels.css(''font-size'', 16*ratio+''px'');\n var pins = $(''.pin'', self.ctx.$container);\n var pinsFontSize = Math.max(9, 12*ratio);\n pins.css(''font-size'', pinsFontSize+''px''); \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"gpioList\": {\n \"title\": \"Gpio leds\",\n \"type\": \"array\",\n \"minItems\" : 1,\n \"items\": {\n \"title\": \"Gpio led\",\n \"type\": \"object\",\n \"properties\": {\n \"pin\": {\n \"title\": \"Pin\",\n \"type\": \"number\"\n },\n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"row\": {\n \"title\": \"Row\",\n \"type\": \"number\"\n },\n \"col\": {\n \"title\": \"Column\",\n \"type\": \"number\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\",\n \"default\": \"red\"\n }\n },\n \"required\": [\"pin\", \"label\", \"row\", \"col\", \"color\"]\n }\n },\n \"ledPanelBackgroundColor\": {\n \"title\": \"LED panel background color\",\n \"type\": \"string\",\n \"default\": \"#008a00\"\n } \n },\n \"required\": [\"gpioList\", \n \"ledPanelBackgroundColor\"]\n },\n \"form\": [\n {\n \"key\": \"gpioList\",\n \"items\": [\n \"gpioList[].pin\",\n \"gpioList[].label\",\n \"gpioList[].row\",\n \"gpioList[].col\",\n {\n \"key\": \"gpioList[].color\",\n \"type\": \"color\"\n }\n ]\n },\n {\n \"key\": \"ledPanelBackgroundColor\",\n \"type\": \"color\"\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"gpioList\":[{\"pin\":1,\"label\":\"3.3V\",\"row\":0,\"col\":0,\"color\":\"#fc9700\",\"_uniqueKey\":0},{\"pin\":2,\"label\":\"5V\",\"row\":0,\"col\":1,\"color\":\"#fb0000\",\"_uniqueKey\":1},{\"pin\":3,\"label\":\"GPIO 2 (I2C1_SDA)\",\"row\":1,\"col\":0,\"color\":\"#02fefb\",\"_uniqueKey\":2},{\"color\":\"#fb0000\",\"pin\":4,\"label\":\"5V\",\"row\":1,\"col\":1},{\"color\":\"#02fefb\",\"pin\":5,\"label\":\"GPIO 3 (I2C1_SCL)\",\"row\":2,\"col\":0},{\"color\":\"#000000\",\"pin\":6,\"label\":\"GND\",\"row\":2,\"col\":1},{\"color\":\"#00fd00\",\"pin\":7,\"label\":\"GPIO 4 (GPCLK0)\",\"row\":3,\"col\":0},{\"color\":\"#fdfb00\",\"pin\":8,\"label\":\"GPIO 14 (UART_TXD)\",\"row\":3,\"col\":1},{\"color\":\"#000000\",\"pin\":9,\"label\":\"GND\",\"row\":4,\"col\":0},{\"color\":\"#fdfb00\",\"pin\":10,\"label\":\"GPIO 15 (UART_RXD)\",\"row\":4,\"col\":1},{\"color\":\"#00fd00\",\"pin\":11,\"label\":\"GPIO 17\",\"row\":5,\"col\":0},{\"color\":\"#00fd00\",\"pin\":12,\"label\":\"GPIO 18\",\"row\":5,\"col\":1},{\"color\":\"#00fd00\",\"pin\":13,\"label\":\"GPIO 27\",\"row\":6,\"col\":0},{\"color\":\"#000000\",\"pin\":14,\"label\":\"GND\",\"row\":6,\"col\":1},{\"color\":\"#00fd00\",\"pin\":15,\"label\":\"GPIO 22\",\"row\":7,\"col\":0},{\"color\":\"#00fd00\",\"pin\":16,\"label\":\"GPIO 23\",\"row\":7,\"col\":1},{\"color\":\"#fc9700\",\"pin\":17,\"label\":\"3.3V\",\"row\":8,\"col\":0},{\"color\":\"#00fd00\",\"pin\":18,\"label\":\"GPIO 24\",\"row\":8,\"col\":1},{\"color\":\"#fd01fd\",\"pin\":19,\"label\":\"GPIO 10 (SPI_MOSI)\",\"row\":9,\"col\":0},{\"color\":\"#000000\",\"pin\":20,\"label\":\"GND\",\"row\":9,\"col\":1},{\"color\":\"#fd01fd\",\"pin\":21,\"label\":\"GPIO 9 (SPI_MISO)\",\"row\":10,\"col\":0},{\"color\":\"#00fd00\",\"pin\":22,\"label\":\"GPIO 25\",\"row\":10,\"col\":1},{\"color\":\"#fd01fd\",\"pin\":23,\"label\":\"GPIO 11 (SPI_SCLK)\",\"row\":11,\"col\":0},{\"color\":\"#fd01fd\",\"pin\":24,\"label\":\"GPIO 8 (SPI_CE0)\",\"row\":11,\"col\":1},{\"color\":\"#000000\",\"pin\":25,\"label\":\"GND\",\"row\":12,\"col\":0},{\"color\":\"#fd01fd\",\"pin\":26,\"label\":\"GPIO 7 (SPI_CE1)\",\"row\":12,\"col\":1},{\"color\":\"#ffffff\",\"pin\":27,\"label\":\"ID_SD\",\"row\":13,\"col\":0},{\"color\":\"#ffffff\",\"pin\":28,\"label\":\"ID_SC\",\"row\":13,\"col\":1},{\"color\":\"#00fd00\",\"pin\":29,\"label\":\"GPIO 5\",\"row\":14,\"col\":0},{\"color\":\"#000000\",\"pin\":30,\"label\":\"GND\",\"row\":14,\"col\":1},{\"color\":\"#00fd00\",\"pin\":31,\"label\":\"GPIO 6\",\"row\":15,\"col\":0},{\"color\":\"#00fd00\",\"pin\":32,\"label\":\"GPIO 12\",\"row\":15,\"col\":1},{\"color\":\"#00fd00\",\"pin\":33,\"label\":\"GPIO 13\",\"row\":16,\"col\":0},{\"color\":\"#000000\",\"pin\":34,\"label\":\"GND\",\"row\":16,\"col\":1},{\"color\":\"#00fd00\",\"pin\":35,\"label\":\"GPIO 19\",\"row\":17,\"col\":0},{\"color\":\"#00fd00\",\"pin\":36,\"label\":\"GPIO 16\",\"row\":17,\"col\":1},{\"color\":\"#00fd00\",\"pin\":37,\"label\":\"GPIO 26\",\"row\":18,\"col\":0},{\"color\":\"#00fd00\",\"pin\":38,\"label\":\"GPIO 20\",\"row\":18,\"col\":1},{\"color\":\"#000000\",\"pin\":39,\"label\":\"GND\",\"row\":19,\"col\":0},{\"color\":\"#00fd00\",\"pin\":40,\"label\":\"GPIO 21\",\"row\":19,\"col\":1}],\"ledPanelBackgroundColor\":\"#008a00\"},\"title\":\"Raspberry Pi GPIO Panel\",\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"7\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.22518255793320163,\"funcBody\":\"var period = time % 1500;\\nreturn period < 500;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"11\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.7008206860666621,\"funcBody\":\"var period = time % 1500;\\nreturn period >= 500 && period < 1000;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"12\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.42600325102193426,\"funcBody\":\"var period = time % 1500;\\nreturn period >= 1000;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"13\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.48362241571415243,\"funcBody\":\"var period = time % 1500;\\nreturn period < 500;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"29\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.7217670147518815,\"funcBody\":\"var period = time % 1500;\\nreturn period >= 500 && period < 1000;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}}}"}',  
175 -'Raspberry Pi GPIO Panel' );  
176 -  
177 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
178 -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'radar_chart_js',  
179 -'{"type":"latest","sizeX":7,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.3.0/Chart.min.js"}],"templateHtml":"<canvas id=\"radarChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n var barData = {\n labels: [],\n datasets: []\n };\n\n var backgroundColor = tinycolor(self.ctx.data[0].dataKey.color);\n backgroundColor.setAlpha(0.2);\n var borderColor = tinycolor(self.ctx.data[0].dataKey.color);\n borderColor.setAlpha(1);\n var dataset = {\n label: self.ctx.datasources[0].name,\n data: [],\n backgroundColor: backgroundColor.toRgbString(),\n borderColor: borderColor.toRgbString(),\n pointBackgroundColor: borderColor.toRgbString(),\n pointBorderColor: borderColor.darken().toRgbString(),\n borderWidth: 1\n }\n \n barData.datasets.push(dataset);\n \n for (var i = 0; i < self.ctx.data.length; i++) {\n var dataKey = self.ctx.data[i].dataKey;\n barData.labels.push(dataKey.label);\n dataset.data.push(0);\n }\n\n var ctx = $(''#radarChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''radar'',\n data: barData,\n options: {\n responsive: false,\n maintainAspectRatio: false\n }\n });\n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n for (var i = 0; i < self.ctx.data.length; i++) {\n var cellData = self.ctx.data[i];\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n var value = tvPair[1];\n self.ctx.chart.data.datasets[0].data[i] = parseFloat(value);\n }\n } \n self.ctx.chart.update();\n}\n\nself.onResize = function() {\n if (self.ctx.height >= 70) {\n self.ctx.chart.resize();\n }\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.545701115289893,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.2592906835158064,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.12880275585455747,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Radar - Chart.js\"}"}',  
180 -'Radar - Chart.js' );  
181 -  
182 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
183 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',  
184 -'neon_gauge_justgage',  
185 -'{"type":"latest","sizeX":5,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":70,\"dashThickness\":1,\"decimals\":1,\"gaugeType\":\"arc\"},\"title\":\"Neon gauge - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
186 -'Neon gauge - justGage' );  
187 -  
188 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
189 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',  
190 -'simple_gauge_justgage',  
191 -'{"type":"latest","sizeX":2,"sizeY":2,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>\n","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"\nself.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#ef6c00\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ffffff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32,\"color\":\"#666666\"},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":0,\"dashThickness\":0,\"decimals\":0,\"gaugeColor\":\"#eeeeee\",\"gaugeType\":\"donut\"},\"title\":\"Simple gauge - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
192 -'Simple gauge - justGage' );  
193 -  
194 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
195 -VALUES ( now ( ), minTimeuuid ( 0 ), 'maps', 'route_map',  
196 -'{"type":"timeseries","sizeX":8.5,"sizeY":6,"resources":[],"templateHtml":"","templateCss":".error {\n color: red;\n}\n.tb-labels {\n color: #222;\n font: 12px/1.5 \"Helvetica Neue\", Arial, Helvetica, sans-serif;\n text-align: center;\n width: 100px;\n white-space: nowrap;\n}","controllerScript":"self.onInit = function() {\n self.ctx.map = new TbMapWidget(''google-map'', true, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"title\": \"Route Map Configuration\",\n \"type\": \"object\",\n \"properties\": {\n \"gmApiKey\": {\n \"title\": \"Google Maps API Key\",\n \"type\": \"string\"\n },\n \"gmDefaultMapType\": {\n \"title\": \"Default map type\",\n \"type\": \"string\",\n \"default\": \"roadmap\"\n },\n \"defaultZoomLevel\": {\n \"title\": \"Default map zoom level (1 - 20)\",\n \"type\": \"number\"\n },\n \"fitMapBounds\": {\n \"title\": \"Fit map bounds to cover all routes\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"routesSettings\": {\n \"title\": \"Routes\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Route settings\",\n \"type\": \"object\",\n \"properties\": {\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"lat\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"lng\"\n },\n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n }, \n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"tooltipPattern\": {\n \"title\": \"Pattern ( for ex. ''Text ${keyName} units.'' or ''${#<key index>} units'' )\",\n \"type\": \"string\",\n \"default\": \"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\"\n },\n \"useColorFunction\": {\n \"title\": \"Use color function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"colorFunction\": {\n \"title\": \"Color function: f(data)\",\n \"type\": \"string\"\n },\n \"markerImage\": {\n \"title\": \"Custom marker image\",\n \"type\": \"string\"\n },\n \"markerImageSize\": {\n \"title\": \"Custom marker image size (px)\",\n \"type\": \"number\",\n \"default\": 34\n },\n \"useMarkerImageFunction\": {\n \"title\": \"Use marker image function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"markerImageFunction\": {\n \"title\": \"Marker image function: f(data, images)\",\n \"type\": \"string\"\n },\n \"markerImages\": {\n \"title\": \"Marker images\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker image\",\n \"type\": \"string\"\n }\n },\n \"strokeWeight\": {\n \"title\": \"Stroke weight\",\n \"type\": \"number\",\n \"default\": 2\n },\n \"strokeOpacity\": {\n \"title\": \"Stroke opacity\",\n \"type\": \"number\",\n \"default\": 1.0\n }\n }\n }\n }\n },\n \"required\": [\n \"gmApiKey\"\n ]\n },\n \"form\": [\n \"gmApiKey\",\n {\n \"key\": \"gmDefaultMapType\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"roadmap\",\n \"label\": \"Roadmap\"\n },\n {\n \"value\": \"satellite\",\n \"label\": \"Satellite\"\n },\n {\n \"value\": \"hybrid\",\n \"label\": \"Hybrid\"\n },\n {\n \"value\": \"terrain\",\n \"label\": \"Terrain\"\n }\n ]\n }, \n \"defaultZoomLevel\",\n \"fitMapBounds\",\n {\n \"key\": \"routesSettings\",\n \"items\": [\n \"routesSettings[].latKeyName\",\n \"routesSettings[].lngKeyName\",\n \"routesSettings[].showLabel\",\n \"routesSettings[].label\",\n \"routesSettings[].tooltipPattern\",\n {\n \"key\": \"routesSettings[].color\",\n \"type\": \"color\"\n },\n \"routesSettings[].useColorFunction\",\n {\n \"key\": \"routesSettings[].colorFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"routesSettings[].markerImage\",\n \"type\": \"image\"\n },\n \"routesSettings[].markerImageSize\",\n \"routesSettings[].useMarkerImageFunction\",\n {\n \"key\": \"routesSettings[].markerImageFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"routesSettings[].markerImages\",\n \"items\": [\n {\n \"key\": \"routesSettings[].markerImages[]\",\n \"type\": \"image\"\n }\n ]\n },\n \"routesSettings[].strokeWeight\",\n \"routesSettings[].strokeOpacity\"\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.3467277073670627,\"funcBody\":\"var lats = [37.7696499,\\n37.7699074,\\n37.7699536,\\n37.7697242,\\n37.7695189,\\n37.7696889,\\n37.7697153,\\n37.7701244,\\n37.7700604,\\n37.7705491,\\n37.7715705,\\n37.771752,\\n37.7707533,\\n37.769866];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lats[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.058309787276281666,\"funcBody\":\"var lons = [-122.4261215,\\n-122.4219157,\\n-122.4199623,\\n-122.4179074,\\n-122.4155876,\\n-122.4155521,\\n-122.4163203,\\n-122.4193876,\\n-122.4210496,\\n-122.422284,\\n-122.4232717,\\n-122.4235138,\\n-122.4247605,\\n-122.4258812];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lons[i];\"}],\"intervalSec\":60},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.14288960550237473,\"funcBody\":\"var value = prevValue;\\nif (time % 500 < 100) {\\n value = value + Math.random() * 40 - 20;\\n if (value < 45) {\\n \\tvalue = 45;\\n } else if (value > 130) {\\n \\tvalue = 130;\\n }\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":30000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"gmApiKey\":\"AIzaSyDoEx2kaGz3PxwbI9T7ccTSg5xjdw8Nw8Q\",\"fitMapBounds\":true,\"routesSettings\":[{\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"color\":\"#1976d2\",\"strokeWeight\":4,\"strokeOpacity\":0.65,\"label\":\"First route\",\"tooltipPattern\":\"<b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Speed:</b> ${Speed} MPH<br/><small>See advanced settings for details</small>\",\"useColorFunction\":true,\"markerImageSize\":34,\"useMarkerImageFunction\":true,\"markerImageFunction\":\"var speed = data[''Speed''];\\nvar res = {\\n url: images[0],\\n size: 55\\n};\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n var index = Math.floor(3 * percent);\\n res.url = images[index];\\n}\\nreturn res;\",\"markerImages\":[\"\",\"\",\"\"],\"colorFunction\":\"var speed = data[''Speed''];\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n if (percent < 0.5) {\\n percent *=2*100; \\n return tinycolor.mix(''green'', ''yellow'', amount = percent).toHexString();\\n } else {\\n percent = (percent - 0.5)*2*100;\\n return tinycolor.mix(''yellow'', ''red'', amount = percent).toHexString();\\n }\\n}\\nreturn ''green'';\"}]},\"title\":\"Route Map\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false}"}',  
197 -'Route Map' );  
198 -  
199 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
200 -VALUES ( now ( ), minTimeuuid ( 0 ), 'analogue_gauges',  
201 -'temperature_gauge_canvas_gauges',  
202 -'{"type":"latest","sizeX":7,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"linearGauge\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbAnalogueLinearGauge(self.ctx, ''linearGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.getSettingsSchema = function() {\n return TbAnalogueLinearGauge.settingsSchema;\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 30 - 15;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"maxValue\":100,\"defaultColor\":\"#e64a19\",\"barStrokeWidth\":2.5,\"colorBar\":\"rgba(255, 255, 255, 0.4)\",\"colorBarEnd\":\"rgba(221, 221, 221, 0.38)\",\"showUnitTitle\":true,\"minorTicks\":2,\"valueBox\":true,\"valueInt\":3,\"colorPlate\":\"#fff\",\"colorMajorTicks\":\"#444\",\"colorMinorTicks\":\"#666\",\"colorNeedleShadowUp\":\"rgba(2,255,255,0.2)\",\"colorNeedleShadowDown\":\"rgba(188,143,143,0.45)\",\"colorValueBoxRect\":\"#888\",\"colorValueBoxRectEnd\":\"#666\",\"colorValueBoxBackground\":\"#babab2\",\"colorValueBoxShadow\":\"rgba(0,0,0,1)\",\"highlightsWidth\":10,\"animation\":true,\"animationDuration\":1500,\"animationRule\":\"linear\",\"showBorder\":false,\"majorTicksCount\":8,\"numbersFont\":{\"family\":\"Arial\",\"size\":18,\"style\":\"normal\",\"weight\":\"normal\",\"color\":\"#263238\"},\"titleFont\":{\"family\":\"Roboto\",\"size\":24,\"style\":\"normal\",\"weight\":\"normal\",\"color\":\"#78909c\"},\"unitsFont\":{\"family\":\"Roboto\",\"size\":26,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#37474f\"},\"valueFont\":{\"family\":\"Roboto\",\"size\":40,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#444\",\"shadowColor\":\"rgba(0,0,0,0.3)\"},\"minValue\":-60,\"highlights\":[{\"from\":-60,\"to\":-40,\"color\":\"#90caf9\"},{\"from\":-40,\"to\":-20,\"color\":\"rgba(144, 202, 249, 0.66)\"},{\"from\":-20,\"to\":0,\"color\":\"rgba(144, 202, 249, 0.33)\"},{\"from\":0,\"to\":20,\"color\":\"rgba(244, 67, 54, 0.2)\"},{\"from\":20,\"to\":40,\"color\":\"rgba(244, 67, 54, 0.4)\"},{\"from\":40,\"to\":60,\"color\":\"rgba(244, 67, 54, 0.6)\"},{\"from\":60,\"to\":80,\"color\":\"rgba(244, 67, 54, 0.8)\"},{\"from\":80,\"to\":100,\"color\":\"#f44336\"}],\"unitTitle\":\"Temperature\",\"units\":\"°C\",\"colorBarProgress\":\"#90caf9\",\"colorBarProgressEnd\":\"#f44336\",\"colorBarStroke\":\"#b0bec5\",\"valueDec\":1},\"title\":\"Temperature gauge - Canvas Gauges\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
203 -'Temperature gauge - Canvas Gauges' );  
204 -  
205 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
206 -VALUES ( now ( ), minTimeuuid ( 0 ), 'maps', 'google_maps',  
207 -'{"type":"latest","sizeX":8.5,"sizeY":6,"resources":[],"templateHtml":"","templateCss":".error {\n color: red;\n}\n.tb-labels {\n color: #222;\n font: 12px/1.5 \"Helvetica Neue\", Arial, Helvetica, sans-serif;\n text-align: center;\n width: 100px;\n white-space: nowrap;\n}","controllerScript":"self.onInit = function() {\n self.ctx.map = new TbMapWidget(''google-map'', false, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"title\": \"Google Map Configuration\",\n \"type\": \"object\",\n \"properties\": {\n \"gmApiKey\": {\n \"title\": \"Google Maps API Key\",\n \"type\": \"string\"\n },\n \"gmDefaultMapType\": {\n \"title\": \"Default map type\",\n \"type\": \"string\",\n \"default\": \"roadmap\"\n },\n \"defaultZoomLevel\": {\n \"title\": \"Default map zoom level (1 - 20)\",\n \"type\": \"number\"\n },\n \"fitMapBounds\": {\n \"title\": \"Fit map bounds to cover all markers\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"markersSettings\": {\n \"title\": \"Markers\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker settings\",\n \"type\": \"object\",\n \"properties\": {\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"lat\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"lng\"\n }, \n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n }, \n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"tooltipPattern\": {\n \"title\": \"Pattern ( for ex. ''Text ${keyName} units.'' or ''${#<key index>} units'' )\",\n \"type\": \"string\",\n \"default\": \"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\"\n },\n \"useColorFunction\": {\n \"title\": \"Use color function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"colorFunction\": {\n \"title\": \"Color function: f(data)\",\n \"type\": \"string\"\n },\n \"markerImage\": {\n \"title\": \"Custom marker image\",\n \"type\": \"string\"\n },\n \"markerImageSize\": {\n \"title\": \"Custom marker image size (px)\",\n \"type\": \"number\",\n \"default\": 34\n },\n \"useMarkerImageFunction\": {\n \"title\": \"Use marker image function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"markerImageFunction\": {\n \"title\": \"Marker image function: f(data, images)\",\n \"type\": \"string\"\n },\n \"markerImages\": {\n \"title\": \"Marker images\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker image\",\n \"type\": \"string\"\n }\n }\n }\n }\n }\n },\n \"required\": [\n \"gmApiKey\"\n ]\n },\n \"form\": [\n \"gmApiKey\",\n {\n \"key\": \"gmDefaultMapType\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"roadmap\",\n \"label\": \"Roadmap\"\n },\n {\n \"value\": \"satellite\",\n \"label\": \"Satellite\"\n },\n {\n \"value\": \"hybrid\",\n \"label\": \"Hybrid\"\n },\n {\n \"value\": \"terrain\",\n \"label\": \"Terrain\"\n }\n ]\n },\n \"defaultZoomLevel\",\n \"fitMapBounds\",\n {\n \"key\": \"markersSettings\",\n \"items\": [\n \"markersSettings[].latKeyName\",\n \"markersSettings[].lngKeyName\",\n \"markersSettings[].showLabel\",\n \"markersSettings[].label\",\n \"markersSettings[].tooltipPattern\",\n {\n \"key\": \"markersSettings[].color\",\n \"type\": \"color\"\n },\n \"markersSettings[].useColorFunction\",\n {\n \"key\": \"markersSettings[].colorFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"markersSettings[].markerImage\",\n \"type\": \"image\"\n },\n \"markersSettings[].markerImageSize\",\n \"markersSettings[].useMarkerImageFunction\",\n {\n \"key\": \"markersSettings[].markerImageFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"markersSettings[].markerImages\",\n \"items\": [\n {\n \"key\": \"markersSettings[].markerImages[]\",\n \"type\": \"image\"\n }\n ]\n }\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"lat\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"lng\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.799863043034289,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"gmApiKey\":\"AIzaSyDoEx2kaGz3PxwbI9T7ccTSg5xjdw8Nw8Q\",\"markersSettings\":[{\"label\":\"First point\",\"color\":\"#1e88e5\",\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"tooltipPattern\":\"<b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"useColorFunction\":true,\"colorFunction\":\"var temperature = data[''temperature''];\\nif (typeof temperature !== undefined) {\\n var percent = (temperature + 60)/120 * 100;\\n return tinycolor.mix(''blue'', ''red'', amount = percent).toHexString();\\n}\\nreturn ''blue'';\",\"markerImages\":[],\"useMarkerImageFunction\":false},{\"label\":\"Second point\",\"color\":\"#fdd835\",\"latKeyName\":\"lat\",\"lngKeyName\":\"lng\",\"showLabel\":true,\"tooltipPattern\":\"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"markerImageSize\":34,\"markerImages\":[\"\",\"\",\"\",\"\"],\"useMarkerImageFunction\":true,\"markerImageFunction\":\"var res = {\\n url: images[0],\\n size: 40\\n}\\nvar temperature = data[''temperature''];\\nif (typeof temperature !== undefined) {\\n var percent = (temperature + 60)/120;\\n var index = Math.floor(4 * percent);\\n res.url = images[index];\\n}\\nreturn res;\",\"useColorFunction\":false}],\"fitMapBounds\":true,\"gmDefaultMapType\":\"roadmap\"},\"title\":\"Google Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false}"}',  
208 -'Google Maps' );  
209 -  
210 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
211 -VALUES ( now ( ), minTimeuuid ( 0 ), 'analogue_gauges',  
212 -'temperature_radial_gauge_canvas_gauges',  
213 -'{"type":"latest","sizeX":6,"sizeY":5,"resources":[],"templateHtml":"<canvas id=\"radialGauge\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbAnalogueRadialGauge(self.ctx, ''radialGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.getSettingsSchema = function() {\n return TbAnalogueRadialGauge.settingsSchema;\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temperature\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"maxValue\":60,\"startAngle\":67.5,\"ticksAngle\":225,\"showBorder\":true,\"defaultColor\":\"#e65100\",\"needleCircleSize\":7,\"highlights\":[{\"from\":-60,\"to\":-50,\"color\":\"#42a5f5\"},{\"from\":-50,\"to\":-40,\"color\":\"rgba(66, 165, 245, 0.83)\"},{\"from\":-40,\"to\":-30,\"color\":\"rgba(66, 165, 245, 0.66)\"},{\"from\":-30,\"to\":-20,\"color\":\"rgba(66, 165, 245, 0.5)\"},{\"from\":-20,\"to\":-10,\"color\":\"rgba(66, 165, 245, 0.33)\"},{\"from\":-10,\"to\":0,\"color\":\"rgba(66, 165, 245, 0.16)\"},{\"from\":0,\"to\":10,\"color\":\"rgba(229, 115, 115, 0.16)\"},{\"from\":10,\"to\":20,\"color\":\"rgba(229, 115, 115, 0.33)\"},{\"from\":20,\"to\":30,\"color\":\"rgba(229, 115, 115, 0.5)\"},{\"from\":30,\"to\":40,\"color\":\"rgba(229, 115, 115, 0.66)\"},{\"from\":40,\"to\":50,\"color\":\"rgba(229, 115, 115, 0.83)\"},{\"from\":50,\"to\":60,\"color\":\"#e57373\"}],\"showUnitTitle\":true,\"colorPlate\":\"#cfd8dc\",\"colorMajorTicks\":\"#444\",\"colorMinorTicks\":\"#666\",\"minorTicks\":2,\"valueInt\":3,\"valueDec\":1,\"highlightsWidth\":15,\"valueBox\":true,\"animation\":true,\"animationDuration\":1000,\"animationRule\":\"bounce\",\"colorNeedleShadowUp\":\"rgba(2, 255, 255, 0)\",\"colorNeedleShadowDown\":\"rgba(188, 143, 143, 0.78)\",\"units\":\"°C\",\"majorTicksCount\":12,\"numbersFont\":{\"family\":\"Roboto\",\"size\":20,\"style\":\"normal\",\"weight\":\"normal\",\"color\":\"#263238\"},\"titleFont\":{\"family\":\"Roboto\",\"size\":24,\"style\":\"normal\",\"weight\":\"normal\",\"color\":\"#263238\"},\"unitsFont\":{\"family\":\"Roboto\",\"size\":28,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#616161\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"size\":30,\"style\":\"normal\",\"weight\":\"normal\",\"shadowColor\":\"rgba(0, 0, 0, 0.49)\",\"color\":\"#444\"},\"colorValueBoxRect\":\"#888\",\"colorValueBoxRectEnd\":\"#666\",\"colorValueBoxBackground\":\"#babab2\",\"colorValueBoxShadow\":\"rgba(0,0,0,1)\",\"unitTitle\":\"Temperature\",\"minValue\":-60},\"title\":\"Temperature radial gauge - Canvas Gauges\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
214 -'Temperature radial gauge - Canvas Gauges' );  
215 -  
216 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
217 -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'attributes_card',  
218 -'{"type":"latest","sizeX":7.5,"sizeY":3,"resources":[],"templateHtml":"","templateCss":"#container {\n overflow: auto;\n}\n\n.tbDatasource-container {\n margin: 5px;\n padding: 8px;\n}\n\n.tbDatasource-title {\n font-size: 1.200rem;\n font-weight: 500;\n padding-bottom: 10px;\n}\n\n.tbDatasource-table {\n width: 100%;\n box-shadow: 0 0 10px #ccc;\n border-collapse: collapse;\n white-space: nowrap;\n font-size: 1.000rem;\n color: #757575;\n}\n\n.tbDatasource-table td {\n position: relative;\n border-top: 1px solid rgba(0, 0, 0, 0.12);\n border-bottom: 1px solid rgba(0, 0, 0, 0.12);\n padding: 0px 18px;\n box-sizing: border-box;\n}","controllerScript":"self.onInit = function() {\n \n self.ctx.datasourceTitleCells = [];\n self.ctx.valueCells = [];\n self.ctx.labelCells = [];\n \n for (var i=0; i < self.ctx.datasources.length; i++) {\n var tbDatasource = self.ctx.datasources[i];\n\n var datasourceId = ''tbDatasource'' + i;\n self.ctx.$container.append(\n \"<div id=''\" + datasourceId +\n \"'' class=''tbDatasource-container''></div>\"\n );\n\n var datasourceContainer = $(''#'' + datasourceId,\n self.ctx.$container);\n\n datasourceContainer.append(\n \"<div class=''tbDatasource-title''>\" +\n tbDatasource.name + \"</div>\"\n );\n \n var datasourceTitleCell = $(''.tbDatasource-title'', datasourceContainer);\n self.ctx.datasourceTitleCells.push(datasourceTitleCell);\n \n var tableId = ''table'' + i;\n datasourceContainer.append(\n \"<table id=''\" + tableId +\n \"'' class=''tbDatasource-table''><col width=''30%''><col width=''70%''></table>\"\n );\n var table = $(''#'' + tableId, self.ctx.$container);\n\n for (var a = 0; a < tbDatasource.dataKeys.length; a++) {\n var dataKey = tbDatasource.dataKeys[a];\n var labelCellId = ''labelCell'' + a;\n var cellId = ''cell'' + a;\n table.append(\"<tr><td id=''\" + labelCellId + \"''>\" + dataKey.label +\n \"</td><td id=''\" + cellId +\n \"''></td></tr>\");\n var labelCell = $(''#'' + labelCellId, table);\n self.ctx.labelCells.push(labelCell);\n var valueCell = $(''#'' + cellId, table);\n self.ctx.valueCells.push(valueCell);\n }\n } \n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n for (var i = 0; i < self.ctx.valueCells.length; i++) {\n var cellData = self.ctx.data[i];\n if (cellData && cellData.data && cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length -\n 1];\n var value = tvPair[1];\n self.ctx.valueCells[i].html(value);\n }\n } \n}\n\nself.onResize = function() {\n var datasoirceTitleFontSize = self.ctx.height/8;\n if (self.ctx.width/self.ctx.height <= 1.5) {\n datasoirceTitleFontSize = self.ctx.width/12;\n }\n datasoirceTitleFontSize = Math.min(datasoirceTitleFontSize, 20);\n for (var i = 0; i < self.ctx.datasourceTitleCells.length; i++) {\n self.ctx.datasourceTitleCells[i].css(''font-size'', datasoirceTitleFontSize+''px'');\n }\n var valueFontSize = self.ctx.height/9;\n var labelFontSize = self.ctx.height/9;\n if (self.ctx.width/self.ctx.height <= 1.5) {\n valueFontSize = self.ctx.width/15;\n labelFontSize = self.ctx.width/15;\n }\n valueFontSize = Math.min(valueFontSize, 18);\n labelFontSize = Math.min(labelFontSize, 18);\n\n for (i = 0; i < self.ctx.valueCells; i++) {\n self.ctx.valueCells[i].css(''font-size'', valueFontSize+''px'');\n self.ctx.valueCells[i].css(''height'', valueFontSize*2.5+''px'');\n self.ctx.valueCells[i].css(''padding'', ''0px '' + valueFontSize + ''px'');\n self.ctx.labelCells[i].css(''font-size'', labelFontSize+''px'');\n self.ctx.labelCells[i].css(''height'', labelFontSize*2.5+''px'');\n self.ctx.labelCells[i].css(''padding'', ''0px '' + labelFontSize + ''px'');\n } \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"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\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Attributes card\"}"}',  
219 -'Attributes card' );  
220 -  
221 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
222 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',  
223 -'digital_thermometer',  
224 -'{"type":"latest","sizeX":3,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < -60) {\\n\\tvalue = 60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":60,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":1,\"levelColors\":[\"#304ffe\",\"#7e57c2\",\"#ff4081\",\"#d32f2f\"],\"refreshAnimationType\":\"<>\",\"refreshAnimationTime\":700,\"startAnimationType\":\"<>\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":18},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"dashThickness\":1.5,\"decimals\":0,\"minValue\":-60,\"units\":\"°C\",\"gaugeColor\":\"#333333\",\"neonGlowBrightness\":35,\"gaugeType\":\"donut\"},\"title\":\"Digital thermometer\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
225 -'Digital thermometer' );  
226 -  
227 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
228 -VALUES ( now ( ), minTimeuuid ( 0 ), 'gpio_widgets',  
229 -'raspberry_pi_gpio_control',  
230 -'{"type":"rpc","sizeX":6,"sizeY":10.5,"resources":[],"templateHtml":"<fieldset class=\"gpio-panel\" ng-disabled=\"!rpcEnabled || executingRpcRequest\" style=\"height: 100%;\">\n <section class=\"gpio-row\" layout=\"row\" ng-repeat=\"row in rows track by $index\" \n ng-style=\"{''height'': prefferedRowHeight+''px''}\">\n <section flex layout=\"row\" ng-repeat=\"cell in row track by $index\">\n <section layout=\"row\" flex ng-if=\"cell\" layout-align=\"{{$index===0 ? ''end center'' : ''start center''}}\">\n <span class=\"gpio-left-label\" ng-show=\"$index===0\">{{ cell.label }}</span>\n <section layout=\"row\" class=\"switch-panel\" layout-align=\"start center\" ng-class=\"$index===0 ? ''col-0'' : ''col-1''\"\n ng-style=\"{''height'': prefferedRowHeight+''px'', ''backgroundColor'': ''{{ switchPanelBackgroundColor }}''}\">\n <span class=\"pin\" ng-show=\"$index===0\">{{cell.pin}}</span>\n <span flex ng-show=\"$index===1\"></span>\n <md-switch\n aria-label=\"{{ cell.label }}\"\n ng-disabled=\"!rpcEnabled || executingRpcRequest\"\n ng-model=\"cell.enabled\" \n ng-change=\"cell.enabled = !cell.enabled\" \n ng-click=\"gpioClick($event, cell)\">\n </md-switch>\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"pin\" ng-show=\"$index===1\">{{cell.pin}}</span>\n </section>\n <span class=\"gpio-right-label\" ng-show=\"$index===1\">{{ cell.label }}</span>\n </section>\n <section layout=\"row\" flex ng-if=\"!cell\">\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"switch-panel\"\n ng-style=\"{''height'': prefferedRowHeight+''px'', ''backgroundColor'': ''{{ switchPanelBackgroundColor }}''}\"></span>\n <span flex ng-show=\"$index===1\"></span>\n </section>\n </section>\n </section> \n <span class=\"error\" style=\"position: absolute; bottom: 5px;\" ng-show=\"rpcErrorText\">{{rpcErrorText}}</span>\n <md-progress-linear ng-show=\"executingRpcRequest\" style=\"position: absolute; bottom: 0;\" md-mode=\"indeterminate\"></md-progress-linear> \n</fieldset>","templateCss":".error {\n font-size: 14px !important;\n color: maroon;/*rgb(250,250,250);*/\n background-color: transparent;\n padding: 6px;\n}\n\n.error span {\n margin: auto;\n}\n\n.gpio-panel {\n padding-top: 10px;\n white-space: nowrap;\n}\n\n.switch-panel {\n margin: 0;\n height: 32px;\n width: 66px;\n min-width: 66px;\n}\n\n.switch-panel md-switch {\n margin: 0;\n width: 36px;\n min-width: 36px;\n}\n\n.switch-panel md-switch > div.md-container {\n margin: 0;\n}\n\n.switch-panel.col-0 md-switch {\n margin-left: 8px;\n margin-right: 4px;\n}\n\n.switch-panel.col-1 md-switch {\n margin-left: 4px;\n margin-right: 8px;\n}\n\n.gpio-row {\n height: 32px;\n}\n\n.pin {\n margin-top: auto;\n margin-bottom: auto;\n color: white;\n font-size: 12px;\n width: 16px;\n min-width: 16px;\n}\n\n.switch-panel.col-0 .pin {\n margin-left: auto;\n padding-left: 2px;\n text-align: right;\n}\n\n.switch-panel.col-1 .pin {\n margin-right: auto;\n \n text-align: left;\n}\n\n.gpio-left-label {\n margin-right: 8px;\n}\n\n.gpio-right-label {\n margin-left: 8px;\n}","controllerScript":"self.onInit = function() {\n \n var i, gpio;\n var scope = self.ctx.$scope;\n var settings = self.ctx.settings;\n scope.gpioList = [];\n for (var g = 0; g < settings.gpioList.length; g++) {\n gpio = settings.gpioList[g];\n scope.gpioList.push(\n {\n row: gpio.row,\n col: gpio.col,\n pin: gpio.pin,\n label: gpio.label,\n enabled: false\n }\n );\n }\n\n scope.requestTimeout = settings.requestTimeout || 1000;\n\n scope.switchPanelBackgroundColor = settings.switchPanelBackgroundColor || tinycolor(''green'').lighten(2).toRgbString();\n\n scope.gpioStatusRequest = {\n method: \"getGpioStatus\",\n paramsBody: \"{}\"\n };\n \n if (settings.gpioStatusRequest) {\n scope.gpioStatusRequest.method = settings.gpioStatusRequest.method || scope.gpioStatusRequest.method;\n scope.gpioStatusRequest.paramsBody = settings.gpioStatusRequest.paramsBody || scope.gpioStatusRequest.paramsBody;\n }\n \n scope.gpioStatusChangeRequest = {\n method: \"setGpioStatus\",\n paramsBody: \"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"\n };\n \n if (settings.gpioStatusChangeRequest) {\n scope.gpioStatusChangeRequest.method = settings.gpioStatusChangeRequest.method || scope.gpioStatusChangeRequest.method;\n scope.gpioStatusChangeRequest.paramsBody = settings.gpioStatusChangeRequest.paramsBody || scope.gpioStatusChangeRequest.paramsBody;\n }\n \n scope.parseGpioStatusFunction = \"return body[pin] === true;\";\n \n if (settings.parseGpioStatusFunction && settings.parseGpioStatusFunction.length > 0) {\n scope.parseGpioStatusFunction = settings.parseGpioStatusFunction;\n }\n \n scope.parseGpioStatusFunction = new Function(\"body, pin\", scope.parseGpioStatusFunction);\n \n function requestGpioStatus() {\n self.ctx.controlApi.sendTwoWayCommand(scope.gpioStatusRequest.method, \n scope.gpioStatusRequest.paramsBody, \n scope.requestTimeout)\n .then(\n function success(responseBody) {\n for (var g = 0; g < scope.gpioList.length; g++) {\n var gpio = scope.gpioList[g];\n var enabled = scope.parseGpioStatusFunction.apply(this, [responseBody, gpio.pin]);\n gpio.enabled = enabled; \n }\n }\n );\n }\n \n function changeGpioStatus(gpio) {\n var pin = gpio.pin + '''';\n var enabled = !gpio.enabled;\n enabled = enabled === true ? ''true'' : ''false'';\n var paramsBody = scope.gpioStatusChangeRequest.paramsBody;\n var requestBody = JSON.parse(paramsBody.replace(\"\\\"{$pin}\\\"\", pin).replace(\"\\\"{$enabled}\\\"\", enabled));\n self.ctx.controlApi.sendTwoWayCommand(scope.gpioStatusChangeRequest.method, \n requestBody, scope.requestTimeout)\n .then(\n function success(responseBody) {\n var enabled = scope.parseGpioStatusFunction.apply(this, [responseBody, gpio.pin]);\n gpio.enabled = enabled;\n }\n );\n }\n \n scope.gpioCells = {};\n var rowCount = 0;\n for (i = 0; i < scope.gpioList.length; i++) {\n gpio = scope.gpioList[i];\n scope.gpioCells[gpio.row+''_''+gpio.col] = gpio;\n rowCount = Math.max(rowCount, gpio.row+1);\n }\n \n scope.prefferedRowHeight = 32;\n scope.rows = [];\n for (i = 0; i < rowCount; i++) {\n var row = [];\n for (var c =0; c<2;c++) {\n if (scope.gpioCells[i+''_''+c]) {\n row[c] = scope.gpioCells[i+''_''+c];\n } else {\n row[c] = null;\n }\n }\n scope.rows.push(row);\n }\n\n scope.gpioClick = function($event, gpio) {\n changeGpioStatus(gpio);\n };\n\n requestGpioStatus(); \n \n self.onResize();\n}\n\nself.onResize = function() {\n var scope = self.ctx.$scope;\n var rowCount = scope.rows.length;\n var prefferedRowHeight = (self.ctx.height - 35)/rowCount;\n prefferedRowHeight = Math.min(32, prefferedRowHeight);\n prefferedRowHeight = Math.max(12, prefferedRowHeight);\n scope.prefferedRowHeight = prefferedRowHeight;\n var ratio = prefferedRowHeight/32;\n var switches = $(''md-switch'', self.ctx.$container);\n switches.css(''height'', 30*ratio+''px'');\n switches.css(''width'', 36*ratio+''px'');\n switches.css(''min-width'', 36*ratio+''px'');\n $(''.md-container'', switches).css(''height'', 24*ratio+''px'');\n $(''.md-container'', switches).css(''width'', 36*ratio+''px'');\n var bars = $(''.md-bar'', self.ctx.$container);\n bars.css(''height'', 14*ratio+''px'');\n bars.css(''width'', 34*ratio+''px'');\n var thumbs = $(''.md-thumb'', self.ctx.$container);\n thumbs.css(''height'', 20*ratio+''px'');\n thumbs.css(''width'', 20*ratio+''px'');\n \n var leftLabels = $(''.gpio-left-label'', self.ctx.$container);\n leftLabels.css(''font-size'', 16*ratio+''px'');\n var rightLabels = $(''.gpio-right-label'', self.ctx.$container);\n rightLabels.css(''font-size'', 16*ratio+''px'');\n var pins = $(''.pin'', self.ctx.$container);\n var pinsFontSize = Math.max(9, 12*ratio);\n pins.css(''font-size'', pinsFontSize+''px''); \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"gpioList\": {\n \"title\": \"Gpio switches\",\n \"type\": \"array\",\n \"minItems\" : 1,\n \"items\": {\n \"title\": \"Gpio switch\",\n \"type\": \"object\",\n \"properties\": {\n \"pin\": {\n \"title\": \"Pin\",\n \"type\": \"number\"\n },\n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"row\": {\n \"title\": \"Row\",\n \"type\": \"number\"\n },\n \"col\": {\n \"title\": \"Column\",\n \"type\": \"number\"\n }\n },\n \"required\": [\"pin\", \"label\", \"row\", \"col\"]\n }\n },\n \"requestTimeout\": {\n \"title\": \"RPC request timeout\",\n \"type\": \"number\",\n \"default\": 500\n },\n \"switchPanelBackgroundColor\": {\n \"title\": \"Switches panel background color\",\n \"type\": \"string\",\n \"default\": \"#008a00\"\n },\n \"gpioStatusRequest\": {\n \"title\": \"GPIO status request\",\n \"type\": \"object\",\n \"properties\": {\n \"method\": {\n \"title\": \"Method name\",\n \"type\": \"string\",\n \"default\": \"getGpioStatus\"\n },\n \"paramsBody\": {\n \"title\": \"Method body\",\n \"type\": \"string\",\n \"default\": \"{}\"\n }\n },\n \"required\": [\"method\", \"paramsBody\"]\n },\n \"gpioStatusChangeRequest\": {\n \"title\": \"GPIO status change request\",\n \"type\": \"object\",\n \"properties\": {\n \"method\": {\n \"title\": \"Method name\",\n \"type\": \"string\",\n \"default\": \"setGpioStatus\"\n },\n \"paramsBody\": {\n \"title\": \"Method body\",\n \"type\": \"string\",\n \"default\": \"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"\n }\n },\n \"required\": [\"method\", \"paramsBody\"]\n },\n \"parseGpioStatusFunction\": {\n \"title\": \"Parse gpio status function\",\n \"type\": \"string\",\n \"default\": \"return body[pin] === true;\"\n } \n },\n \"required\": [\"gpioList\", \n \"requestTimeout\",\n \"switchPanelBackgroundColor\",\n \"gpioStatusRequest\",\n \"gpioStatusChangeRequest\",\n \"parseGpioStatusFunction\"]\n },\n \"form\": [\n \"gpioList\",\n \"requestTimeout\",\n {\n \"key\": \"switchPanelBackgroundColor\",\n \"type\": \"color\"\n },\n {\n \"key\": \"gpioStatusRequest\",\n \"items\": [\n \"gpioStatusRequest.method\",\n {\n \"key\": \"gpioStatusRequest.paramsBody\",\n \"type\": \"json\"\n }\n ]\n },\n {\n \"key\": \"gpioStatusChangeRequest\",\n \"items\": [\n \"gpioStatusChangeRequest.method\",\n {\n \"key\": \"gpioStatusChangeRequest.paramsBody\",\n \"type\": \"json\"\n }\n ]\n },\n {\n \"key\": \"parseGpioStatusFunction\",\n \"type\": \"javascript\"\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"targetDeviceAliases\":[],\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"parseGpioStatusFunction\":\"return body[pin] === true;\",\"gpioStatusChangeRequest\":{\"method\":\"setGpioStatus\",\"paramsBody\":\"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"},\"requestTimeout\":500,\"switchPanelBackgroundColor\":\"#008a00\",\"gpioStatusRequest\":{\"method\":\"getGpioStatus\",\"paramsBody\":\"{}\"},\"gpioList\":[{\"pin\":7,\"label\":\"GPIO 4 (GPCLK0)\",\"row\":3,\"col\":0,\"_uniqueKey\":0},{\"pin\":11,\"label\":\"GPIO 17\",\"row\":5,\"col\":0,\"_uniqueKey\":1},{\"pin\":12,\"label\":\"GPIO 18\",\"row\":5,\"col\":1,\"_uniqueKey\":2},{\"_uniqueKey\":3,\"pin\":13,\"label\":\"GPIO 27\",\"row\":6,\"col\":0},{\"_uniqueKey\":4,\"pin\":15,\"label\":\"GPIO 22\",\"row\":7,\"col\":0},{\"_uniqueKey\":5,\"pin\":16,\"label\":\"GPIO 23\",\"row\":7,\"col\":1},{\"_uniqueKey\":6,\"pin\":18,\"label\":\"GPIO 24\",\"row\":8,\"col\":1},{\"_uniqueKey\":7,\"pin\":22,\"label\":\"GPIO 25\",\"row\":10,\"col\":1},{\"_uniqueKey\":8,\"pin\":29,\"label\":\"GPIO 5\",\"row\":14,\"col\":0},{\"_uniqueKey\":9,\"pin\":31,\"label\":\"GPIO 6\",\"row\":15,\"col\":0},{\"_uniqueKey\":10,\"pin\":32,\"label\":\"GPIO 12\",\"row\":15,\"col\":1},{\"_uniqueKey\":11,\"pin\":33,\"label\":\"GPIO 13\",\"row\":16,\"col\":0},{\"_uniqueKey\":12,\"pin\":35,\"label\":\"GPIO 19\",\"row\":17,\"col\":0},{\"_uniqueKey\":13,\"pin\":36,\"label\":\"GPIO 16\",\"row\":17,\"col\":1},{\"_uniqueKey\":14,\"pin\":37,\"label\":\"GPIO 26\",\"row\":18,\"col\":0},{\"_uniqueKey\":15,\"pin\":38,\"label\":\"GPIO 20\",\"row\":18,\"col\":1},{\"_uniqueKey\":16,\"pin\":40,\"label\":\"GPIO 21\",\"row\":19,\"col\":1}]},\"title\":\"Raspberry Pi GPIO Control\"}"}',  
231 -'Raspberry Pi GPIO Control' );  
232 -  
233 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
234 -VALUES ( now ( ), minTimeuuid ( 0 ), 'gpio_widgets',  
235 -'basic_gpio_control',  
236 -'{"type":"rpc","sizeX":4,"sizeY":2,"resources":[],"templateHtml":"<fieldset class=\"gpio-panel\" ng-disabled=\"!rpcEnabled || executingRpcRequest\" style=\"height: 100%;\">\n <section class=\"gpio-row\" layout=\"row\" ng-repeat=\"row in rows track by $index\" \n ng-style=\"{''height'': prefferedRowHeight+''px''}\">\n <section flex layout=\"row\" ng-repeat=\"cell in row track by $index\">\n <section layout=\"row\" flex ng-if=\"cell\" layout-align=\"{{$index===0 ? ''end center'' : ''start center''}}\">\n <span class=\"gpio-left-label\" ng-show=\"$index===0\">{{ cell.label }}</span>\n <section layout=\"row\" class=\"switch-panel\" layout-align=\"start center\" ng-class=\"$index===0 ? ''col-0'' : ''col-1''\"\n ng-style=\"{''height'': prefferedRowHeight+''px'', ''backgroundColor'': ''{{ switchPanelBackgroundColor }}''}\">\n <span class=\"pin\" ng-show=\"$index===0\">{{cell.pin}}</span>\n <span flex ng-show=\"$index===1\"></span>\n <md-switch\n aria-label=\"{{ cell.label }}\"\n ng-disabled=\"!rpcEnabled || executingRpcRequest\"\n ng-model=\"cell.enabled\" \n ng-change=\"cell.enabled = !cell.enabled\" \n ng-click=\"gpioClick($event, cell)\">\n </md-switch>\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"pin\" ng-show=\"$index===1\">{{cell.pin}}</span>\n </section>\n <span class=\"gpio-right-label\" ng-show=\"$index===1\">{{ cell.label }}</span>\n </section>\n <section layout=\"row\" flex ng-if=\"!cell\">\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"switch-panel\"\n ng-style=\"{''height'': prefferedRowHeight+''px'', ''backgroundColor'': ''{{ switchPanelBackgroundColor }}''}\"></span>\n <span flex ng-show=\"$index===1\"></span>\n </section>\n </section>\n </section> \n <span class=\"error\" style=\"position: absolute; bottom: 5px;\" ng-show=\"rpcErrorText\">{{rpcErrorText}}</span>\n <md-progress-linear ng-show=\"executingRpcRequest\" style=\"position: absolute; bottom: 0;\" md-mode=\"indeterminate\"></md-progress-linear> \n</fieldset>","templateCss":".error {\n font-size: 14px !important;\n color: maroon;/*rgb(250,250,250);*/\n background-color: transparent;\n padding: 6px;\n}\n\n.error span {\n margin: auto;\n}\n\n.gpio-panel {\n padding-top: 10px;\n white-space: nowrap;\n}\n\n.switch-panel {\n margin: 0;\n height: 32px;\n width: 66px;\n min-width: 66px;\n}\n\n.switch-panel md-switch {\n margin: 0;\n width: 36px;\n min-width: 36px;\n}\n\n.switch-panel md-switch > div.md-container {\n margin: 0;\n}\n\n.switch-panel.col-0 md-switch {\n padding-left: 8px;\n padding-right: 4px;\n}\n\n.switch-panel.col-1 md-switch {\n padding-left: 4px;\n padding-right: 8px;\n}\n\n.gpio-row {\n height: 32px;\n}\n\n.pin {\n margin-top: auto;\n margin-bottom: auto;\n color: white;\n font-size: 12px;\n width: 16px;\n min-width: 16px;\n}\n\n.switch-panel.col-0 .pin {\n margin-left: auto;\n padding-left: 2px;\n text-align: right;\n}\n\n.switch-panel.col-1 .pin {\n margin-right: auto;\n \n text-align: left;\n}\n\n.gpio-left-label {\n margin-right: 8px;\n}\n\n.gpio-right-label {\n margin-left: 8px;\n}","controllerScript":"self.onInit = function() {\n \n var i, gpio;\n var scope = self.ctx.$scope;\n var settings = self.ctx.settings;\n scope.gpioList = [];\n for (var g = 0; g < settings.gpioList.length; g++) {\n gpio = settings.gpioList[g];\n scope.gpioList.push(\n {\n row: gpio.row,\n col: gpio.col,\n pin: gpio.pin,\n label: gpio.label,\n enabled: false\n }\n );\n }\n\n scope.requestTimeout = settings.requestTimeout || 1000;\n\n scope.switchPanelBackgroundColor = settings.switchPanelBackgroundColor || tinycolor(''green'').lighten(2).toRgbString();\n\n scope.gpioStatusRequest = {\n method: \"getGpioStatus\",\n paramsBody: \"{}\"\n };\n \n if (settings.gpioStatusRequest) {\n scope.gpioStatusRequest.method = settings.gpioStatusRequest.method || scope.gpioStatusRequest.method;\n scope.gpioStatusRequest.paramsBody = settings.gpioStatusRequest.paramsBody || scope.gpioStatusRequest.paramsBody;\n }\n \n scope.gpioStatusChangeRequest = {\n method: \"setGpioStatus\",\n paramsBody: \"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"\n };\n \n if (settings.gpioStatusChangeRequest) {\n scope.gpioStatusChangeRequest.method = settings.gpioStatusChangeRequest.method || scope.gpioStatusChangeRequest.method;\n scope.gpioStatusChangeRequest.paramsBody = settings.gpioStatusChangeRequest.paramsBody || scope.gpioStatusChangeRequest.paramsBody;\n }\n \n scope.parseGpioStatusFunction = \"return body[pin] === true;\";\n \n if (settings.parseGpioStatusFunction && settings.parseGpioStatusFunction.length > 0) {\n scope.parseGpioStatusFunction = settings.parseGpioStatusFunction;\n }\n \n scope.parseGpioStatusFunction = new Function(\"body, pin\", scope.parseGpioStatusFunction);\n \n function requestGpioStatus() {\n self.ctx.controlApi.sendTwoWayCommand(scope.gpioStatusRequest.method, \n scope.gpioStatusRequest.paramsBody, \n scope.requestTimeout)\n .then(\n function success(responseBody) {\n for (var g = 0; g < scope.gpioList.length; g++) {\n var gpio = scope.gpioList[g];\n var enabled = scope.parseGpioStatusFunction.apply(this, [responseBody, gpio.pin]);\n gpio.enabled = enabled; \n }\n }\n );\n }\n \n function changeGpioStatus(gpio) {\n var pin = gpio.pin + '''';\n var enabled = !gpio.enabled;\n enabled = enabled === true ? ''true'' : ''false'';\n var paramsBody = scope.gpioStatusChangeRequest.paramsBody;\n var requestBody = JSON.parse(paramsBody.replace(\"\\\"{$pin}\\\"\", pin).replace(\"\\\"{$enabled}\\\"\", enabled));\n self.ctx.controlApi.sendTwoWayCommand(scope.gpioStatusChangeRequest.method, \n requestBody, scope.requestTimeout)\n .then(\n function success(responseBody) {\n var enabled = scope.parseGpioStatusFunction.apply(this, [responseBody, gpio.pin]);\n gpio.enabled = enabled;\n }\n );\n }\n \n scope.gpioCells = {};\n var rowCount = 0;\n for (i = 0; i < scope.gpioList.length; i++) {\n gpio = scope.gpioList[i];\n scope.gpioCells[gpio.row+''_''+gpio.col] = gpio;\n rowCount = Math.max(rowCount, gpio.row+1);\n }\n \n scope.prefferedRowHeight = 32;\n scope.rows = [];\n for (i = 0; i < rowCount; i++) {\n var row = [];\n for (var c =0; c<2;c++) {\n if (scope.gpioCells[i+''_''+c]) {\n row[c] = scope.gpioCells[i+''_''+c];\n } else {\n row[c] = null;\n }\n }\n scope.rows.push(row);\n }\n\n scope.gpioClick = function($event, gpio) {\n changeGpioStatus(gpio);\n };\n\n requestGpioStatus(); \n \n self.onResize();\n}\n\nself.onResize = function() {\n var scope = self.ctx.$scope;\n var rowCount = scope.rows.length;\n var prefferedRowHeight = (self.ctx.height - 35)/rowCount;\n prefferedRowHeight = Math.min(32, prefferedRowHeight);\n prefferedRowHeight = Math.max(12, prefferedRowHeight);\n scope.prefferedRowHeight = prefferedRowHeight;\n var ratio = prefferedRowHeight/32;\n var switches = $(''md-switch'', self.ctx.$container);\n switches.css(''height'', 30*ratio+''px'');\n switches.css(''width'', 36*ratio+''px'');\n switches.css(''min-width'', 36*ratio+''px'');\n $(''.md-container'', switches).css(''height'', 24*ratio+''px'');\n $(''.md-container'', switches).css(''width'', 36*ratio+''px'');\n var bars = $(''.md-bar'', self.ctx.$container);\n bars.css(''height'', 14*ratio+''px'');\n bars.css(''width'', 34*ratio+''px'');\n var thumbs = $(''.md-thumb'', self.ctx.$container);\n thumbs.css(''height'', 20*ratio+''px'');\n thumbs.css(''width'', 20*ratio+''px'');\n \n var leftLabels = $(''.gpio-left-label'', self.ctx.$container);\n leftLabels.css(''font-size'', 16*ratio+''px'');\n var rightLabels = $(''.gpio-right-label'', self.ctx.$container);\n rightLabels.css(''font-size'', 16*ratio+''px'');\n var pins = $(''.pin'', self.ctx.$container);\n var pinsFontSize = Math.max(9, 12*ratio);\n pins.css(''font-size'', pinsFontSize+''px''); \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"gpioList\": {\n \"title\": \"Gpio switches\",\n \"type\": \"array\",\n \"minItems\" : 1,\n \"items\": {\n \"title\": \"Gpio switch\",\n \"type\": \"object\",\n \"properties\": {\n \"pin\": {\n \"title\": \"Pin\",\n \"type\": \"number\"\n },\n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"row\": {\n \"title\": \"Row\",\n \"type\": \"number\"\n },\n \"col\": {\n \"title\": \"Column\",\n \"type\": \"number\"\n }\n },\n \"required\": [\"pin\", \"label\", \"row\", \"col\"]\n }\n },\n \"requestTimeout\": {\n \"title\": \"RPC request timeout\",\n \"type\": \"number\",\n \"default\": 500\n },\n \"switchPanelBackgroundColor\": {\n \"title\": \"Switches panel background color\",\n \"type\": \"string\",\n \"default\": \"#008a00\"\n },\n \"gpioStatusRequest\": {\n \"title\": \"GPIO status request\",\n \"type\": \"object\",\n \"properties\": {\n \"method\": {\n \"title\": \"Method name\",\n \"type\": \"string\",\n \"default\": \"getGpioStatus\"\n },\n \"paramsBody\": {\n \"title\": \"Method body\",\n \"type\": \"string\",\n \"default\": \"{}\"\n }\n },\n \"required\": [\"method\", \"paramsBody\"]\n },\n \"gpioStatusChangeRequest\": {\n \"title\": \"GPIO status change request\",\n \"type\": \"object\",\n \"properties\": {\n \"method\": {\n \"title\": \"Method name\",\n \"type\": \"string\",\n \"default\": \"setGpioStatus\"\n },\n \"paramsBody\": {\n \"title\": \"Method body\",\n \"type\": \"string\",\n \"default\": \"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"\n }\n },\n \"required\": [\"method\", \"paramsBody\"]\n },\n \"parseGpioStatusFunction\": {\n \"title\": \"Parse gpio status function\",\n \"type\": \"string\",\n \"default\": \"return body[pin] === true;\"\n } \n },\n \"required\": [\"gpioList\", \n \"requestTimeout\",\n \"switchPanelBackgroundColor\",\n \"gpioStatusRequest\",\n \"gpioStatusChangeRequest\",\n \"parseGpioStatusFunction\"]\n },\n \"form\": [\n \"gpioList\",\n \"requestTimeout\",\n {\n \"key\": \"switchPanelBackgroundColor\",\n \"type\": \"color\"\n },\n {\n \"key\": \"gpioStatusRequest\",\n \"items\": [\n \"gpioStatusRequest.method\",\n {\n \"key\": \"gpioStatusRequest.paramsBody\",\n \"type\": \"json\"\n }\n ]\n },\n {\n \"key\": \"gpioStatusChangeRequest\",\n \"items\": [\n \"gpioStatusChangeRequest.method\",\n {\n \"key\": \"gpioStatusChangeRequest.paramsBody\",\n \"type\": \"json\"\n }\n ]\n },\n {\n \"key\": \"parseGpioStatusFunction\",\n \"type\": \"javascript\"\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"targetDeviceAliases\":[],\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"parseGpioStatusFunction\":\"return body[pin] === true;\",\"gpioStatusChangeRequest\":{\"method\":\"setGpioStatus\",\"paramsBody\":\"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"},\"requestTimeout\":500,\"switchPanelBackgroundColor\":\"#b71c1c\",\"gpioStatusRequest\":{\"method\":\"getGpioStatus\",\"paramsBody\":\"{}\"},\"gpioList\":[{\"pin\":1,\"label\":\"GPIO 1\",\"row\":0,\"col\":0,\"_uniqueKey\":0},{\"pin\":2,\"label\":\"GPIO 2\",\"row\":0,\"col\":1,\"_uniqueKey\":1},{\"pin\":3,\"label\":\"GPIO 3\",\"row\":1,\"col\":0,\"_uniqueKey\":2}]},\"title\":\"Basic GPIO Control\"}"}',  
237 -'Basic GPIO Control' );  
238 -  
239 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
240 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',  
241 -'horizontal_bar_justgage',  
242 -'{"type":"latest","sizeX":7,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>\n","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ffffff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#999999\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"500\",\"size\":18,\"color\":\"#666666\"},\"minMaxFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#666666\"},\"neonGlowBrightness\":0,\"decimals\":0,\"dashThickness\":0,\"gaugeColor\":\"#eeeeee\",\"showTitle\":true,\"gaugeType\":\"horizontalBar\"},\"title\":\"Horizontal bar - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
243 -'Horizontal bar - justGage' );  
244 -  
245 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
246 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',  
247 -'gauge_justgage',  
248 -'{"type":"latest","sizeX":4,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ffffff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#999999\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"500\",\"size\":36,\"color\":\"#666666\"},\"minMaxFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#666666\"},\"neonGlowBrightness\":0,\"decimals\":0,\"dashThickness\":0,\"gaugeColor\":\"#eeeeee\",\"showTitle\":true,\"gaugeType\":\"arc\"},\"title\":\"Gauge - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
249 -'Gauge - justGage' );  
250 -  
251 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
252 -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges', 'lcd_gauge',  
253 -'{"type":"latest","sizeX":5,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 180) {\\n\\tvalue = 180;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#babab2\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":180,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\"linear\",\"refreshAnimationTime\":700,\"startAnimationType\":\"linear\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":0,\"dashThickness\":1.5,\"decimals\":0,\"unitTitle\":\"MPH\",\"showUnitTitle\":true,\"defaultColor\":\"#444444\",\"gaugeType\":\"arc\"},\"title\":\"LCD gauge\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
254 -'LCD gauge' );  
255 -  
256 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
257 -VALUES ( now ( ), minTimeuuid ( 0 ), 'maps', 'openstreetmap',  
258 -'{"type":"latest","sizeX":8.5,"sizeY":6,"resources":[],"templateHtml":"","templateCss":".leaflet-zoom-box {\n\tz-index: 9;\n}\n\n.leaflet-pane { z-index: 4; }\n\n.leaflet-tile-pane { z-index: 2; }\n.leaflet-overlay-pane { z-index: 4; }\n.leaflet-shadow-pane { z-index: 5; }\n.leaflet-marker-pane { z-index: 6; }\n.leaflet-tooltip-pane { z-index: 7; }\n.leaflet-popup-pane { z-index: 8; }\n\n.leaflet-map-pane canvas { z-index: 1; }\n.leaflet-map-pane svg { z-index: 2; }\n\n.leaflet-control {\n\tz-index: 9;\n}\n.leaflet-top,\n.leaflet-bottom {\n\tz-index: 11;\n}\n\n.tb-marker-label {\n border: none;\n background: none;\n box-shadow: none;\n}\n\n.tb-marker-label:before {\n border: none;\n background: none;\n}\n","controllerScript":"self.onInit = function() {\n self.ctx.map = new TbMapWidget(''openstreet-map'', false, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"title\": \"Google Map Configuration\",\n \"type\": \"object\",\n \"properties\": {\n \"defaultZoomLevel\": {\n \"title\": \"Default map zoom level (1 - 20)\",\n \"type\": \"number\"\n },\n \"fitMapBounds\": {\n \"title\": \"Fit map bounds to cover all markers\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"markersSettings\": {\n \"title\": \"Markers\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker settings\",\n \"type\": \"object\",\n \"properties\": {\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"lat\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"lng\"\n }, \n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n }, \n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"tooltipPattern\": {\n \"title\": \"Pattern ( for ex. ''Text ${keyName} units.'' or ''${#<key index>} units'' )\",\n \"type\": \"string\",\n \"default\": \"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\"\n },\n \"useColorFunction\": {\n \"title\": \"Use color function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"colorFunction\": {\n \"title\": \"Color function: f(data)\",\n \"type\": \"string\"\n },\n \"markerImage\": {\n \"title\": \"Custom marker image\",\n \"type\": \"string\"\n },\n \"markerImageSize\": {\n \"title\": \"Custom marker image size (px)\",\n \"type\": \"number\",\n \"default\": 34\n },\n \"useMarkerImageFunction\": {\n \"title\": \"Use marker image function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"markerImageFunction\": {\n \"title\": \"Marker image function: f(data, images)\",\n \"type\": \"string\"\n },\n \"markerImages\": {\n \"title\": \"Marker images\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker image\",\n \"type\": \"string\"\n }\n }\n }\n }\n }\n },\n \"required\": [\n ]\n },\n \"form\": [\n \"defaultZoomLevel\",\n \"fitMapBounds\",\n {\n \"key\": \"markersSettings\",\n \"items\": [\n \"markersSettings[].latKeyName\",\n \"markersSettings[].lngKeyName\",\n \"markersSettings[].showLabel\",\n \"markersSettings[].label\",\n \"markersSettings[].tooltipPattern\",\n {\n \"key\": \"markersSettings[].color\",\n \"type\": \"color\"\n },\n \"markersSettings[].useColorFunction\",\n {\n \"key\": \"markersSettings[].colorFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"markersSettings[].markerImage\",\n \"type\": \"image\"\n },\n \"markersSettings[].markerImageSize\",\n \"markersSettings[].useMarkerImageFunction\",\n {\n \"key\": \"markersSettings[].markerImageFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"markersSettings[].markerImages\",\n \"items\": [\n {\n \"key\": \"markersSettings[].markerImages[]\",\n \"type\": \"image\"\n }\n ]\n }\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"lat\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"lng\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.21553274887887564,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"markersSettings\":[{\"label\":\"First point\",\"color\":\"#1e88e5\",\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"tooltipPattern\":\"<b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"useColorFunction\":true,\"colorFunction\":\"var temperature = data[''temperature''];\\nif (typeof temperature !== undefined) {\\n var percent = (temperature + 60)/120 * 100;\\n return tinycolor.mix(''blue'', ''red'', amount = percent).toHexString();\\n}\\nreturn ''blue'';\"},{\"label\":\"Second point\",\"color\":\"#fdd835\",\"latKeyName\":\"lat\",\"lngKeyName\":\"lng\",\"showLabel\":true,\"tooltipPattern\":\"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"markerImageSize\":34,\"useMarkerImageFunction\":true,\"markerImages\":[\"\",\"\",\"\",\"\"],\"markerImageFunction\":\"var res = {\\n url: images[0],\\n size: 40\\n}\\nvar temperature = data[''temperature''];\\nif (typeof temperature !== undefined) {\\n var percent = (temperature + 60)/120;\\n var index = Math.floor(4 * percent);\\n res.url = images[index];\\n}\\nreturn res;\"}],\"fitMapBounds\":true},\"title\":\"OpenStreetMap\"}"}',  
259 -'OpenStreetMap' );  
260 -  
261 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
262 -VALUES ( now ( ), minTimeuuid ( 0 ), 'analogue_gauges',  
263 -'speed_gauge_canvas_gauges',  
264 -'{"type":"latest","sizeX":7,"sizeY":5,"resources":[],"templateHtml":"<canvas id=\"radialGauge\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbAnalogueRadialGauge(self.ctx, ''radialGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.getSettingsSchema = function() {\n return TbAnalogueRadialGauge.settingsSchema;\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 50 - 25;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 220) {\\n\\tvalue = 220;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"maxValue\":180,\"startAngle\":45,\"ticksAngle\":270,\"showBorder\":false,\"defaultColor\":\"#e65100\",\"needleCircleSize\":7,\"highlights\":[{\"from\":80,\"to\":120,\"color\":\"#fdd835\"},{\"color\":\"#e57373\",\"from\":120,\"to\":180}],\"showUnitTitle\":false,\"colorPlate\":\"#fff\",\"colorMajorTicks\":\"#444\",\"colorMinorTicks\":\"#666\",\"minorTicks\":2,\"valueInt\":3,\"minValue\":0,\"valueDec\":0,\"highlightsWidth\":15,\"valueBox\":true,\"animation\":true,\"animationDuration\":1500,\"animationRule\":\"linear\",\"colorNeedleShadowUp\":\"rgba(2, 255, 255, 0)\",\"colorNeedleShadowDown\":\"rgba(188, 143, 143, 0.78)\",\"units\":\"MPH\",\"majorTicksCount\":9,\"numbersFont\":{\"family\":\"Roboto\",\"size\":22,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#616161\"},\"titleFont\":{\"family\":\"Roboto\",\"size\":24,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#888\"},\"unitsFont\":{\"family\":\"Roboto\",\"size\":28,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#616161\"},\"valueFont\":{\"size\":32,\"style\":\"normal\",\"weight\":\"normal\",\"shadowColor\":\"rgba(0, 0, 0, 0.49)\",\"color\":\"#444\",\"family\":\"Segment7Standard\"},\"colorValueBoxRect\":\"#888\",\"colorValueBoxRectEnd\":\"#666\",\"colorValueBoxBackground\":\"#babab2\",\"colorValueBoxShadow\":\"rgba(0,0,0,1)\"},\"title\":\"Speed gauge - Canvas Gauges\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
265 -'Speed gauge - Canvas Gauges' );  
266 -  
267 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
268 -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'pie',  
269 -'{"type":"latest","sizeX":8,"sizeY":5,"resources":[],"templateHtml":"","templateCss":".legend {\n font-size: 13px;\n line-height: 10px;\n}\n\n.legend table { \n border-spacing: 0px;\n border-collapse: separate;\n}\n\n.pie-label {\n font-size: 12px;\n font-family: ''Roboto'';\n font-weight: bold;\n text-align: center;\n padding: 2px;\n color: white;\n}\n","controllerScript":"self.onInit = function() {\n self.ctx.flot = new TbFlot(self.ctx, ''pie''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.flot.update();\n}\n\nself.onResize = function() {\n self.ctx.flot.resize();\n}\n\nself.onEditModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.getSettingsSchema = function() {\n return TbFlot.pieSettingsSchema;\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbFlot.pieDatakeySettingsSchema;\n}\n\nself.onDestroy = function() {\n self.ctx.flot.destroy();\n}\n","settingsSchema":"{}\n","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.6114638304362894,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.9955906536344441,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.9430835931647599,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"radius\":1,\"fontColor\":\"#545454\",\"fontSize\":10,\"decimals\":1,\"legend\":{\"show\":true,\"position\":\"nw\",\"labelBoxBorderColor\":\"#CCCCCC\",\"backgroundColor\":\"#F0F0F0\",\"backgroundOpacity\":0.85},\"innerRadius\":0,\"showLabels\":true,\"stroke\":{\"width\":5},\"tilt\":1,\"animatedPie\":false},\"title\":\"Pie - Flot\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',  
270 -'Pie - Flot' );  
271 -  
272 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
273 -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'timeseries_bars_flot',  
274 -'{"type":"timeseries","sizeX":8,"sizeY":5,"resources":[],"templateHtml":"","templateCss":".legend {\n font-size: 13px;\n line-height: 10px;\n}\n\n.legend table { \n border-spacing: 0px;\n border-collapse: separate;\n}\n\n.mouse-events .flot-overlay {\n cursor: crosshair; \n}\n\n","controllerScript":"self.onInit = function() {\n self.ctx.flot = new TbFlot(self.ctx, ''bar''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.flot.update();\n}\n\nself.onResize = function() {\n self.ctx.flot.resize();\n}\n\nself.onEditModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.getSettingsSchema = function() {\n return TbFlot.settingsSchema;\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbFlot.datakeySettingsSchema(false);\n}\n\nself.onDestroy = function() {\n self.ctx.flot.destroy();\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{\"showLines\":false,\"fillLines\":false,\"showPoints\":false},\"_hash\":0.8587686344902596,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#ffc107\",\"settings\":{\"showLines\":false,\"fillLines\":false,\"showPoints\":false},\"_hash\":0.12775350966079668,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000},\"aggregation\":{\"limit\":200,\"type\":\"AVG\"}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"shadowSize\":4,\"fontColor\":\"#545454\",\"fontSize\":10,\"xaxis\":{\"showLabels\":true,\"color\":\"#545454\"},\"yaxis\":{\"showLabels\":true,\"color\":\"#545454\"},\"grid\":{\"color\":\"#545454\",\"tickColor\":\"#DDDDDD\",\"verticalLines\":true,\"horizontalLines\":true,\"outlineWidth\":1},\"legend\":{\"show\":true,\"position\":\"nw\",\"backgroundColor\":\"#f0f0f0\",\"backgroundOpacity\":0.85,\"labelBoxBorderColor\":\"rgba(1, 1, 1, 0.45)\"},\"decimals\":1,\"stack\":true,\"tooltipIndividual\":false},\"title\":\"Timeseries Bars - Flot\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"mobileHeight\":null}"}',  
275 -'Timeseries Bars - Flot' );  
276 -  
277 -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )  
278 -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'basic_timeseries',  
279 -'{"type":"timeseries","sizeX":8,"sizeY":5,"resources":[],"templateHtml":"","templateCss":".legend {\n font-size: 13px;\n line-height: 10px;\n}\n\n.legend table { \n border-spacing: 0px;\n border-collapse: separate;\n}\n\n.mouse-events .flot-overlay {\n cursor: crosshair; \n}\n\n","controllerScript":"self.onInit = function() {\n self.ctx.flot = new TbFlot(self.ctx); \n}\n\nself.onDataUpdated = function() {\n self.ctx.flot.update();\n}\n\nself.onResize = function() {\n self.ctx.flot.resize();\n}\n\nself.onEditModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.getSettingsSchema = function() {\n return TbFlot.settingsSchema;\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbFlot.datakeySettingsSchema(true);\n}\n\nself.onDestroy = function() {\n self.ctx.flot.destroy();\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{\"showLines\":true,\"fillLines\":true,\"showPoints\":false},\"_hash\":0.8587686344902596,\"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;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#ffc107\",\"settings\":{\"showLines\":true,\"fillLines\":false,\"showPoints\":false},\"_hash\":0.12775350966079668,\"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\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"shadowSize\":4,\"fontColor\":\"#545454\",\"fontSize\":10,\"xaxis\":{\"showLabels\":true,\"color\":\"#545454\"},\"yaxis\":{\"showLabels\":true,\"color\":\"#545454\"},\"grid\":{\"color\":\"#545454\",\"tickColor\":\"#DDDDDD\",\"verticalLines\":true,\"horizontalLines\":true,\"outlineWidth\":1},\"legend\":{\"show\":true,\"position\":\"nw\",\"backgroundColor\":\"#f0f0f0\",\"backgroundOpacity\":0.85,\"labelBoxBorderColor\":\"rgba(1, 1, 1, 0.45)\"},\"decimals\":1,\"stack\":false,\"tooltipIndividual\":false},\"title\":\"Timeseries - Flot\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"mobileHeight\":null}"}',  
280 -'Timeseries - Flot' );  
281 -  
282 -/** SYSTEM **/ 44 +}' );
@@ -228,12 +228,13 @@ CREATE TABLE IF NOT EXISTS rule_node ( @@ -228,12 +228,13 @@ CREATE TABLE IF NOT EXISTS rule_node (
228 search_text varchar(255) 228 search_text varchar(255)
229 ); 229 );
230 230
231 -CREATE TABLE IF NOT EXISTS entity_views ( 231 +CREATE TABLE IF NOT EXISTS entity_view (
232 id varchar(31) NOT NULL CONSTRAINT entity_view_pkey PRIMARY KEY, 232 id varchar(31) NOT NULL CONSTRAINT entity_view_pkey PRIMARY KEY,
233 entity_id varchar(31), 233 entity_id varchar(31),
234 entity_type varchar(255), 234 entity_type varchar(255),
235 tenant_id varchar(31), 235 tenant_id varchar(31),
236 customer_id varchar(31), 236 customer_id varchar(31),
  237 + type varchar(255),
237 name varchar(255), 238 name varchar(255),
238 keys varchar(255), 239 keys varchar(255),
239 start_ts bigint, 240 start_ts bigint,
@@ -19,4 +19,4 @@ DROP TABLE IF EXISTS widget_type; @@ -19,4 +19,4 @@ DROP TABLE IF EXISTS widget_type;
19 DROP TABLE IF EXISTS widgets_bundle; 19 DROP TABLE IF EXISTS widgets_bundle;
20 DROP TABLE IF EXISTS rule_node; 20 DROP TABLE IF EXISTS rule_node;
21 DROP TABLE IF EXISTS rule_chain; 21 DROP TABLE IF EXISTS rule_chain;
22 -DROP TABLE IF EXISTS entity_views; 22 +DROP TABLE IF EXISTS entity_view;
1 1
2 DOCKER_REPO=local-maven-build 2 DOCKER_REPO=local-maven-build
  3 +
  4 +JS_EXECUTOR_DOCKER_NAME=tb-js-executor
  5 +TB_NODE_DOCKER_NAME=tb-node
  6 +WEB_UI_DOCKER_NAME=tb-web-ui
  7 +
3 TB_VERSION=2.2.0-SNAPSHOT 8 TB_VERSION=2.2.0-SNAPSHOT
4 9
5 KAFKA_TOPICS=js.eval.requests:100:1 10 KAFKA_TOPICS=js.eval.requests:100:1
@@ -15,7 +15,7 @@ @@ -15,7 +15,7 @@
15 # 15 #
16 16
17 17
18 -version: '2' 18 +version: '2.2'
19 19
20 services: 20 services:
21 zookeeper: 21 zookeeper:
@@ -40,7 +40,8 @@ services: @@ -40,7 +40,8 @@ services:
40 - zookeeper 40 - zookeeper
41 tb-js-executor: 41 tb-js-executor:
42 restart: always 42 restart: always
43 - image: "${DOCKER_REPO}/tb-js-executor:${TB_VERSION}" 43 + image: "${DOCKER_REPO}/${JS_EXECUTOR_DOCKER_NAME}:${TB_VERSION}"
  44 + scale: 20
44 environment: 45 environment:
45 TB_KAFKA_SERVERS: kafka:9092 46 TB_KAFKA_SERVERS: kafka:9092
46 env_file: 47 env_file:
@@ -49,11 +50,16 @@ services: @@ -49,11 +50,16 @@ services:
49 - kafka 50 - kafka
50 tb: 51 tb:
51 restart: always 52 restart: always
52 - image: "${DOCKER_REPO}/tb-node:${TB_VERSION}" 53 + image: "${DOCKER_REPO}/${TB_NODE_DOCKER_NAME}:${TB_VERSION}"
53 ports: 54 ports:
54 - "8080" 55 - "8080"
55 - "1883:1883" 56 - "1883:1883"
56 - "5683:5683/udp" 57 - "5683:5683/udp"
  58 + logging:
  59 + driver: "json-file"
  60 + options:
  61 + max-size: "200m"
  62 + max-file: "30"
57 env_file: 63 env_file:
58 - tb-node.env 64 - tb-node.env
59 environment: 65 environment:
@@ -68,7 +74,7 @@ services: @@ -68,7 +74,7 @@ services:
68 - kafka 74 - kafka
69 tb-web-ui1: 75 tb-web-ui1:
70 restart: always 76 restart: always
71 - image: "${DOCKER_REPO}/tb-web-ui:${TB_VERSION}" 77 + image: "${DOCKER_REPO}/${WEB_UI_DOCKER_NAME}:${TB_VERSION}"
72 ports: 78 ports:
73 - "8080" 79 - "8080"
74 environment: 80 environment:
@@ -78,7 +84,7 @@ services: @@ -78,7 +84,7 @@ services:
78 - tb-web-ui.env 84 - tb-web-ui.env
79 tb-web-ui2: 85 tb-web-ui2:
80 restart: always 86 restart: always
81 - image: "${DOCKER_REPO}/tb-web-ui:${TB_VERSION}" 87 + image: "${DOCKER_REPO}/${WEB_UI_DOCKER_NAME}:${TB_VERSION}"
82 ports: 88 ports:
83 - "8080" 89 - "8080"
84 environment: 90 environment:
@@ -17,4 +17,4 @@ @@ -17,4 +17,4 @@
17 17
18 ./check-dirs.sh 18 ./check-dirs.sh
19 19
20 -docker-compose up -d --scale tb-js-executor=20 20 +docker-compose up -d
@@ -15,4 +15,5 @@ @@ -15,4 +15,5 @@
15 # limitations under the License. 15 # limitations under the License.
16 # 16 #
17 17
  18 +docker-compose pull $@
18 docker-compose up -d --no-deps --build $@ 19 docker-compose up -d --no-deps --build $@
@@ -35,8 +35,9 @@ frontend http-in @@ -35,8 +35,9 @@ frontend http-in
35 35
36 reqadd X-Forwarded-Proto:\ http 36 reqadd X-Forwarded-Proto:\ http
37 37
  38 + acl transport_http_acl path_beg /api/v1/
38 acl letsencrypt_http_acl path_beg /.well-known/acme-challenge/ 39 acl letsencrypt_http_acl path_beg /.well-known/acme-challenge/
39 - redirect scheme https if !letsencrypt_http_acl 40 + redirect scheme https if !letsencrypt_http_acl !transport_http_acl
40 use_backend letsencrypt_http if letsencrypt_http_acl 41 use_backend letsencrypt_http if letsencrypt_http_acl
41 42
42 default_backend tb-web-backend 43 default_backend tb-web-backend
@@ -19,6 +19,6 @@ export JAVA_OPTS="$JAVA_OPTS -Xloggc:/var/log/thingsboard/gc.log -XX:+IgnoreUnre @@ -19,6 +19,6 @@ export JAVA_OPTS="$JAVA_OPTS -Xloggc:/var/log/thingsboard/gc.log -XX:+IgnoreUnre
19 export JAVA_OPTS="$JAVA_OPTS -XX:+PrintHeapAtGC -XX:+PrintTenuringDistribution -XX:+PrintGCApplicationStoppedTime -XX:+UseGCLogFileRotation -XX:NumberOfGCLogFiles=10" 19 export JAVA_OPTS="$JAVA_OPTS -XX:+PrintHeapAtGC -XX:+PrintTenuringDistribution -XX:+PrintGCApplicationStoppedTime -XX:+UseGCLogFileRotation -XX:NumberOfGCLogFiles=10"
20 export JAVA_OPTS="$JAVA_OPTS -XX:GCLogFileSize=10M -XX:-UseBiasedLocking -XX:+UseTLAB -XX:+ResizeTLAB -XX:+PerfDisableSharedMem -XX:+UseCondCardMark" 20 export JAVA_OPTS="$JAVA_OPTS -XX:GCLogFileSize=10M -XX:-UseBiasedLocking -XX:+UseTLAB -XX:+ResizeTLAB -XX:+PerfDisableSharedMem -XX:+UseCondCardMark"
21 export JAVA_OPTS="$JAVA_OPTS -XX:CMSWaitDuration=10000 -XX:+UseParNewGC -XX:+UseConcMarkSweepGC -XX:+CMSParallelRemarkEnabled -XX:+CMSParallelInitialMarkEnabled" 21 export JAVA_OPTS="$JAVA_OPTS -XX:CMSWaitDuration=10000 -XX:+UseParNewGC -XX:+UseConcMarkSweepGC -XX:+CMSParallelRemarkEnabled -XX:+CMSParallelInitialMarkEnabled"
22 -export JAVA_OPTS="$JAVA_OPTS -XX:+CMSEdenChunksRecordAlways -XX:CMSInitiatingOccupancyFraction=75 -XX:+UseCMSInitiatingOccupancyOnly" 22 +export JAVA_OPTS="$JAVA_OPTS -XX:+CMSEdenChunksRecordAlways -XX:CMSInitiatingOccupancyFraction=75 -XX:+UseCMSInitiatingOccupancyOnly -XX:+ExitOnOutOfMemoryError"
23 export LOG_FILENAME=thingsboard.out 23 export LOG_FILENAME=thingsboard.out
24 export LOADER_PATH=/usr/share/thingsboard/conf,/usr/share/thingsboard/extensions 24 export LOADER_PATH=/usr/share/thingsboard/conf,/usr/share/thingsboard/extensions
@@ -23,4 +23,6 @@ RUN chmod a+x /tmp/*.sh \ @@ -23,4 +23,6 @@ RUN chmod a+x /tmp/*.sh \
23 23
24 RUN dpkg -i /tmp/${pkg.name}.deb 24 RUN dpkg -i /tmp/${pkg.name}.deb
25 25
  26 +RUN update-rc.d ${pkg.name} disable
  27 +
26 CMD ["start-js-executor.sh"] 28 CMD ["start-js-executor.sh"]
@@ -35,6 +35,7 @@ @@ -35,6 +35,7 @@
35 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> 35 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
36 <main.dir>${basedir}/../..</main.dir> 36 <main.dir>${basedir}/../..</main.dir>
37 <pkg.name>tb-js-executor</pkg.name> 37 <pkg.name>tb-js-executor</pkg.name>
  38 + <docker.name>tb-js-executor</docker.name>
38 <pkg.user>thingsboard</pkg.user> 39 <pkg.user>thingsboard</pkg.user>
39 <pkg.unixLogFolder>/var/log/${pkg.name}</pkg.unixLogFolder> 40 <pkg.unixLogFolder>/var/log/${pkg.name}</pkg.unixLogFolder>
40 <pkg.installFolder>/usr/share/${pkg.name}</pkg.installFolder> 41 <pkg.installFolder>/usr/share/${pkg.name}</pkg.installFolder>
@@ -290,7 +291,7 @@ @@ -290,7 +291,7 @@
290 </executions> 291 </executions>
291 <configuration> 292 <configuration>
292 <skip>${dockerfile.skip}</skip> 293 <skip>${dockerfile.skip}</skip>
293 - <repository>${docker.repo}/${pkg.name}</repository> 294 + <repository>${docker.repo}/${docker.name}</repository>
294 <tag>${project.version}</tag> 295 <tag>${project.version}</tag>
295 <verbose>true</verbose> 296 <verbose>true</verbose>
296 <googleContainerRegistryEnabled>false</googleContainerRegistryEnabled> 297 <googleContainerRegistryEnabled>false</googleContainerRegistryEnabled>
@@ -2,5 +2,5 @@ @@ -2,5 +2,5 @@
2 2
3 chown -R ${pkg.user}: ${pkg.logFolder} 3 chown -R ${pkg.user}: ${pkg.logFolder}
4 chown -R ${pkg.user}: ${pkg.installFolder} 4 chown -R ${pkg.user}: ${pkg.installFolder}
5 -# update-rc.d ${pkg.name} defaults 5 +update-rc.d ${pkg.name} defaults
6 6
@@ -35,6 +35,7 @@ @@ -35,6 +35,7 @@
35 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> 35 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
36 <main.dir>${basedir}/../..</main.dir> 36 <main.dir>${basedir}/../..</main.dir>
37 <pkg.name>thingsboard</pkg.name> 37 <pkg.name>thingsboard</pkg.name>
  38 + <docker.name>tb-node</docker.name>
38 <pkg.user>thingsboard</pkg.user> 39 <pkg.user>thingsboard</pkg.user>
39 <pkg.unixLogFolder>/var/log/${pkg.name}</pkg.unixLogFolder> 40 <pkg.unixLogFolder>/var/log/${pkg.name}</pkg.unixLogFolder>
40 <pkg.installFolder>/usr/share/${pkg.name}</pkg.installFolder> 41 <pkg.installFolder>/usr/share/${pkg.name}</pkg.installFolder>
@@ -114,7 +115,7 @@ @@ -114,7 +115,7 @@
114 </executions> 115 </executions>
115 <configuration> 116 <configuration>
116 <skip>${dockerfile.skip}</skip> 117 <skip>${dockerfile.skip}</skip>
117 - <repository>${docker.repo}/tb-node</repository> 118 + <repository>${docker.repo}/${docker.name}</repository>
118 <tag>${project.version}</tag> 119 <tag>${project.version}</tag>
119 <verbose>true</verbose> 120 <verbose>true</verbose>
120 <googleContainerRegistryEnabled>false</googleContainerRegistryEnabled> 121 <googleContainerRegistryEnabled>false</googleContainerRegistryEnabled>
@@ -23,4 +23,6 @@ RUN chmod a+x /tmp/*.sh \ @@ -23,4 +23,6 @@ RUN chmod a+x /tmp/*.sh \
23 23
24 RUN dpkg -i /tmp/${pkg.name}.deb 24 RUN dpkg -i /tmp/${pkg.name}.deb
25 25
  26 +RUN update-rc.d ${pkg.name} disable
  27 +
26 CMD ["start-web-ui.sh"] 28 CMD ["start-web-ui.sh"]
@@ -35,6 +35,7 @@ @@ -35,6 +35,7 @@
35 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> 35 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
36 <main.dir>${basedir}/../..</main.dir> 36 <main.dir>${basedir}/../..</main.dir>
37 <pkg.name>tb-web-ui</pkg.name> 37 <pkg.name>tb-web-ui</pkg.name>
  38 + <docker.name>tb-web-ui</docker.name>
38 <pkg.user>thingsboard</pkg.user> 39 <pkg.user>thingsboard</pkg.user>
39 <pkg.unixLogFolder>/var/log/${pkg.name}</pkg.unixLogFolder> 40 <pkg.unixLogFolder>/var/log/${pkg.name}</pkg.unixLogFolder>
40 <pkg.installFolder>/usr/share/${pkg.name}</pkg.installFolder> 41 <pkg.installFolder>/usr/share/${pkg.name}</pkg.installFolder>
@@ -314,7 +315,7 @@ @@ -314,7 +315,7 @@
314 </executions> 315 </executions>
315 <configuration> 316 <configuration>
316 <skip>${dockerfile.skip}</skip> 317 <skip>${dockerfile.skip}</skip>
317 - <repository>${docker.repo}/${pkg.name}</repository> 318 + <repository>${docker.repo}/${docker.name}</repository>
318 <tag>${project.version}</tag> 319 <tag>${project.version}</tag>
319 <verbose>true</verbose> 320 <verbose>true</verbose>
320 <googleContainerRegistryEnabled>false</googleContainerRegistryEnabled> 321 <googleContainerRegistryEnabled>false</googleContainerRegistryEnabled>
@@ -2,5 +2,5 @@ @@ -2,5 +2,5 @@
2 2
3 chown -R ${pkg.user}: ${pkg.logFolder} 3 chown -R ${pkg.user}: ${pkg.logFolder}
4 chown -R ${pkg.user}: ${pkg.installFolder} 4 chown -R ${pkg.user}: ${pkg.installFolder}
5 -# update-rc.d ${pkg.name} defaults 5 +update-rc.d ${pkg.name} defaults
6 6
@@ -48,7 +48,7 @@ import static org.thingsboard.rule.engine.api.TbRelationTypes.SUCCESS; @@ -48,7 +48,7 @@ import static org.thingsboard.rule.engine.api.TbRelationTypes.SUCCESS;
48 @Slf4j 48 @Slf4j
49 @RuleNode( 49 @RuleNode(
50 type = ComponentType.ACTION, 50 type = ComponentType.ACTION,
51 - name = "copy attributes", 51 + name = "copy to view",
52 configClazz = EmptyNodeConfiguration.class, 52 configClazz = EmptyNodeConfiguration.class,
53 nodeDescription = "Copy attributes from asset/device to entity view and changes message originator to related entity view", 53 nodeDescription = "Copy attributes from asset/device to entity view and changes message originator to related entity view",
54 nodeDetails = "Copy attributes from asset/device to related entity view according to entity view configuration. \n " + 54 nodeDetails = "Copy attributes from asset/device to related entity view according to entity view configuration. \n " +
@@ -37,7 +37,7 @@ import java.net.InetSocketAddress; @@ -37,7 +37,7 @@ import java.net.InetSocketAddress;
37 import java.net.UnknownHostException; 37 import java.net.UnknownHostException;
38 38
39 @Service("CoapTransportService") 39 @Service("CoapTransportService")
40 -@ConditionalOnProperty(prefix = "coap", value = "enabled", havingValue = "true", matchIfMissing = true) 40 +@ConditionalOnProperty(prefix = "transport.coap", value = "enabled", havingValue = "true")
41 @Slf4j 41 @Slf4j
42 public class CoapTransportService { 42 public class CoapTransportService {
43 43
@@ -18,6 +18,7 @@ package org.thingsboard.server.transport.http; @@ -18,6 +18,7 @@ package org.thingsboard.server.transport.http;
18 import lombok.extern.slf4j.Slf4j; 18 import lombok.extern.slf4j.Slf4j;
19 import org.springframework.beans.factory.annotation.Autowired; 19 import org.springframework.beans.factory.annotation.Autowired;
20 import org.springframework.beans.factory.annotation.Value; 20 import org.springframework.beans.factory.annotation.Value;
  21 +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
21 import org.springframework.http.HttpStatus; 22 import org.springframework.http.HttpStatus;
22 import org.springframework.http.ResponseEntity; 23 import org.springframework.http.ResponseEntity;
23 import org.springframework.web.bind.annotation.*; 24 import org.springframework.web.bind.annotation.*;
@@ -34,6 +35,7 @@ import javax.servlet.http.HttpServletRequest; @@ -34,6 +35,7 @@ import javax.servlet.http.HttpServletRequest;
34 * @author Andrew Shvayka 35 * @author Andrew Shvayka
35 */ 36 */
36 @RestController 37 @RestController
  38 +@ConditionalOnProperty(prefix = "transport.http", value = "enabled", havingValue = "true")
37 @RequestMapping("/api/v1") 39 @RequestMapping("/api/v1")
38 @Slf4j 40 @Slf4j
39 public class DeviceApiController { 41 public class DeviceApiController {
@@ -23,6 +23,7 @@ import lombok.extern.slf4j.Slf4j; @@ -23,6 +23,7 @@ import lombok.extern.slf4j.Slf4j;
23 import org.apache.commons.lang3.RandomStringUtils; 23 import org.apache.commons.lang3.RandomStringUtils;
24 import org.springframework.beans.factory.annotation.Autowired; 24 import org.springframework.beans.factory.annotation.Autowired;
25 import org.springframework.beans.factory.annotation.Value; 25 import org.springframework.beans.factory.annotation.Value;
  26 +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
26 import org.springframework.context.annotation.Lazy; 27 import org.springframework.context.annotation.Lazy;
27 import org.springframework.stereotype.Component; 28 import org.springframework.stereotype.Component;
28 import org.springframework.util.StringUtils; 29 import org.springframework.util.StringUtils;
@@ -41,6 +42,7 @@ import java.util.concurrent.Executors; @@ -41,6 +42,7 @@ import java.util.concurrent.Executors;
41 * Created by ashvayka on 04.10.18. 42 * Created by ashvayka on 04.10.18.
42 */ 43 */
43 @Slf4j 44 @Slf4j
  45 +@ConditionalOnProperty(prefix = "transport.mqtt", value = "enabled", havingValue = "true", matchIfMissing = true)
44 @Component 46 @Component
45 @Data 47 @Data
46 public class MqttTransportContext { 48 public class MqttTransportContext {
@@ -59,7 +61,7 @@ public class MqttTransportContext { @@ -59,7 +61,7 @@ public class MqttTransportContext {
59 @Autowired 61 @Autowired
60 private MqttTransportAdaptor adaptor; 62 private MqttTransportAdaptor adaptor;
61 63
62 - @Value("${mqtt.netty.max_payload_size}") 64 + @Value("${transport.mqtt.netty.max_payload_size}")
63 private Integer maxPayloadSize; 65 private Integer maxPayloadSize;
64 66
65 @Value("${cluster.node_id:#{null}}") 67 @Value("${cluster.node_id:#{null}}")
@@ -55,6 +55,7 @@ import org.thingsboard.server.gen.transport.TransportProtos.ValidateDeviceX509Ce @@ -55,6 +55,7 @@ import org.thingsboard.server.gen.transport.TransportProtos.ValidateDeviceX509Ce
55 import org.thingsboard.server.transport.mqtt.adaptors.MqttTransportAdaptor; 55 import org.thingsboard.server.transport.mqtt.adaptors.MqttTransportAdaptor;
56 import org.thingsboard.server.transport.mqtt.session.DeviceSessionCtx; 56 import org.thingsboard.server.transport.mqtt.session.DeviceSessionCtx;
57 import org.thingsboard.server.transport.mqtt.session.GatewaySessionHandler; 57 import org.thingsboard.server.transport.mqtt.session.GatewaySessionHandler;
  58 +import org.thingsboard.server.transport.mqtt.session.MqttTopicMatcher;
58 import org.thingsboard.server.transport.mqtt.util.SslUtil; 59 import org.thingsboard.server.transport.mqtt.util.SslUtil;
59 60
60 import javax.net.ssl.SSLPeerUnverifiedException; 61 import javax.net.ssl.SSLPeerUnverifiedException;
@@ -93,7 +94,7 @@ public class MqttTransportHandler extends ChannelInboundHandlerAdapter implement @@ -93,7 +94,7 @@ public class MqttTransportHandler extends ChannelInboundHandlerAdapter implement
93 private final TransportService transportService; 94 private final TransportService transportService;
94 private final QuotaService quotaService; 95 private final QuotaService quotaService;
95 private final SslHandler sslHandler; 96 private final SslHandler sslHandler;
96 - private final ConcurrentMap<String, Integer> mqttQoSMap; 97 + private final ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap;
97 98
98 private volatile SessionInfoProto sessionInfo; 99 private volatile SessionInfoProto sessionInfo;
99 private volatile InetSocketAddress address; 100 private volatile InetSocketAddress address;
@@ -295,7 +296,7 @@ public class MqttTransportHandler extends ChannelInboundHandlerAdapter implement @@ -295,7 +296,7 @@ public class MqttTransportHandler extends ChannelInboundHandlerAdapter implement
295 296
296 private void registerSubQoS(String topic, List<Integer> grantedQoSList, MqttQoS reqQoS) { 297 private void registerSubQoS(String topic, List<Integer> grantedQoSList, MqttQoS reqQoS) {
297 grantedQoSList.add(getMinSupportedQos(reqQoS)); 298 grantedQoSList.add(getMinSupportedQos(reqQoS));
298 - mqttQoSMap.put(topic, getMinSupportedQos(reqQoS)); 299 + mqttQoSMap.put(new MqttTopicMatcher(topic), getMinSupportedQos(reqQoS));
299 } 300 }
300 301
301 private void processUnsubscribe(ChannelHandlerContext ctx, MqttUnsubscribeMessage mqttMsg) { 302 private void processUnsubscribe(ChannelHandlerContext ctx, MqttUnsubscribeMessage mqttMsg) {
@@ -304,7 +305,7 @@ public class MqttTransportHandler extends ChannelInboundHandlerAdapter implement @@ -304,7 +305,7 @@ public class MqttTransportHandler extends ChannelInboundHandlerAdapter implement
304 } 305 }
305 log.trace("[{}] Processing subscription [{}]!", sessionId, mqttMsg.variableHeader().messageId()); 306 log.trace("[{}] Processing subscription [{}]!", sessionId, mqttMsg.variableHeader().messageId());
306 for (String topicName : mqttMsg.payload().topics()) { 307 for (String topicName : mqttMsg.payload().topics()) {
307 - mqttQoSMap.remove(topicName); 308 + mqttQoSMap.remove(new MqttTopicMatcher(topicName));
308 try { 309 try {
309 switch (topicName) { 310 switch (topicName) {
310 case MqttTopics.DEVICE_ATTRIBUTES_TOPIC: { 311 case MqttTopics.DEVICE_ATTRIBUTES_TOPIC: {
@@ -44,25 +44,22 @@ import javax.annotation.PreDestroy; @@ -44,25 +44,22 @@ import javax.annotation.PreDestroy;
44 * @author Andrew Shvayka 44 * @author Andrew Shvayka
45 */ 45 */
46 @Service("MqttTransportService") 46 @Service("MqttTransportService")
47 -@ConditionalOnProperty(prefix = "mqtt", value = "enabled", havingValue = "true", matchIfMissing = false) 47 +@ConditionalOnProperty(prefix = "transport.mqtt", value = "enabled", havingValue = "true", matchIfMissing = true)
48 @Slf4j 48 @Slf4j
49 public class MqttTransportService { 49 public class MqttTransportService {
50 50
51 - private static final String V1 = "v1";  
52 - private static final String DEVICE = "device";  
53 -  
54 - @Value("${mqtt.bind_address}") 51 + @Value("${transport.mqtt.bind_address}")
55 private String host; 52 private String host;
56 - @Value("${mqtt.bind_port}") 53 + @Value("${transport.mqtt.bind_port}")
57 private Integer port; 54 private Integer port;
58 - @Value("${mqtt.adaptor}") 55 + @Value("${transport.mqtt.adaptor}")
59 private String adaptorName; 56 private String adaptorName;
60 57
61 - @Value("${mqtt.netty.leak_detector_level}") 58 + @Value("${transport.mqtt.netty.leak_detector_level}")
62 private String leakDetectorLevel; 59 private String leakDetectorLevel;
63 - @Value("${mqtt.netty.boss_group_thread_count}") 60 + @Value("${transport.mqtt.netty.boss_group_thread_count}")
64 private Integer bossGroupThreadCount; 61 private Integer bossGroupThreadCount;
65 - @Value("${mqtt.netty.worker_group_thread_count}") 62 + @Value("${transport.mqtt.netty.worker_group_thread_count}")
66 private Integer workerGroupThreadCount; 63 private Integer workerGroupThreadCount;
67 64
68 @Autowired 65 @Autowired
@@ -36,19 +36,10 @@ public class DeviceSessionCtx extends MqttDeviceAwareSessionContext { @@ -36,19 +36,10 @@ public class DeviceSessionCtx extends MqttDeviceAwareSessionContext {
36 private ChannelHandlerContext channel; 36 private ChannelHandlerContext channel;
37 private AtomicInteger msgIdSeq = new AtomicInteger(0); 37 private AtomicInteger msgIdSeq = new AtomicInteger(0);
38 38
39 - public DeviceSessionCtx(UUID sessionId, ConcurrentMap<String, Integer> mqttQoSMap) { 39 + public DeviceSessionCtx(UUID sessionId, ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap) {
40 super(sessionId, mqttQoSMap); 40 super(sessionId, mqttQoSMap);
41 } 41 }
42 42
43 - private void logAndWrap(AdaptorException e) throws SessionException {  
44 - log.warn("Failed to convert msg: {}", e.getMessage(), e);  
45 - throw new SessionException(e);  
46 - }  
47 -  
48 - private void pushToNetwork(MqttMessage msg) {  
49 - channel.writeAndFlush(msg);  
50 - }  
51 -  
52 public void setChannel(ChannelHandlerContext channel) { 43 public void setChannel(ChannelHandlerContext channel) {
53 this.channel = channel; 44 this.channel = channel;
54 } 45 }
@@ -33,7 +33,7 @@ public class GatewayDeviceSessionCtx extends MqttDeviceAwareSessionContext imple @@ -33,7 +33,7 @@ public class GatewayDeviceSessionCtx extends MqttDeviceAwareSessionContext imple
33 private final GatewaySessionHandler parent; 33 private final GatewaySessionHandler parent;
34 private final SessionInfoProto sessionInfo; 34 private final SessionInfoProto sessionInfo;
35 35
36 - public GatewayDeviceSessionCtx(GatewaySessionHandler parent, DeviceInfoProto deviceInfo, ConcurrentMap<String, Integer> mqttQoSMap) { 36 + public GatewayDeviceSessionCtx(GatewaySessionHandler parent, DeviceInfoProto deviceInfo, ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap) {
37 super(UUID.randomUUID(), mqttQoSMap); 37 super(UUID.randomUUID(), mqttQoSMap);
38 this.parent = parent; 38 this.parent = parent;
39 this.sessionInfo = SessionInfoProto.newBuilder() 39 this.sessionInfo = SessionInfoProto.newBuilder()
@@ -68,7 +68,7 @@ public class GatewaySessionHandler { @@ -68,7 +68,7 @@ public class GatewaySessionHandler {
68 private final DeviceInfoProto gateway; 68 private final DeviceInfoProto gateway;
69 private final UUID sessionId; 69 private final UUID sessionId;
70 private final Map<String, GatewayDeviceSessionCtx> devices; 70 private final Map<String, GatewayDeviceSessionCtx> devices;
71 - private final ConcurrentMap<String, Integer> mqttQoSMap; 71 + private final ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap;
72 private final ChannelHandlerContext channel; 72 private final ChannelHandlerContext channel;
73 private final DeviceSessionCtx deviceSessionCtx; 73 private final DeviceSessionCtx deviceSessionCtx;
74 74
@@ -16,35 +16,38 @@ @@ -16,35 +16,38 @@
16 package org.thingsboard.server.transport.mqtt.session; 16 package org.thingsboard.server.transport.mqtt.session;
17 17
18 import io.netty.handler.codec.mqtt.MqttQoS; 18 import io.netty.handler.codec.mqtt.MqttQoS;
19 -import org.thingsboard.server.common.data.Device;  
20 -import org.thingsboard.server.common.transport.SessionMsgProcessor;  
21 -import org.thingsboard.server.common.transport.auth.DeviceAuthService;  
22 import org.thingsboard.server.common.transport.session.DeviceAwareSessionContext; 19 import org.thingsboard.server.common.transport.session.DeviceAwareSessionContext;
23 20
  21 +import java.util.List;
24 import java.util.Map; 22 import java.util.Map;
25 import java.util.UUID; 23 import java.util.UUID;
26 import java.util.concurrent.ConcurrentMap; 24 import java.util.concurrent.ConcurrentMap;
  25 +import java.util.stream.Collectors;
27 26
28 /** 27 /**
29 * Created by ashvayka on 30.08.18. 28 * Created by ashvayka on 30.08.18.
30 */ 29 */
31 public abstract class MqttDeviceAwareSessionContext extends DeviceAwareSessionContext { 30 public abstract class MqttDeviceAwareSessionContext extends DeviceAwareSessionContext {
32 31
33 - private final ConcurrentMap<String, Integer> mqttQoSMap; 32 + private final ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap;
34 33
35 - public MqttDeviceAwareSessionContext(UUID sessionId, ConcurrentMap<String, Integer> mqttQoSMap) { 34 + public MqttDeviceAwareSessionContext(UUID sessionId, ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap) {
36 super(sessionId); 35 super(sessionId);
37 this.mqttQoSMap = mqttQoSMap; 36 this.mqttQoSMap = mqttQoSMap;
38 } 37 }
39 38
40 - public ConcurrentMap<String, Integer> getMqttQoSMap() { 39 + public ConcurrentMap<MqttTopicMatcher, Integer> getMqttQoSMap() {
41 return mqttQoSMap; 40 return mqttQoSMap;
42 } 41 }
43 42
44 public MqttQoS getQoSForTopic(String topic) { 43 public MqttQoS getQoSForTopic(String topic) {
45 - Integer qos = mqttQoSMap.get(topic);  
46 - if (qos != null) {  
47 - return MqttQoS.valueOf(qos); 44 + List<Integer> qosList = mqttQoSMap.entrySet()
  45 + .stream()
  46 + .filter(entry -> entry.getKey().matches(topic))
  47 + .map(Map.Entry::getValue)
  48 + .collect(Collectors.toList());
  49 + if (!qosList.isEmpty()) {
  50 + return MqttQoS.valueOf(qosList.get(0));
48 } else { 51 } else {
49 return MqttQoS.AT_LEAST_ONCE; 52 return MqttQoS.AT_LEAST_ONCE;
50 } 53 }
  1 +/**
  2 + * Copyright © 2016-2018 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.mqtt.session;
  17 +
  18 +import java.util.regex.Pattern;
  19 +
  20 +public class MqttTopicMatcher {
  21 +
  22 + private final String topic;
  23 + private final Pattern topicRegex;
  24 +
  25 + public MqttTopicMatcher(String topic) {
  26 + if(topic == null){
  27 + throw new NullPointerException("topic");
  28 + }
  29 + this.topic = topic;
  30 + this.topicRegex = Pattern.compile(topic.replace("+", "[^/]+").replace("#", ".+") + "$");
  31 + }
  32 +
  33 + public String getTopic() {
  34 + return topic;
  35 + }
  36 +
  37 + public boolean matches(String topic){
  38 + return this.topicRegex.matcher(topic).matches();
  39 + }
  40 +
  41 + @Override
  42 + public boolean equals(Object o) {
  43 + if (this == o) return true;
  44 + if (o == null || getClass() != o.getClass()) return false;
  45 +
  46 + MqttTopicMatcher that = (MqttTopicMatcher) o;
  47 +
  48 + return topic.equals(that.topic);
  49 + }
  50 +
  51 + @Override
  52 + public int hashCode() {
  53 + return topic.hashCode();
  54 + }
  55 +}
@@ -17,32 +17,31 @@ @@ -17,32 +17,31 @@
17 spring.main.web-application-type: none 17 spring.main.web-application-type: none
18 18
19 # MQTT server parameters 19 # MQTT server parameters
20 -mqtt:  
21 - # Enable/disable mqtt transport protocol.  
22 - enabled: "${MQTT_ENABLED:true}"  
23 - bind_address: "${MQTT_BIND_ADDRESS:0.0.0.0}"  
24 - bind_port: "${MQTT_BIND_PORT:1883}"  
25 - adaptor: "${MQTT_ADAPTOR_NAME:JsonMqttAdaptor}"  
26 - timeout: "${MQTT_TIMEOUT:10000}"  
27 - netty:  
28 - leak_detector_level: "${NETTY_LEASK_DETECTOR_LVL:DISABLED}"  
29 - boss_group_thread_count: "${NETTY_BOSS_GROUP_THREADS:1}"  
30 - worker_group_thread_count: "${NETTY_WORKER_GROUP_THREADS:12}"  
31 - max_payload_size: "${NETTY_MAX_PAYLOAD_SIZE:65536}"  
32 - # MQTT SSL configuration  
33 - ssl:  
34 - # Enable/disable SSL support  
35 - enabled: "${MQTT_SSL_ENABLED:false}"  
36 - # SSL protocol: See http://docs.oracle.com/javase/8/docs/technotes/guides/security/StandardNames.html#SSLContext  
37 - protocol: "${MQTT_SSL_PROTOCOL:TLSv1.2}"  
38 - # Path to the key store that holds the SSL certificate  
39 - key_store: "${MQTT_SSL_KEY_STORE:mqttserver.jks}"  
40 - # Password used to access the key store  
41 - key_store_password: "${MQTT_SSL_KEY_STORE_PASSWORD:server_ks_password}"  
42 - # Password used to access the key  
43 - key_password: "${MQTT_SSL_KEY_PASSWORD:server_key_password}"  
44 - # Type of the key store  
45 - key_store_type: "${MQTT_SSL_KEY_STORE_TYPE:JKS}" 20 +transport:
  21 + mqtt:
  22 + bind_address: "${MQTT_BIND_ADDRESS:0.0.0.0}"
  23 + bind_port: "${MQTT_BIND_PORT:1883}"
  24 + adaptor: "${MQTT_ADAPTOR_NAME:JsonMqttAdaptor}"
  25 + timeout: "${MQTT_TIMEOUT:10000}"
  26 + netty:
  27 + leak_detector_level: "${NETTY_LEASK_DETECTOR_LVL:DISABLED}"
  28 + boss_group_thread_count: "${NETTY_BOSS_GROUP_THREADS:1}"
  29 + worker_group_thread_count: "${NETTY_WORKER_GROUP_THREADS:12}"
  30 + max_payload_size: "${NETTY_MAX_PAYLOAD_SIZE:65536}"
  31 + # MQTT SSL configuration
  32 + ssl:
  33 + # Enable/disable SSL support
  34 + enabled: "${MQTT_SSL_ENABLED:false}"
  35 + # SSL protocol: See http://docs.oracle.com/javase/8/docs/technotes/guides/security/StandardNames.html#SSLContext
  36 + protocol: "${MQTT_SSL_PROTOCOL:TLSv1.2}"
  37 + # Path to the key store that holds the SSL certificate
  38 + key_store: "${MQTT_SSL_KEY_STORE:mqttserver.jks}"
  39 + # Password used to access the key store
  40 + key_store_password: "${MQTT_SSL_KEY_STORE_PASSWORD:server_ks_password}"
  41 + # Password used to access the key
  42 + key_password: "${MQTT_SSL_KEY_PASSWORD:server_key_password}"
  43 + # Type of the key store
  44 + key_store_type: "${MQTT_SSL_KEY_STORE_TYPE:JKS}"
46 45
47 #Quota parameters 46 #Quota parameters
48 quota: 47 quota:
@@ -56,7 +55,7 @@ quota: @@ -56,7 +55,7 @@ quota:
56 # Interval for scheduled task that cleans expired records. TTL is used for expiring 55 # Interval for scheduled task that cleans expired records. TTL is used for expiring
57 cleanPeriodMs: "${QUOTA_HOST_CLEAN_PERIOD_MS:300000}" 56 cleanPeriodMs: "${QUOTA_HOST_CLEAN_PERIOD_MS:300000}"
58 # Enable Host API Limits 57 # Enable Host API Limits
59 - enabled: "${QUOTA_HOST_ENABLED:false}" 58 + enabled: "${QUOTA_HOST_ENABLED:true}"
60 # Array of whitelist hosts 59 # Array of whitelist hosts
61 whitelist: "${QUOTA_HOST_WHITELIST:localhost,127.0.0.1}" 60 whitelist: "${QUOTA_HOST_WHITELIST:localhost,127.0.0.1}"
62 # Array of blacklist hosts 61 # Array of blacklist hosts
@@ -533,6 +533,21 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device @@ -533,6 +533,21 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
533 } 533 }
534 ); 534 );
535 break; 535 break;
  536 + case types.aliasFilterType.entityViewType.value:
  537 + getEntitiesByNameFilter(types.entityType.entityView, filter.entityViewNameFilter, maxItems, {ignoreLoading: true}, filter.entityViewType).then(
  538 + function success(entities) {
  539 + if (entities && entities.length || !failOnEmpty) {
  540 + result.entities = entitiesToEntitiesInfo(entities);
  541 + deferred.resolve(result);
  542 + } else {
  543 + deferred.reject();
  544 + }
  545 + },
  546 + function fail() {
  547 + deferred.reject();
  548 + }
  549 + );
  550 + break;
536 case types.aliasFilterType.relationsQuery.value: 551 case types.aliasFilterType.relationsQuery.value:
537 result.stateEntity = filter.rootStateEntity; 552 result.stateEntity = filter.rootStateEntity;
538 var rootEntityType; 553 var rootEntityType;
@@ -578,6 +593,7 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device @@ -578,6 +593,7 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
578 break; 593 break;
579 case types.aliasFilterType.assetSearchQuery.value: 594 case types.aliasFilterType.assetSearchQuery.value:
580 case types.aliasFilterType.deviceSearchQuery.value: 595 case types.aliasFilterType.deviceSearchQuery.value:
  596 + case types.aliasFilterType.entityViewSearchQuery.value:
581 result.stateEntity = filter.rootStateEntity; 597 result.stateEntity = filter.rootStateEntity;
582 if (result.stateEntity && stateEntityId) { 598 if (result.stateEntity && stateEntityId) {
583 rootEntityType = stateEntityId.entityType; 599 rootEntityType = stateEntityId.entityType;
@@ -604,6 +620,9 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device @@ -604,6 +620,9 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
604 } else if (filter.type == types.aliasFilterType.deviceSearchQuery.value) { 620 } else if (filter.type == types.aliasFilterType.deviceSearchQuery.value) {
605 searchQuery.deviceTypes = filter.deviceTypes; 621 searchQuery.deviceTypes = filter.deviceTypes;
606 findByQueryPromise = deviceService.findByQuery(searchQuery, false, {ignoreLoading: true}); 622 findByQueryPromise = deviceService.findByQuery(searchQuery, false, {ignoreLoading: true});
  623 + } else if (filter.type == types.aliasFilterType.entityViewSearchQuery.value) {
  624 + searchQuery.entityViewTypes = filter.entityViewTypes;
  625 + findByQueryPromise = entityViewService.findByQuery(searchQuery, false, {ignoreLoading: true});
607 } 626 }
608 findByQueryPromise.then( 627 findByQueryPromise.then(
609 function success(entities) { 628 function success(entities) {
@@ -646,6 +665,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device @@ -646,6 +665,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
646 return entityTypes.indexOf(types.entityType.asset) > -1 ? true : false; 665 return entityTypes.indexOf(types.entityType.asset) > -1 ? true : false;
647 case types.aliasFilterType.deviceType.value: 666 case types.aliasFilterType.deviceType.value:
648 return entityTypes.indexOf(types.entityType.device) > -1 ? true : false; 667 return entityTypes.indexOf(types.entityType.device) > -1 ? true : false;
  668 + case types.aliasFilterType.entityViewType.value:
  669 + return entityTypes.indexOf(types.entityType.entityView) > -1 ? true : false;
649 case types.aliasFilterType.relationsQuery.value: 670 case types.aliasFilterType.relationsQuery.value:
650 if (filter.filters && filter.filters.length) { 671 if (filter.filters && filter.filters.length) {
651 var match = false; 672 var match = false;
@@ -671,6 +692,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device @@ -671,6 +692,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
671 return entityTypes.indexOf(types.entityType.asset) > -1 ? true : false; 692 return entityTypes.indexOf(types.entityType.asset) > -1 ? true : false;
672 case types.aliasFilterType.deviceSearchQuery.value: 693 case types.aliasFilterType.deviceSearchQuery.value:
673 return entityTypes.indexOf(types.entityType.device) > -1 ? true : false; 694 return entityTypes.indexOf(types.entityType.device) > -1 ? true : false;
  695 + case types.aliasFilterType.entityViewSearchQuery.value:
  696 + return entityTypes.indexOf(types.entityType.entityView) > -1 ? true : false;
674 } 697 }
675 } 698 }
676 return false; 699 return false;
@@ -690,12 +713,16 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device @@ -690,12 +713,16 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
690 return entityType === types.entityType.asset; 713 return entityType === types.entityType.asset;
691 case types.aliasFilterType.deviceType.value: 714 case types.aliasFilterType.deviceType.value:
692 return entityType === types.entityType.device; 715 return entityType === types.entityType.device;
  716 + case types.aliasFilterType.entityViewType.value:
  717 + return entityType === types.entityType.entityView;
693 case types.aliasFilterType.relationsQuery.value: 718 case types.aliasFilterType.relationsQuery.value:
694 return true; 719 return true;
695 case types.aliasFilterType.assetSearchQuery.value: 720 case types.aliasFilterType.assetSearchQuery.value:
696 return entityType === types.entityType.asset; 721 return entityType === types.entityType.asset;
697 case types.aliasFilterType.deviceSearchQuery.value: 722 case types.aliasFilterType.deviceSearchQuery.value:
698 return entityType === types.entityType.device; 723 return entityType === types.entityType.device;
  724 + case types.aliasFilterType.entityViewSearchQuery.value:
  725 + return entityType === types.entityType.entityView;
699 } 726 }
700 return false; 727 return false;
701 } 728 }
@@ -1046,6 +1073,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device @@ -1046,6 +1073,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
1046 return assetService.deleteAsset(entityId.id); 1073 return assetService.deleteAsset(entityId.id);
1047 } else if (entityId.entityType == types.entityType.device) { 1074 } else if (entityId.entityType == types.entityType.device) {
1048 return deviceService.deleteDevice(entityId.id); 1075 return deviceService.deleteDevice(entityId.id);
  1076 + } else if (entityId.entityType == types.entityType.entityView) {
  1077 + return entityViewService.deleteEntityView(entityId.id);
1049 } 1078 }
1050 } 1079 }
1051 1080
@@ -1151,6 +1180,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device @@ -1151,6 +1180,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
1151 return assetService.saveAsset(entity); 1180 return assetService.saveAsset(entity);
1152 } else if (entityType == types.entityType.device) { 1181 } else if (entityType == types.entityType.device) {
1153 return deviceService.saveDevice(entity); 1182 return deviceService.saveDevice(entity);
  1183 + } else if (entityType == types.entityType.entityView) {
  1184 + return entityViewService.saveEntityView(entity);
1154 } 1185 }
1155 } 1186 }
1156 1187
@@ -1279,6 +1310,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device @@ -1279,6 +1310,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
1279 searchQuery.assetTypes = entitySubTypes; 1310 searchQuery.assetTypes = entitySubTypes;
1280 } else if (entityType == types.entityType.device) { 1311 } else if (entityType == types.entityType.device) {
1281 searchQuery.deviceTypes = entitySubTypes; 1312 searchQuery.deviceTypes = entitySubTypes;
  1313 + } else if (entityType == types.entityType.entityView) {
  1314 + searchQuery.entityViewTypes = entitySubTypes;
1282 } else { 1315 } else {
1283 return null; //Not supported 1316 return null; //Not supported
1284 } 1317 }
@@ -172,9 +172,9 @@ export default class Subscription { @@ -172,9 +172,9 @@ export default class Subscription {
172 if (this.type === this.ctx.types.widgetType.rpc.value) { 172 if (this.type === this.ctx.types.widgetType.rpc.value) {
173 if (this.targetDeviceId) { 173 if (this.targetDeviceId) {
174 entityId = { 174 entityId = {
175 - entityType: this.ctx.entityType.device, 175 + entityType: this.ctx.types.entityType.device,
176 id: this.targetDeviceId 176 id: this.targetDeviceId
177 - } 177 + };
178 entityName = this.targetDeviceName; 178 entityName = this.targetDeviceName;
179 } 179 }
180 } else if (this.type == this.ctx.types.widgetType.alarm.value) { 180 } else if (this.type == this.ctx.types.widgetType.alarm.value) {
@@ -182,7 +182,7 @@ export default class Subscription { @@ -182,7 +182,7 @@ export default class Subscription {
182 entityId = { 182 entityId = {
183 entityType: this.alarmSource.entityType, 183 entityType: this.alarmSource.entityType,
184 id: this.alarmSource.entityId 184 id: this.alarmSource.entityId
185 - } 185 + };
186 entityName = this.alarmSource.entityName; 186 entityName = this.alarmSource.entityName;
187 } 187 }
188 } else { 188 } else {
@@ -192,7 +192,7 @@ export default class Subscription { @@ -192,7 +192,7 @@ export default class Subscription {
192 entityId = { 192 entityId = {
193 entityType: datasource.entityType, 193 entityType: datasource.entityType,
194 id: datasource.entityId 194 id: datasource.entityId
195 - } 195 + };
196 entityName = datasource.entityName; 196 entityName = datasource.entityName;
197 break; 197 break;
198 } 198 }
@@ -26,15 +26,17 @@ const MIN_INTERVAL = SECOND; @@ -26,15 +26,17 @@ const MIN_INTERVAL = SECOND;
26 const MAX_INTERVAL = 365 * 20 * DAY; 26 const MAX_INTERVAL = 365 * 20 * DAY;
27 27
28 const MIN_LIMIT = 10; 28 const MIN_LIMIT = 10;
29 -const AVG_LIMIT = 200;  
30 -const MAX_LIMIT = 500; 29 +//const AVG_LIMIT = 200;
  30 +//const MAX_LIMIT = 500;
31 31
32 /*@ngInject*/ 32 /*@ngInject*/
33 -function TimeService($translate, types) { 33 +function TimeService($translate, $http, $q, types) {
34 34
35 var predefIntervals; 35 var predefIntervals;
  36 + var maxDatapointsLimit;
36 37
37 var service = { 38 var service = {
  39 + loadMaxDatapointsLimit: loadMaxDatapointsLimit,
38 minIntervalLimit: minIntervalLimit, 40 minIntervalLimit: minIntervalLimit,
39 maxIntervalLimit: maxIntervalLimit, 41 maxIntervalLimit: maxIntervalLimit,
40 boundMinInterval: boundMinInterval, 42 boundMinInterval: boundMinInterval,
@@ -45,20 +47,38 @@ function TimeService($translate, types) { @@ -45,20 +47,38 @@ function TimeService($translate, types) {
45 defaultTimewindow: defaultTimewindow, 47 defaultTimewindow: defaultTimewindow,
46 toHistoryTimewindow: toHistoryTimewindow, 48 toHistoryTimewindow: toHistoryTimewindow,
47 createSubscriptionTimewindow: createSubscriptionTimewindow, 49 createSubscriptionTimewindow: createSubscriptionTimewindow,
48 - avgAggregationLimit: function () {  
49 - return AVG_LIMIT; 50 + getMaxDatapointsLimit: function () {
  51 + return maxDatapointsLimit;
  52 + },
  53 + getMinDatapointsLimit: function () {
  54 + return MIN_LIMIT;
50 } 55 }
51 } 56 }
52 57
53 return service; 58 return service;
54 59
  60 + function loadMaxDatapointsLimit() {
  61 + var deferred = $q.defer();
  62 + var url = '/api/dashboard/maxDatapointsLimit';
  63 + $http.get(url, {ignoreLoading: true}).then(function success(response) {
  64 + maxDatapointsLimit = response.data;
  65 + if (!maxDatapointsLimit || maxDatapointsLimit <= MIN_LIMIT) {
  66 + maxDatapointsLimit = MIN_LIMIT + 1;
  67 + }
  68 + deferred.resolve();
  69 + }, function fail() {
  70 + deferred.reject();
  71 + });
  72 + return deferred.promise;
  73 + }
  74 +
55 function minIntervalLimit(timewindow) { 75 function minIntervalLimit(timewindow) {
56 - var min = timewindow / MAX_LIMIT; 76 + var min = timewindow / 500;
57 return boundMinInterval(min); 77 return boundMinInterval(min);
58 } 78 }
59 79
60 function avgInterval(timewindow) { 80 function avgInterval(timewindow) {
61 - var avg = timewindow / AVG_LIMIT; 81 + var avg = timewindow / 200;
62 return boundMinInterval(avg); 82 return boundMinInterval(avg);
63 } 83 }
64 84
@@ -230,7 +250,7 @@ function TimeService($translate, types) { @@ -230,7 +250,7 @@ function TimeService($translate, types) {
230 }, 250 },
231 aggregation: { 251 aggregation: {
232 type: types.aggregation.avg.value, 252 type: types.aggregation.avg.value,
233 - limit: AVG_LIMIT 253 + limit: Math.floor(maxDatapointsLimit / 2)
234 } 254 }
235 } 255 }
236 return timewindow; 256 return timewindow;
@@ -246,22 +266,27 @@ function TimeService($translate, types) { @@ -246,22 +266,27 @@ function TimeService($translate, types) {
246 } 266 }
247 267
248 var aggType; 268 var aggType;
  269 + var limit;
249 if (timewindow.aggregation) { 270 if (timewindow.aggregation) {
250 aggType = timewindow.aggregation.type || types.aggregation.avg.value; 271 aggType = timewindow.aggregation.type || types.aggregation.avg.value;
  272 + limit = timewindow.aggregation.limit || maxDatapointsLimit;
251 } else { 273 } else {
252 aggType = types.aggregation.avg.value; 274 aggType = types.aggregation.avg.value;
  275 + limit = maxDatapointsLimit;
253 } 276 }
254 277
  278 +
255 var historyTimewindow = { 279 var historyTimewindow = {
256 history: { 280 history: {
257 fixedTimewindow: { 281 fixedTimewindow: {
258 startTimeMs: startTimeMs, 282 startTimeMs: startTimeMs,
259 endTimeMs: endTimeMs 283 endTimeMs: endTimeMs
260 }, 284 },
261 - interval: boundIntervalToTimewindow(endTimeMs - startTimeMs, interval, aggType) 285 + interval: boundIntervalToTimewindow(endTimeMs - startTimeMs, interval, types.aggregation.avg.value)
262 }, 286 },
263 aggregation: { 287 aggregation: {
264 - type: aggType 288 + type: aggType,
  289 + limit: limit
265 } 290 }
266 } 291 }
267 292
@@ -275,7 +300,7 @@ function TimeService($translate, types) { @@ -275,7 +300,7 @@ function TimeService($translate, types) {
275 realtimeWindowMs: null, 300 realtimeWindowMs: null,
276 aggregation: { 301 aggregation: {
277 interval: SECOND, 302 interval: SECOND,
278 - limit: AVG_LIMIT, 303 + limit: maxDatapointsLimit,
279 type: types.aggregation.avg.value 304 type: types.aggregation.avg.value
280 } 305 }
281 }; 306 };
@@ -283,14 +308,14 @@ function TimeService($translate, types) { @@ -283,14 +308,14 @@ function TimeService($translate, types) {
283 if (stateData) { 308 if (stateData) {
284 subscriptionTimewindow.aggregation = { 309 subscriptionTimewindow.aggregation = {
285 interval: SECOND, 310 interval: SECOND,
286 - limit: MAX_LIMIT, 311 + limit: maxDatapointsLimit,
287 type: types.aggregation.none.value, 312 type: types.aggregation.none.value,
288 stateData: true 313 stateData: true
289 }; 314 };
290 } else { 315 } else {
291 subscriptionTimewindow.aggregation = { 316 subscriptionTimewindow.aggregation = {
292 interval: SECOND, 317 interval: SECOND,
293 - limit: AVG_LIMIT, 318 + limit: maxDatapointsLimit,
294 type: types.aggregation.avg.value 319 type: types.aggregation.avg.value
295 }; 320 };
296 } 321 }
@@ -298,7 +323,7 @@ function TimeService($translate, types) { @@ -298,7 +323,7 @@ function TimeService($translate, types) {
298 if (angular.isDefined(timewindow.aggregation) && !stateData) { 323 if (angular.isDefined(timewindow.aggregation) && !stateData) {
299 subscriptionTimewindow.aggregation = { 324 subscriptionTimewindow.aggregation = {
300 type: timewindow.aggregation.type || types.aggregation.avg.value, 325 type: timewindow.aggregation.type || types.aggregation.avg.value,
301 - limit: timewindow.aggregation.limit || AVG_LIMIT 326 + limit: timewindow.aggregation.limit || maxDatapointsLimit
302 }; 327 };
303 } 328 }
304 if (angular.isDefined(timewindow.realtime)) { 329 if (angular.isDefined(timewindow.realtime)) {
@@ -22,7 +22,7 @@ export default angular.module('thingsboard.api.user', [thingsboardApiLogin, @@ -22,7 +22,7 @@ export default angular.module('thingsboard.api.user', [thingsboardApiLogin,
22 .name; 22 .name;
23 23
24 /*@ngInject*/ 24 /*@ngInject*/
25 -function UserService($http, $q, $rootScope, adminService, dashboardService, loginService, toast, store, jwtHelper, $translate, $state, $location) { 25 +function UserService($http, $q, $rootScope, adminService, dashboardService, timeService, loginService, toast, store, jwtHelper, $translate, $state, $location) {
26 var currentUser = null, 26 var currentUser = null,
27 currentUserDetails = null, 27 currentUserDetails = null,
28 lastPublicDashboardId = null, 28 lastPublicDashboardId = null,
@@ -390,6 +390,7 @@ function UserService($http, $q, $rootScope, adminService, dashboardService, logi @@ -390,6 +390,7 @@ function UserService($http, $q, $rootScope, adminService, dashboardService, logi
390 function loadSystemParams() { 390 function loadSystemParams() {
391 var promises = []; 391 var promises = [];
392 promises.push(loadIsUserTokenAccessEnabled()); 392 promises.push(loadIsUserTokenAccessEnabled());
  393 + promises.push(timeService.loadMaxDatapointsLimit());
393 return $q.all(promises); 394 return $q.all(promises);
394 } 395 }
395 396
@@ -253,6 +253,10 @@ export default angular.module('thingsboard.types', []) @@ -253,6 +253,10 @@ export default angular.module('thingsboard.types', [])
253 value: 'deviceType', 253 value: 'deviceType',
254 name: 'alias.filter-type-device-type' 254 name: 'alias.filter-type-device-type'
255 }, 255 },
  256 + entityViewType: {
  257 + value: 'entityViewType',
  258 + name: 'alias.filter-type-entity-view-type'
  259 + },
256 relationsQuery: { 260 relationsQuery: {
257 value: 'relationsQuery', 261 value: 'relationsQuery',
258 name: 'alias.filter-type-relations-query' 262 name: 'alias.filter-type-relations-query'
@@ -264,6 +268,10 @@ export default angular.module('thingsboard.types', []) @@ -264,6 +268,10 @@ export default angular.module('thingsboard.types', [])
264 deviceSearchQuery: { 268 deviceSearchQuery: {
265 value: 'deviceSearchQuery', 269 value: 'deviceSearchQuery',
266 name: 'alias.filter-type-device-search-query' 270 name: 'alias.filter-type-device-search-query'
  271 + },
  272 + entityViewSearchQuery: {
  273 + value: 'entityViewSearchQuery',
  274 + name: 'alias.filter-type-entity-view-search-query'
267 } 275 }
268 }, 276 },
269 position: { 277 position: {
@@ -31,6 +31,8 @@ export default function TimewindowPanelController(mdPanelRef, $scope, timeServic @@ -31,6 +31,8 @@ export default function TimewindowPanelController(mdPanelRef, $scope, timeServic
31 vm.maxRealtimeAggInterval = maxRealtimeAggInterval; 31 vm.maxRealtimeAggInterval = maxRealtimeAggInterval;
32 vm.minHistoryAggInterval = minHistoryAggInterval; 32 vm.minHistoryAggInterval = minHistoryAggInterval;
33 vm.maxHistoryAggInterval = maxHistoryAggInterval; 33 vm.maxHistoryAggInterval = maxHistoryAggInterval;
  34 + vm.minDatapointsLimit = minDatapointsLimit;
  35 + vm.maxDatapointsLimit = maxDatapointsLimit;
34 36
35 if (vm.historyOnly) { 37 if (vm.historyOnly) {
36 vm.timewindow.selectedTab = 1; 38 vm.timewindow.selectedTab = 1;
@@ -86,6 +88,14 @@ export default function TimewindowPanelController(mdPanelRef, $scope, timeServic @@ -86,6 +88,14 @@ export default function TimewindowPanelController(mdPanelRef, $scope, timeServic
86 return timeService.maxIntervalLimit(currentHistoryTimewindow()); 88 return timeService.maxIntervalLimit(currentHistoryTimewindow());
87 } 89 }
88 90
  91 + function minDatapointsLimit () {
  92 + return timeService.getMinDatapointsLimit();
  93 + }
  94 +
  95 + function maxDatapointsLimit () {
  96 + return timeService.getMaxDatapointsLimit();
  97 + }
  98 +
89 function currentHistoryTimewindow() { 99 function currentHistoryTimewindow() {
90 if (vm.timewindow.history.historyType === 0) { 100 if (vm.timewindow.history.historyType === 0) {
91 return vm.timewindow.history.timewindowMs; 101 return vm.timewindow.history.timewindowMs;
@@ -60,19 +60,22 @@ @@ -60,19 +60,22 @@
60 </md-option> 60 </md-option>
61 </md-select> 61 </md-select>
62 </md-input-container> 62 </md-input-container>
63 - <md-slider-container ng-show="vm.showLimit()"> 63 + <md-slider-container ng-if="vm.showLimit()">
64 <span translate>aggregation.limit</span> 64 <span translate>aggregation.limit</span>
65 - <md-slider flex min="10" max="500" ng-model="vm.timewindow.aggregation.limit" aria-label="limit" id="limit-slider"> 65 + <md-slider flex min="{{vm.minDatapointsLimit()}}" max="{{vm.maxDatapointsLimit()}}" step="1"
  66 + ng-model="vm.timewindow.aggregation.limit" aria-label="limit" id="limit-slider">
66 </md-slider> 67 </md-slider>
67 - <md-input-container>  
68 - <input flex type="number" ng-model="vm.timewindow.aggregation.limit" aria-label="limit" aria-controls="limit-slider"> 68 + <md-input-container style="max-width: 80px;">
  69 + <input flex type="number" step="1"
  70 + min="{{vm.minDatapointsLimit()}}" max="{{vm.maxDatapointsLimit()}}"
  71 + ng-model="vm.timewindow.aggregation.limit" aria-label="limit" aria-controls="limit-slider">
69 </md-input-container> 72 </md-input-container>
70 </md-slider-container> 73 </md-slider-container>
71 - <tb-timeinterval ng-show="vm.showRealtimeAggInterval()" min="vm.minRealtimeAggInterval()" max="vm.maxRealtimeAggInterval()" 74 + <tb-timeinterval ng-if="vm.showRealtimeAggInterval()" min="vm.minRealtimeAggInterval()" max="vm.maxRealtimeAggInterval()"
72 predefined-name="'aggregation.group-interval'" 75 predefined-name="'aggregation.group-interval'"
73 ng-model="vm.timewindow.realtime.interval"> 76 ng-model="vm.timewindow.realtime.interval">
74 </tb-timeinterval> 77 </tb-timeinterval>
75 - <tb-timeinterval ng-show="vm.showHistoryAggInterval()" min="vm.minHistoryAggInterval()" max="vm.maxHistoryAggInterval()" 78 + <tb-timeinterval ng-if="vm.showHistoryAggInterval()" min="vm.minHistoryAggInterval()" max="vm.maxHistoryAggInterval()"
76 predefined-name="'aggregation.group-interval'" 79 predefined-name="'aggregation.group-interval'"
77 ng-model="vm.timewindow.history.interval"> 80 ng-model="vm.timewindow.history.interval">
78 </tb-timeinterval> 81 </tb-timeinterval>
@@ -228,7 +228,7 @@ function Timewindow($compile, $templateCache, $filter, $mdPanel, $document, $mdM @@ -228,7 +228,7 @@ function Timewindow($compile, $templateCache, $filter, $mdPanel, $document, $mdM
228 if (angular.isDefined(value.aggregation.type) && value.aggregation.type.length > 0) { 228 if (angular.isDefined(value.aggregation.type) && value.aggregation.type.length > 0) {
229 model.aggregation.type = value.aggregation.type; 229 model.aggregation.type = value.aggregation.type;
230 } 230 }
231 - model.aggregation.limit = value.aggregation.limit || timeService.avgAggregationLimit(); 231 + model.aggregation.limit = value.aggregation.limit || Math.floor(timeService.getMaxDatapointsLimit() / 2);
232 } 232 }
233 } 233 }
234 scope.updateDisplayValue(); 234 scope.updateDisplayValue();
@@ -52,12 +52,22 @@ @@ -52,12 +52,22 @@
52 <div translate ng-message="required">entity-view.name-required</div> 52 <div translate ng-message="required">entity-view.name-required</div>
53 </div> 53 </div>
54 </md-input-container> 54 </md-input-container>
55 - <tb-entity-select flex ng-disabled="!isEdit"  
56 - the-form="theForm"  
57 - tb-required="true"  
58 - allowed-entity-types="allowedEntityTypes"  
59 - ng-model="entityView.entityId">  
60 - </tb-entity-select> 55 + <tb-entity-subtype-autocomplete
  56 + ng-disabled="$root.loading || !isEdit"
  57 + tb-required="true"
  58 + the-form="theForm"
  59 + ng-model="entityView.type"
  60 + entity-type="types.entityType.entityView">
  61 + </tb-entity-subtype-autocomplete>
  62 + <section layout="column">
  63 + <label translate class="tb-title no-padding">entity-view.related-entity</label>
  64 + <tb-entity-select flex ng-disabled="!isEdit"
  65 + the-form="theForm"
  66 + tb-required="true"
  67 + allowed-entity-types="allowedEntityTypes"
  68 + ng-model="entityView.entityId">
  69 + </tb-entity-select>
  70 + </section>
61 <md-input-container class="md-block"> 71 <md-input-container class="md-block">
62 <label translate>entity-view.description</label> 72 <label translate>entity-view.description</label>
63 <textarea ng-model="entityView.additionalInfo.description" rows="2"></textarea> 73 <textarea ng-model="entityView.additionalInfo.description" rows="2"></textarea>
@@ -42,7 +42,7 @@ export default function EntityViewRoutes($stateProvider, types) { @@ -42,7 +42,7 @@ export default function EntityViewRoutes($stateProvider, types) {
42 pageTitle: 'entity-view.entity-views' 42 pageTitle: 'entity-view.entity-views'
43 }, 43 },
44 ncyBreadcrumb: { 44 ncyBreadcrumb: {
45 - label: '{"icon": "view_stream", "label": "entity-view.entity-views"}' 45 + label: '{"icon": "view_quilt", "label": "entity-view.entity-views"}'
46 } 46 }
47 }) 47 })
48 .state('home.customers.entityViews', { 48 .state('home.customers.entityViews', {
@@ -65,7 +65,7 @@ export default function EntityViewRoutes($stateProvider, types) { @@ -65,7 +65,7 @@ export default function EntityViewRoutes($stateProvider, types) {
65 pageTitle: 'customer.entity-views' 65 pageTitle: 'customer.entity-views'
66 }, 66 },
67 ncyBreadcrumb: { 67 ncyBreadcrumb: {
68 - label: '{"icon": "view_stream", "label": "{{ vm.customerEntityViewsTitle }}", "translate": "false"}' 68 + label: '{"icon": "view_quilt", "label": "{{ vm.customerEntityViewsTitle }}", "translate": "false"}'
69 } 69 }
70 }); 70 });
71 71
@@ -77,6 +77,15 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q, @@ -77,6 +77,15 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q,
77 scope.filterDisplayValue = $translate.instant('alias.filter-type-device-type-description', {deviceType: deviceType}); 77 scope.filterDisplayValue = $translate.instant('alias.filter-type-device-type-description', {deviceType: deviceType});
78 } 78 }
79 break; 79 break;
  80 + case types.aliasFilterType.entityViewType.value:
  81 + var entityViewType = scope.filter.entityViewType;
  82 + prefix = scope.filter.entityViewNameFilter;
  83 + if (prefix && prefix.length) {
  84 + scope.filterDisplayValue = $translate.instant('alias.filter-type-entity-view-type-and-name-description', {entityViewType: entityViewType, prefix: prefix});
  85 + } else {
  86 + scope.filterDisplayValue = $translate.instant('alias.filter-type-entity-view-type-description', {entityViewType: entityViewType});
  87 + }
  88 + break;
80 case types.aliasFilterType.relationsQuery.value: 89 case types.aliasFilterType.relationsQuery.value:
81 var rootEntityText; 90 var rootEntityText;
82 var directionText; 91 var directionText;
@@ -134,6 +143,7 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q, @@ -134,6 +143,7 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q,
134 break; 143 break;
135 case types.aliasFilterType.assetSearchQuery.value: 144 case types.aliasFilterType.assetSearchQuery.value:
136 case types.aliasFilterType.deviceSearchQuery.value: 145 case types.aliasFilterType.deviceSearchQuery.value:
  146 + case types.aliasFilterType.entityViewSearchQuery.value:
137 allEntitiesText = $translate.instant('alias.all-entities'); 147 allEntitiesText = $translate.instant('alias.all-entities');
138 anyRelationText = $translate.instant('alias.any-relation'); 148 anyRelationText = $translate.instant('alias.any-relation');
139 if (scope.filter.rootStateEntity) { 149 if (scope.filter.rootStateEntity) {
@@ -165,7 +175,7 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q, @@ -165,7 +175,7 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q,
165 scope.filterDisplayValue = $translate.instant('alias.filter-type-asset-search-query-description', 175 scope.filterDisplayValue = $translate.instant('alias.filter-type-asset-search-query-description',
166 translationValues 176 translationValues
167 ); 177 );
168 - } else { 178 + } else if (scope.filter.type == types.aliasFilterType.deviceSearchQuery.value) {
169 var deviceTypesQuoted = []; 179 var deviceTypesQuoted = [];
170 scope.filter.deviceTypes.forEach(function(deviceType) { 180 scope.filter.deviceTypes.forEach(function(deviceType) {
171 deviceTypesQuoted.push("'"+deviceType+"'"); 181 deviceTypesQuoted.push("'"+deviceType+"'");
@@ -175,6 +185,16 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q, @@ -175,6 +185,16 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q,
175 scope.filterDisplayValue = $translate.instant('alias.filter-type-device-search-query-description', 185 scope.filterDisplayValue = $translate.instant('alias.filter-type-device-search-query-description',
176 translationValues 186 translationValues
177 ); 187 );
  188 + } else if (scope.filter.type == types.aliasFilterType.entityViewSearchQuery.value) {
  189 + var entityViewTypesQuoted = [];
  190 + scope.filter.entityViewTypes.forEach(function(entityViewType) {
  191 + entityViewTypesQuoted.push("'"+entityViewType+"'");
  192 + });
  193 + var entityViewTypesText = entityViewTypesQuoted.join(', ');
  194 + translationValues.entityViewTypes = entityViewTypesText;
  195 + scope.filterDisplayValue = $translate.instant('alias.filter-type-entity-view-search-query-description',
  196 + translationValues
  197 + );
178 } 198 }
179 break; 199 break;
180 default: 200 default:
@@ -69,9 +69,14 @@ export default function EntityFilterDirective($compile, $templateCache, $q, $doc @@ -69,9 +69,14 @@ export default function EntityFilterDirective($compile, $templateCache, $q, $doc
69 filter.deviceType = null; 69 filter.deviceType = null;
70 filter.deviceNameFilter = ''; 70 filter.deviceNameFilter = '';
71 break; 71 break;
  72 + case types.aliasFilterType.entityViewType.value:
  73 + filter.entityViewType = null;
  74 + filter.entityViewNameFilter = '';
  75 + break;
72 case types.aliasFilterType.relationsQuery.value: 76 case types.aliasFilterType.relationsQuery.value:
73 case types.aliasFilterType.assetSearchQuery.value: 77 case types.aliasFilterType.assetSearchQuery.value:
74 case types.aliasFilterType.deviceSearchQuery.value: 78 case types.aliasFilterType.deviceSearchQuery.value:
  79 + case types.aliasFilterType.entityViewSearchQuery.value:
75 filter.rootStateEntity = false; 80 filter.rootStateEntity = false;
76 filter.stateEntityParamName = null; 81 filter.stateEntityParamName = null;
77 filter.defaultStateEntity = null; 82 filter.defaultStateEntity = null;
@@ -86,6 +91,9 @@ export default function EntityFilterDirective($compile, $templateCache, $q, $doc @@ -86,6 +91,9 @@ export default function EntityFilterDirective($compile, $templateCache, $q, $doc
86 } else if (filter.type === types.aliasFilterType.deviceSearchQuery.value) { 91 } else if (filter.type === types.aliasFilterType.deviceSearchQuery.value) {
87 filter.relationType = null; 92 filter.relationType = null;
88 filter.deviceTypes = []; 93 filter.deviceTypes = [];
  94 + } else if (filter.type === types.aliasFilterType.entityViewSearchQuery.value) {
  95 + filter.relationType = null;
  96 + filter.entityViewTypes = [];
89 } 97 }
90 break; 98 break;
91 } 99 }
@@ -112,6 +112,20 @@ @@ -112,6 +112,20 @@
112 aria-label="{{ 'device.name-starts-with' | translate }}"> 112 aria-label="{{ 'device.name-starts-with' | translate }}">
113 </md-input-container> 113 </md-input-container>
114 </section> 114 </section>
  115 + <section layout="column" ng-if="filter.type == types.aliasFilterType.entityViewType.value" id="entityViewTypeFilter">
  116 + <tb-entity-subtype-autocomplete
  117 + tb-required="true"
  118 + the-form="theForm"
  119 + ng-model="filter.entityViewType"
  120 + entity-type="types.entityType.entityView">
  121 + </tb-entity-subtype-autocomplete>
  122 + <md-input-container class="md-block">
  123 + <label translate>entity-view.name-starts-with</label>
  124 + <input name="entityViewNameFilter"
  125 + ng-model="filter.entityViewNameFilter"
  126 + aria-label="{{ 'entity-view.name-starts-with' | translate }}">
  127 + </md-input-container>
  128 + </section>
115 <section layout="column" ng-if="filter.type == types.aliasFilterType.relationsQuery.value" id="relationsQueryFilter"> 129 <section layout="column" ng-if="filter.type == types.aliasFilterType.relationsQuery.value" id="relationsQueryFilter">
116 <label class="tb-small">{{ 'alias.root-entity' | translate }}</label> 130 <label class="tb-small">{{ 'alias.root-entity' | translate }}</label>
117 <section class="tb-root-state-entity-switch" layout="row" layout-align="start center" style="padding-left: 0px;"> 131 <section class="tb-root-state-entity-switch" layout="row" layout-align="start center" style="padding-left: 0px;">
@@ -311,4 +325,73 @@ @@ -311,4 +325,73 @@
311 ng-model="filter.deviceTypes"> 325 ng-model="filter.deviceTypes">
312 </tb-entity-subtype-list> 326 </tb-entity-subtype-list>
313 </section> 327 </section>
  328 + <section layout="column" ng-if="filter.type == types.aliasFilterType.entityViewSearchQuery.value" id="entityViewSearchQueryFilter">
  329 + <label class="tb-small">{{ 'alias.root-entity' | translate }}</label>
  330 + <section class="tb-root-state-entity-switch" layout="row" layout-align="start center" style="padding-left: 0px;">
  331 + <md-switch class="root-state-entity-switch" ng-model="filter.rootStateEntity"
  332 + aria-label="{{ 'alias.root-state-entity' | translate }}">
  333 + </md-switch>
  334 + <label class="tb-small root-state-entity-label" translate>alias.root-state-entity</label>
  335 + </section>
  336 + <div flex layout="row" ng-if="!filter.rootStateEntity">
  337 + <tb-entity-select flex
  338 + the-form="theForm"
  339 + tb-required="!filter.rootStateEntity"
  340 + ng-disabled="filter.rootStateEntity"
  341 + use-alias-entity-types="true"
  342 + ng-model="filter.rootEntity">
  343 + </tb-entity-select>
  344 + </div>
  345 + <div flex layout="row" ng-if="filter.rootStateEntity">
  346 + <md-input-container class="md-block" style="margin-top: 32px;">
  347 + <label translate>alias.state-entity-parameter-name</label>
  348 + <input name="stateEntityParamName"
  349 + placeholder="{{ 'alias.default-entity-parameter-name' | translate }}"
  350 + ng-model="filter.stateEntityParamName"
  351 + aria-label="{{ 'alias.state-entity-parameter-name' | translate }}">
  352 + </md-input-container>
  353 + <div flex layout="column">
  354 + <label class="tb-small">{{ 'alias.default-state-entity' | translate }}</label>
  355 + <tb-entity-select flex
  356 + the-form="theForm"
  357 + tb-required="false"
  358 + use-alias-entity-types="true"
  359 + ng-model="filter.defaultStateEntity">
  360 + </tb-entity-select>
  361 + </div>
  362 + </div>
  363 + <div flex layout="row">
  364 + <md-input-container class="md-block" style="min-width: 100px;">
  365 + <label translate>relation.direction</label>
  366 + <md-select required ng-model="filter.direction">
  367 + <md-option ng-repeat="direction in types.entitySearchDirection" ng-value="direction">
  368 + {{ ('relation.search-direction.' + direction) | translate}}
  369 + </md-option>
  370 + </md-select>
  371 + </md-input-container>
  372 + <md-input-container flex class="md-block">
  373 + <label translate>alias.max-relation-level</label>
  374 + <input name="maxRelationLevel"
  375 + type="number"
  376 + min="1"
  377 + step="1"
  378 + placeholder="{{ 'alias.unlimited-level' | translate }}"
  379 + ng-model="filter.maxLevel"
  380 + aria-label="{{ 'alias.max-relation-level' | translate }}">
  381 + </md-input-container>
  382 + </div>
  383 + <div class="md-caption" style="color: rgba(0,0,0,0.57);" translate>relation.relation-type</div>
  384 + <tb-relation-type-autocomplete flex
  385 + hide-label
  386 + the-form="theForm"
  387 + ng-model="filter.relationType"
  388 + tb-required="false">
  389 + </tb-relation-type-autocomplete>
  390 + <div class="md-caption tb-required" style="color: rgba(0,0,0,0.57);" translate>entity-view.entity-view-types</div>
  391 + <tb-entity-subtype-list
  392 + tb-required="true"
  393 + entity-type="types.entityType.entityView"
  394 + ng-model="filter.entityViewTypes">
  395 + </tb-entity-subtype-list>
  396 + </section>
314 </div> 397 </div>
@@ -22,7 +22,7 @@ import entitySubtypeAutocompleteTemplate from './entity-subtype-autocomplete.tpl @@ -22,7 +22,7 @@ import entitySubtypeAutocompleteTemplate from './entity-subtype-autocomplete.tpl
22 /* eslint-enable import/no-unresolved, import/default */ 22 /* eslint-enable import/no-unresolved, import/default */
23 23
24 /*@ngInject*/ 24 /*@ngInject*/
25 -export default function EntitySubtypeAutocomplete($compile, $templateCache, $q, $filter, assetService, deviceService, types) { 25 +export default function EntitySubtypeAutocomplete($compile, $templateCache, $q, $filter, assetService, deviceService, entityViewService, types) {
26 26
27 var linker = function (scope, element, attrs, ngModelCtrl) { 27 var linker = function (scope, element, attrs, ngModelCtrl) {
28 var template = $templateCache.get(entitySubtypeAutocompleteTemplate); 28 var template = $templateCache.get(entitySubtypeAutocompleteTemplate);
@@ -96,6 +96,8 @@ export default function EntitySubtypeAutocomplete($compile, $templateCache, $q, @@ -96,6 +96,8 @@ export default function EntitySubtypeAutocomplete($compile, $templateCache, $q,
96 entitySubtypesPromise = assetService.getAssetTypes({ignoreLoading: true}); 96 entitySubtypesPromise = assetService.getAssetTypes({ignoreLoading: true});
97 } else if (scope.entityType == types.entityType.device) { 97 } else if (scope.entityType == types.entityType.device) {
98 entitySubtypesPromise = deviceService.getDeviceTypes({ignoreLoading: true}); 98 entitySubtypesPromise = deviceService.getDeviceTypes({ignoreLoading: true});
  99 + } else if (scope.entityType == types.entityType.entityView) {
  100 + entitySubtypesPromise = entityViewService.getEntityViewTypes({ignoreLoading: true});
99 } 101 }
100 if (entitySubtypesPromise) { 102 if (entitySubtypesPromise) {
101 entitySubtypesPromise.then( 103 entitySubtypesPromise.then(
@@ -134,6 +136,13 @@ export default function EntitySubtypeAutocomplete($compile, $templateCache, $q, @@ -134,6 +136,13 @@ export default function EntitySubtypeAutocomplete($compile, $templateCache, $q,
134 scope.$on('deviceSaved', function() { 136 scope.$on('deviceSaved', function() {
135 scope.entitySubtypes = null; 137 scope.entitySubtypes = null;
136 }); 138 });
  139 + } else if (scope.entityType == types.entityType.entityView) {
  140 + scope.selectEntitySubtypeText = 'entity-view.select-entity-view-type';
  141 + scope.entitySubtypeText = 'entity-view.entity-view-type';
  142 + scope.entitySubtypeRequiredText = 'entity-view.entity-view-type-required';
  143 + scope.$on('entityViewSaved', function() {
  144 + scope.entitySubtypes = null;
  145 + });
137 } 146 }
138 } 147 }
139 148
@@ -22,7 +22,7 @@ import entitySubtypeListTemplate from './entity-subtype-list.tpl.html'; @@ -22,7 +22,7 @@ import entitySubtypeListTemplate from './entity-subtype-list.tpl.html';
22 import './entity-subtype-list.scss'; 22 import './entity-subtype-list.scss';
23 23
24 /*@ngInject*/ 24 /*@ngInject*/
25 -export default function EntitySubtypeListDirective($compile, $templateCache, $q, $mdUtil, $translate, $filter, types, assetService, deviceService) { 25 +export default function EntitySubtypeListDirective($compile, $templateCache, $q, $mdUtil, $translate, $filter, types, assetService, deviceService, entityViewService) {
26 26
27 var linker = function (scope, element, attrs, ngModelCtrl) { 27 var linker = function (scope, element, attrs, ngModelCtrl) {
28 28
@@ -47,6 +47,12 @@ export default function EntitySubtypeListDirective($compile, $templateCache, $q, @@ -47,6 +47,12 @@ export default function EntitySubtypeListDirective($compile, $templateCache, $q,
47 scope.secondaryPlaceholder = '+' + $translate.instant('device.device-type'); 47 scope.secondaryPlaceholder = '+' + $translate.instant('device.device-type');
48 scope.noSubtypesMathingText = 'device.no-device-types-matching'; 48 scope.noSubtypesMathingText = 'device.no-device-types-matching';
49 scope.subtypeListEmptyText = 'device.device-type-list-empty'; 49 scope.subtypeListEmptyText = 'device.device-type-list-empty';
  50 + } else if (scope.entityType == types.entityType.entityView) {
  51 + scope.placeholder = scope.tbRequired ? $translate.instant('entity-view.enter-entity-view-type')
  52 + : $translate.instant('entity-view.any-entity-view');
  53 + scope.secondaryPlaceholder = '+' + $translate.instant('entity-view.entity-view-type');
  54 + scope.noSubtypesMathingText = 'entity-view.no-entity-view-types-matching';
  55 + scope.subtypeListEmptyText = 'entity-view.entity-view-type-list-empty';
50 } 56 }
51 57
52 scope.$watch('tbRequired', function () { 58 scope.$watch('tbRequired', function () {
@@ -97,6 +103,8 @@ export default function EntitySubtypeListDirective($compile, $templateCache, $q, @@ -97,6 +103,8 @@ export default function EntitySubtypeListDirective($compile, $templateCache, $q,
97 entitySubtypesPromise = assetService.getAssetTypes({ignoreLoading: true}); 103 entitySubtypesPromise = assetService.getAssetTypes({ignoreLoading: true});
98 } else if (scope.entityType == types.entityType.device) { 104 } else if (scope.entityType == types.entityType.device) {
99 entitySubtypesPromise = deviceService.getDeviceTypes({ignoreLoading: true}); 105 entitySubtypesPromise = deviceService.getDeviceTypes({ignoreLoading: true});
  106 + } else if (scope.entityType == types.entityType.entityView) {
  107 + entitySubtypesPromise = entityViewService.getEntityViewTypes({ignoreLoading: true});
100 } 108 }
101 if (entitySubtypesPromise) { 109 if (entitySubtypesPromise) {
102 entitySubtypesPromise.then( 110 entitySubtypesPromise.then(
@@ -22,7 +22,7 @@ import entitySubtypeSelectTemplate from './entity-subtype-select.tpl.html'; @@ -22,7 +22,7 @@ import entitySubtypeSelectTemplate from './entity-subtype-select.tpl.html';
22 /* eslint-enable import/no-unresolved, import/default */ 22 /* eslint-enable import/no-unresolved, import/default */
23 23
24 /*@ngInject*/ 24 /*@ngInject*/
25 -export default function EntitySubtypeSelect($compile, $templateCache, $translate, assetService, deviceService, types) { 25 +export default function EntitySubtypeSelect($compile, $templateCache, $translate, assetService, deviceService, entityViewService, types) {
26 26
27 var linker = function (scope, element, attrs, ngModelCtrl) { 27 var linker = function (scope, element, attrs, ngModelCtrl) {
28 var template = $templateCache.get(entitySubtypeSelectTemplate); 28 var template = $templateCache.get(entitySubtypeSelectTemplate);
@@ -75,6 +75,8 @@ export default function EntitySubtypeSelect($compile, $templateCache, $translate @@ -75,6 +75,8 @@ export default function EntitySubtypeSelect($compile, $templateCache, $translate
75 entitySubtypesPromise = assetService.getAssetTypes({ignoreLoading: true}); 75 entitySubtypesPromise = assetService.getAssetTypes({ignoreLoading: true});
76 } else if (scope.entityType == types.entityType.device) { 76 } else if (scope.entityType == types.entityType.device) {
77 entitySubtypesPromise = deviceService.getDeviceTypes({ignoreLoading: true}); 77 entitySubtypesPromise = deviceService.getDeviceTypes({ignoreLoading: true});
  78 + } else if (scope.entityType == types.entityType.entityView) {
  79 + entitySubtypesPromise = entityViewService.getEntityViewTypes({ignoreLoading: true});
78 } 80 }
79 if (entitySubtypesPromise) { 81 if (entitySubtypesPromise) {
80 entitySubtypesPromise.then( 82 entitySubtypesPromise.then(
@@ -100,6 +102,9 @@ export default function EntitySubtypeSelect($compile, $templateCache, $translate @@ -100,6 +102,9 @@ export default function EntitySubtypeSelect($compile, $templateCache, $translate
100 } else if (scope.entityType == types.entityType.device) { 102 } else if (scope.entityType == types.entityType.device) {
101 scope.entitySubtypeTitle = 'device.device-type'; 103 scope.entitySubtypeTitle = 'device.device-type';
102 scope.entitySubtypeRequiredText = 'device.device-type-required'; 104 scope.entitySubtypeRequiredText = 'device.device-type-required';
  105 + } else if (scope.entityType == types.entityType.entityView) {
  106 + scope.entitySubtypeTitle = 'entity-view.entity-view-type';
  107 + scope.entitySubtypeRequiredText = 'entity-view.entity-view-type-required';
103 } 108 }
104 scope.entitySubtypes.length = 0; 109 scope.entitySubtypes.length = 0;
105 if (scope.entitySubtypesList && scope.entitySubtypesList.length) { 110 if (scope.entitySubtypesList && scope.entitySubtypesList.length) {
@@ -116,6 +121,10 @@ export default function EntitySubtypeSelect($compile, $templateCache, $translate @@ -116,6 +121,10 @@ export default function EntitySubtypeSelect($compile, $templateCache, $translate
116 scope.$on('deviceSaved', function() { 121 scope.$on('deviceSaved', function() {
117 loadSubTypes(); 122 loadSubTypes();
118 }); 123 });
  124 + } else if (scope.entityType == types.entityType.entityView) {
  125 + scope.$on('entityViewSaved', function() {
  126 + loadSubTypes();
  127 + });
119 } 128 }
120 } 129 }
121 } 130 }
@@ -96,6 +96,7 @@ export default angular.module('thingsboard.help', []) @@ -96,6 +96,7 @@ export default angular.module('thingsboard.help', [])
96 customers: helpBaseUrl + "/docs/user-guide/ui/customers", 96 customers: helpBaseUrl + "/docs/user-guide/ui/customers",
97 assets: helpBaseUrl + "/docs/user-guide/ui/assets", 97 assets: helpBaseUrl + "/docs/user-guide/ui/assets",
98 devices: helpBaseUrl + "/docs/user-guide/ui/devices", 98 devices: helpBaseUrl + "/docs/user-guide/ui/devices",
  99 + entityViews: helpBaseUrl + "/docs/user-guide/ui/entity-views",
99 dashboards: helpBaseUrl + "/docs/user-guide/ui/dashboards", 100 dashboards: helpBaseUrl + "/docs/user-guide/ui/dashboards",
100 users: helpBaseUrl + "/docs/user-guide/ui/users", 101 users: helpBaseUrl + "/docs/user-guide/ui/users",
101 widgetsBundles: helpBaseUrl + "/docs/user-guide/ui/widget-library#bundles", 102 widgetsBundles: helpBaseUrl + "/docs/user-guide/ui/widget-library#bundles",
  1 +{
  2 + "access": {
  3 + "unauthorized": "Yetkisiz",
  4 + "unauthorized-access": "Yetkisiz Eriþim",
  5 + "unauthorized-access-text": "Bu kaynaða eriþmek için giriþ yapmalýsýnýz!",
  6 + "access-forbidden": "Eriþim Yasaklandý",
  7 + "access-forbidden-text": "Bu yere eriþim haklarýna sahip deðilsiniz! <br/> Bu konuma eriþmek istiyorsanýz yine de farklý kullanýcýlarla giriþ yapmayý deneyin.",
  8 + "refresh-token-expired": "Oturum süresi doldu",
  9 + "refresh-token-failed": "Seans yenilenemiyor"
  10 + },
  11 + "action": {
  12 + "activate": "Aktive Et",
  13 + "suspend": "Askýya Alma",
  14 + "save": "Kayýt etmek",
  15 + "saveAs": "Farklý kaydet",
  16 + "cancel": "Ýptal etmek",
  17 + "ok": "TAMAM",
  18 + "delete": "Sil",
  19 + "add": "Eklemek",
  20 + "yes": "Evet",
  21 + "no": "Yok hayýr",
  22 + "update": "Güncelleþtirme",
  23 + "remove": "Kaldýr",
  24 + "search": "Arama",
  25 + "clear-search": "Aramayý Temizle",
  26 + "assign": "Atamak",
  27 + "unassign": "Atamayý kaldýrma",
  28 + "share": "Pay",
  29 + "make-private": "Özel yap",
  30 + "apply": "Uygulamak",
  31 + "apply-changes": "Deðiþiklikleri uygula",
  32 + "edit-mode": "Düzenleme modu",
  33 + "enter-edit-mode": "Düzenleme moduna girin",
  34 + "decline-changes": "Deðiþiklikleri reddet",
  35 + "close": "Kapat",
  36 + "back": "Geri",
  37 + "run": "Koþmak",
  38 + "sign-in": "Oturum aç!",
  39 + "edit": "Düzenle",
  40 + "view": "Görünüm",
  41 + "create": "Yaratmak",
  42 + "drag": "Sürüklemek",
  43 + "refresh": "Yenile",
  44 + "undo": "Geri alma",
  45 + "copy": "Kopyala",
  46 + "paste": "Yapýþtýrmak",
  47 + "copy-reference": "Referansý kopyala",
  48 + "paste-reference": "Referansý yapýþtýr",
  49 + "import": "Ýthalat",
  50 + "export": "Ýhracat",
  51 + "share-via": "{{Provider}} aracýlýðýyla paylaþ"
  52 + },
  53 + "aggregation": {
  54 + "aggregation": "Toplama",
  55 + "function": "Veri toplama iþlevi",
  56 + "limit": "Maksimum deðerler",
  57 + "group-interval": "Gruplama aralýðý",
  58 + "min": "Min",
  59 + "max": "Max",
  60 + "avg": "Ortalama",
  61 + "sum": "Sum",
  62 + "count": "Sayý",
  63 + "none": "Yok"
  64 + },
  65 + "admin": {
  66 + "general": "Genel",
  67 + "general-settings": "Genel Ayarlar",
  68 + "outgoing-mail": "Giden posta",
  69 + "outgoing-mail-settings": "Giden Posta Ayarlarý",
  70 + "system-settings": "Sistem ayarlarý",
  71 + "test-mail-sent": "Test postasý baþarýyla gönderildi!",
  72 + "base-url": "Temel URL",
  73 + "base-url-required": "Temel URL gerekli.",
  74 + "mail-from": "Mail þu kiþiden geldi",
  75 + "mail-from-required": "Mail From gereklidir.",
  76 + "smtp-protocol": "SMTP protokolü",
  77 + "smtp-host": "SMTP ana bilgisayarý",
  78 + "smtp-host-required": "SMTP ana bilgisayarý gerekli.",
  79 + "smtp-port": "SMTP portu",
  80 + "smtp-port-required": "Bir smtp portu saðlamalýsýnýz.",
  81 + "smtp-port-invalid": "Bu geçerli bir smtp portuna benzemiyor.",
  82 + "timeout-msec": "Zaman aþýmý (msn)",
  83 + "timeout-required": "Zaman aþýmý gerekli.",
  84 + "timeout-invalid": "Bu geçerli bir zaman aþýmý gibi görünmüyor.",
  85 + "enable-tls": "TLS'yi etkinleþtir",
  86 + "send-test-mail": "Test postasý gönder"
  87 + },
  88 + "alarm": {
  89 + "alarm": "Alarm",
  90 + "alarms": "Alarmlar",
  91 + "select-alarm": "Alarmý seç",
  92 + "no-alarms-matching": " '{{Entity}}' ile eþleþen hiçbir alarm bulunamadý. ",
  93 + "alarm-required": "Alarm gerekli",
  94 + "alarm-status": "Alarm durumu",
  95 + "search-status": {
  96 + "ANY": "Herhangi biri",
  97 + "ACTIVE": "Aktif",
  98 + "CLEARED": "Temizlendi",
  99 + "ACK": "Kabul edilen",
  100 + "UNACK": "Unacknowledged"
  101 + },
  102 + "display-status": {
  103 + "ACTIVE_UNACK": "Aktif Olmamýþ",
  104 + "ACTIVE_ACK": "Aktif Kabul",
  105 + "CLEARED_UNACK": "Onaylanmamýþ Onaylandý",
  106 + "CLEARED_ACK": "Onaylandý onaylandý"
  107 + },
  108 + "no-alarms-prompt": "Alarm bulunamadý",
  109 + "created-time": "Zaman yaratýldý",
  110 + "type": "Tür",
  111 + "severity": "Önem",
  112 + "originator": "Originator",
  113 + "originator-type": "Gönderen türü",
  114 + "details": "Ayrýntýlar",
  115 + "status": "Durum",
  116 + "alarm-details": "Alarm detaylarý",
  117 + "start-time": "Baþlama zamaný",
  118 + "end-time": "Bitiþ zamaný",
  119 + "ack-time": "Onaylanmýþ zaman",
  120 + "clear-time": "Temizlenmiþ zaman",
  121 + "severity-critical": "Kritik",
  122 + "severity-major": "Majör",
  123 + "severity-minor": "Minör",
  124 + "severity-warning": "Uyarý",
  125 + "severity-indeterminate": "Belirsiz",
  126 + "acknowledge": "Onay",
  127 + "clear": "Açýk",
  128 + "search": "Arama alarmlarý",
  129 + "selected-alarms": "{count, çoðul, 1 {1 alarm} diðer {# alarm}} seçildi",
  130 + "no-data": "Gösterilecek bilgi yok",
  131 + "polling-interval": "Alarmlar sorgulama aralýðý (sn)",
  132 + "polling-interval-required": "Alarmlar sorgulama aralýðý gerekli.",
  133 + "min-polling-interval-message": "En az 1 san yoklama aralýðýna izin verilir.",
  134 + "aknowledge-alarms-title": "Kabul et { count, plural, 1 {1 alarm} other {# alarms}} selected ",
  135 + "aknowledge-alarms-text": "{Count, plural, 1 {1 alarm} dother {# alarm}} kimliðini kabul etmek istediðinizden emin misiniz?",
  136 + "clear-alarms-title": "Temizle {count, plural, 1 {1 alarm} other {# alarm}}",
  137 + "clear-alarms-text": "count, plural, 1 {1 alarm} diðer {# alarms}} silmek istediðinizden emin misiniz?"
  138 + },
  139 + "alias": {
  140 + "add": "Takma ad ekle",
  141 + "edit": "Takma adý düzenle",
  142 + "name": "Takma ad",
  143 + "name-required": "Takma ad gerekli",
  144 + "duplicate-alias": "Ayný ada sahip diðer ad zaten var.",
  145 + "filter-type-single-entity": "Tek varlýk",
  146 + "filter-type-entity-list": "Varlýk listesi",
  147 + "filter-type-entity-name": "Varlýk adý",
  148 + "filter-type-state-entity": "Gösterge panosu durumundan varlýk",
  149 + "filter-type-state-entity-description": "Gösterge panosu durum parametrelerinden alýnan varlýk",
  150 + "filter-type-asset-type": "Varlýk türü",
  151 + "filter-type-asset-type-description": "Öðe varlýklarý {{assetType}} '",
  152 + "filter-type-asset-type-and-name-description": "{{AssetType}} türündeki varlýklar ve {{prefix}} ile baþlayan adla ",
  153 + "filter-type-device-type": "Cihaz tipi",
  154 + "filter-type-device-type-description": "Türlü cihazlar {{deviceType}} ",
  155 + "filter-type-device-type-and-name-description": "{{DeviceType}} türündeki cihazlar ve {{prefix}} ile baþlayan adla ",
  156 + "filter-type-relations-query": "Ýliþkiler sorgusu",
  157 + "filter-type-relations-query-description": "{{relationType}} iliþkisine sahip {{direction}} {{rootEntity}} olan {{entities}}",
  158 + "filter-type-asset-search-query": "Öðe arama sorgusu",
  159 + "filter-type-asset-search-query-description": "{{RelationType}} iliþkisine sahip {{direction}}} türlerine sahip öðeler {{direction}} {{rootEntity}} ",
  160 + "filter-type-device-search-query": "Cihaz arama sorgusu",
  161 + "filter-type-device-search-query-description": "{{RelationType}} iliþkisi olan {{direction}} {{rootEntity}} içeren {{deviceTypes}} türüne sahip cihazlar",
  162 + "entity-filter": "Varlýk filtresi",
  163 + "resolve-multiple": "Birden çok varlýk olarak çöz",
  164 + "filter-type": "Filtre tipi",
  165 + "filter-type-required": "Filtre türü gerekli.",
  166 + "entity-filter-no-entity-matched": "Belirtilen filtreyle eþleþen varlýk bulunamadý.",
  167 + "no-entity-filter-specified": "Hiçbir varlýk filtresi belirtilmemiþ",
  168 + "root-state-entity": "Gösterge panosu durum varlýðýný root olarak kullan",
  169 + "root-entity": "Kök varlýk",
  170 + "state-entity-parameter-name": "Durum öðesi parametre adý",
  171 + "default-state-entity": "Varsayýlan durum varlýðý",
  172 + "default-entity-parameter-name": "Varsayýlan olarak",
  173 + "max-relation-level": "Maksimum iliþki seviyesi",
  174 + "unlimited-level": "Sýnýrsýz seviye",
  175 + "state-entity": "Gösterge panosu durumu",
  176 + "all-entities": "Tüm varlýklar",
  177 + "any-relation": "Herhangi"
  178 + },
  179 + "asset": {
  180 + "asset": "Varlýk",
  181 + "assets": "Varlýklar",
  182 + "management": "Varlýk Yönetimi",
  183 + "view-assets": "Varlýklarý Görüntüle",
  184 + "add": "Öðe Ekle",
  185 + "assign-to-customer": "Müþteriye atama",
  186 + "assign-asset-to-customer": "Müþteriye Varlýk (lar) Atama",
  187 + "assign-asset-to-customer-text": "Lütfen müþteriye atamak için varlýklarý seçin",
  188 + "no-assets-text": "Öðe bulunamadý",
  189 + "assign-to-customer-text": "Lütfen varlýk (lar) a atamak için müþteriyi seçin",
  190 + "public": "Halka açýk",
  191 + "assignedToCustomer": "Müþteriye atandý",
  192 + "make-public": "Varlýðý herkese açýk yap",
  193 + "make-private": "Varlýk özel yap",
  194 + "unassign-from-customer": "Müþteriden atama",
  195 + "delete": "Öðeyi sil",
  196 + "asset-public": "Varlýk halka açýk",
  197 + "asset-type": "Varlýk türü",
  198 + "asset-type-required": "Öðe türü gerekli.",
  199 + "select-asset-type": "Varlýk türünü seç",
  200 + "enter-asset-type": "Varlýk türünü girin",
  201 + "any-asset": "Herhangi bir varlýk",
  202 + "no-asset-types-matching": "{{EntitySubtype}} ile eþleþen öðe türü bulunamadý. ",
  203 + "asset-type-list-empty": "Hiçbir öðe türü seçilmedi.",
  204 + "asset-types": "Varlýk türleri",
  205 + "name": "Ad",
  206 + "name-required": "Ýsim gerekli.",
  207 + "description": "Açýklama",
  208 + "type": "Tür",
  209 + "type-required": "Tür gerekli.",
  210 + "details": "Ayrýntýlar",
  211 + "events": "Etkinlikler",
  212 + "add-asset-text": "Yeni varlýk ekle",
  213 + "asset-details": "Varlýk ayrýntýlarý",
  214 + "assign-assets": "Öðeleri atama",
  215 + "assign-assets-text": "Müþteriye {count, çoðul, 1 {1 asset}} diðer {# asset}} atayýn ",
  216 + "delete-assets": "Öðeleri sil",
  217 + "unassign-assets": "Varlýklarý atama",
  218 + "unassign-assets-action-title": "Müþteriden tekil, sayým, çoðul, 1 {1 öðe} diðer {# varlýk}} atama ",
  219 + "assign-new-asset": "Yeni varlýk atama",
  220 + "delete-asset-title": "Öðeyi {{assetName}} silmek istediðinizden emin misiniz?",
  221 + "delete-asset-text": "Dikkatli olun, onaylandýktan sonra varlýk ve ilgili tüm veriler kurtarýlamaz.",
  222 + "delete-assets-title": "{Count, çoðul, 1 {1 asset} diðer {# asset}} silmek istediðinizden emin misiniz?",
  223 + "delete-assets-action-title": "Sil {count, çoðul, 1 {1 asset} diðer {# asset}}",
  224 + "delete-assets-text": "Dikkatli olun, onaylandýktan sonra tüm seçilmiþ varlýklar kaldýrýlacak ve ilgili tüm veriler kurtarýlamayacaktýr.",
  225 + "make-public-asset-title": "Varlýðý {{assetName}} herkese açýk yapmak istediðinizden emin misiniz? ",
  226 + "make-public-asset-text": "Onaydan sonra varlýk ve tüm verileri kamuya açýk ve baþkalarý tarafýndan eriþilebilir olacak.",
  227 + "make-private-asset-title": "Öðeyi {{assetName}} özel yapmak istediðinizden emin misiniz? ",
  228 + "make-private-asset-text": "Onaydan sonra varlýk ve tüm verileri gizli tutulacak ve baþkalarý tarafýndan eriþilemeyecektir.",
  229 + "unassign-asset-title": "Öðeyi {{assetName}} öðesinin atamasýný kaldýrmak istediðinizden emin misiniz?",
  230 + "unassign-asset-text": "Onaydan sonra varlýk atama yapýlmayacak ve müþteri tarafýndan eriþilemeyecektir.",
  231 + "unassign-asset": "Atama atamayý kaldýr",
  232 + "unassign-assets-title": "count, plural, 1 {1 öðe} other {# asset}} atama atamak istediðinizden emin misiniz?",
  233 + "unassign-assets-text": "Onaylandýktan sonra, seçilen tüm öðeler atamadan kaldýrýlacak ve müþteri tarafýndan eriþilemeyecektir.",
  234 + "copyId": "Öðe Kimliði Kopyala",
  235 + "idCopiedMessage": "Öðe Kimliði panoya kopyalandý",
  236 + "select-asset": "Varlýk seç",
  237 + "no-assets-matching": "{{Entity}} ile eþleþen hiçbir öðe bulunamadý. ",
  238 + "asset-required": "Varlýk gerekli",
  239 + "name-starts-with": "Öðe adý ile baþlar"
  240 + },
  241 + "attribute": {
  242 + "attributes": "Öznitellikler",
  243 + "latest-telemetry": "Son telemetri",
  244 + "attributes-scope": "Varlýk öznitelikleri kapsamý",
  245 + "scope-latest-telemetry": "Son telemetri",
  246 + "scope-client": "Müþteri özellikleri",
  247 + "scope-server": "Sunucu öznitelikleri",
  248 + "scope-shared": "Paylaþýlan özellikler",
  249 + "add": "Özellik ekle",
  250 + "key": "Anahtar",
  251 + "last-update-time": "Son güncelleme zamaný",
  252 + "key-required": "Özellik anahtarý gerekli.",
  253 + "value": "Deðer",
  254 + "value-required": "Özellik deðeri gerekli.",
  255 + "delete-attributes-title": "{Count, çoðul, 1 {1 özellik} diðer {# attributes}} silmek istediðinizden emin misiniz?",
  256 + "delete-attributes-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen özellikler kaldýrýlacak.",
  257 + "delete-attributes": "Nitelikleri sil",
  258 + "enter-attribute-value": "Özellik deðerini girin",
  259 + "show-on-widget": "Widget’ta göster",
  260 + "widget-mode": "Widget modu",
  261 + "next-widget": "Sonraki pencere öðesi",
  262 + "prev-widget": "Önceki pencere öðesi",
  263 + "add-to-dashboard": "Gösterge tablosuna ekle",
  264 + "add-widget-to-dashboard": "Gösterge tablosuna widget ekle",
  265 + "selected-attributes": "{count, çoðul, 1 {1 özellik} diðer {# attributes}} seçildi",
  266 + "selected-telemetry": "{count, çoðul, 1 {1 telemetri birimi} diðer {# telemetri birimleri}} seçildi"
  267 + },
  268 + "audit-log": {
  269 + "audit": "Denetim",
  270 + "audit-logs": "Denetim Kayýtlarý",
  271 + "timestamp": "Zaman Damgasý",
  272 + "entity-type": "Varlýk Türü",
  273 + "entity-name": "Varlýk adý",
  274 + "user": "Kullanýcý",
  275 + "type": "Tür",
  276 + "status": "Durum",
  277 + "details": "Ayrýntýlar",
  278 + "type-added": "Katma",
  279 + "type-deleted": "Silindi",
  280 + "type-updated": "Güncellenmiþ",
  281 + "type-attributes-updated": "Öznitelikler güncellendi",
  282 + "type-attributes-deleted": "Öznitelikler silindi",
  283 + "type-rpc-call": "RPC çaðrýsý",
  284 + "type-credentials-updated": "Kimlik bilgileri güncellendi",
  285 + "type-assigned-to-customer": "Müþteriye Atanan",
  286 + "type-unassigned-from-customer": "Müþteriden Atanmamýþ",
  287 + "type-activated": "Aktif",
  288 + "type-suspended": "Askýya alýndý",
  289 + "type-credentials-read": "Kimlik bilgileri okundu",
  290 + "type-attributes-read": "Nitelikler oku",
  291 + "type-relation-add-or-update": "Ýliþki güncellendi",
  292 + "type-relation-delete": "Ýliþki silindi",
  293 + "type-relations-delete": "Tüm iliþki silindi",
  294 + "type-alarm-ack": "Kabul edilen",
  295 + "type-alarm-clear": "Temizlendi",
  296 + "status-success": "Baþarý",
  297 + "status-failure": "Baþarýsýzlýk",
  298 + "audit-log-details": "Denetim kaydý ayrýntýlarý",
  299 + "no-audit-logs-prompt": "Kayýt bulunamadý",
  300 + "action-data": "Eylem verileri",
  301 + "failure-details": "Arýza detaylarý",
  302 + "search": "Denetim günlüklerini ara",
  303 + "clear-search": "Aramayý Temizle"
  304 + },
  305 + "confirm-on-exit": {
  306 + "message": "Kaydedilmemiþ deðiþiklikleriniz var. Bu sayfadan ayrýlmak istediðinizden emin misiniz?",
  307 + "html-message": "Kaydedilmemiþ deðiþiklikleriniz var. <br/> Bu sayfadan ayrýlmak istediðinizden emin misiniz?",
  308 + "title": "Kaydedilmemiþ deðiþiklikler"
  309 + },
  310 + "contact": {
  311 + "country": "Ülke",
  312 + "city": "Kent",
  313 + "state": "Eyalet / Ýl",
  314 + "postal-code": "Posta Kodu",
  315 + "postal-code-invalid": "Geçersiz Posta Kodu / Posta Kodu biçimi.",
  316 + "address": "Adres",
  317 + "address2": "Adres 2",
  318 + "phone": "Telefon",
  319 + "email": "E-posta",
  320 + "no-address": "Adres yok"
  321 + },
  322 + "common": {
  323 + "username": "Kullanýcý adý",
  324 + "password": "Parola",
  325 + "enter-username": "Kullanýcý adý girin",
  326 + "enter-password": "Parolaný Gir",
  327 + "enter-search": "Arama girin"
  328 + },
  329 + "content-type": {
  330 + "json": "Json",
  331 + "text": "Metin",
  332 + "binary": "Ýkili (Base64)"
  333 + },
  334 + "customer": {
  335 + "customer": "Müþteri",
  336 + "customers": "Müþteriler",
  337 + "management": "Müþteri yönetimi",
  338 + "dashboard": "Müþteri Kontrol Paneli",
  339 + "dashboards": "Müþteri Kontrol Panelleri",
  340 + "devices": "Müþteri Cihazlarý",
  341 + "entity-views": "Müþteri Varlýðý Görüntüleme Sayýsý",
  342 + "assets": "Müþteri Varlýklarý",
  343 + "public-dashboards": "Genel Panolar",
  344 + "public-devices": "Kamu Aygýtlarý",
  345 + "public-assets": "Kamu Varlýklarý",
  346 + "public-entity-views": "Kamu Varlýk Görüntüleme Sayýsý",
  347 + "add": "Müþteri Ekle",
  348 + "delete": "Müþteriyi sil",
  349 + "manage-customer-users": "Müþteri kullanýcýlarýný yönet",
  350 + "manage-customer-devices": "Müþteri cihazlarýný yönet",
  351 + "manage-customer-dashboards": "Müþteri panolarýný yönet",
  352 + "manage-public-devices": "Genel cihazlarý yönet",
  353 + "manage-public-dashboards": "Genel panolarý yönet",
  354 + "manage-customer-assets": "Müþteri varlýklarýný yönet",
  355 + "manage-public-assets": "Kamu varlýklarýný yönet",
  356 + "add-customer-text": "Yeni müþteri ekle",
  357 + "no-customers-text": "Müþteri bulunamadý",
  358 + "customer-details": "Müþteri detaylarý",
  359 + "delete-customer-title": "Müþteriyi silmek istediðinizden emin misiniz? {{CustomerTitle}} ? ",
  360 + "delete-customer-text": "Dikkatli olun, onaylandýktan sonra müþteri ve ilgili tüm veriler kurtarýlamaz.",
  361 + "delete-customers-title": "{Count, çoðul, 1 {1 müþteri} diðer {# customers}} silmek istediðinizden emin misiniz?",
  362 + "delete-customers-action-title": "Sil {count, çoðul, 1 {1 müþteri} diðer {# customers}}",
  363 + "delete-customers-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen müþteriler kaldýrýlacak ve ilgili tüm veriler kurtarýlamayacaktýr.",
  364 + "manage-users": "Kullanýcýlarý Yönet",
  365 + "manage-assets": "Varlýklarý yönet",
  366 + "manage-devices": "Cihazlarý yönet",
  367 + "manage-dashboards": "Gösterge tablolarýný yönet",
  368 + "title": "Baþlýk",
  369 + "title-required": "Baþlýk gerekli.",
  370 + "description": "Açýklama",
  371 + "details": "Ayrýntýlar",
  372 + "events": "Etkinlikler",
  373 + "copyId": "Müþteri kimliðini kopyala",
  374 + "idCopiedMessage": "Müþteri Kimliði panoya kopyalandý",
  375 + "select-customer": "Müþteriyi seç",
  376 + "no-customers-matching": "{{Entity}} ile eþleþen hiçbir müþteri bulunamadý. ",
  377 + "customer-required": "Müþteri gerekli",
  378 + "select-default-customer": "Varsayýlan müþteriyi seç",
  379 + "default-customer": "Varsayýlan müþteri",
  380 + "default-customer-required": "Kiracý düzeyinde gösterge tablosunda hata ayýklamak için varsayýlan müþteri gerekiyor"
  381 + },
  382 + "datetime": {
  383 + "date-from": "Tarih",
  384 + "time-from": "Zaman",
  385 + "date-to": "Tarih",
  386 + "time-to": "Zaman"
  387 + },
  388 + "dashboard": {
  389 + "dashboard": "Pano",
  390 + "dashboards": "Gösterge tablolarý",
  391 + "management": "Gösterge tablosu yönetimi",
  392 + "view-dashboards": "Gösterge Panolarý",
  393 + "add": "Gösterge Tablosu Ekle",
  394 + "assign-dashboard-to-customer": "Gösterge Tablosunu / Müþterilerini Müþteriye Atama",
  395 + "assign-dashboard-to-customer-text": "Lütfen müþteriye atamak için kontrol panellerini seçin",
  396 + "assign-to-customer-text": "Gösterge panellerini atamak için lütfen müþteriyi seçiniz",
  397 + "assign-to-customer": "Müþteriye atama",
  398 + "unassign-from-customer": "Müþteriden atama",
  399 + "make-public": "Gösterge panosunu herkese açýk yap",
  400 + "make-private": "Gösterge panosunu özel yap",
  401 + "manage-assigned-customers": "Atanan müþterileri yönet",
  402 + "assigned-customers": "Atanan müþteriler",
  403 + "assign-to-customers": "Gösterge Tablosunu / Müþterilerini Müþterilere Atama",
  404 + "assign-to-customers-text": "Lütfen gösterge panosunu atamak için müþterileri seçin",
  405 + "unassign-from-customers": "Müþterilerden Gösterge Tablosunu (Notlarýný) Atama",
  406 + "unassign-from-customers-text": "Lütfen gösterge tablosundan atamak için müþterileri seçin",
  407 + "no-dashboards-text": "Gösterge panelleri bulunamadý",
  408 + "no-widgets": "Hiçbir widget yapýlandýrýlmamýþ",
  409 + "add-widget": "Yeni pencere öðesi ekle",
  410 + "title": "Baþlýk",
  411 + "select-widget-title": "Widget seç",
  412 + "select-widget-subtitle": "Kullanýlabilir pencere öðesi türleri",
  413 + "delete": "Gösterge tablosunu sil",
  414 + "title-required": "Baþlýk gerekli.",
  415 + "description": "Açýklama",
  416 + "details": "Ayrýntýlar",
  417 + "dashboard-details": "Gösterge tablosu ayrýntýlarý",
  418 + "add-dashboard-text": "Yeni gösterge tablosu ekle",
  419 + "assign-dashboards": "Gösterge tablolarý atama",
  420 + "assign-new-dashboard": "Yeni kontrol paneli atama",
  421 + "assign-dashboards-text": "Müþterilere {count, plural, 1 {1 gösterge tablosu} other {# dashboard}}} atayýn ",
  422 + "unassign-dashboards-action-text": "Müþterilerden atama sayýsý, çoðul, 1 {1 gösterge tablosu} diðer {# panolar}}",
  423 + "delete-dashboards": "Gösterge tablolarýný sil",
  424 + "unassign-dashboards": "Atanmamýþ gösterge panolarý",
  425 + "unassign-dashboards-action-title": "Müþteriden atama sayým, çoðul, 1 {1 gösterge tablosu} diðer {# pano}}",
  426 + "delete-dashboard-title": "'{{DashboardTitle}}' gösterge panosunu silmek istediðinizden emin misiniz?",
  427 + "delete-dashboard-text": "Doðrulamadan sonra, kontrol panelinden ve ilgili tüm veriler kurtarýlamayacak.",
  428 + "delete-dashboards-title": "{Count, çoðul, 1 {1 dashboard} diðer {# dashboards}} silmek istediðinizden emin misiniz?",
  429 + "delete-dashboards-action-title": "Sil {count, çoðul, 1 {1 gösterge tablosu} diðer {# dashboard}}",
  430 + "delete-dashboards-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen panolar silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  431 + "unassign-dashboard-title": "{{DashboardTitle}} gösterge tablosunun atamasýný kaldýrmak istediðinizden emin misiniz? ",
  432 + "unassign-dashboard-text": "Onaydan sonra gösterge panosu atanmamýþ olacak ve müþteri tarafýndan eriþilemeyecektir.",
  433 + "unassign-dashboard": "Atanmamýþ gösterge panosu",
  434 + "unassign-dashboards-title": "Sayým, çoðul, 1 {1 gösterge tablosu} diðer {# panolar}} atama atamak istediðinizden emin misiniz?",
  435 + "unassign-dashboards-text": "Onaydan sonra tüm seçilen gösterge tablolarý atanmamýþ olacak ve müþteri tarafýndan eriþilemeyecektir.",
  436 + "public-dashboard-title": "Gösterge panosu artýk herkese açýk",
  437 + "public-dashboard-text": "Gösterge tablonuz <b> {{dashboardTitle}} </ b> artýk herkese açýk ve bir sonraki herkese açýk <a href='{{publicLink}} 'target='_blank'> link </a> üzerinden eriþilebilir:",
  438 + "public-dashboard-notice": "<b> Not: </ b> Verilerine eriþmek için ilgili cihazlarý herkese açýk yapmayý unutmayýn.",
  439 + "make-private-dashboard-title": "Gösterge panosunu {{dashboardTitle}} özel yapmak istediðinizden emin misiniz?",
  440 + "make-private-dashboard-text": "Onaydan sonra gösterge panosu özel hale getirilecek ve baþkalarý tarafýndan eriþilemeyecektir.",
  441 + "make-private-dashboard": "Gösterge panosunu özel yap",
  442 + "socialshare-text": "'{{dashboardTitle}} ThingsBoard tarafýndan desteklenmektedir ",
  443 + "socialshare-title": "'{{dashboardTitle}} ThingsBoard tarafýndan desteklenmektedir ",
  444 + "select-dashboard": "Gösterge tablosu seç",
  445 + "no-dashboards-matching": "{{Entity}} ile eþleþen hiçbir gösterge paneli bulunamadý. ",
  446 + "dashboard-required": "Gösterge tablosu gerekli.",
  447 + "select-existing": "Mevcut kontrol panelini seç",
  448 + "create-new": "Yeni gösterge tablosu oluþtur",
  449 + "new-dashboard-title": "Yeni kontrol paneli baþlýðý",
  450 + "open-dashboard": "Açýk kontrol paneli",
  451 + "set-background": "Arka planý ayarla",
  452 + "background-color": "Arka plan rengi",
  453 + "background-image": "Arka plan görüntüsü",
  454 + "background-size-mode": "Arka plan boyut modu",
  455 + "no-image": "Görüntü seçilmedi",
  456 + "drop-image": "Bir resim býrakýn veya yüklenecek bir dosya seçmek için týklayýn.",
  457 + "settings": "Ayarlar",
  458 + "columns-count": "Sütun sayýsý",
  459 + "columns-count-required": "Sütun sayýsý gerekli.",
  460 + "min-columns-count-message": "Sadece 10 minimum sütun sayýsýna izin verilir.",
  461 + "max-columns-count-message": "Sadece maksimum 1000 sütun sayýsýna izin verilir.",
  462 + "widgets-margins": "Widget arasýnda marj",
  463 + "horizontal-margin": "Yatay kenar boþluðu",
  464 + "horizontal-margin-required": "Yatay kenar boþluðu deðeri gerekiyor.",
  465 + "min-horizontal-margin-message": "Minimum yatay kenar boþluðu deðeri olarak yalnýzca 0'a izin verilir.",
  466 + "max-horizontal-margin-message": "Maksimum yatay kenar boþluðu deðeri olarak yalnýzca 50'ye izin verilir.",
  467 + "vertical-margin": "Dikey kenar boþluðu",
  468 + "vertical-margin-required": "Dikey kenar boþluðu deðeri gereklidir.",
  469 + "min-vertical-margin-message": "En az dikey kenar boþluðu deðeri olarak yalnýzca 0'a izin verilir.",
  470 + "max-vertical-margin-message": "Maksimum dikey kenar boþluðu deðeri olarak yalnýzca 50'ye izin verilir.",
  471 + "autofill-height": "Otomatik dolgu düzeni yüksekliði",
  472 + "mobile-layout": "Mobil düzen ayarlarý",
  473 + "mobile-row-height": "Mobil satýr yüksekliði, px",
  474 + "mobile-row-height-required": "Mobil satýr yüksekliði deðeri gerekli.",
  475 + "min-mobile-row-height-message": "Minimum mobil satýr yüksekliði deðeri olarak yalnýzca 5 piksele izin verilir.",
  476 + "max-mobile-row-height-message": "Maksimum mobil satýr yüksekliði deðeri olarak yalnýzca 200 piksele izin verilir.",
  477 + "display-title": "Gösterge tablosu baþlýðý",
  478 + "toolbar-always-open": "Araç çubuðunu açýk tut",
  479 + "title-color": "Baþlýk rengi",
  480 + "display-dashboards-selection": "Gösterge panolarý seçimi",
  481 + "display-entities-selection": "Öðe varlýklarý seçimi",
  482 + "display-dashboard-timewindow": "Zaman penceresi göster",
  483 + "display-dashboard-export": "Görüntülü dýþa aktarma",
  484 + "import": "Gösterge paneli",
  485 + "export": "Dýþ gösterge panosu",
  486 + "export-failed-error": "Gösterge tablosu gönderilemiyor: {{error}}",
  487 + "create-new-dashboard": "Yeni gösterge tablosu oluþtur",
  488 + "dashboard-file": "Gösterge tablosu dosyasý",
  489 + "invalid-dashboard-file-error": "Gösterge tablosu alýnamadý: Geçersiz pano veri yapýsý.",
  490 + "dashboard-import-missing-aliases-title": "Alýnan pano tarafýndan kullanýlan takma adlarý yapýlandýr",
  491 + "create-new-widget": "Yeni pencere öðesi oluþtur",
  492 + "import-widget": "Widget'ý içe aktar",
  493 + "widget-file": "Widget dosyasý",
  494 + "invalid-widget-file-error": "Widget içe aktarýlamýyor: Geçersiz pencere öðesi yapýsý.",
  495 + "widget-import-missing-aliases-title": "Ýçe aktarýlan pencere aracý tarafýndan kullanýlan takma adlarý yapýlandýr",
  496 + "open-toolbar": "Gösterge tablosu araç çubuðunu aç",
  497 + "close-toolbar": "Araç çubuðunu kapat",
  498 + "configuration-error": "Yapýlandýrma hatasý",
  499 + "alias-resolution-error-title": "Gösterge tablosu takma ad hatasý",
  500 + "invalid-aliases-config": "Diðer ad filtrelerinin biriyle eþleþen herhangi bir cihaz bulunamadý. <br/> Lütfen bu sorunu çözmek için yöneticinize baþvurun.",
  501 + "select-devices": "Cihaz seç",
  502 + "assignedToCustomer": "Müþteriye atandý",
  503 + "assignedToCustomers": "Müþterilere atandý",
  504 + "public": "Halka açýk",
  505 + "public-link": "Genel baðlantý",
  506 + "copy-public-link": "Genel baðlantýyý kopyala",
  507 + "public-link-copied-message": "Pano genel baðlantýsý panoya kopyalandý",
  508 + "manage-states": "Gösterge panosu durumlarýný yönet",
  509 + "states": "Gösterge durumu",
  510 + "search-states": "Gösterge panosu durumlarý",
  511 + "selected-states": "{count, çoðul, 1 {1 gösterge tablosu durumu} diðer {# dashboard durumlarý}} seçildi",
  512 + "edit-state": "Gösterge panosu durumunu düzenle",
  513 + "delete-state": "Gösterge panosu durumunu sil",
  514 + "add-state": "Gösterge paneli durumu",
  515 + "state": "Gösterge paneli durumu",
  516 + "state-name": "Ad",
  517 + "state-name-required": "Gösterge panosu durum adý gerekli.",
  518 + "state-id": "Durum Kimliði",
  519 + "state-id-required": "Gösterge durumu kimliði gerekiyor.",
  520 + "state-id-exists": "Ayný kimliðe sahip gösterge tablosu zaten var.",
  521 + "is-root-state": "Kök devlet",
  522 + "delete-state-title": "Gösterge panosu durumunu sil",
  523 + "delete-state-text": "Gösterge panosu durumunu '{{stateName}}' adýyla silmek istediðinizden emin misiniz?",
  524 + "show-details": "Detaylarý göster",
  525 + "hide-details": "Detaylarý gizle",
  526 + "select-state": "Hedef durumu seç",
  527 + "state-controller": "Durum kontrolörü"
  528 + },
  529 + "datakey": {
  530 + "settings": "Ayarlar",
  531 + "advanced": "Ýleri",
  532 + "label": "Etiket",
  533 + "color": "Renk",
  534 + "units": "Deðerin yanýnda gösterilecek özel sembol",
  535 + "decimals": "Kayan noktadan sonraki basamak sayýsý",
  536 + "data-generation-func": "Veri oluþturma iþlevi",
  537 + "use-data-post-processing-func": "Veri iþleme sonrasý iþlevini kullan",
  538 + "configuration": "Veri anahtarý yapýlandýrmasý",
  539 + "timeseries": "Zaman serisi",
  540 + "attributes": "Öznitellikler",
  541 + "alarm": "Alarm alanlarý",
  542 + "timeseries-required": "Varlýk zamanlamalarý gerekli.",
  543 + "timeseries-or-attributes-required": "Varlýk zaman çizelgeleri / öznitelikler gereklidir.",
  544 + "maximum-timeseries-or-attributes": "Maksimum {count, çoðul, 1 {1 timeseries / attribute.} Diðer {# timeseries / attributes {} izin verilir}}",
  545 + "alarm-fields-required": "Alarm alanlarý gerekli.",
  546 + "function-types": "Ýþlev türleri",
  547 + "function-types-required": "Ýþlev tipleri gereklidir.",
  548 + "maximum-function-types": "Maksimum {sayým, çoðul, 1 {1 iþlev türüne izin verilir.} Diðer {# iþlev türlerine izin verilir}}"
  549 + },
  550 + "datasource": {
  551 + "type": "Veri kaynaðý türü",
  552 + "name": "Ad",
  553 + "add-datasource-prompt": "Lütfen veri kaynaðý ekle"
  554 + },
  555 + "details": {
  556 + "edit-mode": "Düzenleme modu",
  557 + "toggle-edit-mode": "Düzenleme modunu deðiþtir"
  558 + },
  559 + "device": {
  560 + "device": "Cihaz",
  561 + "device-required": "Cihaz gerekli.",
  562 + "devices": "Cihazlar",
  563 + "management": "Cihaz yönetimi",
  564 + "view-devices": "Cihazlarý Görüntüle",
  565 + "device-alias": "Cihaz takma adý",
  566 + "aliases": "Cihaz takma adlarý",
  567 + "no-alias-matching": "'{{alias}} bulunamadý. ",
  568 + "no-aliases-found": "Takma ad bulunamadý",
  569 + "no-key-matching": "'{{anahtar bulunamadý.",
  570 + "no-keys-found": "Anahtar bulunamadý.",
  571 + "create-new-alias": "Yeni bir tane oluþtur!",
  572 + "create-new-key": "Yeni bir tane oluþtur!",
  573 + "duplicate-alias-error": "Yinelenen takma ad bulundu {{alias}}.. <br> Cihaz takma adlarý, kontrol panelinde benzersiz olmalýdýr. ",
  574 + "configure-alias": "Yapýlandýrma {{alias}} takma ad",
  575 + "no-devices-matching": "{{Entity}} ile eþleþen hiçbir cihaz bulunamadý. ",
  576 + "alias": "Alias",
  577 + "alias-required": "Cihaz takma adý gerekiyor.",
  578 + "remove-alias": "Cihaz takma adýný kaldýr",
  579 + "add-alias": "Cihaz takma adý ekle",
  580 + "name-starts-with": "Cihaz adý ile baþlýyor",
  581 + "device-list": "Aygýt listesi",
  582 + "use-device-name-filter": "Filtre kullan",
  583 + "device-list-empty": "Cihaz seçilmedi.",
  584 + "device-name-filter-required": "Cihaz adý filtresi gerekli.",
  585 + "device-name-filter-no-device-matched": "{{Device}} ile baþlayan hiçbir cihaz bulunamadý. ",
  586 + "add": "Cihaz ekle",
  587 + "assign-to-customer": "Müþteriye atama",
  588 + "assign-device-to-customer": "Aygýtý / Aygýtlarý Müþteriye Atama",
  589 + "assign-device-to-customer-text": "Lütfen müþteriye atamak istediðiniz cihazlarý seçin",
  590 + "make-public": "Cihazý herkese açýk yap",
  591 + "make-private": "Cihazý özel yap",
  592 + "no-devices-text": "Hiçbir cihaz bulunamadý",
  593 + "assign-to-customer-text": "Lütfen cihazý atamak için müþteriyi seçin",
  594 + "device-details": "Cihaz detaylarý",
  595 + "add-device-text": "Yeni cihaz ekle",
  596 + "credentials": "Kimlik bilgileri",
  597 + "manage-credentials": "Kimlik bilgilerini yönet",
  598 + "delete": "Cihazý sil",
  599 + "assign-devices": "Aygýtlarý atama",
  600 + "assign-devices-text": "Müþteriye {count, çoðul, 1 {1 cihaz} diðer {# devices}} atayýn ",
  601 + "delete-devices": "Cihazlarý sil",
  602 + "unassign-from-customer": "Müþteriden atama",
  603 + "unassign-devices": "Atanmamýþ cihazlarý",
  604 + "unassign-devices-action-title": "Müþteriden atayýn, çoðul, 1 {1 cihaz} diðer {# devices}} atama ",
  605 + "assign-new-device": "Yeni cihaz atama",
  606 + "make-public-device-title": "Cihazý {{deviceName}} herkese açýk yapmak istediðinizden emin misiniz?",
  607 + "make-public-device-text": "Onaydan sonra cihaz ve tüm verileri kamuya açýk ve baþkalarý tarafýndan eriþilebilir olacak.",
  608 + "make-private-device-title": "Cihazý {{deviceName}} özel yapmak istediðinizden emin misiniz?",
  609 + "make-private-device-text": "Onaylandýktan sonra cihaz ve tüm verileri gizli tutulacak ve baþkalarý tarafýndan eriþilemeyecektir.",
  610 + "view-credentials": "Kimlik bilgilerini görüntüle",
  611 + "delete-device-title": "Cihazý {{deviceName}} silmek istediðinizden emin misiniz? ",
  612 + "delete-device-text": "Dikkatli olun, onaylamadan sonra cihaz ve tüm ilgili veriler kurtarýlamaz.",
  613 + "delete-devices-title": "{Count, çoðul, 1 {1 cihaz} diðer {# devices}} silmek istediðinizden emin misiniz?",
  614 + "delete-devices-action-title": "Sil {count, çoðul, 1 {1 cihaz} diðer {# devices}}",
  615 + "delete-devices-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen cihazlar silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  616 + "unassign-device-title": "Cihazýn atamasýný kaldýrmak istediðinizden emin misiniz? {{DeviceName}} ? ",
  617 + "unassign-device-text": "Onaydan sonra cihaz atanmamýþ olacak ve müþteri tarafýndan eriþilemeyecektir.",
  618 + "unassign-device": "Atanmamýþ cihaz",
  619 + "unassign-devices-title": "Sayým, çoðul, 1 {1 cihaz} diðer {# devices}} atama atamak istediðinizden emin misiniz?",
  620 + "unassign-devices-text": "Onaylandýktan sonra tüm seçilen cihazlar atanmamýþ olacak ve müþteri tarafýndan eriþilemeyecektir.",
  621 + "device-credentials": "Cihaz Kimlik Bilgileri",
  622 + "credentials-type": "Kimlik bilgileri türü",
  623 + "access-token": "Eriþim belirteci",
  624 + "access-token-required": "Eriþim belirteci gerekli.",
  625 + "access-token-invalid": "Eriþim belirteci uzunluðu 1 ile 20 karakter arasýnda olmalýdýr.",
  626 + "rsa-key": "RSA ortak anahtarý",
  627 + "rsa-key-required": "RSA ortak anahtarý gerekli.",
  628 + "secret": "Gizli",
  629 + "secret-required": "Gizli gerekli",
  630 + "device-type": "Cihaz tipi",
  631 + "device-type-required": "Cihaz tipi gerekli.",
  632 + "select-device-type": "Cihaz türünü seç",
  633 + "enter-device-type": "Cihaz türünü girin",
  634 + "any-device": "Herhangi bir cihaz",
  635 + "no-device-types-matching": "{{EntitySubtype}} ile eþleþen cihaz türü bulunamadý. ",
  636 + "device-type-list-empty": "Hiçbir cihaz türü seçilmedi.",
  637 + "device-types": "Cihaz türleri",
  638 + "name": "Ad",
  639 + "name-required": "Ýsim gerekli.",
  640 + "description": "Açýklama",
  641 + "events": "Etkinlikler",
  642 + "details": "Ayrýntýlar",
  643 + "copyId": "Cihaz kimliðini kopyala",
  644 + "copyAccessToken": "Eriþim belirteci kopyala",
  645 + "idCopiedMessage": "Cihaz Kimliði panoya kopyalandý",
  646 + "accessTokenCopiedMessage": "Cihaz eriþim belirteci panoya kopyalandý",
  647 + "assignedToCustomer": "Müþteriye atandý",
  648 + "unable-delete-device-alias-title": "Cihaz takma adý silinemiyor",
  649 + "unable-delete-device-alias-text": "Cihaz takma adý {{deviceAlias}} ', þu widget (lar) tarafýndan kullanýldýðý þekliyle silinemiyor: <br/> {{widgetsList}} ",
  650 + "is-gateway": "Að geçidi",
  651 + "public": "Halka açýk",
  652 + "device-public": "Cihaz herkese açýk",
  653 + "select-device": "Cihaz seç"
  654 + },
  655 + "dialog": {
  656 + "close": "Ýletiþim kutusunu kapat"
  657 + },
  658 + "error": {
  659 + "unable-to-connect": "Sunucuya baðlanýlamýyor! Lütfen Ýnternet baðlantýnýzý kontrol edin.",
  660 + "unhandled-error-code": "Ýþlenmemiþ hata kodu: {{errorCode}}",
  661 + "unknown-error": "Bilinmeyen hata"
  662 + },
  663 + "entity": {
  664 + "entity": "Varlýk",
  665 + "entities": "Varlýklarý",
  666 + "aliases": "Varlýk takma adlarý",
  667 + "entity-alias": "Varlýk takma adý",
  668 + "unable-delete-entity-alias-title": "Varlýk takma adý silinemiyor",
  669 + "unable-delete-entity-alias-text": "Varlýk takma adý {{entityAlias}} ', þu widget (lar) tarafýndan kullanýldýðý þekliyle silinemez: <br/> {{widgetsList}} ",
  670 + "duplicate-alias-error": "Yinelenen takma ad bulundu {{alias}} '... Entity takma adlar, gösterge panosunda benzersiz olmalýdýr. ",
  671 + "missing-entity-filter-error": "Diðer adlar için filtre eksik {{alias}}. ",
  672 + "configure-alias": "Yapýlandýrma {{alias}} takma ad",
  673 + "alias": "Alias",
  674 + "alias-required": "Varlýk takma adý gerekiyor.",
  675 + "remove-alias": "Varlýk takma adýný kaldýrma",
  676 + "add-alias": "Varlýk takma adý ekle",
  677 + "entity-list": "Varlýk listesi",
  678 + "entity-type": "Varlýk türü",
  679 + "entity-types": "Varlýk türleri",
  680 + "entity-type-list": "Varlýk türü listesi",
  681 + "any-entity": "Herhangi bir varlýk",
  682 + "enter-entity-type": "Varlýk türü girin",
  683 + "no-entities-matching": "{{Entity}} ile eþleþen hiçbir varlýk bulunamadý. ",
  684 + "no-entity-types-matching": "{{EntityType}} ile eþleþen hiçbir varlýk türü bulunamadý. ",
  685 + "name-starts-with": "Ýsim ile baþlar",
  686 + "use-entity-name-filter": "Filtre kullan",
  687 + "entity-list-empty": "Hiçbir varlýk seçilmedi.",
  688 + "entity-type-list-empty": "Hiçbir varlýk türü seçilmedi.",
  689 + "entity-name-filter-required": "Varlýk adý filtresi gerekli.",
  690 + "entity-name-filter-no-entity-matched": "{{Entity}} ile baþlayan hiçbir varlýk bulunamadý. ",
  691 + "all-subtypes": "Herþey",
  692 + "select-entities": "Öðeleri seç",
  693 + "no-aliases-found": "Takma ad bulunamadý",
  694 + "no-alias-matching": "'{{alias}} bulunamadý. ",
  695 + "create-new-alias": "Yeni bir tane oluþtur!",
  696 + "key": "Anahtar",
  697 + "key-name": "Anahtar adý",
  698 + "no-keys-found": "Anahtar bulunamadý.",
  699 + "no-key-matching": "'{{anahtar bulunamadý.",
  700 + "create-new-key": "Yeni bir tane oluþtur!",
  701 + "type": "Tür",
  702 + "type-required": "Varlýk türü gerekli.",
  703 + "type-device": "Cihaz",
  704 + "type-devices": "Cihazlar",
  705 + "list-of-devices": "{sayým, çoðul, 1 {Bir cihaz} diðer {# cihazlarýn listesi}}",
  706 + "device-name-starts-with": "Ýsimleri '{{prefix}} ile baþlayan cihazlar ",
  707 + "type-asset": "Varlýk",
  708 + "type-assets": "Varlýklar",
  709 + "list-of-assets": "{count, plural, 1 {One asset} diðer {# asset}}",
  710 + "asset-name-starts-with": "Adlarý {{prefix}} ile baþlayan varlýklar ",
  711 + "type-entity-view": "Varlýk Görünümü",
  712 + "type-entity-views": "Varlýk Görünümleri",
  713 + "list-of-entity-views": "{count, çoðul, 1 {Bir varlýk görünümü} diðer {# varlýk görüntüleme}} listesi",
  714 + "entity-view-name-starts-with": "Adý {{önek}} ile baþlayan varlýk görünümleri",
  715 + "type-rule": "Kural",
  716 + "type-rules": "Kurallar",
  717 + "list-of-rules": "{count, çoðul, 1 {Bir kural} diðer {# kurallarýn}} listesi",
  718 + "rule-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan kurallar",
  719 + "type-plugin": "Eklenti",
  720 + "type-plugins": "Eklentiler",
  721 + "list-of-plugins": "{count, çoðul, 1 {Bir eklenti} diðer {# eklenti listesi}}",
  722 + "plugin-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan eklentiler",
  723 + "type-tenant": "Kiracý",
  724 + "type-tenants": "Kiracýlar",
  725 + "list-of-tenants": "{count, çoðul, 1 {Bir kiracý} diðer {# kiracýlarýn listesi}}",
  726 + "tenant-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan kiracýlar, ",
  727 + "type-customer": "Müþteri",
  728 + "type-customers": "Müþteriler",
  729 + "list-of-customers": "{count, çoðul, 1 {Bir müþteri} diðer {# müþteri}} listesi",
  730 + "customer-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan müþteriler,",
  731 + "type-user": "Kullanýcý",
  732 + "type-users": "Kullanýcýlar",
  733 + "list-of-users": "{count, çoðul, 1 {Bir kullanýcý} diðer {# user}} listesi",
  734 + "user-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan kullanýcýlar",
  735 + "type-dashboard": "Pano",
  736 + "type-dashboards": "Gösterge tablolarý",
  737 + "list-of-dashboards": "{count, çoðul, 1 {Bir pano} Diðer {# panolarýn}}} listesi",
  738 + "dashboard-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan panolar",
  739 + "type-alarm": "Alarm",
  740 + "type-alarms": "Alarmlar",
  741 + "list-of-alarms": "{count, çoðul, 1 {Bir alarm} diðer {{# alarm}} listesi",
  742 + "alarm-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan alarmlar",
  743 + "type-rulechain": "Kural zinciri",
  744 + "type-rulechains": "Kural zincirleri",
  745 + "list-of-rulechains": "{count, çoðul, 1 {Bir kural zinciri} diðer {# kural zincirinin listesi}}",
  746 + "rulechain-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan kural zincirleri",
  747 + "type-rulenode": "Kural düðümü",
  748 + "type-rulenodes": "Kural düðümleri",
  749 + "list-of-rulenodes": "{count, çoðul, 1 {Bir kural node} diðer {# kural düðümünün listesi}}",
  750 + "rulenode-name-starts-with": "Ýsimleri '{{prefix}} ile baþlayan kural düðümleri",
  751 + "type-current-customer": "Mevcut Müþteri",
  752 + "search": "Varlýklar ara",
  753 + "selected-entities": "{count, çoðul, 1 {1 varlýk} diðer {# entities}} seçildi",
  754 + "entity-name": "Varlýk adý",
  755 + "details": "Varlýk ayrýntýlarý",
  756 + "no-entities-prompt": "Hiçbir varlýk bulunamadý",
  757 + "no-data": "Gösterilecek bilgi yok"
  758 + },
  759 + "entity-view": {
  760 + "entity-view": "Varlýk Görünümü",
  761 + "entity-views": "Varlýk Görünümleri",
  762 + "management": "Varlýk Görünümü yönetimi",
  763 + "view-entity-views": "Varlýk Görünümlerini Görüntüle",
  764 + "entity-view-alias": "Varlýk Görünümü takma adý",
  765 + "aliases": "Varlýk Görünümü takma adlarý",
  766 + "no-alias-matching": "'{{alias}} bulunamadý. ",
  767 + "no-aliases-found": "Takma ad bulunamadý",
  768 + "no-key-matching": "'{{anahtar bulunamadý.",
  769 + "no-keys-found": "Anahtar bulunamadý.",
  770 + "create-new-alias": "Yeni bir tane oluþtur!",
  771 + "create-new-key": "Yeni bir tane oluþtur!",
  772 + "duplicate-alias-error": "Yinelenen takma ad bulundu {{alias}} '.. Entity View diðer adlar, gösterge panosunda benzersiz olmalýdýr. ",
  773 + "configure-alias": "Yapýlandýrma {{alias}} takma ad",
  774 + "no-entity-views-matching": "{{Entity}} ile eþleþen hiçbir varlýk yorumu bulunamadý. ",
  775 + "alias": "Alias",
  776 + "alias-required": "Varlýk Görünümü takma adý gerekiyor.",
  777 + "remove-alias": "Varlýk görünümü takma adýný kaldýr",
  778 + "add-alias": "Varlýk görünümü takma adý ekle",
  779 + "name-starts-with": "Varlýk Görünümü adý ile baþlýyor",
  780 + "entity-view-list": "Varlýk Görünümü listesi",
  781 + "use-entity-view-name-filter": "Filtre kullan",
  782 + "entity-view-list-empty": "Hiçbir varlýk görüþü seçilmedi.",
  783 + "entity-view-name-filter-required": "Varlýk görünüm adý filtresi gerekli.",
  784 + "entity-view-name-filter-no-entity-view-matched": "{{EntityView}} ile baþlayan hiçbir varlýk sayýsý bulunamadý.",
  785 + "add": "Varlýk Görünümü Ekle",
  786 + "assign-to-customer": "Müþteriye atama",
  787 + "assign-entity-view-to-customer": "Varlýk Görünümlerini Müþteriye Atama",
  788 + "assign-entity-view-to-customer-text": "Lütfen müþteriye atamak için varlýk görünümlerini seçin",
  789 + "no-entity-views-text": "Varlýk görüþü bulunamadý",
  790 + "assign-to-customer-text": "Lütfen varlýk görünümlerini atamak için müþteriyi seçin",
  791 + "entity-view-details": "Varlýk görünümü ayrýntýlarý",
  792 + "add-entity-view-text": "Yeni varlýk görünümü ekle",
  793 + "delete": "Varlýk görünümünü sil",
  794 + "assign-entity-views": "Varlýk görünümleri atama",
  795 + "assign-entity-views-text": "Müþteriye {count, çoðul, 1 {1 entityView} diðer {# entityViews}} atayýn ",
  796 + "delete-entity-views": "Varlýk görünümlerini sil",
  797 + "unassign-from-customer": "Müþteriden atama",
  798 + "unassign-entity-views": "Varlýk görünümlerini atama",
  799 + "unassign-entity-views-action-title": "Müþteriden atama sayým, çoðul, 1 {1 entityView} diðer {# entityViews}}",
  800 + "assign-new-entity-view": "Yeni varlýk görünümü atama",
  801 + "delete-entity-view-title": "Varlýk görünümünü silmek istediðinizden emin misiniz?, {{EntityViewName}} '? ",
  802 + "delete-entity-view-text": "Dikkatli olun, onaylandýktan sonra varlýk görünümü ve ilgili tüm veriler kurtarýlamayacak.",
  803 + "delete-entity-views-title": "{Count, çoðul, 1 {1 entityView} diðer {# entityViews}} varlýk görünümüne sahip olmak istediðinizden emin misiniz?",
  804 + "delete-entity-views-action-title": "Sil {count, çoðul, 1 {1 entityView} diðer {# entityViews}}",
  805 + "delete-entity-views-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen görünümler kaldýrýlacak ve ilgili tüm veriler kurtarýlamayacaktýr.",
  806 + "unassign-entity-view-title": "Varlýk görünümünün atamasýný kaldýrmak istediðinizden emin misiniz? {{EntityViewName}} '? ",
  807 + "unassign-entity-view-text": "Onaydan sonra varlýk görünümü atanmamýþ olacak ve müþteri tarafýndan eriþilemeyecektir.",
  808 + "unassign-entity-view": "Varlýk görünümünün atamasýný kaldýr",
  809 + "unassign-entity-views-title": "Sayým, çoðul, 1 {1 entityView} diðer {# entityViews}} hesabýnýn atamasýný kaldýrmak istediðinizden emin misiniz?",
  810 + "unassign-entity-views-text": "Onaylandýktan sonra, seçilen tüm öðe görünümleri atamadan kaldýrýlacak ve müþteri tarafýndan eriþilemeyecektir.",
  811 + "entity-view-type": "Varlýk Görünümü türü",
  812 + "entity-view-type-required": "Varlýk Görünümü türü gerekli.",
  813 + "select-entity-view-type": "Varlýk görüntüleme türünü seç",
  814 + "enter-entity-view-type": "Varlýk görüntüleme türünü girin",
  815 + "any-entity-view": "Herhangi bir varlýk görünümü",
  816 + "no-entity-view-types-matching": "{{EntitySubtype}} ile eþleþen hiçbir varlýk görüntüleme türü bulunamadý. ",
  817 + "entity-view-type-list-empty": "Hiçbir varlýk görünümü türü seçilmemiþ.",
  818 + "entity-view-types": "Varlýk Görünümü türleri",
  819 + "name": "Ad",
  820 + "name-required": "Ýsim gerekli.",
  821 + "description": "Açýklama",
  822 + "events": "Etkinlikler",
  823 + "details": "Ayrýntýlar",
  824 + "copyId": "Varlýk görüntüleme kimliðini kopyala",
  825 + "assignedToCustomer": "Müþteriye atandý",
  826 + "unable-entity-view-device-alias-title": "Varlýk görünümü takma adý silinemiyor",
  827 + "unable-entity-view-device-alias-text": "Cihaz takma adý {{entityViewAlias}} ', aþaðýdaki widget (lar) tarafýndan kullanýldýðý þekliyle silinemez: <br/> {{widgetsList}} ",
  828 + "select-entity-view": "Varlýk görünümünü seç",
  829 + "make-public": "Varlýðý herkese görünür yap",
  830 + "start-ts": "Ts",
  831 + "end-ts": "End ts"
  832 + },
  833 + "event": {
  834 + "event-type": "Etkinlik tipi",
  835 + "type-error": "Hata",
  836 + "type-lc-event": "Yaþam döngüsü etkinliði",
  837 + "type-stats": "Ýstatistik",
  838 + "type-debug-rule-node": "Hata ayýklama",
  839 + "type-debug-rule-chain": "Hata ayýklama",
  840 + "no-events-prompt": "Etkinlik bulunamadý",
  841 + "error": "Hata",
  842 + "alarm": "Alarm",
  843 + "event-time": "Etkinlik zamaný",
  844 + "server": "Sunucu",
  845 + "body": "Vücut",
  846 + "method": "Yöntem",
  847 + "type": "Tür",
  848 + "entity": "Varlýk",
  849 + "message-id": "Mesaj Kimliði",
  850 + "message-type": "Mesaj tipi",
  851 + "data-type": "Veri tipi",
  852 + "relation-type": "Ýliþki Türü",
  853 + "metadata": "Meta veri",
  854 + "data": "Veri",
  855 + "event": "Etkinlik",
  856 + "status": "Durum",
  857 + "success": "Baþarý",
  858 + "failed": "Baþarýsýz oldu",
  859 + "messages-processed": "Mesajlar iþlendi",
  860 + "errors-occurred": "Hatalar oluþtu"
  861 + },
  862 + "extension": {
  863 + "extensions": "Uzantýlar",
  864 + "selected-extensions": "{count, çoðul, 1 {1 uzantý} diðer {# extensions}} seçildi",
  865 + "type": "Tür",
  866 + "key": "Anahtar",
  867 + "value": "Deðer",
  868 + "id": "ÝD",
  869 + "extension-id": "Uzantý kimliði",
  870 + "extension-type": "Uzatma tipi",
  871 + "transformer-json": "JSON *",
  872 + "unique-id-required": "Mevcut uzantý kimliði zaten mevcut.",
  873 + "delete": "Uzantýyý sil",
  874 + "add": "Uzantý eklemek",
  875 + "edit": "Uzantýyý düzenle",
  876 + "delete-extension-title": "{{ExtensionId}} uzantýsýný silmek istediðinizden emin misiniz? ",
  877 + "delete-extension-text": "Dikkatli olun, onaylamadan sonra uzantý ve ilgili tüm veriler kurtarýlamaz.",
  878 + "delete-extensions-title": "{Count, çoðul, 1 {1 uzantý} diðer {# extensions}} silmek istediðinizden emin misiniz?",
  879 + "delete-extensions-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen uzantýlar kaldýrýlacak.",
  880 + "converters": "Dönüþtürücü",
  881 + "converter-id": "Dönüþtürücü kimliði",
  882 + "configuration": "Yapýlandýrma",
  883 + "converter-configurations": "Dönüþtürücü yapýlandýrmalarý",
  884 + "token": "Güvenlik belirteci",
  885 + "add-converter": "Dönüþtürücü ekle",
  886 + "add-config": "Dönüþtürücü yapýlandýrmasý ekle",
  887 + "device-name-expression": "Cihaz adý ifadesi",
  888 + "device-type-expression": "Cihaz tipi ifadesi",
  889 + "custom": "Özel",
  890 + "to-double": "Çifte",
  891 + "transformer": "Transformer",
  892 + "json-required": "Trafo jsonu gerekli.",
  893 + "json-parse": "Trafo json ayrýþtýrýlamýyor.",
  894 + "attributes": "Öznitellikler",
  895 + "add-attribute": "Özellik ekle",
  896 + "add-map": "Eþleme elemaný ekle",
  897 + "timeseries": "Zaman serisi",
  898 + "add-timeseries": "Zaman çizelgeleri ekle",
  899 + "field-required": "Alan gereklidir",
  900 + "brokers": "Komisyoncular",
  901 + "add-broker": "Broker ekle",
  902 + "host": "Host",
  903 + "port": "Liman",
  904 + "port-range": "Liman 1'den 65535'e kadar olmalýdýr.",
  905 + "ssl": "SSL",
  906 + "credentials": "Kimlik bilgileri",
  907 + "username": "Kullanýcý adý",
  908 + "password": "Parola",
  909 + "retry-interval": "Milisaniye cinsinden tekrar deneme aralýðý",
  910 + "anonymous": "Anonim",
  911 + "basic": "Temel",
  912 + "pem": "PEM",
  913 + "ca-cert": "CA sertifika dosyasý *",
  914 + "private-key": "Özel anahtar dosya *",
  915 + "cert": "Sertifika dosyasý *",
  916 + "no-file": "Dosya seçilmedi.",
  917 + "drop-file": "Bir dosya býrakýn veya yüklenecek bir dosya seçmek için týklayýn.",
  918 + "mapping": "Mapping",
  919 + "topic-filter": "Konu filtresi",
  920 + "converter-type": "Dönüþtürücü tipi",
  921 + "converter-json": "Json",
  922 + "json-name-expression": "Cihaz adý json ifadesi",
  923 + "topic-name-expression": "Cihaz adý konu ifadesi",
  924 + "json-type-expression": "Cihaz tipi json ifadesi",
  925 + "topic-type-expression": "Cihaz tipi konu ifadesi",
  926 + "attribute-key-expression": "Öznitelik anahtar ifadesi",
  927 + "attr-json-key-expression": "Öznitelik anahtar json ifadesi",
  928 + "attr-topic-key-expression": "Öznitelik anahtar konu ifadesi",
  929 + "request-id-expression": "Kimlik ifadesi iste",
  930 + "request-id-json-expression": "Kimlik json ifadesi iste",
  931 + "request-id-topic-expression": "Kimlik konu ifadesini isteyin",
  932 + "response-topic-expression": "Yanýt konusu ifadesi",
  933 + "value-expression": "Deðer ifadesi",
  934 + "topic": "Konu",
  935 + "timeout": "Zaman aþýmý milisaniye cinsinden",
  936 + "converter-json-required": "Dönüþtürücü json gerekli.",
  937 + "converter-json-parse": "Dönüþtürücü json ayrýþtýrýlamýyor.",
  938 + "filter-expression": "Filtre ifadesi",
  939 + "connect-requests": "Ýstekleri baðla",
  940 + "add-connect-request": "Baðlantý talebi ekle",
  941 + "disconnect-requests": "Ýstekleri kes",
  942 + "add-disconnect-request": "Baðlantýyý kes isteði ekle",
  943 + "attribute-requests": "Özellik istekleri",
  944 + "add-attribute-request": "Özellik isteði ekle",
  945 + "attribute-updates": "Öznitelik güncellemeleri",
  946 + "add-attribute-update": "Özellik güncellemesi ekle",
  947 + "server-side-rpc": "Sunucu tarafý RPC",
  948 + "add-server-side-rpc-request": "Sunucu tarafý RPC isteði ekle",
  949 + "device-name-filter": "Cihaz adý filtresi",
  950 + "attribute-filter": "Özellik filtresi",
  951 + "method-filter": "Yöntem filtresi",
  952 + "request-topic-expression": "Konu ifadesi iste",
  953 + "response-timeout": "Milisaniye cinsinden yanýt zaman aþýmý",
  954 + "topic-expression": "Konu ifadesi",
  955 + "client-scope": "Müþteri kapsamý",
  956 + "add-device": "Cihaz ekle",
  957 + "opc-server": "Sunucular",
  958 + "opc-add-server": "Sunucu ekle",
  959 + "opc-add-server-prompt": "Lütfen sunucu ekle",
  960 + "opc-application-name": "Uygulama Adý",
  961 + "opc-application-uri": "Uygulama uri",
  962 + "opc-scan-period-in-seconds": "Saniyeler içinde tarama süresi",
  963 + "opc-security": "Güvenlik",
  964 + "opc-identity": "Kimlik",
  965 + "opc-keystore": "Keystore",
  966 + "opc-type": "Tür",
  967 + "opc-keystore-type": "Tür",
  968 + "opc-keystore-location": "Yer *",
  969 + "opc-keystore-password": "Parola",
  970 + "opc-keystore-alias": "Alias",
  971 + "opc-keystore-key-password": "Anahtar þifre",
  972 + "opc-device-node-pattern": "Cihaz düðümü modeli",
  973 + "opc-device-name-pattern": "Cihaz adý deseni",
  974 + "modbus-server": "Sunucular / köle",
  975 + "modbus-add-server": "Sunucu ekle / köle",
  976 + "modbus-add-server-prompt": "Lütfen sunucu / slave ekle",
  977 + "modbus-transport": "Taþýma",
  978 + "modbus-port-name": "Seri port adý",
  979 + "modbus-encoding": "Kodlama",
  980 + "modbus-parity": "Parite",
  981 + "modbus-baudrate": "Baud hýzý",
  982 + "modbus-databits": "Veri bitleri",
  983 + "modbus-stopbits": "Bitleri durdur",
  984 + "modbus-databits-range": "Veri bitleri 7 ila 8 arasýnda olmalýdýr",
  985 + "modbus-stopbits-range": "Durma bitleri 1'den 2'ye kadar olmalýdýr.",
  986 + "modbus-unit-id": "Birim Kimliði",
  987 + "modbus-unit-id-range": "Birim numarasý 1 ile 247 arasýnda olmalýdýr.",
  988 + "modbus-device-name": "Cihaz adý",
  989 + "modbus-poll-period": "Anket dönemi (ms)",
  990 + "modbus-attributes-poll-period": "Nitelikler yoklama süresi (ms)",
  991 + "modbus-timeseries-poll-period": "Timeseries anket süresi (ms)",
  992 + "modbus-poll-period-range": "Anket dönemi pozitif deðer olmalý",
  993 + "modbus-tag": "Etiket",
  994 + "modbus-function": "Ýþlev",
  995 + "modbus-register-address": "Kayýt adresi",
  996 + "modbus-register-address-range": "Kayýt adresi 0 ile 65535 arasýnda olmalýdýr.",
  997 + "modbus-register-bit-index": "Bit endeksi",
  998 + "modbus-register-bit-index-range": "Bit endeksi 0 ile 15 arasýnda olmalýdýr",
  999 + "modbus-register-count": "Kayýt sayýsý",
  1000 + "modbus-register-count-range": "Kayýt sayýsý pozitif bir deðer olmalýdýr.",
  1001 + "modbus-byte-order": "Bayt sýrasý",
  1002 + "sync": {
  1003 + "status": "Durum",
  1004 + "sync": "Senkronizasyon",
  1005 + "not-sync": "Eþitleme",
  1006 + "last-sync-time": "Son senkronizasyon zamaný",
  1007 + "not-available": "Müsait deðil"
  1008 + },
  1009 + "export-extensions-configuration": "Ýhracat uzantýlarý yapýlandýrmasý",
  1010 + "import-extensions-configuration": "Uzantýlarýný içe aktarma yapýlandýrmasý",
  1011 + "import-extensions": "Uzantýlarý içe aktar",
  1012 + "import-extension": "Uzantý içe aktar",
  1013 + "export-extension": "Ýhracat uzantýsý",
  1014 + "file": "Uzantýlar dosyasý",
  1015 + "invalid-file-error": "Geçersiz uzantý dosyasý"
  1016 + },
  1017 + "fullscreen": {
  1018 + "expand": "Tam ekrana geniþlet",
  1019 + "exit": "Tam ekrandan çýk",
  1020 + "toggle": "Tam ekran modunu deðiþtir",
  1021 + "fullscreen": "Tam ekran"
  1022 + },
  1023 + "function": {
  1024 + "function": "Ýþlev"
  1025 + },
  1026 + "grid": {
  1027 + "delete-item-title": "Bu maddeyi silmek istediðinden emin misin?",
  1028 + "delete-item-text": "Dikkatli olun, onaylandýktan sonra bu öðe ve ilgili tüm veriler kurtarýlamaz.",
  1029 + "delete-items-title": "{Count, çoðul, 1 {1 item} diðer {# items}} silmek istediðinizden emin misiniz?",
  1030 + "delete-items-action-title": "Sil {count, çoðul, 1 {1 öðe} diðer {# items}}",
  1031 + "delete-items-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen öðeler silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1032 + "add-item-text": "Yeni öðe ekle",
  1033 + "no-items-text": "Hiç bir öðe bulunamadý",
  1034 + "item-details": "Ürün detaylarý",
  1035 + "delete-item": "Öðeyi silmek",
  1036 + "delete-items": "Ürünleri sil",
  1037 + "scroll-to-top": "Baþa kaydýr"
  1038 + },
  1039 + "help": {
  1040 + "goto-help-page": "Yardým sayfasýna git"
  1041 + },
  1042 + "home": {
  1043 + "home": "Ev",
  1044 + "profile": "Profil",
  1045 + "logout": "Çýkýþ Yap",
  1046 + "menu": "Menü",
  1047 + "avatar": "Avatar",
  1048 + "open-user-menu": "Kullanýcý menüsünü aç"
  1049 + },
  1050 + "import": {
  1051 + "no-file": "Dosya seçilmedi",
  1052 + "drop-file": "Bir JSON dosyasýný indirin veya yüklemek için bir dosya seçmek için týklayýn."
  1053 + },
  1054 + "item": {
  1055 + "selected": "Seçilmiþ"
  1056 + },
  1057 + "js-func": {
  1058 + "no-return-error": "Ýþlev deðer döndürmeli!",
  1059 + "return-type-mismatch": "Ýþlev '{{type}}' tipinin deðerini döndürmelidir!",
  1060 + "tidy": "Düzenli"
  1061 + },
  1062 + "key-val": {
  1063 + "key": "Anahtar",
  1064 + "value": "Deðer",
  1065 + "remove-entry": "Giriþi kaldýr",
  1066 + "add-entry": "Giriþ ekle",
  1067 + "no-data": "Giriþ yok"
  1068 + },
  1069 + "layout": {
  1070 + "layout": "Düzen",
  1071 + "manage": "Düzenleri yönet",
  1072 + "settings": "Yerleþim ayarlarý",
  1073 + "color": "Renk",
  1074 + "main": "Ana",
  1075 + "right": "Sað",
  1076 + "select": "Hedef yerleþimini seç"
  1077 + },
  1078 + "legend": {
  1079 + "position": "Efsane pozisyonu",
  1080 + "show-max": "Maksimum deðeri göster",
  1081 + "show-min": "Min deðerini göster",
  1082 + "show-avg": "Ortalama deðeri göster",
  1083 + "show-total": "Toplam deðeri göster",
  1084 + "settings": "Açýklama ayarlarý",
  1085 + "min": "min",
  1086 + "max": "max",
  1087 + "avg": "avg",
  1088 + "total": "Genel Toplam"
  1089 + },
  1090 + "login": {
  1091 + "login": "Oturum aç",
  1092 + "request-password-reset": "Þifre sýfýrlama isteði",
  1093 + "reset-password": "Þifreyi yenile",
  1094 + "create-password": "Þifre oluþtur",
  1095 + "passwords-mismatch-error": "Girilen þifreler ayný olmalýdýr!",
  1096 + "password-again": "Þifre Tekrar",
  1097 + "sign-in": "Lütfen giriþ yapýn",
  1098 + "username": "Kullanýcý adý (email)",
  1099 + "remember-me": "Beni hatýrla",
  1100 + "forgot-password": "Parolanýzý mý unuttunuz?",
  1101 + "password-reset": "Parola sýfýrlama",
  1102 + "new-password": "Yeni Þifre",
  1103 + "new-password-again": "Yeni Þifre Tekrar",
  1104 + "password-link-sent-message": "Þifre sýfýrlama baðlantýsý baþarýyla gönderildi!",
  1105 + "email": "E-posta"
  1106 + },
  1107 + "position": {
  1108 + "top": "Üst",
  1109 + "bottom": "Alt",
  1110 + "left": "Ayrýldý",
  1111 + "right": "Sað"
  1112 + },
  1113 + "profile": {
  1114 + "profile": "Profil",
  1115 + "change-password": "Þifre deðiþtir",
  1116 + "current-password": "Þimdiki Þifre"
  1117 + },
  1118 + "relation": {
  1119 + "relations": "Ýliþkiler",
  1120 + "direction": "Yön",
  1121 + "search-direction": {
  1122 + "FROM": "Kimden",
  1123 + "TO": "Alýcý"
  1124 + },
  1125 + "direction-type": {
  1126 + "FROM": "den",
  1127 + "TO": "to"
  1128 + },
  1129 + "from-relations": "Giden iliþkiler",
  1130 + "to-relations": "Gelen iliþkiler",
  1131 + "selected-relations": "{count, çoðul, 1 {1 iliþki} diðer {# relations}} seçildi",
  1132 + "type": "Tür",
  1133 + "to-entity-type": "Varlýk türüne",
  1134 + "to-entity-name": "Varlýk adýna",
  1135 + "from-entity-type": "Varlýk türünden",
  1136 + "from-entity-name": "Varlýk adýndan",
  1137 + "to-entity": "Varlýk",
  1138 + "from-entity": "Varlýktan",
  1139 + "delete": "Ýliþkisi sil",
  1140 + "relation-type": "Ýliþki türü",
  1141 + "relation-type-required": "Ýliþki türü gerekli.",
  1142 + "any-relation-type": "Her hangi bir tür",
  1143 + "add": "Ýliþki ekle",
  1144 + "edit": "Ýliþkisi düzenle",
  1145 + "delete-to-relation-title": "{{EntityName}} varlýðýna iliþkin iliþkiyi silmek istediðinizden emin misiniz? ",
  1146 + "delete-to-relation-text": "Doðrulamadan sonra, varlýðýn {{entityName}} öðesinin mevcut varlýktan alakasýz olacaðýndan emin olun. ",
  1147 + "delete-to-relations-title": "{Count, çoðul, 1 {1 iliþki} diðer {# relations}} silmek istediðinizden emin misiniz?",
  1148 + "delete-to-relations-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen iliþkiler kaldýrýlacak ve ilgili varlýklar mevcut varlýk ile ilgisiz olacaktýr.",
  1149 + "delete-from-relation-title": "{{EntityName}} varlýðýndan iliþkiyi silmek istediðinizden emin misiniz? ",
  1150 + "delete-from-relation-text": "Doðrulama varlýðýnýn, {{entityName}} kuruluþundan alakasýz olacaðýndan emin olun.",
  1151 + "delete-from-relations-title": "{Count, çoðul, 1 {1 iliþki} diðer {# relations}} silmek istediðinizden emin misiniz?",
  1152 + "delete-from-relations-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen iliþkiler kaldýrýlacak ve mevcut varlýk ilgili varlýklardan ilgisiz olacaktýr.",
  1153 + "remove-relation-filter": "Ýliþki filtresini kaldýr",
  1154 + "add-relation-filter": "Ýliþki filtresi ekle",
  1155 + "any-relation": "Herhangi bir iliþki",
  1156 + "relation-filters": "Ýliþki filtreleri",
  1157 + "additional-info": "Ek bilgi (JSON)",
  1158 + "invalid-additional-info": "Ek bilgi json ayrýþtýrýlamýyor."
  1159 + },
  1160 + "rulechain": {
  1161 + "rulechain": "Kural zinciri",
  1162 + "rulechains": "Kural zincirleri",
  1163 + "root": "Kök",
  1164 + "delete": "Kural zincirini sil",
  1165 + "name": "Ad",
  1166 + "name-required": "Ýsim gerekli.",
  1167 + "description": "Açýklama",
  1168 + "add": "Kural Zinciri Ekleme",
  1169 + "set-root": "Kural zincirinin kökü yap",
  1170 + "set-root-rulechain-title": "Kural zincirini {{ruleChainName}} root? Yapmak istediðinizden emin misiniz?",
  1171 + "set-root-rulechain-text": "Onaydan sonra kural zinciri kökleþecek ve gelen tüm iletilerle ilgilenecek.",
  1172 + "delete-rulechain-title": "Kural zincirini {{ruleChainName}} silmek istediðinizden emin misiniz?",
  1173 + "delete-rulechain-text": "Dikkatli olun, onaylamadan sonra kural zinciri ve ilgili tüm veriler kurtarýlamaz.",
  1174 + "delete-rulechains-title": "{Count, çoðul, 1 {1 kural zinciri} diðer {# kural zincirleri}} silmek istediðinizden emin misiniz?",
  1175 + "delete-rulechains-action-title": "Sil {count, çoðul, 1 {1 kural zinciri} diðer {# kural zincirleri}}",
  1176 + "delete-rulechains-text": "Dikkatli olun, onaylandýktan sonra seçilen tüm kural zincirleri silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1177 + "add-rulechain-text": "Yeni kural zinciri ekle",
  1178 + "no-rulechains-text": "Kural zinciri bulunamadý",
  1179 + "rulechain-details": "Kural zinciri detaylarý",
  1180 + "details": "Ayrýntýlar",
  1181 + "events": "Etkinlikler",
  1182 + "system": "Sistem",
  1183 + "import": "Kural zincirini içe aktar",
  1184 + "export": "Kural zinciri dýþa aktar",
  1185 + "export-failed-error": "Kural zinciri dýþa aktarýlamadý: {{error}}",
  1186 + "create-new-rulechain": "Yeni kural zinciri oluþtur",
  1187 + "rulechain-file": "Kural zinciri dosyasý",
  1188 + "invalid-rulechain-file-error": "Kural zinciri içe aktarýlamýyor: Geçersiz kural zinciri veri yapýsý.",
  1189 + "copyId": "Kural zinciri kimliðini kopyala",
  1190 + "idCopiedMessage": "Kural zinciri kimliði panoya kopyalandý",
  1191 + "select-rulechain": "Kural zincirini seç",
  1192 + "no-rulechains-matching": "{{Entity}} ile eþleþen kural zinciri bulunamadý. ",
  1193 + "rulechain-required": "Kural zinciri gerekli",
  1194 + "management": "Kural yönetimi",
  1195 + "debug-mode": "Hata ayýklama modu"
  1196 + },
  1197 + "rulenode": {
  1198 + "details": "Ayrýntýlar",
  1199 + "events": "Etkinlikler",
  1200 + "search": "Arama düðümleri",
  1201 + "open-node-library": "Düðüm kütüphanesini aç",
  1202 + "add": "Kural düðümü ekle",
  1203 + "name": "Ad",
  1204 + "name-required": "Ýsim gerekli.",
  1205 + "type": "Tür",
  1206 + "description": "Açýklama",
  1207 + "delete": "Kural düðümünü sil",
  1208 + "select-all-objects": "Tüm düðümleri ve baðlantýlarý seç",
  1209 + "deselect-all-objects": "Tüm düðümlerin ve baðlantýlarýn seçimini kaldýrýn",
  1210 + "delete-selected-objects": "Seçilen düðümleri ve baðlantýlarý sil",
  1211 + "delete-selected": "Silme seçildi",
  1212 + "select-all": "Hepsini seç",
  1213 + "copy-selected": "Seçilenleri kopyala",
  1214 + "deselect-all": "Hiçbirini seçme",
  1215 + "rulenode-details": "Kural düðümü ayrýntýlarý",
  1216 + "debug-mode": "Hata ayýklama modu",
  1217 + "configuration": "Yapýlandýrma",
  1218 + "link": "Baðlantý",
  1219 + "link-details": "Kural düðüm baðlantý detaylarý",
  1220 + "add-link": "Link ekle",
  1221 + "link-label": "Baðlantý etiketi",
  1222 + "link-label-required": "Baðlantý etiketi gerekli.",
  1223 + "custom-link-label": "Özel baðlantý etiketi",
  1224 + "custom-link-label-required": "Özel baðlantý etiketi gerekli.",
  1225 + "link-labels": "Link etiketleri",
  1226 + "link-labels-required": "Link etiketleri gerekli.",
  1227 + "no-link-labels-found": "Baðlantý etiketi bulunamadý",
  1228 + "no-link-label-matching": "{{label}} bulunamadý. ",
  1229 + "create-new-link-label": "Yeni bir tane oluþtur!",
  1230 + "type-filter": "Filtre",
  1231 + "type-filter-details": "Gelen iletileri yapýlandýrýlmýþ koþullara göre filtrele",
  1232 + "type-enrichment": "Zenginleþtirme",
  1233 + "type-enrichment-details": "Mesaj Meta Verilerine ek bilgi",
  1234 + "type-transformation": "Dönüþüm",
  1235 + "type-transformation-details": "Mesaj yükünü ve Meta Verileri Deðiþtir",
  1236 + "type-action": "Aksiyon",
  1237 + "type-action-details": "Özel eylem gerçekleþtir",
  1238 + "type-external": "Dýþ",
  1239 + "type-external-details": "Dýþ sistemle etkileþir",
  1240 + "type-rule-chain": "Kural Zinciri",
  1241 + "type-rule-chain-details": "Belirtilen Kural Zincirine gelen mesajlarý ilet",
  1242 + "type-input": "Giriþ",
  1243 + "type-input-details": "Kural Zinciri'nin mantýksal girdisi, bir sonraki ilgili Kural Düðümüne gelen iletileri iletme",
  1244 + "type-unknown": "Bilinmeyen",
  1245 + "type-unknown-details": "Çözümlenmemiþ Kural Düðümü",
  1246 + "directive-is-not-loaded": "Tanýmlanmýþ yapýlandýrma yönergesi {{directiveName}} 'mevcut deðil. ",
  1247 + "ui-resources-load-error": "Yapýlandýrma kullanýcý arayüzü kaynaklarý yüklenemedi.",
  1248 + "invalid-target-rulechain": "Hedef kural zinciri çözülemiyor!",
  1249 + "test-script-function": "Test komut dosyasý iþlevi",
  1250 + "message": "Mesaj",
  1251 + "message-type": "Mesaj tipi",
  1252 + "select-message-type": "Mesaj tipini seç",
  1253 + "message-type-required": "Mesaj türü gerekli",
  1254 + "metadata": "Meta veri",
  1255 + "metadata-required": "Meta veri giriþleri boþ býrakýlamaz.",
  1256 + "output": "Çýktý",
  1257 + "test": "Ölçek",
  1258 + "help": "Yardým et"
  1259 + },
  1260 + "tenant": {
  1261 + "tenant": "Kiracý",
  1262 + "tenants": "Kiracýlar",
  1263 + "management": "Kiracý yönetimi",
  1264 + "add": "Kiracý ekle",
  1265 + "admins": "Yöneticiler",
  1266 + "manage-tenant-admins": "Kiracý yöneticileri yönet",
  1267 + "delete": "Kiracýyý sil",
  1268 + "add-tenant-text": "Yeni kiracý ekle",
  1269 + "no-tenants-text": "Kiracý bulunamadý",
  1270 + "tenant-details": "Kiracý detaylarý",
  1271 + "delete-tenant-title": "Kiracýyý silmek istediðinizden emin misiniz? {{TenantTitle}}? ",
  1272 + "delete-tenant-text": "Dikkatli olun, onayýndan sonra kiracý ve ilgili tüm veriler kurtarýlamaz.",
  1273 + "delete-tenants-title": "{Count, çoðul, 1 {1 kiracý} diðer {# kiracý}} silmek istediðinizden emin misiniz?",
  1274 + "delete-tenants-action-title": "Sil {count, çoðul, 1 {1 kiracý} diðer {# kiracý}}",
  1275 + "delete-tenants-text": "Dikkatli olun, onaylandýktan sonra tüm kiracýlar silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1276 + "title": "Baþlýk",
  1277 + "title-required": "Baþlýk gerekli.",
  1278 + "description": "Açýklama",
  1279 + "details": "Ayrýntýlar",
  1280 + "events": "Etkinlikler",
  1281 + "copyId": "Kiracý Kimliði Kimliði",
  1282 + "idCopiedMessage": "Kiracý Kimliði panoya kopyalandý",
  1283 + "select-tenant": "Kiracý seç",
  1284 + "no-tenants-matching": "{{Entity}} ile eþleþen kiracý bulunamadý. ",
  1285 + "tenant-required": "Kiracý gerekli"
  1286 + },
  1287 + "timeinterval": {
  1288 + "seconds-interval": "{saniye, çoðul, 1 {1 saniye} diðer {# seconds}}",
  1289 + "minutes-interval": "{dakika, çoðul, 1 {1 dakika} diðer {# dakika}}",
  1290 + "hours-interval": "{saat, çoðul, 1 {1 saat} diðer {# hours}}",
  1291 + "days-interval": "{günler, çoðul, 1 {1 gün} diðer {# gün}}",
  1292 + "days": "Gün",
  1293 + "hours": "Saatler",
  1294 + "minutes": "Dakika",
  1295 + "seconds": "Saniye",
  1296 + "advanced": "Ýleri"
  1297 + },
  1298 + "timewindow": {
  1299 + "days": "{günler, çoðul, 1 {gün} diðer {# gün}}",
  1300 + "hours": "{saat, çoðul, 0 {saat} 1 {1 saat} diðer {# hours}}",
  1301 + "minutes": "{dakika, çoðul, 0 {dakika} 1 {1 dakika} diðer {# dakika}}",
  1302 + "seconds": "{saniye, çoðul, 0 {saniye} 1 {1 saniye} diðer {# seconds}}",
  1303 + "realtime": "Gerçek zaman",
  1304 + "history": "Tarihçe",
  1305 + "last-prefix": "son",
  1306 + "period": "{{startTime}} - {{endTime}} arasýnda ",
  1307 + "edit": "Zaman penceresini düzenle",
  1308 + "date-range": "Tarih aralýðý",
  1309 + "last": "Son",
  1310 + "time-period": "Zaman dilimi"
  1311 + },
  1312 + "user": {
  1313 + "user": "Kullanýcý",
  1314 + "users": "Kullanýcýlar",
  1315 + "customer-users": "Müþteri Kullanýcýlarý",
  1316 + "tenant-admins": "Kiracý Admins",
  1317 + "sys-admin": "Sistem yöneticisi",
  1318 + "tenant-admin": "Kiracý yöneticisi",
  1319 + "customer": "Müþteri",
  1320 + "anonymous": "Anonim",
  1321 + "add": "Kullanýcý Ekle",
  1322 + "delete": "Kullanýcýyý sil",
  1323 + "add-user-text": "Yeni kullanýcý Ekle",
  1324 + "no-users-text": "Kullanýcý bulunamadý",
  1325 + "user-details": "Kullanýcý detaylarý",
  1326 + "delete-user-title": "Kullanýcýyý silmek istediðinizden emin misiniz? {{UserEmail}} '? ",
  1327 + "delete-user-text": "Dikkatli olun, onaylandýktan sonra kullanýcý ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1328 + "delete-users-title": "{Count, çoðul, 1 {1 kullanýcý} diðer {# users}} silmek istediðinizden emin misiniz?",
  1329 + "delete-users-action-title": "Sil {count, çoðul, 1 {1 kullanýcý} diðer {# users}}",
  1330 + "delete-users-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen kullanýcýlar silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1331 + "activation-email-sent-message": "Etkinleþtirme e-postasý baþarýyla gönderildi!",
  1332 + "resend-activation": "Etkinleþtirmeyi tekrar gönder",
  1333 + "email": "E-posta",
  1334 + "email-required": "Email gereklidir.",
  1335 + "invalid-email-format": "Geçersiz e-posta formatý.",
  1336 + "first-name": "Ýsim",
  1337 + "last-name": "Soyadý",
  1338 + "description": "Açýklama",
  1339 + "default-dashboard": "Varsayýlan gösterge paneli",
  1340 + "always-fullscreen": "Her zaman tam ekran",
  1341 + "select-user": "Kullanýcý seç",
  1342 + "no-users-matching": "{{Entity}} ile eþleþen kullanýcý bulunamadý. ",
  1343 + "user-required": "Kullanýcý gerekli",
  1344 + "activation-method": "Aktivasyon yöntemi",
  1345 + "display-activation-link": "Etkinleþtirme baðlantýsý göster",
  1346 + "send-activation-mail": "Etkinleþtirme postasý gönder",
  1347 + "activation-link": "Kullanýcý aktivasyon linki",
  1348 + "activation-link-text": "Kullanýcýyý aktif hale getirmek için aþaðýdaki <a href='{{activationLink}} target='_blank'> aktivasyon linki </a> kullanýn.",
  1349 + "copy-activation-link": "Etkinleþtirme baðlantýsýný kopyala",
  1350 + "activation-link-copied-message": "Kullanýcý aktivasyon linki panoya kopyalandý",
  1351 + "details": "Ayrýntýlar",
  1352 + "login-as-tenant-admin": "Kiracý Yönetici Giriþi",
  1353 + "login-as-customer-user": "Müþteri olarak giriþ yap"
  1354 + },
  1355 + "value": {
  1356 + "type": "Deðer türü",
  1357 + "string": "Dize",
  1358 + "string-value": "Dize deðeri",
  1359 + "integer": "Integer",
  1360 + "integer-value": "Tamsayý deðeri",
  1361 + "invalid-integer-value": "Geçersiz tam sayý",
  1362 + "double": "Çift",
  1363 + "double-value": "Çift deðer",
  1364 + "boolean": "Boole",
  1365 + "boolean-value": "Boole deðeri",
  1366 + "false": "Yanlýþ",
  1367 + "true": "Doðru",
  1368 + "long": "Uzun"
  1369 + },
  1370 + "widget": {
  1371 + "widget-library": "Widgets Kitaplýðý",
  1372 + "widget-bundle": "Widget Paketi",
  1373 + "select-widgets-bundle": "Widget paketini seç",
  1374 + "management": "Widget yönetimi",
  1375 + "editor": "Widget Düzenleyici",
  1376 + "widget-type-not-found": "Sorun yükleme pencere öðesi yapýlandýrmasý. <br> Muhtemelen iliþkili /n pencere öðesi türü kaldýrýldý.",
  1377 + "widget-type-load-error": "Widget, aþaðýdaki hatalardan dolayý yüklenmedi:",
  1378 + "remove": "Widget'ý kaldýr",
  1379 + "edit": "Widget düzenle",
  1380 + "remove-widget-title": "Widget'ý kaldýrmak istediðinizden emin misiniz? {{WidgetTitle}}? ",
  1381 + "remove-widget-text": "Onaydan sonra widget ve ilgili tüm veriler kurtarýlamaz.",
  1382 + "timeseries": "Zaman serisi",
  1383 + "search-data": "Arama verileri",
  1384 + "no-data-found": "Veri bulunamadý",
  1385 + "latest-values": "Son deðerler",
  1386 + "rpc": "Kontrol aracý",
  1387 + "alarm": "Alarm gereci",
  1388 + "static": "Statik pencere öðesi",
  1389 + "select-widget-type": "Widget tipi seç",
  1390 + "missing-widget-title-error": "Widget baþlýðý belirtilmelidir!",
  1391 + "widget-saved": "Widget kaydedildi",
  1392 + "unable-to-save-widget-error": "Widget kaydedilemiyor! Widget'ýn hatalarý var!",
  1393 + "save": "Widget kaydet",
  1394 + "saveAs": "Widget olarak kaydet",
  1395 + "save-widget-type-as": "Widget türünü kaydet",
  1396 + "save-widget-type-as-text": "Lütfen yeni pencere öðesi baþlýðýný girin ve / veya hedef widget'larý seçin",
  1397 + "toggle-fullscreen": "Tam ekrana geç",
  1398 + "run": "Widget'ý çalýþtýr",
  1399 + "title": "Widget Baþlýðý",
  1400 + "title-required": "Widget baþlýðý gerekiyor.",
  1401 + "type": "Widget türü",
  1402 + "resources": "Kaynaklar",
  1403 + "resource-url": "JavaScript / CSS URL",
  1404 + "remove-resource": "Kaynaðý kaldýr",
  1405 + "add-resource": "Kaynak ekle",
  1406 + "html": "HTML",
  1407 + "tidy": "Düzenli",
  1408 + "css": "CSS",
  1409 + "settings-schema": "Ayarlar þemasý",
  1410 + "datakey-settings-schema": "Veri anahtarý ayarlarý þemasý",
  1411 + "javascript": "JavaScript",
  1412 + "remove-widget-type-title": "'{{WidgetName}}' widget türünü kaldýrmak istediðinizden emin misiniz?",
  1413 + "remove-widget-type-text": "Onaydan sonra widget tipi ve ilgili tüm veriler kurtarýlamayacak.",
  1414 + "remove-widget-type": "Widget türünü kaldýr",
  1415 + "add-widget-type": "Yeni widget türü ekle",
  1416 + "widget-type-load-failed-error": "Widget türü yüklenemedi!",
  1417 + "widget-template-load-failed-error": "Widget þablonu yüklenemedi!",
  1418 + "add": "Widget Ekle",
  1419 + "undo": "Widget deðiþikliklerini geri al",
  1420 + "export": "Widget'ý dýþa aktar"
  1421 + },
  1422 + "widget-action": {
  1423 + "header-button": "Widget baþlýk düðmesi",
  1424 + "open-dashboard-state": "Yeni gösterge panosuna git",
  1425 + "update-dashboard-state": "Mevcut kontrol paneli durumunu güncelle",
  1426 + "open-dashboard": "Diðer kontrol paneline git",
  1427 + "custom": "Özel eylem",
  1428 + "target-dashboard-state": "Hedef gösterge panosu durumu",
  1429 + "target-dashboard-state-required": "Hedef gösterge tablosu gerekli",
  1430 + "set-entity-from-widget": "Öðeyi pencere öðesinden ayarla",
  1431 + "target-dashboard": "Hedef gösterge panosu",
  1432 + "open-right-layout": "Saðdaki gösterge tablosu düzeni (mobil görünüm)"
  1433 + },
  1434 + "widgets-bundle": {
  1435 + "current": "Mevcut paket",
  1436 + "widgets-bundles": "Widget Paketleri",
  1437 + "add": "Widget Paketi Ekle",
  1438 + "delete": "Widget paketini sil",
  1439 + "title": "Baþlýk",
  1440 + "title-required": "Baþlýk gerekli.",
  1441 + "add-widgets-bundle-text": "Yeni widget'lar paketi ekle",
  1442 + "no-widgets-bundles-text": "Widget paketi bulunamadý",
  1443 + "empty": "Widget'ler paketi boþ",
  1444 + "details": "Ayrýntýlar",
  1445 + "widgets-bundle-details": "Widget'lar paket ayrýntýlarýný",
  1446 + "delete-widgets-bundle-title": "Widget paketini {{widgetsBundleTitle}} silmek istediðinizden emin misiniz?",
  1447 + "delete-widgets-bundle-text": "Dikkatli olun, onaylandýktan sonra widget'lar paketi ve tüm ilgili veriler kurtarýlamayacak.",
  1448 + "delete-widgets-bundles-title": "{Count, çoðul, 1 {1 widgets bundle} diðer {# widgets bundles}} silmek istediðinizden emin misiniz?",
  1449 + "delete-widgets-bundles-action-title": "Sil {count, çoðul, 1 {1 widgets bundle} diðer {# widgets bundles}}",
  1450 + "delete-widgets-bundles-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen widget'lar paketler kaldýrýlacak ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1451 + "no-widgets-bundles-matching": "{{WidgetsBundle}} ile eþleþen hiçbir widget grubu bulunamadý. ",
  1452 + "widgets-bundle-required": "Widget paketi gerekli.",
  1453 + "system": "Sistem",
  1454 + "import": "Widget paketlerini içe aktar",
  1455 + "export": "Widget paketini dýþa aktar",
  1456 + "export-failed-error": "Widget grubu dýþa aktarýlamadý: {{error}}",
  1457 + "create-new-widgets-bundle": "Yeni widget'lar paketi oluþtur",
  1458 + "widgets-bundle-file": "Widgets paket dosyasý",
  1459 + "invalid-widgets-bundle-file-error": "Widget grubu içe aktarýlamýyor: Geçersiz widget'lar veri yapýsýný paketliyor."
  1460 + },
  1461 + "widget-config": {
  1462 + "data": "Veri",
  1463 + "settings": "Ayarlar",
  1464 + "advanced": "Ýleri",
  1465 + "title": "Baþlýk",
  1466 + "general-settings": "Genel Ayarlar",
  1467 + "display-title": "Baþlýk",
  1468 + "drop-shadow": "Düþen gölge",
  1469 + "enable-fullscreen": "Tam ekraný etkinleþtir",
  1470 + "background-color": "Arka plan rengi",
  1471 + "text-color": "Metin rengi",
  1472 + "padding": "Dolgu malzemesi",
  1473 + "margin": "Kenar",
  1474 + "widget-style": "Widget stili",
  1475 + "title-style": "Baþlýk stili",
  1476 + "mobile-mode-settings": "Mobil mod ayarlarý",
  1477 + "order": "Sipariþ",
  1478 + "height": "Yükseklik",
  1479 + "units": "Deðerin yanýnda gösterilecek özel sembol",
  1480 + "decimals": "Kayan noktadan sonraki basamak sayýsý",
  1481 + "timewindow": "Timewindow",
  1482 + "use-dashboard-timewindow": "Gösterge panosu zaman tüneli",
  1483 + "display-legend": "Gösterge efsanesi",
  1484 + "datasources": "Veri kaynaklarý",
  1485 + "maximum-datasources": "Maksimum {sayým, çoðul, 1 {1 veri kaynaðý izinli.} Diðer {# veri kaynaklarýna izin verilir}}",
  1486 + "datasource-type": "Tür",
  1487 + "datasource-parameters": "Parametreler",
  1488 + "remove-datasource": "Veri kaynaðýný kaldýr",
  1489 + "add-datasource": "Veri kaynaðý ekle",
  1490 + "target-device": "Hedef cihaz",
  1491 + "alarm-source": "Alarm kaynaðý",
  1492 + "actions": "Ýþlemler",
  1493 + "action": "Aksiyon",
  1494 + "add-action": "Eylem ekle",
  1495 + "search-actions": "Arama iþlemleri",
  1496 + "action-source": "Eylem kaynaðý",
  1497 + "action-source-required": "Eylem kaynaðý gerekli.",
  1498 + "action-name": "Ad",
  1499 + "action-name-required": "Eylem adý gerekli.",
  1500 + "action-name-not-unique": "Ayný ada sahip baþka bir eylem zaten var. <br/> Eylem adý, ayný eylem kaynaðý içinde benzersiz olmalýdýr.",
  1501 + "action-icon": "Simge",
  1502 + "action-type": "Tür",
  1503 + "action-type-required": "Eylem türü gerekli",
  1504 + "edit-action": "Eylemi düzenle",
  1505 + "delete-action": "Eylemi sil",
  1506 + "delete-action-title": "Widget eylemini sil",
  1507 + "delete-action-text": "Widget eylemini {{actionName}} adýyla silmek istediðinizden emin misiniz?"
  1508 + },
  1509 + "widget-type": {
  1510 + "import": "Widget türü içe aktar",
  1511 + "export": "Widget türünü dýþa aktar",
  1512 + "export-failed-error": "Widget türü verilemiyor: {{error}}",
  1513 + "create-new-widget-type": "Yeni pencere öðesi türü oluþtur",
  1514 + "widget-type-file": "Widget tipi dosya",
  1515 + "invalid-widget-type-file-error": "Widget türü içe aktarýlamýyor: Geçersiz pencere öðesi türü yapýsý."
  1516 + },
  1517 + "icon": {
  1518 + "icon": "Simge",
  1519 + "select-icon": "Simge seç",
  1520 + "material-icons": "Malzeme simgeleri",
  1521 + "show-all": "Tüm simgeleri göster"
  1522 + },
  1523 + "custom": {
  1524 + "widget-action": {
  1525 + "action-cell-button": "Eylem hücresi düðmesi",
  1526 + "row-click": "Satýr týklamasý",
  1527 + "marker-click": "Ýþaretçi týklamasýnda ",
  1528 + "tooltip-tag-action": "Araç ipucu etiketi iþlemi"
  1529 + }
  1530 + },
  1531 + "language": {
  1532 + "language": "Dil",
  1533 + "locales": {
  1534 + "fr_FR": "Fransýzca",
  1535 + "zh_CN": "Çince",
  1536 + "en_US": "Ýngilizce",
  1537 + "it_IT": "Ýtalyan",
  1538 + "ko_KR": "Koreli",
  1539 + "ru_RU": "Rusça",
  1540 + "es_ES": "Ýspanyol",
  1541 + "ja_JA": "Japonca",
  1542 + "TR": "Türk"
  1543 + }
  1544 + }
  1545 +}
@@ -158,12 +158,17 @@ @@ -158,12 +158,17 @@
158 "filter-type-device-type": "Device type", 158 "filter-type-device-type": "Device type",
159 "filter-type-device-type-description": "Devices of type '{{deviceType}}'", 159 "filter-type-device-type-description": "Devices of type '{{deviceType}}'",
160 "filter-type-device-type-and-name-description": "Devices of type '{{deviceType}}' and with name starting with '{{prefix}}'", 160 "filter-type-device-type-and-name-description": "Devices of type '{{deviceType}}' and with name starting with '{{prefix}}'",
  161 + "filter-type-entity-view-type": "Entity View type",
  162 + "filter-type-entity-view-type-description": "Entity Views of type '{{entityView}}'",
  163 + "filter-type-entity-view-type-and-name-description": "Entity Views of type '{{entityView}}' and with name starting with '{{prefix}}'",
161 "filter-type-relations-query": "Relations query", 164 "filter-type-relations-query": "Relations query",
162 "filter-type-relations-query-description": "{{entities}} that have {{relationType}} relation {{direction}} {{rootEntity}}", 165 "filter-type-relations-query-description": "{{entities}} that have {{relationType}} relation {{direction}} {{rootEntity}}",
163 "filter-type-asset-search-query": "Asset search query", 166 "filter-type-asset-search-query": "Asset search query",
164 "filter-type-asset-search-query-description": "Assets with types {{assetTypes}} that have {{relationType}} relation {{direction}} {{rootEntity}}", 167 "filter-type-asset-search-query-description": "Assets with types {{assetTypes}} that have {{relationType}} relation {{direction}} {{rootEntity}}",
165 "filter-type-device-search-query": "Device search query", 168 "filter-type-device-search-query": "Device search query",
166 "filter-type-device-search-query-description": "Devices with types {{deviceTypes}} that have {{relationType}} relation {{direction}} {{rootEntity}}", 169 "filter-type-device-search-query-description": "Devices with types {{deviceTypes}} that have {{relationType}} relation {{direction}} {{rootEntity}}",
  170 + "filter-type-entity-view-search-query": "Entity view search query",
  171 + "filter-type-entity-view-search-query-description": "Entity views with types {{entityViewTypes}} that have {{relationType}} relation {{direction}} {{rootEntity}}",
167 "entity-filter": "Entity filter", 172 "entity-filter": "Entity filter",
168 "resolve-multiple": "Resolve as multiple entities", 173 "resolve-multiple": "Resolve as multiple entities",
169 "filter-type": "Filter type", 174 "filter-type": "Filter type",
@@ -839,7 +844,8 @@ @@ -839,7 +844,8 @@
839 "client-attributes": "Client attributes", 844 "client-attributes": "Client attributes",
840 "shared-attributes": "Shared attributes", 845 "shared-attributes": "Shared attributes",
841 "server-attributes": "Server attributes", 846 "server-attributes": "Server attributes",
842 - "latest-timeseries": "Latest timeseries" 847 + "latest-timeseries": "Latest timeseries",
  848 + "related-entity": "Related entity"
843 }, 849 },
844 "event": { 850 "event": {
845 "event-type": "Event type", 851 "event-type": "Event type",
@@ -1551,7 +1557,8 @@ @@ -1551,7 +1557,8 @@
1551 "ko_KR": "Korean", 1557 "ko_KR": "Korean",
1552 "ru_RU": "Russian", 1558 "ru_RU": "Russian",
1553 "es_ES": "Spanish", 1559 "es_ES": "Spanish",
1554 - "ja_JA": "Japanese" 1560 + "ja_JA": "Japanese",
  1561 + "TR": "Turkish"
1555 } 1562 }
1556 } 1563 }
1557 } 1564 }