-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathIndiAutoConnect.ts
111 lines (92 loc) · 3.97 KB
/
IndiAutoConnect.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
import CancellationToken from 'cancellationtoken';
import Log from './Log';
import IndiManager from './IndiManager';
const logger = Log.logger(__filename);
// Evaluate f function, but if fail, return def
function noErr<T>(f:()=>T, def: T)
{
try {
return f();
} catch(e) {
return def;
}
}
export default class IndiAutoConnect {
// Device => connectattempted
memory:{[id:string]:boolean} = {};
indiManager: IndiManager;
constructor(indiManager:IndiManager) {
this.indiManager = indiManager;
// Change of the config flag for any will trigger recompute
indiManager.appStateManager.addSynchronizer(['indiManager', 'configuration', 'indiServer', 'devices', null, 'options', 'autoConnect'],
this.check, false);
indiManager.appStateManager.addSynchronizer(['indiManager', 'deviceTree', null, 'CONNECTION', 'childs', 'CONNECT'],
this.check, false);
// Any change of the connection status will trigger recompute
}
private check=()=>{
// Check all the devices with flag set to true
// if connection status is :
// - missing or idle (connected), clear the memory
// - busy, set the memory to done
// - idle disconnected, connect
// Remove the unknown devices from the memory
logger.debug('Recheck');
const configDevices = noErr(()=>this.indiManager.currentStatus.configuration.indiServer.devices, undefined) || {};
const configuredDevices:{[id:string]:boolean} = {};
for(let devId of Object.keys(configDevices))
{
const dev = configDevices[devId];
if (!noErr(()=>dev.options.autoConnect, false)) {
continue;
}
logger.debug('Configured device', {devId});
configuredDevices[devId] = true;
}
const unseenDevices = {...this.memory};
if (this.indiManager.connection) {
const c = this.indiManager.connection;
const deviceIds = c.getAvailableDeviceIdsWith(['CONNECTION']);
logger.debug('valid devices', {deviceIds});
for(let devId of deviceIds) {
if (!Object.prototype.hasOwnProperty.call(configuredDevices, devId)) {
continue;
}
delete(unseenDevices[devId]);
// Check the connection state of the device
// Take memory state from the state of the connection vector
const memoryState = Object.prototype.hasOwnProperty.call(this.memory, devId) ? this.memory[devId] : null;
if (memoryState === true) {
continue;
}
const vector = c.getDevice(devId).getVector('CONNECTION');
if (vector.getState() === 'Busy') {
this.memory[devId] = true;
continue;
}
const val = vector.getPropertyValueIfExists('CONNECT');
if (val === null) {
// Doesn't really exists...
unseenDevices[devId] = true;
continue;
}
this.memory[devId] = true;
if (val === 'Off') {
(async ()=> {
try {
logger.info('Auto connecting device', {devId});
await this.indiManager.connectDevice(CancellationToken.CONTINUE, {device: devId});
logger.info('Done connecting device', {devId});
} catch(e) {
logger.error('Failed to autostart', {devId});
}
})();
}
}
}
for(let devId of Object.keys(unseenDevices)) {
logger.debug('Forget about ', {devId});
delete this.memory[devId];
}
}
}