iobroker.zigbee 1.5.6 → 1.6.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (64) hide show
  1. package/.devcontainer/devcontainer.json +35 -35
  2. package/.devcontainer/docker-compose.yml +51 -51
  3. package/.devcontainer/iobroker/Dockerfile +1 -1
  4. package/.devcontainer/nginx/nginx.conf +32 -32
  5. package/.devcontainer/parcel/Dockerfile +8 -8
  6. package/.devcontainer/parcel/run.sh +6 -6
  7. package/.eslintignore +1 -1
  8. package/.eslintrc.json +36 -36
  9. package/.github/FUNDING.yml +3 -3
  10. package/.github/stale.yml +13 -13
  11. package/.github/workflows/test-and-release.yml +151 -151
  12. package/.travis/wiki.sh +27 -27
  13. package/LICENSE +21 -21
  14. package/README.md +385 -353
  15. package/admin/adapter-settings.js +244 -208
  16. package/admin/admin.js +2704 -2714
  17. package/admin/img/R7060.png +0 -0
  18. package/admin/img/WHD02.png +0 -0
  19. package/admin/img/ikea_E1812.png +0 -0
  20. package/admin/img/philips_hue_lom001.png +0 -0
  21. package/admin/img/tuya_rb280.png +0 -0
  22. package/admin/index.html +159 -159
  23. package/admin/index_m.html +1055 -965
  24. package/admin/moment.min.js +1 -1
  25. package/admin/shuffle.min.js +2 -2
  26. package/admin/tab_m.html +1025 -934
  27. package/admin/vis-network.min.js +26 -26
  28. package/admin/words.js +106 -106
  29. package/docs/de/readme.md +27 -27
  30. package/docs/en/readme.md +30 -30
  31. package/docs/flashing_via_arduino_(en).md +110 -110
  32. package/docs/ru/readme.md +28 -28
  33. package/docs/tutorial/groups-1.png +0 -0
  34. package/docs/tutorial/groups-2.png +0 -0
  35. package/docs/tutorial/tab-dev-1.png +0 -0
  36. package/docs/tutorial/zigbee.png +0 -0
  37. package/io-package.json +317 -290
  38. package/lib/backup.js +132 -132
  39. package/lib/binding.js +325 -325
  40. package/lib/colors.js +460 -460
  41. package/lib/commands.js +435 -434
  42. package/lib/developer.js +148 -144
  43. package/lib/devices.js +3119 -3109
  44. package/lib/exclude.js +168 -168
  45. package/lib/exposes.js +204 -51
  46. package/lib/groups.js +316 -316
  47. package/lib/json.js +60 -60
  48. package/lib/networkmap.js +56 -56
  49. package/lib/ota.js +153 -153
  50. package/lib/rgb.js +225 -225
  51. package/lib/seriallist.js +37 -37
  52. package/lib/states.js +6381 -6322
  53. package/lib/statescontroller.js +502 -495
  54. package/lib/tools.js +54 -54
  55. package/lib/utils.js +151 -132
  56. package/lib/zbBaseExtension.js +31 -27
  57. package/lib/zbDelayedAction.js +151 -146
  58. package/lib/zbDeviceAvailability.js +306 -304
  59. package/lib/zbDeviceConfigure.js +148 -143
  60. package/lib/zbDeviceEvent.js +43 -43
  61. package/lib/zigbeecontroller.js +856 -822
  62. package/main.js +113 -39
  63. package/package.json +74 -73
  64. package/support/docgen.js +93 -93
