incyclist-devices 1.4.97 → 1.4.100
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.
- package/LICENSE +0 -0
- package/lib/CyclingMode.d.ts +76 -76
- package/lib/CyclingMode.js +79 -79
- package/lib/Device.d.ts +92 -92
- package/lib/Device.js +71 -71
- package/lib/DeviceProtocol.d.ts +74 -74
- package/lib/DeviceProtocol.js +41 -41
- package/lib/DeviceRegistry.d.ts +8 -8
- package/lib/DeviceRegistry.js +33 -33
- package/lib/DeviceSupport.d.ts +34 -34
- package/lib/DeviceSupport.js +78 -78
- package/lib/ant/AntAdapter.d.ts +50 -50
- package/lib/ant/AntAdapter.js +109 -109
- package/lib/ant/AntScanner.d.ts +60 -60
- package/lib/ant/AntScanner.js +651 -651
- package/lib/ant/antfe/AntFEAdapter.d.ts +83 -78
- package/lib/ant/antfe/AntFEAdapter.js +652 -611
- package/lib/ant/antfe/ant-fe-adv-st-mode.d.ts +9 -9
- package/lib/ant/antfe/ant-fe-adv-st-mode.js +51 -51
- package/lib/ant/antfe/ant-fe-erg-mode.d.ts +6 -6
- package/lib/ant/antfe/ant-fe-erg-mode.js +14 -14
- package/lib/ant/antfe/ant-fe-st-mode.d.ts +5 -5
- package/lib/ant/antfe/ant-fe-st-mode.js +13 -13
- package/lib/ant/anthrm/AntHrmAdapter.d.ts +16 -16
- package/lib/ant/anthrm/AntHrmAdapter.js +130 -130
- package/lib/ant/antpwr/pwr-adapter.d.ts +49 -49
- package/lib/ant/antpwr/pwr-adapter.js +251 -251
- package/lib/ant/utils.d.ts +1 -1
- package/lib/ant/utils.js +23 -23
- package/lib/ble/ble-device.d.ts +63 -63
- package/lib/ble/ble-device.js +442 -442
- package/lib/ble/ble-erg-mode.d.ts +18 -18
- package/lib/ble/ble-erg-mode.js +132 -127
- package/lib/ble/ble-interface.d.ts +100 -99
- package/lib/ble/ble-interface.js +717 -712
- package/lib/ble/ble-peripheral.d.ts +36 -36
- package/lib/ble/ble-peripheral.js +200 -200
- package/lib/ble/ble-st-mode.d.ts +15 -15
- package/lib/ble/ble-st-mode.js +102 -102
- package/lib/ble/ble.d.ts +129 -129
- package/lib/ble/ble.js +86 -86
- package/lib/ble/consts.d.ts +14 -14
- package/lib/ble/consts.js +17 -17
- package/lib/ble/fm.d.ts +125 -125
- package/lib/ble/fm.js +739 -739
- package/lib/ble/hrm.d.ts +48 -48
- package/lib/ble/hrm.js +134 -134
- package/lib/ble/incyclist-protocol.d.ts +31 -31
- package/lib/ble/incyclist-protocol.js +147 -147
- package/lib/ble/pwr.d.ts +89 -89
- package/lib/ble/pwr.js +321 -321
- package/lib/ble/tacx.d.ts +90 -90
- package/lib/ble/tacx.js +731 -730
- package/lib/ble/wahoo-kickr.d.ts +98 -98
- package/lib/ble/wahoo-kickr.js +496 -496
- package/lib/calculations.d.ts +13 -13
- package/lib/calculations.js +150 -150
- package/lib/daum/DaumAdapter.d.ts +66 -66
- package/lib/daum/DaumAdapter.js +396 -396
- package/lib/daum/DaumPowerMeterCyclingMode.d.ts +8 -8
- package/lib/daum/DaumPowerMeterCyclingMode.js +21 -21
- package/lib/daum/ERGCyclingMode.d.ts +26 -26
- package/lib/daum/ERGCyclingMode.js +201 -201
- package/lib/daum/SmartTrainerCyclingMode.d.ts +41 -41
- package/lib/daum/SmartTrainerCyclingMode.js +344 -344
- package/lib/daum/classic/DaumClassicAdapter.d.ts +18 -18
- package/lib/daum/classic/DaumClassicAdapter.js +146 -146
- package/lib/daum/classic/DaumClassicCyclingMode.d.ts +13 -13
- package/lib/daum/classic/DaumClassicCyclingMode.js +97 -97
- package/lib/daum/classic/DaumClassicProtocol.d.ts +27 -27
- package/lib/daum/classic/DaumClassicProtocol.js +185 -185
- package/lib/daum/classic/bike.d.ts +64 -64
- package/lib/daum/classic/bike.js +456 -456
- package/lib/daum/classic/utils.d.ts +13 -13
- package/lib/daum/classic/utils.js +143 -143
- package/lib/daum/constants.d.ts +19 -19
- package/lib/daum/constants.js +22 -22
- package/lib/daum/premium/DaumClassicCyclingMode.d.ts +14 -14
- package/lib/daum/premium/DaumClassicCyclingMode.js +86 -86
- package/lib/daum/premium/DaumPremiumAdapter.d.ts +12 -12
- package/lib/daum/premium/DaumPremiumAdapter.js +131 -131
- package/lib/daum/premium/DaumPremiumProtocol.d.ts +32 -32
- package/lib/daum/premium/DaumPremiumProtocol.js +207 -207
- package/lib/daum/premium/bike.d.ts +123 -123
- package/lib/daum/premium/bike.js +894 -894
- package/lib/daum/premium/tcpserial.d.ts +33 -33
- package/lib/daum/premium/tcpserial.js +123 -123
- package/lib/daum/premium/utils.d.ts +62 -62
- package/lib/daum/premium/utils.js +376 -376
- package/lib/kettler/comms.d.ts +59 -59
- package/lib/kettler/comms.js +242 -242
- package/lib/kettler/ergo-racer/ERGCyclingMode.d.ts +25 -25
- package/lib/kettler/ergo-racer/ERGCyclingMode.js +144 -145
- package/lib/kettler/ergo-racer/adapter.d.ts +101 -101
- package/lib/kettler/ergo-racer/adapter.js +639 -639
- package/lib/kettler/ergo-racer/protocol.d.ts +41 -41
- package/lib/kettler/ergo-racer/protocol.js +203 -203
- package/lib/modes/power-base.d.ts +20 -20
- package/lib/modes/power-base.js +70 -70
- package/lib/modes/power-meter.d.ts +20 -20
- package/lib/modes/power-meter.js +78 -78
- package/lib/modes/simulator.d.ts +29 -29
- package/lib/modes/simulator.js +140 -140
- package/lib/simulator/Simulator.d.ts +69 -69
- package/lib/simulator/Simulator.js +288 -288
- package/lib/types/command.d.ts +8 -8
- package/lib/types/command.js +2 -2
- package/lib/types/route.d.ts +24 -24
- package/lib/types/route.js +9 -9
- package/lib/types/user.d.ts +11 -11
- package/lib/types/user.js +9 -9
- package/lib/utils.d.ts +14 -14
- package/lib/utils.js +114 -114
- package/package.json +46 -46
- package/lib/ant/antfe/ant-fe-st-mode copy.d.ts +0 -7
- package/lib/ant/antfe/ant-fe-st-mode copy.js +0 -54
- package/lib/ant/antpwr/AntPWRAdapter.d.ts +0 -24
- package/lib/ant/antpwr/AntPWRAdapter.js +0 -252
- package/lib/daum/PowerMeterCyclingMode.d.ts +0 -18
- package/lib/daum/PowerMeterCyclingMode.js +0 -78
- package/lib/daum/classic/ERGCyclingMode.d.ts +0 -23
- package/lib/daum/classic/ERGCyclingMode.js +0 -171
- package/lib/daum/indoorbike.d.ts +0 -24
- package/lib/daum/indoorbike.js +0 -178
- package/lib/kettler/ergo-racer/modes/power-meter.d.ts +0 -18
- package/lib/kettler/ergo-racer/modes/power-meter.js +0 -86
- package/lib/simulator/simulator-mode.d.ts +0 -28
- package/lib/simulator/simulator-mode.js +0 -120
|
@@ -1,639 +1,639 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
-
if (k2 === undefined) k2 = k;
|
|
4
|
-
Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
|
|
5
|
-
}) : (function(o, m, k, k2) {
|
|
6
|
-
if (k2 === undefined) k2 = k;
|
|
7
|
-
o[k2] = m[k];
|
|
8
|
-
}));
|
|
9
|
-
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
10
|
-
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
11
|
-
}) : function(o, v) {
|
|
12
|
-
o["default"] = v;
|
|
13
|
-
});
|
|
14
|
-
var __importStar = (this && this.__importStar) || function (mod) {
|
|
15
|
-
if (mod && mod.__esModule) return mod;
|
|
16
|
-
var result = {};
|
|
17
|
-
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
|
18
|
-
__setModuleDefault(result, mod);
|
|
19
|
-
return result;
|
|
20
|
-
};
|
|
21
|
-
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
22
|
-
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
23
|
-
return new (P || (P = Promise))(function (resolve, reject) {
|
|
24
|
-
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
25
|
-
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
26
|
-
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
27
|
-
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
28
|
-
});
|
|
29
|
-
};
|
|
30
|
-
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
31
|
-
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
32
|
-
};
|
|
33
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
34
|
-
const Device_1 = __importStar(require("../../Device"));
|
|
35
|
-
const gd_eventlog_1 = require("gd-eventlog");
|
|
36
|
-
const comms_1 = __importDefault(require("../comms"));
|
|
37
|
-
const utils_1 = require("../../utils");
|
|
38
|
-
const power_meter_1 = __importDefault(require("../../modes/power-meter"));
|
|
39
|
-
const ERGCyclingMode_1 = __importDefault(require("./ERGCyclingMode"));
|
|
40
|
-
class KettlerRacerAdapter extends Device_1.default {
|
|
41
|
-
constructor(protocol, settings) {
|
|
42
|
-
super(protocol, settings);
|
|
43
|
-
this.requests = [];
|
|
44
|
-
this.logger = new gd_eventlog_1.EventLogger('KettlerRacer');
|
|
45
|
-
this.ignoreHrm = false;
|
|
46
|
-
this.ignorePower = false;
|
|
47
|
-
this.ignoreBike = false;
|
|
48
|
-
this.paused = false;
|
|
49
|
-
this.iv = null;
|
|
50
|
-
this.comms = new comms_1.default({ protocol, port: this.settings.port, logger: this.logger });
|
|
51
|
-
}
|
|
52
|
-
isBike() { return true; }
|
|
53
|
-
isPower() { return true; }
|
|
54
|
-
isHrm() { return true; }
|
|
55
|
-
isSame(device) {
|
|
56
|
-
if (!(device instanceof KettlerRacerAdapter))
|
|
57
|
-
return false;
|
|
58
|
-
const adapter = device;
|
|
59
|
-
return (adapter.getName() === this.getName() && adapter.getPort() === this.getPort());
|
|
60
|
-
}
|
|
61
|
-
setID(id) {
|
|
62
|
-
this.id = id;
|
|
63
|
-
}
|
|
64
|
-
getID() {
|
|
65
|
-
return this.id;
|
|
66
|
-
}
|
|
67
|
-
getName() {
|
|
68
|
-
return this.settings.name || this.getProtocolName();
|
|
69
|
-
}
|
|
70
|
-
getPort() {
|
|
71
|
-
return this.settings.port;
|
|
72
|
-
}
|
|
73
|
-
setIgnoreHrm(ignore) {
|
|
74
|
-
this.ignoreHrm = ignore;
|
|
75
|
-
}
|
|
76
|
-
setIgnorePower(ignore) {
|
|
77
|
-
this.ignorePower = ignore;
|
|
78
|
-
}
|
|
79
|
-
setIgnoreBike(ignore) {
|
|
80
|
-
this.ignoreBike = ignore;
|
|
81
|
-
}
|
|
82
|
-
_getComms() {
|
|
83
|
-
return this.comms;
|
|
84
|
-
}
|
|
85
|
-
_setComms(comms) {
|
|
86
|
-
this.comms = comms;
|
|
87
|
-
}
|
|
88
|
-
getLogger() {
|
|
89
|
-
return this.logger;
|
|
90
|
-
}
|
|
91
|
-
getUserSettings() {
|
|
92
|
-
return this.settings.userSettings || { weight: Device_1.DEFAULT_USER_WEIGHT };
|
|
93
|
-
}
|
|
94
|
-
getWeight() {
|
|
95
|
-
let userWeight = Device_1.DEFAULT_USER_WEIGHT;
|
|
96
|
-
let bikeWeight = Device_1.DEFAULT_BIKE_WEIGHT;
|
|
97
|
-
if (this.settings.userSettings && this.settings.userSettings.weight) {
|
|
98
|
-
userWeight = this.settings.userSettings.weight;
|
|
99
|
-
}
|
|
100
|
-
if (this.settings.bikeSettings && this.settings.bikeSettings.weight) {
|
|
101
|
-
userWeight = this.settings.bikeSettings.weight;
|
|
102
|
-
}
|
|
103
|
-
return bikeWeight + userWeight;
|
|
104
|
-
}
|
|
105
|
-
setComputerMode() {
|
|
106
|
-
return this.send('setComputerMode', 'CP').then(response => {
|
|
107
|
-
this.logger.logEvent({ response });
|
|
108
|
-
if (response === 'ACK' || response === 'RUN') {
|
|
109
|
-
return true;
|
|
110
|
-
}
|
|
111
|
-
else {
|
|
112
|
-
return false;
|
|
113
|
-
}
|
|
114
|
-
});
|
|
115
|
-
}
|
|
116
|
-
setClientMode() {
|
|
117
|
-
return this.send('setClientMode', 'CM').then(response => {
|
|
118
|
-
this.logger.logEvent({ response });
|
|
119
|
-
if (response === 'ACK' || response === 'RUN') {
|
|
120
|
-
return true;
|
|
121
|
-
}
|
|
122
|
-
else {
|
|
123
|
-
return false;
|
|
124
|
-
}
|
|
125
|
-
});
|
|
126
|
-
}
|
|
127
|
-
reset() {
|
|
128
|
-
return this.send('reset', 'RS').then(response => {
|
|
129
|
-
this.logger.logEvent({ response });
|
|
130
|
-
if (response === 'ACK' || response === 'RUN') {
|
|
131
|
-
return true;
|
|
132
|
-
}
|
|
133
|
-
else {
|
|
134
|
-
return false;
|
|
135
|
-
}
|
|
136
|
-
});
|
|
137
|
-
}
|
|
138
|
-
getIdentifier() {
|
|
139
|
-
return this.send('getIdentifier', 'ID').then(response => {
|
|
140
|
-
this.logger.logEvent({ response });
|
|
141
|
-
return response.substring(0, 3);
|
|
142
|
-
});
|
|
143
|
-
}
|
|
144
|
-
getInterface() {
|
|
145
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
146
|
-
const res = yield this.send('getInterface', 'KI');
|
|
147
|
-
this.logger.logEvent({ interface: res });
|
|
148
|
-
return res;
|
|
149
|
-
});
|
|
150
|
-
}
|
|
151
|
-
getVersion() {
|
|
152
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
153
|
-
const res = yield this.send('getVersion', 'VE');
|
|
154
|
-
this.logger.logEvent({ version: res });
|
|
155
|
-
return res;
|
|
156
|
-
});
|
|
157
|
-
}
|
|
158
|
-
getCalibration() {
|
|
159
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
160
|
-
return yield this.send('getCalibration', 'CA');
|
|
161
|
-
});
|
|
162
|
-
}
|
|
163
|
-
startTraining() {
|
|
164
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
165
|
-
return yield this.send('startTraining', 'LB');
|
|
166
|
-
});
|
|
167
|
-
}
|
|
168
|
-
unknownSN() {
|
|
169
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
170
|
-
return yield this.send('SN', 'SN');
|
|
171
|
-
});
|
|
172
|
-
}
|
|
173
|
-
setBaudrate(baudrate) {
|
|
174
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
175
|
-
return yield this.send(`setBaudrate(${baudrate})`, `BR${baudrate}`);
|
|
176
|
-
});
|
|
177
|
-
}
|
|
178
|
-
setPower(power) {
|
|
179
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
180
|
-
return this.send(`setPower(${power})`, `PW${power}`).then(response => {
|
|
181
|
-
const data = this.parseStatus(response);
|
|
182
|
-
return data;
|
|
183
|
-
});
|
|
184
|
-
});
|
|
185
|
-
}
|
|
186
|
-
getExtendedStatus() {
|
|
187
|
-
return this.send('getExtendedStatus', 'ES1').then(response => {
|
|
188
|
-
const data = this.parseExtendedStatus(response);
|
|
189
|
-
return data;
|
|
190
|
-
});
|
|
191
|
-
}
|
|
192
|
-
getStatus() {
|
|
193
|
-
return this.send('getStatus', 'ST').then(response => {
|
|
194
|
-
const data = this.parseStatus(response);
|
|
195
|
-
return data;
|
|
196
|
-
});
|
|
197
|
-
}
|
|
198
|
-
getDB() {
|
|
199
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
200
|
-
return yield this.send('getDB', 'DB');
|
|
201
|
-
});
|
|
202
|
-
}
|
|
203
|
-
send(logStr, message, timeout) {
|
|
204
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
205
|
-
return new Promise((resolve, reject) => __awaiter(this, void 0, void 0, function* () {
|
|
206
|
-
try {
|
|
207
|
-
const opened = yield this.waitForOpened();
|
|
208
|
-
if (!opened) {
|
|
209
|
-
reject(new Error('connection error'));
|
|
210
|
-
}
|
|
211
|
-
}
|
|
212
|
-
catch (err) {
|
|
213
|
-
reject(err);
|
|
214
|
-
}
|
|
215
|
-
this.comms.send({ logStr, message, onResponse: resolve, onError: reject, timeout });
|
|
216
|
-
}));
|
|
217
|
-
});
|
|
218
|
-
}
|
|
219
|
-
parseExtendedStatus(data) {
|
|
220
|
-
const result = {};
|
|
221
|
-
return result;
|
|
222
|
-
}
|
|
223
|
-
parseStatus(data) {
|
|
224
|
-
const states = data.split('\t');
|
|
225
|
-
const result = {};
|
|
226
|
-
if (states.length === 8) {
|
|
227
|
-
const hr = parseInt(states[0]);
|
|
228
|
-
if (!isNaN(hr)) {
|
|
229
|
-
result.heartrate = hr;
|
|
230
|
-
}
|
|
231
|
-
var cadence = parseInt(states[1]);
|
|
232
|
-
if (!isNaN(cadence)) {
|
|
233
|
-
result.cadence = cadence;
|
|
234
|
-
}
|
|
235
|
-
const speed = parseInt(states[2]);
|
|
236
|
-
if (!isNaN(speed)) {
|
|
237
|
-
result.speed = speed * 0.1;
|
|
238
|
-
}
|
|
239
|
-
const distance = parseInt(states[3]);
|
|
240
|
-
if (!isNaN(distance)) {
|
|
241
|
-
result.distance = distance * 100;
|
|
242
|
-
}
|
|
243
|
-
const requestedPower = parseInt(states[4]);
|
|
244
|
-
if (!isNaN(requestedPower)) {
|
|
245
|
-
result.requestedPower = requestedPower;
|
|
246
|
-
}
|
|
247
|
-
const energy = parseInt(states[5]);
|
|
248
|
-
if (!isNaN(energy)) {
|
|
249
|
-
result.energy = energy;
|
|
250
|
-
}
|
|
251
|
-
const timeStr = states[6];
|
|
252
|
-
const time = timeStr.split(':');
|
|
253
|
-
const hours = parseInt(time[0]);
|
|
254
|
-
const minutes = parseInt(time[1]);
|
|
255
|
-
if (!isNaN(hours) && !isNaN(minutes)) {
|
|
256
|
-
result.time = hours * 60 + minutes;
|
|
257
|
-
}
|
|
258
|
-
const power = parseInt(states[7]);
|
|
259
|
-
if (!isNaN(power)) {
|
|
260
|
-
result.power = power;
|
|
261
|
-
}
|
|
262
|
-
result.timestamp = Date.now();
|
|
263
|
-
}
|
|
264
|
-
return result;
|
|
265
|
-
}
|
|
266
|
-
check() {
|
|
267
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
268
|
-
var info = {};
|
|
269
|
-
return new Promise((resolve, reject) => __awaiter(this, void 0, void 0, function* () {
|
|
270
|
-
this.logger.logEvent({ message: "check()", port: this.getPort() });
|
|
271
|
-
let iv = undefined;
|
|
272
|
-
try {
|
|
273
|
-
if (!info.opened)
|
|
274
|
-
info.opened = yield this.waitForOpened();
|
|
275
|
-
iv = setTimeout(() => {
|
|
276
|
-
this.logger.logEvent({ message: "check() timeout", port: this.getPort() });
|
|
277
|
-
reject(new Error(`timeout`));
|
|
278
|
-
}, 5000);
|
|
279
|
-
if (!info.pcMode)
|
|
280
|
-
info.pcMode = yield this.setClientMode();
|
|
281
|
-
if (!info.id)
|
|
282
|
-
info.id = yield this.getIdentifier();
|
|
283
|
-
if (!info.version)
|
|
284
|
-
info.version = yield this.getVersion();
|
|
285
|
-
try {
|
|
286
|
-
yield this.getInterface();
|
|
287
|
-
}
|
|
288
|
-
catch (e) {
|
|
289
|
-
this.logger.logEvent({ message: 'Error', error: e.message });
|
|
290
|
-
}
|
|
291
|
-
clearTimeout(iv);
|
|
292
|
-
resolve(info);
|
|
293
|
-
}
|
|
294
|
-
catch (err) {
|
|
295
|
-
this.logger.logEvent({ message: 'Error', error: err.message });
|
|
296
|
-
if (iv)
|
|
297
|
-
clearTimeout(iv);
|
|
298
|
-
iv = undefined;
|
|
299
|
-
reject(err);
|
|
300
|
-
}
|
|
301
|
-
}));
|
|
302
|
-
});
|
|
303
|
-
}
|
|
304
|
-
start(props) {
|
|
305
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
306
|
-
this.logger.logEvent({ message: 'start()' });
|
|
307
|
-
var info = {};
|
|
308
|
-
yield this.waitForOpened(true);
|
|
309
|
-
return (0, utils_1.runWithRetries)(() => __awaiter(this, void 0, void 0, function* () {
|
|
310
|
-
try {
|
|
311
|
-
if (!info.checkDone) {
|
|
312
|
-
info.checkDone = yield this.check();
|
|
313
|
-
}
|
|
314
|
-
try {
|
|
315
|
-
if (!info.started) {
|
|
316
|
-
info.started = yield this.startTraining();
|
|
317
|
-
}
|
|
318
|
-
}
|
|
319
|
-
catch (e) {
|
|
320
|
-
this.logger.logEvent({ message: 'Error', error: e.message });
|
|
321
|
-
}
|
|
322
|
-
try {
|
|
323
|
-
yield this.setPower(100);
|
|
324
|
-
}
|
|
325
|
-
catch (e) {
|
|
326
|
-
this.logger.logEvent({ message: 'Error', error: e.message });
|
|
327
|
-
}
|
|
328
|
-
if (!info.data) {
|
|
329
|
-
yield this.update();
|
|
330
|
-
info.data = this.data;
|
|
331
|
-
}
|
|
332
|
-
return info.data;
|
|
333
|
-
}
|
|
334
|
-
catch (err) {
|
|
335
|
-
try {
|
|
336
|
-
yield this.reset();
|
|
337
|
-
}
|
|
338
|
-
catch (e) {
|
|
339
|
-
this.logger.logEvent({ message: 'Error', error: e.message });
|
|
340
|
-
}
|
|
341
|
-
throw (new Error(`could not start device, reason:${err.message}`));
|
|
342
|
-
}
|
|
343
|
-
}), 5, 1000)
|
|
344
|
-
.then((data) => {
|
|
345
|
-
this.startUpdatePull();
|
|
346
|
-
return data;
|
|
347
|
-
});
|
|
348
|
-
});
|
|
349
|
-
}
|
|
350
|
-
startUpdatePull() {
|
|
351
|
-
if (this.iv)
|
|
352
|
-
return;
|
|
353
|
-
this.logger.logEvent({ message: 'start regular device update' });
|
|
354
|
-
if (this.ignoreBike && this.ignoreHrm && this.ignorePower)
|
|
355
|
-
return;
|
|
356
|
-
const ivSync = setInterval(() => {
|
|
357
|
-
this.bikeSync();
|
|
358
|
-
}, 1000);
|
|
359
|
-
const ivUpdate = setInterval(() => {
|
|
360
|
-
this.sendData();
|
|
361
|
-
this.refreshRequests();
|
|
362
|
-
}, 1000);
|
|
363
|
-
this.iv = {
|
|
364
|
-
sync: ivSync,
|
|
365
|
-
update: ivUpdate
|
|
366
|
-
};
|
|
367
|
-
}
|
|
368
|
-
stop() {
|
|
369
|
-
this.logger.logEvent({ message: 'stop request' });
|
|
370
|
-
return new Promise((resolve, reject) => __awaiter(this, void 0, void 0, function* () {
|
|
371
|
-
try {
|
|
372
|
-
if (this.iv) {
|
|
373
|
-
if (this.iv.sync)
|
|
374
|
-
clearInterval(this.iv.sync);
|
|
375
|
-
if (this.iv.update)
|
|
376
|
-
clearInterval(this.iv.update);
|
|
377
|
-
this.iv = undefined;
|
|
378
|
-
}
|
|
379
|
-
yield this.waitForClosed();
|
|
380
|
-
this.logger.logEvent({ message: 'stop request completed' });
|
|
381
|
-
this.paused = undefined;
|
|
382
|
-
resolve(true);
|
|
383
|
-
}
|
|
384
|
-
catch (err) {
|
|
385
|
-
this.logger.logEvent({ message: 'stop error', error: err.message });
|
|
386
|
-
this.paused = undefined;
|
|
387
|
-
reject(err);
|
|
388
|
-
}
|
|
389
|
-
}));
|
|
390
|
-
}
|
|
391
|
-
pause() {
|
|
392
|
-
this.logger.logEvent({ message: 'pause' });
|
|
393
|
-
return new Promise(resolve => {
|
|
394
|
-
this.paused = true;
|
|
395
|
-
resolve(true);
|
|
396
|
-
});
|
|
397
|
-
}
|
|
398
|
-
resume() {
|
|
399
|
-
this.logger.logEvent({ message: 'resume' });
|
|
400
|
-
return new Promise(resolve => {
|
|
401
|
-
this.paused = false;
|
|
402
|
-
resolve(true);
|
|
403
|
-
});
|
|
404
|
-
}
|
|
405
|
-
mapData(bikeData) {
|
|
406
|
-
let data = {};
|
|
407
|
-
data.isPedalling = bikeData.cadence > 0;
|
|
408
|
-
data.power = bikeData.power;
|
|
409
|
-
data.pedalRpm = bikeData.cadence;
|
|
410
|
-
data.speed = bikeData.speed;
|
|
411
|
-
data.heartrate = bikeData.heartrate;
|
|
412
|
-
data.distanceInternal = bikeData.distance;
|
|
413
|
-
data.time = bikeData.time;
|
|
414
|
-
return data;
|
|
415
|
-
}
|
|
416
|
-
transformData(internalData, bikeData) {
|
|
417
|
-
let data = {};
|
|
418
|
-
const prevDistance = this.prevDistance || 0;
|
|
419
|
-
let distance = internalData.distanceInternal - prevDistance;
|
|
420
|
-
if (distance < 0)
|
|
421
|
-
distance = internalData.distanceInternal < 100 ? internalData.distanceInternal : 0;
|
|
422
|
-
data.heartrate = internalData.heartrate;
|
|
423
|
-
data.timestamp = Date.now();
|
|
424
|
-
data.deviceTime = bikeData.time;
|
|
425
|
-
if (!this.ignoreBike) {
|
|
426
|
-
data.speed = internalData.speed;
|
|
427
|
-
data.power = internalData.power;
|
|
428
|
-
data.cadence = internalData.pedalRpm;
|
|
429
|
-
data.distance = distance;
|
|
430
|
-
data.deviceDistanceCounter = bikeData.distance;
|
|
431
|
-
data.internalDistanceCounter = internalData.distanceInternal;
|
|
432
|
-
this.prevDistance = internalData.distanceInternal;
|
|
433
|
-
}
|
|
434
|
-
if (this.ignoreHrm)
|
|
435
|
-
delete this.data.heartrate;
|
|
436
|
-
if (this.ignorePower) {
|
|
437
|
-
delete this.data.power;
|
|
438
|
-
delete this.data.cadence;
|
|
439
|
-
}
|
|
440
|
-
return data;
|
|
441
|
-
}
|
|
442
|
-
update() {
|
|
443
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
444
|
-
this.updateBusy = true;
|
|
445
|
-
return this.getStatus()
|
|
446
|
-
.then((bikeData) => {
|
|
447
|
-
if (bikeData) {
|
|
448
|
-
try {
|
|
449
|
-
this.kettlerData = bikeData;
|
|
450
|
-
let data = this.mapData(bikeData);
|
|
451
|
-
data = this.getCyclingMode().updateData(data);
|
|
452
|
-
this.internalData = data;
|
|
453
|
-
this.data = this.transformData(data, bikeData);
|
|
454
|
-
}
|
|
455
|
-
catch (err) {
|
|
456
|
-
this.logger.logEvent({ message: 'bike update error', error: err.message });
|
|
457
|
-
}
|
|
458
|
-
}
|
|
459
|
-
this.updateBusy = false;
|
|
460
|
-
})
|
|
461
|
-
.catch(err => {
|
|
462
|
-
this.logger.logEvent({ message: 'bike update error', error: err.message });
|
|
463
|
-
this.updateBusy = false;
|
|
464
|
-
});
|
|
465
|
-
});
|
|
466
|
-
}
|
|
467
|
-
sendRequest(request) {
|
|
468
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
469
|
-
this.requestBusy = true;
|
|
470
|
-
try {
|
|
471
|
-
this.logger.logEvent({ message: 'sendRequest', request });
|
|
472
|
-
const isReset = (!request || request.reset || Object.keys(request).length === 0);
|
|
473
|
-
if (isReset) {
|
|
474
|
-
this.requestBusy = false;
|
|
475
|
-
return {};
|
|
476
|
-
}
|
|
477
|
-
if (request.slope !== undefined) {
|
|
478
|
-
this.data.slope = request.slope;
|
|
479
|
-
}
|
|
480
|
-
if (request.targetPower !== undefined) {
|
|
481
|
-
yield this.setPower(request.targetPower);
|
|
482
|
-
}
|
|
483
|
-
this.requestBusy = false;
|
|
484
|
-
return request;
|
|
485
|
-
}
|
|
486
|
-
catch (err) {
|
|
487
|
-
this.requestBusy = false;
|
|
488
|
-
this.logger.logEvent({ message: 'error', fn: 'sendRequest()', error: err.message || err });
|
|
489
|
-
return;
|
|
490
|
-
}
|
|
491
|
-
});
|
|
492
|
-
}
|
|
493
|
-
sendRequests() {
|
|
494
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
495
|
-
if (this.requests.length > 0) {
|
|
496
|
-
const processing = [...this.requests];
|
|
497
|
-
const cnt = processing.length;
|
|
498
|
-
processing.forEach((request, idx) => __awaiter(this, void 0, void 0, function* () {
|
|
499
|
-
if (cnt > 1 && idx < cnt - 1) {
|
|
500
|
-
this.logger.logEvent({ message: 'ignoring bike update request', request });
|
|
501
|
-
this.requests.shift();
|
|
502
|
-
return;
|
|
503
|
-
}
|
|
504
|
-
}));
|
|
505
|
-
const request = processing[0];
|
|
506
|
-
try {
|
|
507
|
-
yield this.sendRequest(request);
|
|
508
|
-
this.requests.shift();
|
|
509
|
-
}
|
|
510
|
-
catch (err) {
|
|
511
|
-
this.logger.logEvent({ message: 'bike update error', error: err.message, stack: err.stack, request });
|
|
512
|
-
}
|
|
513
|
-
}
|
|
514
|
-
});
|
|
515
|
-
}
|
|
516
|
-
bikeSync() {
|
|
517
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
518
|
-
if (this.paused) {
|
|
519
|
-
return;
|
|
520
|
-
}
|
|
521
|
-
if (this.updateBusy || this.requestBusy) {
|
|
522
|
-
return;
|
|
523
|
-
}
|
|
524
|
-
this.logger.logEvent({ message: 'bikeSync' });
|
|
525
|
-
if (!this.ignoreBike) {
|
|
526
|
-
yield this.sendRequests();
|
|
527
|
-
}
|
|
528
|
-
yield this.update();
|
|
529
|
-
});
|
|
530
|
-
}
|
|
531
|
-
sendUpdate(request) {
|
|
532
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
533
|
-
if (this.paused)
|
|
534
|
-
return;
|
|
535
|
-
this.logger.logEvent({ message: 'sendUpdate', request, waiting: this.requests.length });
|
|
536
|
-
return yield this.processClientRequest(request);
|
|
537
|
-
});
|
|
538
|
-
}
|
|
539
|
-
sendData() {
|
|
540
|
-
if (this.onDataFn)
|
|
541
|
-
this.onDataFn(this.data);
|
|
542
|
-
}
|
|
543
|
-
refreshRequests() {
|
|
544
|
-
if (this.kettlerData.cadence === 0)
|
|
545
|
-
return;
|
|
546
|
-
let bikeRequest = this.getCyclingMode().sendBikeUpdate({ refresh: true }) || {};
|
|
547
|
-
const prev = this.requests[this.requests.length - 1] || {};
|
|
548
|
-
if (bikeRequest.targetPower !== undefined && bikeRequest.targetPower !== prev.targetPower) {
|
|
549
|
-
this.logger.logEvent({ message: 'add request', request: bikeRequest });
|
|
550
|
-
this.requests.push(bikeRequest);
|
|
551
|
-
}
|
|
552
|
-
}
|
|
553
|
-
processClientRequest(request) {
|
|
554
|
-
if (request.slope !== undefined) {
|
|
555
|
-
this.data.slope = request.slope;
|
|
556
|
-
}
|
|
557
|
-
return new Promise((resolve) => __awaiter(this, void 0, void 0, function* () {
|
|
558
|
-
let bikeRequest = this.getCyclingMode().sendBikeUpdate(request);
|
|
559
|
-
this.logger.logEvent({ message: 'add request', request: bikeRequest });
|
|
560
|
-
this.requests.push(bikeRequest);
|
|
561
|
-
resolve(bikeRequest);
|
|
562
|
-
}));
|
|
563
|
-
}
|
|
564
|
-
waitForOpened(retries = false) {
|
|
565
|
-
const run = (resolve, reject) => {
|
|
566
|
-
try {
|
|
567
|
-
if (this.comms.isConnected()) {
|
|
568
|
-
resolve(true);
|
|
569
|
-
return;
|
|
570
|
-
}
|
|
571
|
-
const cleanup = () => {
|
|
572
|
-
this.comms.removeAllListeners();
|
|
573
|
-
};
|
|
574
|
-
const onOpen = () => {
|
|
575
|
-
resolve(true);
|
|
576
|
-
cleanup();
|
|
577
|
-
};
|
|
578
|
-
const onError = (err) => { reject(err); cleanup(); };
|
|
579
|
-
const onClose = () => { cleanup(); };
|
|
580
|
-
this.comms.on('opened', onOpen);
|
|
581
|
-
this.comms.on('closed', onClose);
|
|
582
|
-
this.comms.on('error', onError);
|
|
583
|
-
this.logger.logEvent({ message: 'opening', port: this.getPort() });
|
|
584
|
-
this.comms.open();
|
|
585
|
-
}
|
|
586
|
-
catch (err) {
|
|
587
|
-
this.logger.logEvent({ message: 'error', fn: 'waitForOpened()', error: err.message || err });
|
|
588
|
-
reject(err);
|
|
589
|
-
}
|
|
590
|
-
};
|
|
591
|
-
if (!retries) {
|
|
592
|
-
return new Promise((resolve, reject) => run(resolve, reject));
|
|
593
|
-
}
|
|
594
|
-
return (0, utils_1.runWithRetries)(() => {
|
|
595
|
-
return new Promise((resolve, reject) => run(resolve, reject));
|
|
596
|
-
}, 3, 1000);
|
|
597
|
-
}
|
|
598
|
-
waitForClosed() {
|
|
599
|
-
return new Promise((resolve, reject) => {
|
|
600
|
-
try {
|
|
601
|
-
if (!this.comms.isConnected()) {
|
|
602
|
-
resolve(true);
|
|
603
|
-
return;
|
|
604
|
-
}
|
|
605
|
-
const cleanup = () => {
|
|
606
|
-
this.comms.removeAllListeners();
|
|
607
|
-
};
|
|
608
|
-
const onClose = () => {
|
|
609
|
-
resolve(true);
|
|
610
|
-
cleanup();
|
|
611
|
-
};
|
|
612
|
-
const onError = (err) => { reject(err); cleanup(); };
|
|
613
|
-
const onOpen = () => { cleanup(); };
|
|
614
|
-
this.comms.on('closed', onClose);
|
|
615
|
-
this.comms.on('opened', onOpen);
|
|
616
|
-
this.comms.on('error', onError);
|
|
617
|
-
this.logger.logEvent({ message: 'closing', port: this.getPort() });
|
|
618
|
-
this.comms.close();
|
|
619
|
-
}
|
|
620
|
-
catch (err) {
|
|
621
|
-
this.logger.logEvent({ message: 'error', fn: 'waitForClosed()', error: err.message || err });
|
|
622
|
-
reject(err);
|
|
623
|
-
}
|
|
624
|
-
});
|
|
625
|
-
}
|
|
626
|
-
getSupportedCyclingModes() {
|
|
627
|
-
return [power_meter_1.default, ERGCyclingMode_1.default];
|
|
628
|
-
}
|
|
629
|
-
getDefaultCyclingMode() {
|
|
630
|
-
return new ERGCyclingMode_1.default(this);
|
|
631
|
-
}
|
|
632
|
-
setUserSettings(userSettings) {
|
|
633
|
-
this.settings.userSettings = userSettings;
|
|
634
|
-
}
|
|
635
|
-
setBikeSettings(bikeSettings) {
|
|
636
|
-
this.settings.bikeSettings = bikeSettings;
|
|
637
|
-
}
|
|
638
|
-
}
|
|
639
|
-
exports.default = KettlerRacerAdapter;
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
|
|
5
|
+
}) : (function(o, m, k, k2) {
|
|
6
|
+
if (k2 === undefined) k2 = k;
|
|
7
|
+
o[k2] = m[k];
|
|
8
|
+
}));
|
|
9
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
10
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
11
|
+
}) : function(o, v) {
|
|
12
|
+
o["default"] = v;
|
|
13
|
+
});
|
|
14
|
+
var __importStar = (this && this.__importStar) || function (mod) {
|
|
15
|
+
if (mod && mod.__esModule) return mod;
|
|
16
|
+
var result = {};
|
|
17
|
+
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
|
18
|
+
__setModuleDefault(result, mod);
|
|
19
|
+
return result;
|
|
20
|
+
};
|
|
21
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
22
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
23
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
24
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
25
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
26
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
27
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
28
|
+
});
|
|
29
|
+
};
|
|
30
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
31
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
32
|
+
};
|
|
33
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
34
|
+
const Device_1 = __importStar(require("../../Device"));
|
|
35
|
+
const gd_eventlog_1 = require("gd-eventlog");
|
|
36
|
+
const comms_1 = __importDefault(require("../comms"));
|
|
37
|
+
const utils_1 = require("../../utils");
|
|
38
|
+
const power_meter_1 = __importDefault(require("../../modes/power-meter"));
|
|
39
|
+
const ERGCyclingMode_1 = __importDefault(require("./ERGCyclingMode"));
|
|
40
|
+
class KettlerRacerAdapter extends Device_1.default {
|
|
41
|
+
constructor(protocol, settings) {
|
|
42
|
+
super(protocol, settings);
|
|
43
|
+
this.requests = [];
|
|
44
|
+
this.logger = new gd_eventlog_1.EventLogger('KettlerRacer');
|
|
45
|
+
this.ignoreHrm = false;
|
|
46
|
+
this.ignorePower = false;
|
|
47
|
+
this.ignoreBike = false;
|
|
48
|
+
this.paused = false;
|
|
49
|
+
this.iv = null;
|
|
50
|
+
this.comms = new comms_1.default({ protocol, port: this.settings.port, logger: this.logger });
|
|
51
|
+
}
|
|
52
|
+
isBike() { return true; }
|
|
53
|
+
isPower() { return true; }
|
|
54
|
+
isHrm() { return true; }
|
|
55
|
+
isSame(device) {
|
|
56
|
+
if (!(device instanceof KettlerRacerAdapter))
|
|
57
|
+
return false;
|
|
58
|
+
const adapter = device;
|
|
59
|
+
return (adapter.getName() === this.getName() && adapter.getPort() === this.getPort());
|
|
60
|
+
}
|
|
61
|
+
setID(id) {
|
|
62
|
+
this.id = id;
|
|
63
|
+
}
|
|
64
|
+
getID() {
|
|
65
|
+
return this.id;
|
|
66
|
+
}
|
|
67
|
+
getName() {
|
|
68
|
+
return this.settings.name || this.getProtocolName();
|
|
69
|
+
}
|
|
70
|
+
getPort() {
|
|
71
|
+
return this.settings.port;
|
|
72
|
+
}
|
|
73
|
+
setIgnoreHrm(ignore) {
|
|
74
|
+
this.ignoreHrm = ignore;
|
|
75
|
+
}
|
|
76
|
+
setIgnorePower(ignore) {
|
|
77
|
+
this.ignorePower = ignore;
|
|
78
|
+
}
|
|
79
|
+
setIgnoreBike(ignore) {
|
|
80
|
+
this.ignoreBike = ignore;
|
|
81
|
+
}
|
|
82
|
+
_getComms() {
|
|
83
|
+
return this.comms;
|
|
84
|
+
}
|
|
85
|
+
_setComms(comms) {
|
|
86
|
+
this.comms = comms;
|
|
87
|
+
}
|
|
88
|
+
getLogger() {
|
|
89
|
+
return this.logger;
|
|
90
|
+
}
|
|
91
|
+
getUserSettings() {
|
|
92
|
+
return this.settings.userSettings || { weight: Device_1.DEFAULT_USER_WEIGHT };
|
|
93
|
+
}
|
|
94
|
+
getWeight() {
|
|
95
|
+
let userWeight = Device_1.DEFAULT_USER_WEIGHT;
|
|
96
|
+
let bikeWeight = Device_1.DEFAULT_BIKE_WEIGHT;
|
|
97
|
+
if (this.settings.userSettings && this.settings.userSettings.weight) {
|
|
98
|
+
userWeight = this.settings.userSettings.weight;
|
|
99
|
+
}
|
|
100
|
+
if (this.settings.bikeSettings && this.settings.bikeSettings.weight) {
|
|
101
|
+
userWeight = this.settings.bikeSettings.weight;
|
|
102
|
+
}
|
|
103
|
+
return bikeWeight + userWeight;
|
|
104
|
+
}
|
|
105
|
+
setComputerMode() {
|
|
106
|
+
return this.send('setComputerMode', 'CP').then(response => {
|
|
107
|
+
this.logger.logEvent({ response });
|
|
108
|
+
if (response === 'ACK' || response === 'RUN') {
|
|
109
|
+
return true;
|
|
110
|
+
}
|
|
111
|
+
else {
|
|
112
|
+
return false;
|
|
113
|
+
}
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
setClientMode() {
|
|
117
|
+
return this.send('setClientMode', 'CM').then(response => {
|
|
118
|
+
this.logger.logEvent({ response });
|
|
119
|
+
if (response === 'ACK' || response === 'RUN') {
|
|
120
|
+
return true;
|
|
121
|
+
}
|
|
122
|
+
else {
|
|
123
|
+
return false;
|
|
124
|
+
}
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
reset() {
|
|
128
|
+
return this.send('reset', 'RS').then(response => {
|
|
129
|
+
this.logger.logEvent({ response });
|
|
130
|
+
if (response === 'ACK' || response === 'RUN') {
|
|
131
|
+
return true;
|
|
132
|
+
}
|
|
133
|
+
else {
|
|
134
|
+
return false;
|
|
135
|
+
}
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
getIdentifier() {
|
|
139
|
+
return this.send('getIdentifier', 'ID').then(response => {
|
|
140
|
+
this.logger.logEvent({ response });
|
|
141
|
+
return response.substring(0, 3);
|
|
142
|
+
});
|
|
143
|
+
}
|
|
144
|
+
getInterface() {
|
|
145
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
146
|
+
const res = yield this.send('getInterface', 'KI');
|
|
147
|
+
this.logger.logEvent({ interface: res });
|
|
148
|
+
return res;
|
|
149
|
+
});
|
|
150
|
+
}
|
|
151
|
+
getVersion() {
|
|
152
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
153
|
+
const res = yield this.send('getVersion', 'VE');
|
|
154
|
+
this.logger.logEvent({ version: res });
|
|
155
|
+
return res;
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
getCalibration() {
|
|
159
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
160
|
+
return yield this.send('getCalibration', 'CA');
|
|
161
|
+
});
|
|
162
|
+
}
|
|
163
|
+
startTraining() {
|
|
164
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
165
|
+
return yield this.send('startTraining', 'LB');
|
|
166
|
+
});
|
|
167
|
+
}
|
|
168
|
+
unknownSN() {
|
|
169
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
170
|
+
return yield this.send('SN', 'SN');
|
|
171
|
+
});
|
|
172
|
+
}
|
|
173
|
+
setBaudrate(baudrate) {
|
|
174
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
175
|
+
return yield this.send(`setBaudrate(${baudrate})`, `BR${baudrate}`);
|
|
176
|
+
});
|
|
177
|
+
}
|
|
178
|
+
setPower(power) {
|
|
179
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
180
|
+
return this.send(`setPower(${power})`, `PW${power}`).then(response => {
|
|
181
|
+
const data = this.parseStatus(response);
|
|
182
|
+
return data;
|
|
183
|
+
});
|
|
184
|
+
});
|
|
185
|
+
}
|
|
186
|
+
getExtendedStatus() {
|
|
187
|
+
return this.send('getExtendedStatus', 'ES1').then(response => {
|
|
188
|
+
const data = this.parseExtendedStatus(response);
|
|
189
|
+
return data;
|
|
190
|
+
});
|
|
191
|
+
}
|
|
192
|
+
getStatus() {
|
|
193
|
+
return this.send('getStatus', 'ST').then(response => {
|
|
194
|
+
const data = this.parseStatus(response);
|
|
195
|
+
return data;
|
|
196
|
+
});
|
|
197
|
+
}
|
|
198
|
+
getDB() {
|
|
199
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
200
|
+
return yield this.send('getDB', 'DB');
|
|
201
|
+
});
|
|
202
|
+
}
|
|
203
|
+
send(logStr, message, timeout) {
|
|
204
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
205
|
+
return new Promise((resolve, reject) => __awaiter(this, void 0, void 0, function* () {
|
|
206
|
+
try {
|
|
207
|
+
const opened = yield this.waitForOpened();
|
|
208
|
+
if (!opened) {
|
|
209
|
+
reject(new Error('connection error'));
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
catch (err) {
|
|
213
|
+
reject(err);
|
|
214
|
+
}
|
|
215
|
+
this.comms.send({ logStr, message, onResponse: resolve, onError: reject, timeout });
|
|
216
|
+
}));
|
|
217
|
+
});
|
|
218
|
+
}
|
|
219
|
+
parseExtendedStatus(data) {
|
|
220
|
+
const result = {};
|
|
221
|
+
return result;
|
|
222
|
+
}
|
|
223
|
+
parseStatus(data) {
|
|
224
|
+
const states = data.split('\t');
|
|
225
|
+
const result = {};
|
|
226
|
+
if (states.length === 8) {
|
|
227
|
+
const hr = parseInt(states[0]);
|
|
228
|
+
if (!isNaN(hr)) {
|
|
229
|
+
result.heartrate = hr;
|
|
230
|
+
}
|
|
231
|
+
var cadence = parseInt(states[1]);
|
|
232
|
+
if (!isNaN(cadence)) {
|
|
233
|
+
result.cadence = cadence;
|
|
234
|
+
}
|
|
235
|
+
const speed = parseInt(states[2]);
|
|
236
|
+
if (!isNaN(speed)) {
|
|
237
|
+
result.speed = speed * 0.1;
|
|
238
|
+
}
|
|
239
|
+
const distance = parseInt(states[3]);
|
|
240
|
+
if (!isNaN(distance)) {
|
|
241
|
+
result.distance = distance * 100;
|
|
242
|
+
}
|
|
243
|
+
const requestedPower = parseInt(states[4]);
|
|
244
|
+
if (!isNaN(requestedPower)) {
|
|
245
|
+
result.requestedPower = requestedPower;
|
|
246
|
+
}
|
|
247
|
+
const energy = parseInt(states[5]);
|
|
248
|
+
if (!isNaN(energy)) {
|
|
249
|
+
result.energy = energy;
|
|
250
|
+
}
|
|
251
|
+
const timeStr = states[6];
|
|
252
|
+
const time = timeStr.split(':');
|
|
253
|
+
const hours = parseInt(time[0]);
|
|
254
|
+
const minutes = parseInt(time[1]);
|
|
255
|
+
if (!isNaN(hours) && !isNaN(minutes)) {
|
|
256
|
+
result.time = hours * 60 + minutes;
|
|
257
|
+
}
|
|
258
|
+
const power = parseInt(states[7]);
|
|
259
|
+
if (!isNaN(power)) {
|
|
260
|
+
result.power = power;
|
|
261
|
+
}
|
|
262
|
+
result.timestamp = Date.now();
|
|
263
|
+
}
|
|
264
|
+
return result;
|
|
265
|
+
}
|
|
266
|
+
check() {
|
|
267
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
268
|
+
var info = {};
|
|
269
|
+
return new Promise((resolve, reject) => __awaiter(this, void 0, void 0, function* () {
|
|
270
|
+
this.logger.logEvent({ message: "check()", port: this.getPort() });
|
|
271
|
+
let iv = undefined;
|
|
272
|
+
try {
|
|
273
|
+
if (!info.opened)
|
|
274
|
+
info.opened = yield this.waitForOpened();
|
|
275
|
+
iv = setTimeout(() => {
|
|
276
|
+
this.logger.logEvent({ message: "check() timeout", port: this.getPort() });
|
|
277
|
+
reject(new Error(`timeout`));
|
|
278
|
+
}, 5000);
|
|
279
|
+
if (!info.pcMode)
|
|
280
|
+
info.pcMode = yield this.setClientMode();
|
|
281
|
+
if (!info.id)
|
|
282
|
+
info.id = yield this.getIdentifier();
|
|
283
|
+
if (!info.version)
|
|
284
|
+
info.version = yield this.getVersion();
|
|
285
|
+
try {
|
|
286
|
+
yield this.getInterface();
|
|
287
|
+
}
|
|
288
|
+
catch (e) {
|
|
289
|
+
this.logger.logEvent({ message: 'Error', error: e.message });
|
|
290
|
+
}
|
|
291
|
+
clearTimeout(iv);
|
|
292
|
+
resolve(info);
|
|
293
|
+
}
|
|
294
|
+
catch (err) {
|
|
295
|
+
this.logger.logEvent({ message: 'Error', error: err.message });
|
|
296
|
+
if (iv)
|
|
297
|
+
clearTimeout(iv);
|
|
298
|
+
iv = undefined;
|
|
299
|
+
reject(err);
|
|
300
|
+
}
|
|
301
|
+
}));
|
|
302
|
+
});
|
|
303
|
+
}
|
|
304
|
+
start(props) {
|
|
305
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
306
|
+
this.logger.logEvent({ message: 'start()' });
|
|
307
|
+
var info = {};
|
|
308
|
+
yield this.waitForOpened(true);
|
|
309
|
+
return (0, utils_1.runWithRetries)(() => __awaiter(this, void 0, void 0, function* () {
|
|
310
|
+
try {
|
|
311
|
+
if (!info.checkDone) {
|
|
312
|
+
info.checkDone = yield this.check();
|
|
313
|
+
}
|
|
314
|
+
try {
|
|
315
|
+
if (!info.started) {
|
|
316
|
+
info.started = yield this.startTraining();
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
catch (e) {
|
|
320
|
+
this.logger.logEvent({ message: 'Error', error: e.message });
|
|
321
|
+
}
|
|
322
|
+
try {
|
|
323
|
+
yield this.setPower(100);
|
|
324
|
+
}
|
|
325
|
+
catch (e) {
|
|
326
|
+
this.logger.logEvent({ message: 'Error', error: e.message });
|
|
327
|
+
}
|
|
328
|
+
if (!info.data) {
|
|
329
|
+
yield this.update();
|
|
330
|
+
info.data = this.data;
|
|
331
|
+
}
|
|
332
|
+
return info.data;
|
|
333
|
+
}
|
|
334
|
+
catch (err) {
|
|
335
|
+
try {
|
|
336
|
+
yield this.reset();
|
|
337
|
+
}
|
|
338
|
+
catch (e) {
|
|
339
|
+
this.logger.logEvent({ message: 'Error', error: e.message });
|
|
340
|
+
}
|
|
341
|
+
throw (new Error(`could not start device, reason:${err.message}`));
|
|
342
|
+
}
|
|
343
|
+
}), 5, 1000)
|
|
344
|
+
.then((data) => {
|
|
345
|
+
this.startUpdatePull();
|
|
346
|
+
return data;
|
|
347
|
+
});
|
|
348
|
+
});
|
|
349
|
+
}
|
|
350
|
+
startUpdatePull() {
|
|
351
|
+
if (this.iv)
|
|
352
|
+
return;
|
|
353
|
+
this.logger.logEvent({ message: 'start regular device update' });
|
|
354
|
+
if (this.ignoreBike && this.ignoreHrm && this.ignorePower)
|
|
355
|
+
return;
|
|
356
|
+
const ivSync = setInterval(() => {
|
|
357
|
+
this.bikeSync();
|
|
358
|
+
}, 1000);
|
|
359
|
+
const ivUpdate = setInterval(() => {
|
|
360
|
+
this.sendData();
|
|
361
|
+
this.refreshRequests();
|
|
362
|
+
}, 1000);
|
|
363
|
+
this.iv = {
|
|
364
|
+
sync: ivSync,
|
|
365
|
+
update: ivUpdate
|
|
366
|
+
};
|
|
367
|
+
}
|
|
368
|
+
stop() {
|
|
369
|
+
this.logger.logEvent({ message: 'stop request' });
|
|
370
|
+
return new Promise((resolve, reject) => __awaiter(this, void 0, void 0, function* () {
|
|
371
|
+
try {
|
|
372
|
+
if (this.iv) {
|
|
373
|
+
if (this.iv.sync)
|
|
374
|
+
clearInterval(this.iv.sync);
|
|
375
|
+
if (this.iv.update)
|
|
376
|
+
clearInterval(this.iv.update);
|
|
377
|
+
this.iv = undefined;
|
|
378
|
+
}
|
|
379
|
+
yield this.waitForClosed();
|
|
380
|
+
this.logger.logEvent({ message: 'stop request completed' });
|
|
381
|
+
this.paused = undefined;
|
|
382
|
+
resolve(true);
|
|
383
|
+
}
|
|
384
|
+
catch (err) {
|
|
385
|
+
this.logger.logEvent({ message: 'stop error', error: err.message });
|
|
386
|
+
this.paused = undefined;
|
|
387
|
+
reject(err);
|
|
388
|
+
}
|
|
389
|
+
}));
|
|
390
|
+
}
|
|
391
|
+
pause() {
|
|
392
|
+
this.logger.logEvent({ message: 'pause' });
|
|
393
|
+
return new Promise(resolve => {
|
|
394
|
+
this.paused = true;
|
|
395
|
+
resolve(true);
|
|
396
|
+
});
|
|
397
|
+
}
|
|
398
|
+
resume() {
|
|
399
|
+
this.logger.logEvent({ message: 'resume' });
|
|
400
|
+
return new Promise(resolve => {
|
|
401
|
+
this.paused = false;
|
|
402
|
+
resolve(true);
|
|
403
|
+
});
|
|
404
|
+
}
|
|
405
|
+
mapData(bikeData) {
|
|
406
|
+
let data = {};
|
|
407
|
+
data.isPedalling = bikeData.cadence > 0;
|
|
408
|
+
data.power = bikeData.power;
|
|
409
|
+
data.pedalRpm = bikeData.cadence;
|
|
410
|
+
data.speed = bikeData.speed;
|
|
411
|
+
data.heartrate = bikeData.heartrate;
|
|
412
|
+
data.distanceInternal = bikeData.distance;
|
|
413
|
+
data.time = bikeData.time;
|
|
414
|
+
return data;
|
|
415
|
+
}
|
|
416
|
+
transformData(internalData, bikeData) {
|
|
417
|
+
let data = {};
|
|
418
|
+
const prevDistance = this.prevDistance || 0;
|
|
419
|
+
let distance = internalData.distanceInternal - prevDistance;
|
|
420
|
+
if (distance < 0)
|
|
421
|
+
distance = internalData.distanceInternal < 100 ? internalData.distanceInternal : 0;
|
|
422
|
+
data.heartrate = internalData.heartrate;
|
|
423
|
+
data.timestamp = Date.now();
|
|
424
|
+
data.deviceTime = bikeData.time;
|
|
425
|
+
if (!this.ignoreBike) {
|
|
426
|
+
data.speed = internalData.speed;
|
|
427
|
+
data.power = internalData.power;
|
|
428
|
+
data.cadence = internalData.pedalRpm;
|
|
429
|
+
data.distance = distance;
|
|
430
|
+
data.deviceDistanceCounter = bikeData.distance;
|
|
431
|
+
data.internalDistanceCounter = internalData.distanceInternal;
|
|
432
|
+
this.prevDistance = internalData.distanceInternal;
|
|
433
|
+
}
|
|
434
|
+
if (this.ignoreHrm)
|
|
435
|
+
delete this.data.heartrate;
|
|
436
|
+
if (this.ignorePower) {
|
|
437
|
+
delete this.data.power;
|
|
438
|
+
delete this.data.cadence;
|
|
439
|
+
}
|
|
440
|
+
return data;
|
|
441
|
+
}
|
|
442
|
+
update() {
|
|
443
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
444
|
+
this.updateBusy = true;
|
|
445
|
+
return this.getStatus()
|
|
446
|
+
.then((bikeData) => {
|
|
447
|
+
if (bikeData) {
|
|
448
|
+
try {
|
|
449
|
+
this.kettlerData = bikeData;
|
|
450
|
+
let data = this.mapData(bikeData);
|
|
451
|
+
data = this.getCyclingMode().updateData(data);
|
|
452
|
+
this.internalData = data;
|
|
453
|
+
this.data = this.transformData(data, bikeData);
|
|
454
|
+
}
|
|
455
|
+
catch (err) {
|
|
456
|
+
this.logger.logEvent({ message: 'bike update error', error: err.message });
|
|
457
|
+
}
|
|
458
|
+
}
|
|
459
|
+
this.updateBusy = false;
|
|
460
|
+
})
|
|
461
|
+
.catch(err => {
|
|
462
|
+
this.logger.logEvent({ message: 'bike update error', error: err.message });
|
|
463
|
+
this.updateBusy = false;
|
|
464
|
+
});
|
|
465
|
+
});
|
|
466
|
+
}
|
|
467
|
+
sendRequest(request) {
|
|
468
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
469
|
+
this.requestBusy = true;
|
|
470
|
+
try {
|
|
471
|
+
this.logger.logEvent({ message: 'sendRequest', request });
|
|
472
|
+
const isReset = (!request || request.reset || Object.keys(request).length === 0);
|
|
473
|
+
if (isReset) {
|
|
474
|
+
this.requestBusy = false;
|
|
475
|
+
return {};
|
|
476
|
+
}
|
|
477
|
+
if (request.slope !== undefined) {
|
|
478
|
+
this.data.slope = request.slope;
|
|
479
|
+
}
|
|
480
|
+
if (request.targetPower !== undefined) {
|
|
481
|
+
yield this.setPower(request.targetPower);
|
|
482
|
+
}
|
|
483
|
+
this.requestBusy = false;
|
|
484
|
+
return request;
|
|
485
|
+
}
|
|
486
|
+
catch (err) {
|
|
487
|
+
this.requestBusy = false;
|
|
488
|
+
this.logger.logEvent({ message: 'error', fn: 'sendRequest()', error: err.message || err });
|
|
489
|
+
return;
|
|
490
|
+
}
|
|
491
|
+
});
|
|
492
|
+
}
|
|
493
|
+
sendRequests() {
|
|
494
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
495
|
+
if (this.requests.length > 0) {
|
|
496
|
+
const processing = [...this.requests];
|
|
497
|
+
const cnt = processing.length;
|
|
498
|
+
processing.forEach((request, idx) => __awaiter(this, void 0, void 0, function* () {
|
|
499
|
+
if (cnt > 1 && idx < cnt - 1) {
|
|
500
|
+
this.logger.logEvent({ message: 'ignoring bike update request', request });
|
|
501
|
+
this.requests.shift();
|
|
502
|
+
return;
|
|
503
|
+
}
|
|
504
|
+
}));
|
|
505
|
+
const request = processing[0];
|
|
506
|
+
try {
|
|
507
|
+
yield this.sendRequest(request);
|
|
508
|
+
this.requests.shift();
|
|
509
|
+
}
|
|
510
|
+
catch (err) {
|
|
511
|
+
this.logger.logEvent({ message: 'bike update error', error: err.message, stack: err.stack, request });
|
|
512
|
+
}
|
|
513
|
+
}
|
|
514
|
+
});
|
|
515
|
+
}
|
|
516
|
+
bikeSync() {
|
|
517
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
518
|
+
if (this.paused) {
|
|
519
|
+
return;
|
|
520
|
+
}
|
|
521
|
+
if (this.updateBusy || this.requestBusy) {
|
|
522
|
+
return;
|
|
523
|
+
}
|
|
524
|
+
this.logger.logEvent({ message: 'bikeSync' });
|
|
525
|
+
if (!this.ignoreBike) {
|
|
526
|
+
yield this.sendRequests();
|
|
527
|
+
}
|
|
528
|
+
yield this.update();
|
|
529
|
+
});
|
|
530
|
+
}
|
|
531
|
+
sendUpdate(request) {
|
|
532
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
533
|
+
if (this.paused)
|
|
534
|
+
return;
|
|
535
|
+
this.logger.logEvent({ message: 'sendUpdate', request, waiting: this.requests.length });
|
|
536
|
+
return yield this.processClientRequest(request);
|
|
537
|
+
});
|
|
538
|
+
}
|
|
539
|
+
sendData() {
|
|
540
|
+
if (this.onDataFn)
|
|
541
|
+
this.onDataFn(this.data);
|
|
542
|
+
}
|
|
543
|
+
refreshRequests() {
|
|
544
|
+
if (this.kettlerData.cadence === 0)
|
|
545
|
+
return;
|
|
546
|
+
let bikeRequest = this.getCyclingMode().sendBikeUpdate({ refresh: true }) || {};
|
|
547
|
+
const prev = this.requests[this.requests.length - 1] || {};
|
|
548
|
+
if (bikeRequest.targetPower !== undefined && bikeRequest.targetPower !== prev.targetPower) {
|
|
549
|
+
this.logger.logEvent({ message: 'add request', request: bikeRequest });
|
|
550
|
+
this.requests.push(bikeRequest);
|
|
551
|
+
}
|
|
552
|
+
}
|
|
553
|
+
processClientRequest(request) {
|
|
554
|
+
if (request.slope !== undefined) {
|
|
555
|
+
this.data.slope = request.slope;
|
|
556
|
+
}
|
|
557
|
+
return new Promise((resolve) => __awaiter(this, void 0, void 0, function* () {
|
|
558
|
+
let bikeRequest = this.getCyclingMode().sendBikeUpdate(request);
|
|
559
|
+
this.logger.logEvent({ message: 'add request', request: bikeRequest });
|
|
560
|
+
this.requests.push(bikeRequest);
|
|
561
|
+
resolve(bikeRequest);
|
|
562
|
+
}));
|
|
563
|
+
}
|
|
564
|
+
waitForOpened(retries = false) {
|
|
565
|
+
const run = (resolve, reject) => {
|
|
566
|
+
try {
|
|
567
|
+
if (this.comms.isConnected()) {
|
|
568
|
+
resolve(true);
|
|
569
|
+
return;
|
|
570
|
+
}
|
|
571
|
+
const cleanup = () => {
|
|
572
|
+
this.comms.removeAllListeners();
|
|
573
|
+
};
|
|
574
|
+
const onOpen = () => {
|
|
575
|
+
resolve(true);
|
|
576
|
+
cleanup();
|
|
577
|
+
};
|
|
578
|
+
const onError = (err) => { reject(err); cleanup(); };
|
|
579
|
+
const onClose = () => { cleanup(); };
|
|
580
|
+
this.comms.on('opened', onOpen);
|
|
581
|
+
this.comms.on('closed', onClose);
|
|
582
|
+
this.comms.on('error', onError);
|
|
583
|
+
this.logger.logEvent({ message: 'opening', port: this.getPort() });
|
|
584
|
+
this.comms.open();
|
|
585
|
+
}
|
|
586
|
+
catch (err) {
|
|
587
|
+
this.logger.logEvent({ message: 'error', fn: 'waitForOpened()', error: err.message || err });
|
|
588
|
+
reject(err);
|
|
589
|
+
}
|
|
590
|
+
};
|
|
591
|
+
if (!retries) {
|
|
592
|
+
return new Promise((resolve, reject) => run(resolve, reject));
|
|
593
|
+
}
|
|
594
|
+
return (0, utils_1.runWithRetries)(() => {
|
|
595
|
+
return new Promise((resolve, reject) => run(resolve, reject));
|
|
596
|
+
}, 3, 1000);
|
|
597
|
+
}
|
|
598
|
+
waitForClosed() {
|
|
599
|
+
return new Promise((resolve, reject) => {
|
|
600
|
+
try {
|
|
601
|
+
if (!this.comms.isConnected()) {
|
|
602
|
+
resolve(true);
|
|
603
|
+
return;
|
|
604
|
+
}
|
|
605
|
+
const cleanup = () => {
|
|
606
|
+
this.comms.removeAllListeners();
|
|
607
|
+
};
|
|
608
|
+
const onClose = () => {
|
|
609
|
+
resolve(true);
|
|
610
|
+
cleanup();
|
|
611
|
+
};
|
|
612
|
+
const onError = (err) => { reject(err); cleanup(); };
|
|
613
|
+
const onOpen = () => { cleanup(); };
|
|
614
|
+
this.comms.on('closed', onClose);
|
|
615
|
+
this.comms.on('opened', onOpen);
|
|
616
|
+
this.comms.on('error', onError);
|
|
617
|
+
this.logger.logEvent({ message: 'closing', port: this.getPort() });
|
|
618
|
+
this.comms.close();
|
|
619
|
+
}
|
|
620
|
+
catch (err) {
|
|
621
|
+
this.logger.logEvent({ message: 'error', fn: 'waitForClosed()', error: err.message || err });
|
|
622
|
+
reject(err);
|
|
623
|
+
}
|
|
624
|
+
});
|
|
625
|
+
}
|
|
626
|
+
getSupportedCyclingModes() {
|
|
627
|
+
return [power_meter_1.default, ERGCyclingMode_1.default];
|
|
628
|
+
}
|
|
629
|
+
getDefaultCyclingMode() {
|
|
630
|
+
return new ERGCyclingMode_1.default(this);
|
|
631
|
+
}
|
|
632
|
+
setUserSettings(userSettings) {
|
|
633
|
+
this.settings.userSettings = userSettings;
|
|
634
|
+
}
|
|
635
|
+
setBikeSettings(bikeSettings) {
|
|
636
|
+
this.settings.bikeSettings = bikeSettings;
|
|
637
|
+
}
|
|
638
|
+
}
|
|
639
|
+
exports.default = KettlerRacerAdapter;
|