package/lib/groups.js CHANGED
@@ -1,316 +1,316 @@
1
- 'use strict';
2
-
3
- const statesMapping = require('./devices');
4
-
5
-
6
- class Groups {
7
- constructor(adapter) {
8
- this.adapter = adapter;
9
- this.adapter.on('message', this.onMessage.bind(this));
10
- }
11
-
12
- start(zbController, stController) {
13
- this.zbController = zbController;
14
- this.stController = stController;
15
- this.adapter.getStateAsync('info.groups')
16
- .then((groupsState) => {
17
- const groups = (groupsState && groupsState.val) ? JSON.parse(groupsState.val) : {};
18
- this.syncGroups(groups);
19
- });
20
- }
21
-
22
- stop() {
23
- delete this.zbController;
24
- delete this.stController;
25
- }
26
-
27
- info(msg) {
28
- this.adapter.log.info(msg);
29
- }
30
-
31
- error(msg) {
32
- this.adapter.log.error(msg);
33
- }
34
-
35
- debug(msg) {
36
- this.adapter.log.debug(msg);
37
- }
38
-
39
- warn(msg) {
40
- this.adapter.log.warn(msg);
41
- }
42
-
43
- /**
44
- * @param {ioBroker.Message} obj
45
- */
46
- onMessage(obj) {
47
- if (typeof obj === 'object' && obj.command) {
48
- switch (obj.command) {
49
- case 'getGroups':
50
- this.getGroups(obj);
51
- break;
52
- case 'renameGroup':
53
- // used for renaming AND creating groups
54
- this.renameGroup(obj.from, obj.command, obj.message, obj.callback);
55
- break;
56
- case 'deleteGroup':
57
- this.deleteGroup(obj.from, obj.command, obj.message, obj.callback);
58
- break;
59
- case 'updateGroupMembership':
60
- this.updateGroupMembership(obj.from, obj.command, obj.message, obj.callback);
61
- break;
62
- }
63
- }
64
- }
65
-
66
- async getGroupMembersFromController(id) {
67
- const members = [];
68
- try {
69
- const group = await this.zbController.getGroupByID(id);
70
- if (group) {
71
- const groupmembers = group.members;
72
-
73
- for (const member of groupmembers) {
74
- const nwk = member.deviceNetworkAddress;
75
- const device = this.zbController.getDeviceByNetworkAddress(nwk);
76
- if (device && device.ieeeAddr) members.push( { device:device.ieeeAddr } );
77
- }
78
- }
79
- else {
80
- return undefined;
81
- }
82
-
83
- } catch (error) {
84
- if (error) this.error('getGroupMembersFromController: error is ' + JSON.stringify(error) + ' ' + JSON.stringify(new Error().stack));
85
- else this.error('unidentifed error in getGroupMembersFromController');
86
- }
87
- return members;
88
- }
89
-
90
- async getGroups(obj) {
91
- const response = { groups: {} };
92
- try {
93
- const groupsState = await this.adapter.getStateAsync('info.groups');
94
- const herdsmanGroups = await this.zbController.getGroups();
95
-
96
- const groups = (groupsState && groupsState.val) ? JSON.parse(groupsState.val) : {};
97
-
98
- if (typeof herdsmanGroups === 'object') {
99
- for (const group of herdsmanGroups) {
100
- const gid = group.groupID;
101
- if (gid && groups[gid]=== undefined) {
102
- groups[gid] = `Auto Group ${gid}`;
103
- }
104
- }
105
- }
106
- this.debug('getGroups result: ' + JSON.stringify(groups));
107
- response.groups = groups;
108
- }
109
- catch (error) {
110
- response.error = `getGroups: caught error: ${error}`;
111
- this.error(`getGroups: caught error: ${error}`);
112
- }
113
- finally {
114
- if (obj)
115
- this.adapter.sendTo(obj.from, obj.command, response, obj.callback);
116
-
117
- }
118
- }
119
-
120
- async updateGroupMembership(from, command, message, callback) {
121
- try {
122
- const groups = (message && message.groups ? message.groups : []);
123
- const devId = (message && message.id ? message.id : undefined);
124
- if (devId === undefined) {
125
- this.adapter.sendTo(from, command, {error: 'No device specified'}, callback);
126
- }
127
- const sysid = devId.replace(this.adapter.namespace + '.', '0x');
128
- const id = `${devId}.groups`;
129
-
130
- this.adapter.setState(id, JSON.stringify(groups), true);
131
- let response = await this.zbController.removeDevFromAllGroups(sysid);
132
- if (response && response.error)
133
- {
134
- this.adapter.sendTo(from, command, response, callback);
135
- return;
136
- }
137
- for (const groupId of groups) {
138
- response = await this.zbController.addDevToGroup(sysid, parseInt(groupId));
139
- if (response && response.error)
140
- {
141
- this.adapter.sendTo(from, command, response, callback);
142
- return;
143
- }
144
- }
145
-
146
- } catch (e) {
147
- this.adapter.sendTo(from, command, {error: e}, callback);
148
- return;
149
- }
150
- this.adapter.sendTo(from, command, { }, callback);
151
- }
152
-
153
-
154
- async queryGroupMemberState(groupID, stateDesc) {
155
- const members = await this.getGroupMembersFromController(groupID);
156
- const result = {
157
- unsupported: [],
158
- unread: []
159
- };
160
- for (const member of members) {
161
- const entity = await this.zbController.resolveEntity(member.device);
162
- if (!entity) return false;
163
- this.debug(`entity: ${JSON.stringify(entity)}`);
164
- const mappedModel = entity.mapped;
165
- this.debug('Mapped Model: ' + JSON.stringify(mappedModel));
166
- const converter = mappedModel.toZigbee.find((c) => c && (c.key.includes(stateDesc.prop) || c.key.includes(stateDesc.setattr) || c.key.includes(stateDesc.id)));
167
- if (!converter) {
168
- result.unsupported.push(member.device);
169
- continue;
170
- }
171
- if (converter.hasOwnProperty('convertGet')) {
172
- try {
173
- await converter.convertGet(entity.device.endpoints[0], stateDesc.id, {});
174
- } catch (error) {
175
- result.unread.push(member.device);
176
- }
177
- }
178
- }
179
- if (result.unsupported.length>0) {
180
- // this.warn('unsupported ' + stateDesc.id + ' change for group members ' + JSON.stringify(result.unsupported));
181
- const error = {
182
- code: 134,
183
- message: `unsupported ${stateDesc.id} change for group members ${result.unsupported.join()}`
184
- };
185
- throw error;
186
- }
187
- if (result.unread.length>0) {
188
- this.warn('unread ' + stateDesc.id + ' change for group members ' + JSON.stringify(result.unread));
189
- }
190
- }
191
-
192
- async deleteGroup(from, command, message) {
193
- const members = await this.getGroupMembersFromController(parseInt(message));
194
- if (members && members.length) {
195
- for (const member of members) {
196
- const devName = member.device.substring(2);
197
- const groupEntry = this.adapter.getStateAsync(`${devName}.groups`);
198
- const memberarray = (groupEntry && groupEntry.val) ? JSON.parse(groupEntry.val) : [];
199
- const index = memberarray.indexOf(message.toString());
200
- if (index > -1) {
201
- memberarray.splice(index, 1);
202
- }
203
- if (memberarray.length > 0) {
204
- await this.adapter.setStateAsync(`${devName}.groups`, JSON.stringify(memberarray), true);
205
- }
206
- else {
207
- await this.adapter.setStateAsync(`${devName}.groups`, '', true);
208
- }
209
- }
210
- }
211
- const groupsEntry = await this.adapter.getStateAsync('info.groups');
212
- const objGroups = (groupsEntry && groupsEntry.val ? JSON.parse(groupsEntry.val) : {});
213
- delete objGroups[message.toString()];
214
- await this.adapter.setStateAsync('info.groups', JSON.stringify(objGroups), true);
215
- await this.zbController.removeGroupById(message);
216
- this.stController.deleteDeviceStates(`group_${parseInt(message)}`);
217
- }
218
-
219
- async renameGroup(from, command, message) {
220
- const groupsEntry = await this.adapter.getStateAsync('info.groups');
221
- const objGroups = (groupsEntry && groupsEntry.val ? JSON.parse(groupsEntry.val) : {});
222
- const name = message.name;
223
- const id = `group_${message.id}`;
224
- objGroups[message.id.toString()] = message.name;
225
- await this.adapter.setStateAsync('info.groups', JSON.stringify(objGroups), true);
226
-
227
- const group = await this.adapter.getStateAsync(id);
228
- if (!group) {
229
- // assume we have to create the group
230
- this.adapter.setObjectNotExists(id, {
231
- type: 'device',
232
- common: {name: name, type: 'group'},
233
- native: {id: id}
234
- }, () => {
235
- this.adapter.extendObject(id , {common: {name: name, type: 'group'}});
236
- // create writable states for groups from their devices
237
- for (const stateInd in statesMapping.groupStates) {
238
- if (!statesMapping.groupStates.hasOwnProperty(stateInd)) continue;
239
- const statedesc = statesMapping.groupStates[stateInd];
240
- const common = {
241
- name: statedesc.name,
242
- type: statedesc.type,
243
- unit: statedesc.unit,
244
- read: statedesc.read,
245
- write: statedesc.write,
246
- icon: statedesc.icon,
247
- role: statedesc.role,
248
- min: statedesc.min,
249
- max: statedesc.max,
250
- };
251
- this.stController.updateState(id, statedesc.id, undefined, common);
252
- }
253
- });
254
- }
255
- }
256
-
257
-
258
- syncGroups(groups) {
259
- const chain = [];
260
- const usedGroupsIds = [];
261
- for (const j in groups) {
262
- if (groups.hasOwnProperty(j)) {
263
- const id = `group_${j}`,
264
- name = groups[j];
265
- chain.push(new Promise((resolve) => {
266
- this.adapter.setObjectNotExists(id, {
267
- type: 'device',
268
- common: {name: name, type: 'group'},
269
- native: {id: j}
270
- }, () => {
271
- this.adapter.extendObject(id, {common: {type: 'group'}});
272
- // create writable states for groups from their devices
273
- for (const stateInd in statesMapping.groupStates) {
274
- if (!statesMapping.groupStates.hasOwnProperty(stateInd)) continue;
275
- const statedesc = statesMapping.groupStates[stateInd];
276
- const common = {
277
- name: statedesc.name,
278
- type: statedesc.type,
279
- unit: statedesc.unit,
280
- read: statedesc.read,
281
- write: statedesc.write,
282
- icon: statedesc.icon,
283
- role: statedesc.role,
284
- min: statedesc.min,
285
- max: statedesc.max,
286
- };
287
- this.stController.updateState(id, statedesc.id, undefined, common);
288
- }
289
- resolve();
290
- });
291
- }));
292
- usedGroupsIds.push(parseInt(j));
293
- }
294
- }
295
- chain.push(new Promise((resolve) => {
296
- // remove unused adpter groups
297
- this.adapter.getDevices((err, devices) => {
298
- if (!err) {
299
- devices.forEach((dev) => {
300
- if (dev.common.type === 'group') {
301
- const groupid = parseInt(dev.native.id);
302
- if (!usedGroupsIds.includes(groupid)) {
303
- this.stController.deleteDeviceStates(`group_${groupid}`);
304
- }
305
- }
306
- });
307
- }
308
- resolve();
309
- });
310
- }));
311
- Promise.all(chain);
312
- }
313
-
314
- }
315
-
316
- module.exports = Groups;
1
+ 'use strict';
2
+
3
+ const statesMapping = require('./devices');
4
+
5
+
6
+ class Groups {
7
+ constructor(adapter) {
8
+ this.adapter = adapter;
9
+ this.adapter.on('message', this.onMessage.bind(this));
10
+ }
11
+
12
+ start(zbController, stController) {
13
+ this.zbController = zbController;
14
+ this.stController = stController;
15
+ this.adapter.getStateAsync('info.groups')
16
+ .then((groupsState) => {
17
+ const groups = (groupsState && groupsState.val) ? JSON.parse(groupsState.val) : {};
18
+ this.syncGroups(groups);
19
+ });
20
+ }
21
+
22
+ stop() {
23
+ delete this.zbController;
24
+ delete this.stController;
25
+ }
26
+
27
+ info(msg) {
28
+ this.adapter.log.info(msg);
29
+ }
30
+
31
+ error(msg) {
32
+ this.adapter.log.error(msg);
33
+ }
34
+
35
+ debug(msg) {
36
+ this.adapter.log.debug(msg);
37
+ }
38
+
39
+ warn(msg) {
40
+ this.adapter.log.warn(msg);
41
+ }
42
+
43
+ /**
44
+ * @param {ioBroker.Message} obj
45
+ */
46
+ onMessage(obj) {
47
+ if (typeof obj === 'object' && obj.command) {
48
+ switch (obj.command) {
49
+ case 'getGroups':
50
+ this.getGroups(obj);
51
+ break;
52
+ case 'renameGroup':
53
+ // used for renaming AND creating groups
54
+ this.renameGroup(obj.from, obj.command, obj.message, obj.callback);
55
+ break;
56
+ case 'deleteGroup':
57
+ this.deleteGroup(obj.from, obj.command, obj.message, obj.callback);
58
+ break;
59
+ case 'updateGroupMembership':
60
+ this.updateGroupMembership(obj.from, obj.command, obj.message, obj.callback);
61
+ break;
62
+ }
63
+ }
64
+ }
65
+
66
+ async getGroupMembersFromController(id) {
67
+ const members = [];
68
+ try {
69
+ const group = await this.zbController.getGroupByID(id);
70
+ if (group) {
71
+ const groupmembers = group.members;
72
+
73
+ for (const member of groupmembers) {
74
+ const nwk = member.deviceNetworkAddress;
75
+ const device = this.zbController.getDeviceByNetworkAddress(nwk);
76
+ if (device && device.ieeeAddr) members.push( { device:device.ieeeAddr } );
77
+ }
78
+ }
79
+ else {
80
+ return undefined;
81
+ }
82
+
83
+ } catch (error) {
84
+ if (error) this.error('getGroupMembersFromController: error is ' + JSON.stringify(error) + ' ' + JSON.stringify(new Error().stack));
85
+ else this.error('unidentifed error in getGroupMembersFromController');
86
+ }
87
+ return members;
88
+ }
89
+
90
+ async getGroups(obj) {
91
+ const response = { groups: {} };
92
+ try {
93
+ const groupsState = await this.adapter.getStateAsync('info.groups');
94
+ const herdsmanGroups = await this.zbController.getGroups();
95
+
96
+ const groups = (groupsState && groupsState.val) ? JSON.parse(groupsState.val) : {};
97
+
98
+ if (typeof herdsmanGroups === 'object') {
99
+ for (const group of herdsmanGroups) {
100
+ const gid = group.groupID;
101
+ if (gid && groups[gid]=== undefined) {
102
+ groups[gid] = `Auto Group ${gid}`;
103
+ }
104
+ }
105
+ }
106
+ this.debug('getGroups result: ' + JSON.stringify(groups));
107
+ response.groups = groups;
108
+ }
109
+ catch (error) {
110
+ response.error = `getGroups: caught error: ${error}`;
111
+ this.error(`getGroups: caught error: ${error}`);
112
+ }
113
+ finally {
114
+ if (obj)
115
+ this.adapter.sendTo(obj.from, obj.command, response, obj.callback);
116
+
117
+ }
118
+ }
119
+
120
+ async updateGroupMembership(from, command, message, callback) {
121
+ try {
122
+ const groups = (message && message.groups ? message.groups : []);
123
+ const devId = (message && message.id ? message.id : undefined);
124
+ if (devId === undefined) {
125
+ this.adapter.sendTo(from, command, {error: 'No device specified'}, callback);
126
+ }
127
+ const sysid = devId.replace(this.adapter.namespace + '.', '0x');
128
+ const id = `${devId}.groups`;
129
+
130
+ this.adapter.setState(id, JSON.stringify(groups), true);
131
+ let response = await this.zbController.removeDevFromAllGroups(sysid);
132
+ if (response && response.error)
133
+ {
134
+ this.adapter.sendTo(from, command, response, callback);
135
+ return;
136
+ }
137
+ for (const groupId of groups) {
138
+ response = await this.zbController.addDevToGroup(sysid, parseInt(groupId));
139
+ if (response && response.error)
140
+ {
141
+ this.adapter.sendTo(from, command, response, callback);
142
+ return;
143
+ }
144
+ }
145
+
146
+ } catch (e) {
147
+ this.adapter.sendTo(from, command, {error: e}, callback);
148
+ return;
149
+ }
150
+ this.adapter.sendTo(from, command, { }, callback);
151
+ }
152
+
153
+
154
+ async queryGroupMemberState(groupID, stateDesc) {
155
+ const members = await this.getGroupMembersFromController(groupID);
156
+ const result = {
157
+ unsupported: [],
158
+ unread: []
159
+ };
160
+ for (const member of members) {
161
+ const entity = await this.zbController.resolveEntity(member.device);
162
+ if (!entity) return false;
163
+ this.debug(`entity: ${JSON.stringify(entity)}`);
164
+ const mappedModel = entity.mapped;
165
+ this.debug('Mapped Model: ' + JSON.stringify(mappedModel));
166
+ const converter = mappedModel.toZigbee.find((c) => c && (c.key.includes(stateDesc.prop) || c.key.includes(stateDesc.setattr) || c.key.includes(stateDesc.id)));
167
+ if (!converter) {
168
+ result.unsupported.push(member.device);
169
+ continue;
170
+ }
171
+ if (converter.hasOwnProperty('convertGet')) {
172
+ try {
173
+ await converter.convertGet(entity.device.endpoints[0], stateDesc.id, {});
174
+ } catch (error) {
175
+ result.unread.push(member.device);
176
+ }
177
+ }
178
+ }
179
+ if (result.unsupported.length>0) {
180
+ // this.warn('unsupported ' + stateDesc.id + ' change for group members ' + JSON.stringify(result.unsupported));
181
+ const error = {
182
+ code: 134,
183
+ message: `unsupported ${stateDesc.id} change for group members ${result.unsupported.join()}`
184
+ };
185
+ throw error;
186
+ }
187
+ if (result.unread.length>0) {
188
+ this.warn('unread ' + stateDesc.id + ' change for group members ' + JSON.stringify(result.unread));
189
+ }
190
+ }
191
+
192
+ async deleteGroup(from, command, message) {
193
+ const members = await this.getGroupMembersFromController(parseInt(message));
194
+ if (members && members.length) {
195
+ for (const member of members) {
196
+ const devName = member.device.substring(2);
197
+ const groupEntry = this.adapter.getStateAsync(`${devName}.groups`);
198
+ const memberarray = (groupEntry && groupEntry.val) ? JSON.parse(groupEntry.val) : [];
199
+ const index = memberarray.indexOf(message.toString());
200
+ if (index > -1) {
201
+ memberarray.splice(index, 1);
202
+ }
203
+ if (memberarray.length > 0) {
204
+ await this.adapter.setStateAsync(`${devName}.groups`, JSON.stringify(memberarray), true);
205
+ }
206
+ else {
207
+ await this.adapter.setStateAsync(`${devName}.groups`, '', true);
208
+ }
209
+ }
210
+ }
211
+ const groupsEntry = await this.adapter.getStateAsync('info.groups');
212
+ const objGroups = (groupsEntry && groupsEntry.val ? JSON.parse(groupsEntry.val) : {});
213
+ delete objGroups[message.toString()];
214
+ await this.adapter.setStateAsync('info.groups', JSON.stringify(objGroups), true);
215
+ await this.zbController.removeGroupById(message);
216
+ this.stController.deleteDeviceStates(`group_${parseInt(message)}`);
217
+ }
218
+
219
+ async renameGroup(from, command, message) {
220
+ const groupsEntry = await this.adapter.getStateAsync('info.groups');
221
+ const objGroups = (groupsEntry && groupsEntry.val ? JSON.parse(groupsEntry.val) : {});
222
+ const name = message.name;
223
+ const id = `group_${message.id}`;
224
+ objGroups[message.id.toString()] = message.name;
225
+ await this.adapter.setStateAsync('info.groups', JSON.stringify(objGroups), true);
226
+
227
+ const group = await this.adapter.getStateAsync(id);
228
+ if (!group) {
229
+ // assume we have to create the group
230
+ this.adapter.setObjectNotExists(id, {
231
+ type: 'device',
232
+ common: {name: name, type: 'group'},
233
+ native: {id: id}
234
+ }, () => {
235
+ this.adapter.extendObject(id , {common: {name: name, type: 'group'}});
236
+ // create writable states for groups from their devices
237
+ for (const stateInd in statesMapping.groupStates) {
238
+ if (!statesMapping.groupStates.hasOwnProperty(stateInd)) continue;
239
+ const statedesc = statesMapping.groupStates[stateInd];
240
+ const common = {
241
+ name: statedesc.name,
242
+ type: statedesc.type,
243
+ unit: statedesc.unit,
244
+ read: statedesc.read,
245
+ write: statedesc.write,
246
+ icon: statedesc.icon,
247
+ role: statedesc.role,
248
+ min: statedesc.min,
249
+ max: statedesc.max,
250
+ };
251
+ this.stController.updateState(id, statedesc.id, undefined, common);
252
+ }
253
+ });
254
+ }
255
+ }
256
+
257
+
258
+ syncGroups(groups) {
259
+ const chain = [];
260
+ const usedGroupsIds = [];
261
+ for (const j in groups) {
262
+ if (groups.hasOwnProperty(j)) {
263
+ const id = `group_${j}`,
264
+ name = groups[j];
265
+ chain.push(new Promise((resolve) => {
266
+ this.adapter.setObjectNotExists(id, {
267
+ type: 'device',
268
+ common: {name: name, type: 'group'},
269
+ native: {id: j}
270
+ }, () => {
271
+ this.adapter.extendObject(id, {common: {type: 'group'}});
272
+ // create writable states for groups from their devices
273
+ for (const stateInd in statesMapping.groupStates) {
274
+ if (!statesMapping.groupStates.hasOwnProperty(stateInd)) continue;
275
+ const statedesc = statesMapping.groupStates[stateInd];
276
+ const common = {
277
+ name: statedesc.name,
278
+ type: statedesc.type,
279
+ unit: statedesc.unit,
280
+ read: statedesc.read,
281
+ write: statedesc.write,
282
+ icon: statedesc.icon,
283
+ role: statedesc.role,
284
+ min: statedesc.min,
285
+ max: statedesc.max,
286
+ };
287
+ this.stController.updateState(id, statedesc.id, undefined, common);
288
+ }
289
+ resolve();
290
+ });
291
+ }));
292
+ usedGroupsIds.push(parseInt(j));
293
+ }
294
+ }
295
+ chain.push(new Promise((resolve) => {
296
+ // remove unused adpter groups
297
+ this.adapter.getDevices((err, devices) => {
298
+ if (!err) {
299
+ devices.forEach((dev) => {
300
+ if (dev.common.type === 'group') {
301
+ const groupid = parseInt(dev.native.id);
302
+ if (!usedGroupsIds.includes(groupid)) {
303
+ this.stController.deleteDeviceStates(`group_${groupid}`);
304
+ }
305
+ }
306
+ });
307
+ }
308
+ resolve();
309
+ });
310
+ }));
311
+ Promise.all(chain);
312
+ }
313
+
314
+ }
315
+
316
+ module.exports = Groups;