incyclist-devices 1.4.102 → 1.5.0
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/DeviceSupport.d.ts +36 -34
- package/lib/DeviceSupport.js +82 -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 -83
- package/lib/ant/antfe/AntFEAdapter.js +652 -652
- 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/antv2/adapter-factory.d.ts +11 -0
- package/lib/antv2/adapter-factory.js +40 -0
- package/lib/antv2/ant-binding.d.ts +13 -0
- package/lib/antv2/ant-binding.js +27 -0
- package/lib/antv2/ant-device.d.ts +51 -0
- package/lib/antv2/ant-device.js +115 -0
- package/lib/antv2/ant-interface.d.ts +37 -0
- package/lib/antv2/ant-interface.js +239 -0
- package/lib/antv2/fe.d.ts +28 -0
- package/lib/antv2/fe.js +237 -0
- package/lib/antv2/hr.d.ts +18 -0
- package/lib/antv2/hr.js +93 -0
- package/lib/antv2/incyclist-protocol.d.ts +37 -0
- package/lib/antv2/incyclist-protocol.js +126 -0
- package/lib/antv2/pwr.d.ts +28 -0
- package/lib/antv2/pwr.js +163 -0
- package/lib/antv2/sensor-factory.d.ts +5 -0
- package/lib/antv2/sensor-factory.js +20 -0
- package/lib/ble/ble-device.d.ts +63 -63
- package/lib/ble/ble-device.js +444 -444
- package/lib/ble/ble-erg-mode.d.ts +18 -18
- package/lib/ble/ble-erg-mode.js +132 -132
- package/lib/ble/ble-interface.d.ts +100 -100
- package/lib/ble/ble-interface.js +721 -719
- 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 +95 -95
- 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 +745 -745
- 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 -731
- 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/{CyclingMode.d.ts → cycling-mode.d.ts} +76 -76
- package/lib/{CyclingMode.js → cycling-mode.js} +79 -79
- 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/{Device.d.ts → device.d.ts} +92 -92
- package/lib/{Device.js → device.js} +71 -71
- 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 -144
- 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/{DeviceProtocol.d.ts → protocol.d.ts} +74 -74
- package/lib/{DeviceProtocol.js → protocol.js} +41 -41
- package/lib/{DeviceRegistry.d.ts → registry.d.ts} +8 -8
- package/lib/{DeviceRegistry.js → registry.js} +33 -33
- 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 +47 -46
package/lib/ble/wahoo-kickr.js
CHANGED
|
@@ -1,496 +1,496 @@
|
|
|
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
|
-
exports.WahooAdvancedFmAdapter = void 0;
|
|
35
|
-
const ble_interface_1 = __importDefault(require("./ble-interface"));
|
|
36
|
-
const ble_1 = require("./ble");
|
|
37
|
-
const
|
|
38
|
-
const gd_eventlog_1 = require("gd-eventlog");
|
|
39
|
-
const fm_1 = __importStar(require("./fm"));
|
|
40
|
-
const consts_1 = require("./consts");
|
|
41
|
-
const WAHOO_ADVANCED_FTMS = 'a026ee0b';
|
|
42
|
-
const cwABike = {
|
|
43
|
-
race: 0.35,
|
|
44
|
-
triathlon: 0.29,
|
|
45
|
-
mountain: 0.57
|
|
46
|
-
};
|
|
47
|
-
const cRR = 0.0036;
|
|
48
|
-
const ErgWriteDelay = 2000;
|
|
49
|
-
class WahooAdvancedFitnessMachineDevice extends fm_1.default {
|
|
50
|
-
constructor(props) {
|
|
51
|
-
super(props);
|
|
52
|
-
this.prevCrankData = undefined;
|
|
53
|
-
this.currentCrankData = undefined;
|
|
54
|
-
this.timeOffset = 0;
|
|
55
|
-
this.tsPrevWrite = undefined;
|
|
56
|
-
this.prevSlope = undefined;
|
|
57
|
-
this.isRequestControlBusy = false;
|
|
58
|
-
this.weight =
|
|
59
|
-
this.data = {};
|
|
60
|
-
this.wahooCP = consts_1.WAHOO_ADVANCED_TRAINER_CP;
|
|
61
|
-
}
|
|
62
|
-
isMatching(characteristics) {
|
|
63
|
-
if (!characteristics)
|
|
64
|
-
return false;
|
|
65
|
-
const hasWahooCP = characteristics.find(c => (0, ble_1.matches)(c, consts_1.WAHOO_ADVANCED_TRAINER_CP)) !== undefined;
|
|
66
|
-
const hasFTMS = characteristics.find(c => (0, ble_1.matches)(c, consts_1.FTMS_CP)) !== undefined;
|
|
67
|
-
return hasWahooCP && !hasFTMS;
|
|
68
|
-
}
|
|
69
|
-
init() {
|
|
70
|
-
const _super = Object.create(null, {
|
|
71
|
-
initDevice: { get: () => super.initDevice }
|
|
72
|
-
});
|
|
73
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
74
|
-
try {
|
|
75
|
-
yield this.subscribeWriteResponse(this.wahooCP);
|
|
76
|
-
try {
|
|
77
|
-
if (this.wahooCP !== consts_1.WAHOO_ADVANCED_TRAINER_CP_FULL.toLowerCase())
|
|
78
|
-
yield this.subscribeWriteResponse(consts_1.WAHOO_ADVANCED_TRAINER_CP_FULL.toLowerCase());
|
|
79
|
-
}
|
|
80
|
-
catch (err) {
|
|
81
|
-
}
|
|
82
|
-
yield _super.initDevice.call(this);
|
|
83
|
-
return true;
|
|
84
|
-
}
|
|
85
|
-
catch (err) {
|
|
86
|
-
this.logEvent({ message: 'error', fn: 'WahooAdvancedFitnessMachineDevice.init()', error: err.message || err, stack: err.stack });
|
|
87
|
-
return false;
|
|
88
|
-
}
|
|
89
|
-
});
|
|
90
|
-
}
|
|
91
|
-
setCharacteristicUUIDs(uuids) {
|
|
92
|
-
this.logEvent({ message: 'set uuids', uuids });
|
|
93
|
-
uuids.forEach(c => {
|
|
94
|
-
if ((0, ble_1.matches)(c, consts_1.WAHOO_ADVANCED_TRAINER_CP))
|
|
95
|
-
this.wahooCP = c;
|
|
96
|
-
});
|
|
97
|
-
}
|
|
98
|
-
getProfile() {
|
|
99
|
-
return 'Wahoo Smart Trainer';
|
|
100
|
-
}
|
|
101
|
-
getServiceUUids() {
|
|
102
|
-
return WahooAdvancedFitnessMachineDevice.services;
|
|
103
|
-
}
|
|
104
|
-
isBike() {
|
|
105
|
-
return true;
|
|
106
|
-
}
|
|
107
|
-
isPower() {
|
|
108
|
-
return true;
|
|
109
|
-
}
|
|
110
|
-
isHrm() {
|
|
111
|
-
return this.hasService('180d');
|
|
112
|
-
}
|
|
113
|
-
parseCrankData(crankData) {
|
|
114
|
-
if (!this.prevCrankData)
|
|
115
|
-
this.prevCrankData = { revolutions: 0, time: 0, cntUpdateMissing: -1 };
|
|
116
|
-
const c = this.currentCrankData = crankData;
|
|
117
|
-
const p = this.prevCrankData;
|
|
118
|
-
let rpm = this.data.cadence;
|
|
119
|
-
let hasUpdate = c.time !== p.time;
|
|
120
|
-
if (hasUpdate) {
|
|
121
|
-
let time = c.time - p.time;
|
|
122
|
-
let revs = c.revolutions - p.revolutions;
|
|
123
|
-
if (c.time < p.time) {
|
|
124
|
-
time += 0x10000;
|
|
125
|
-
this.timeOffset += 0x10000;
|
|
126
|
-
}
|
|
127
|
-
if (c.revolutions < p.revolutions)
|
|
128
|
-
revs += 0x10000;
|
|
129
|
-
rpm = 1024 * 60 * revs / time;
|
|
130
|
-
}
|
|
131
|
-
else {
|
|
132
|
-
if (p.cntUpdateMissing < 0 || p.cntUpdateMissing > 2) {
|
|
133
|
-
rpm = 0;
|
|
134
|
-
}
|
|
135
|
-
}
|
|
136
|
-
const cntUpdateMissing = p.cntUpdateMissing;
|
|
137
|
-
this.prevCrankData = this.currentCrankData;
|
|
138
|
-
if (hasUpdate)
|
|
139
|
-
this.prevCrankData.cntUpdateMissing = 0;
|
|
140
|
-
else
|
|
141
|
-
this.prevCrankData.cntUpdateMissing = cntUpdateMissing + 1;
|
|
142
|
-
return { rpm, time: this.timeOffset + c.time };
|
|
143
|
-
}
|
|
144
|
-
parsePower(_data) {
|
|
145
|
-
const data = Buffer.from(_data);
|
|
146
|
-
try {
|
|
147
|
-
let offset = 4;
|
|
148
|
-
const flags = data.readUInt16LE(0);
|
|
149
|
-
this.data.instantaneousPower = data.readUInt16LE(2);
|
|
150
|
-
if (flags & 0x1)
|
|
151
|
-
data.readUInt8(offset++);
|
|
152
|
-
if (flags & 0x4) {
|
|
153
|
-
data.readUInt16LE(offset);
|
|
154
|
-
offset += 2;
|
|
155
|
-
}
|
|
156
|
-
if (flags & 0x20) {
|
|
157
|
-
const crankData = {
|
|
158
|
-
revolutions: data.readUInt16LE(offset),
|
|
159
|
-
time: data.readUInt16LE(offset + 2)
|
|
160
|
-
};
|
|
161
|
-
const { rpm, time } = this.parseCrankData(crankData);
|
|
162
|
-
this.data.cadence = rpm;
|
|
163
|
-
this.data.time = time;
|
|
164
|
-
offset += 4;
|
|
165
|
-
}
|
|
166
|
-
}
|
|
167
|
-
catch (err) {
|
|
168
|
-
}
|
|
169
|
-
const { instantaneousPower, cadence, time } = this.data;
|
|
170
|
-
return { instantaneousPower, cadence, time, raw: data.toString('hex') };
|
|
171
|
-
}
|
|
172
|
-
onData(characteristic, data) {
|
|
173
|
-
const hasData = super.onData(characteristic, data);
|
|
174
|
-
if (!hasData)
|
|
175
|
-
return false;
|
|
176
|
-
const uuid = characteristic.toLowerCase();
|
|
177
|
-
let res = undefined;
|
|
178
|
-
switch (uuid) {
|
|
179
|
-
case '2a63':
|
|
180
|
-
res = this.parsePower(data);
|
|
181
|
-
break;
|
|
182
|
-
case '2ad2':
|
|
183
|
-
res = this.parseIndoorBikeData(data);
|
|
184
|
-
break;
|
|
185
|
-
case '2a37':
|
|
186
|
-
res = this.parseHrm(data);
|
|
187
|
-
break;
|
|
188
|
-
case '2ada':
|
|
189
|
-
res = this.parseFitnessMachineStatus(data);
|
|
190
|
-
break;
|
|
191
|
-
default:
|
|
192
|
-
this.logEvent({ message: 'data', uuid, data: data.toString('hex') });
|
|
193
|
-
break;
|
|
194
|
-
}
|
|
195
|
-
if (res) {
|
|
196
|
-
this.emit('data', res);
|
|
197
|
-
return false;
|
|
198
|
-
}
|
|
199
|
-
return true;
|
|
200
|
-
}
|
|
201
|
-
writeWahooFtmsMessage(requestedOpCode, data, props) {
|
|
202
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
203
|
-
try {
|
|
204
|
-
const opcode = Buffer.alloc(1);
|
|
205
|
-
opcode.writeUInt8(requestedOpCode, 0);
|
|
206
|
-
const message = Buffer.concat([opcode, data]);
|
|
207
|
-
this.logEvent({ message: 'wahoo cp:write', data: message.toString('hex') });
|
|
208
|
-
const res = yield this.write(this.wahooCP, message, props);
|
|
209
|
-
const responseData = Buffer.from(res);
|
|
210
|
-
const result = responseData.readUInt8(0);
|
|
211
|
-
return result === 1;
|
|
212
|
-
}
|
|
213
|
-
catch (err) {
|
|
214
|
-
this.logEvent({ message: 'wahoo cp:write failed', opCode: requestedOpCode, reason: err.message });
|
|
215
|
-
return false;
|
|
216
|
-
}
|
|
217
|
-
});
|
|
218
|
-
}
|
|
219
|
-
requestControl() {
|
|
220
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
221
|
-
if (this.hasControl)
|
|
222
|
-
return true;
|
|
223
|
-
this.logEvent({ message: 'requestControl' });
|
|
224
|
-
if (this.isRequestControlBusy)
|
|
225
|
-
return false;
|
|
226
|
-
this.isRequestControlBusy = true;
|
|
227
|
-
try {
|
|
228
|
-
const data = Buffer.alloc(2);
|
|
229
|
-
data.writeUInt8(0xEE, 0);
|
|
230
|
-
data.writeUInt8(0xFC, 1);
|
|
231
|
-
const res = yield this.writeWahooFtmsMessage(32, data, { timeout: 10000 });
|
|
232
|
-
if (res === true) {
|
|
233
|
-
this.hasControl = true;
|
|
234
|
-
}
|
|
235
|
-
else {
|
|
236
|
-
this.logEvent({ message: 'requestControl failed' });
|
|
237
|
-
}
|
|
238
|
-
}
|
|
239
|
-
catch (err) {
|
|
240
|
-
this.logEvent({ message: 'error', fn: 'requestControl()', error: err.message || err, stack: err.stack });
|
|
241
|
-
}
|
|
242
|
-
this.isRequestControlBusy = false;
|
|
243
|
-
return this.hasControl;
|
|
244
|
-
});
|
|
245
|
-
}
|
|
246
|
-
setPowerAdjusting() {
|
|
247
|
-
this.tsPrevWrite = Date.now();
|
|
248
|
-
}
|
|
249
|
-
isPowerAdjusting() {
|
|
250
|
-
if (this.tsPrevWrite === undefined)
|
|
251
|
-
return false;
|
|
252
|
-
if (this.tsPrevWrite < Date.now() - ErgWriteDelay) {
|
|
253
|
-
this.tsPrevWrite = undefined;
|
|
254
|
-
return false;
|
|
255
|
-
}
|
|
256
|
-
return true;
|
|
257
|
-
}
|
|
258
|
-
setErgMode(power) {
|
|
259
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
260
|
-
this.logger.logEvent({ message: 'setErgMode', power });
|
|
261
|
-
try {
|
|
262
|
-
if (this.isPowerAdjusting())
|
|
263
|
-
return false;
|
|
264
|
-
const data = Buffer.alloc(2);
|
|
265
|
-
data.writeInt16LE(Math.round(power), 0);
|
|
266
|
-
const res = yield this.writeWahooFtmsMessage(66, data);
|
|
267
|
-
if (res === true) {
|
|
268
|
-
this.setPowerAdjusting();
|
|
269
|
-
this.data.targetPower = power;
|
|
270
|
-
this.isSimMode = false;
|
|
271
|
-
this.simModeSettings = undefined;
|
|
272
|
-
}
|
|
273
|
-
return res;
|
|
274
|
-
}
|
|
275
|
-
catch (err) {
|
|
276
|
-
this.logEvent({ message: 'error', fn: 'setErgMode', error: err.message || err, stack: err.stack });
|
|
277
|
-
return false;
|
|
278
|
-
}
|
|
279
|
-
});
|
|
280
|
-
}
|
|
281
|
-
setSimMode(weight, crr, cw) {
|
|
282
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
283
|
-
this.logger.logEvent({ message: 'setSimMode', weight, crr, cw });
|
|
284
|
-
try {
|
|
285
|
-
if (this.isSimMode && this.simModeSettings) {
|
|
286
|
-
if (weight === this.simModeSettings.weight &&
|
|
287
|
-
crr === this.simModeSettings.crr &&
|
|
288
|
-
cw === this.simModeSettings.cw)
|
|
289
|
-
return true;
|
|
290
|
-
}
|
|
291
|
-
const hasControl = yield this.requestControl();
|
|
292
|
-
if (!hasControl) {
|
|
293
|
-
this.logEvent({ message: 'setSimMode failed', reason: 'control is disabled' });
|
|
294
|
-
return false;
|
|
295
|
-
}
|
|
296
|
-
this.weight = weight;
|
|
297
|
-
this.crr = crr;
|
|
298
|
-
this.cw = cw;
|
|
299
|
-
const data = Buffer.alloc(6);
|
|
300
|
-
data.writeInt16LE(Math.round(weight * 100), 0);
|
|
301
|
-
data.writeInt16LE(Math.round(crr * 10000), 2);
|
|
302
|
-
data.writeInt16LE(Math.round(cw * 1000), 4);
|
|
303
|
-
const res = yield this.writeWahooFtmsMessage(67, data);
|
|
304
|
-
this.isSimMode = true;
|
|
305
|
-
this.simModeSettings = { weight, crr, cw };
|
|
306
|
-
return res;
|
|
307
|
-
}
|
|
308
|
-
catch (err) {
|
|
309
|
-
this.logEvent({ message: 'error', fn: 'setSimMode', error: err.message || err, stack: err.stack });
|
|
310
|
-
return false;
|
|
311
|
-
}
|
|
312
|
-
});
|
|
313
|
-
}
|
|
314
|
-
setSimCRR(crr) {
|
|
315
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
316
|
-
this.logger.logEvent({ message: 'setSimCRR', crr });
|
|
317
|
-
try {
|
|
318
|
-
const data = Buffer.alloc(2);
|
|
319
|
-
data.writeInt16LE(Math.round(crr * 10000), 0);
|
|
320
|
-
const res = yield this.writeWahooFtmsMessage(68, data);
|
|
321
|
-
return res;
|
|
322
|
-
}
|
|
323
|
-
catch (err) {
|
|
324
|
-
this.logEvent({ message: 'error', fn: 'setSimCRR', error: err.message || err, stack: err.stack });
|
|
325
|
-
return false;
|
|
326
|
-
}
|
|
327
|
-
});
|
|
328
|
-
}
|
|
329
|
-
setSimWindResistance(cw) {
|
|
330
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
331
|
-
this.logger.logEvent({ message: 'setSimWindResistance', cw });
|
|
332
|
-
try {
|
|
333
|
-
const data = Buffer.alloc(2);
|
|
334
|
-
data.writeInt16LE(Math.round(cw * 1000), 0);
|
|
335
|
-
const res = yield this.writeWahooFtmsMessage(69, data);
|
|
336
|
-
return res;
|
|
337
|
-
}
|
|
338
|
-
catch (err) {
|
|
339
|
-
this.logEvent({ message: 'error', fn: 'setSimWindResistance', error: err.message || err, stack: err.stack });
|
|
340
|
-
return false;
|
|
341
|
-
}
|
|
342
|
-
});
|
|
343
|
-
}
|
|
344
|
-
setSimGrade(slope) {
|
|
345
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
346
|
-
this.logger.logEvent({ message: 'setSimGrade', slope });
|
|
347
|
-
try {
|
|
348
|
-
let s = slope;
|
|
349
|
-
if (s < -100)
|
|
350
|
-
s = -100;
|
|
351
|
-
if (s > 100)
|
|
352
|
-
s = 100;
|
|
353
|
-
const slopeVal = Math.min(Math.round((1 + s / 100) * 65535 / 2.0), 65535);
|
|
354
|
-
const data = Buffer.alloc(2);
|
|
355
|
-
data.writeUInt16LE(slopeVal, 0);
|
|
356
|
-
const res = yield this.writeWahooFtmsMessage(70, data);
|
|
357
|
-
return res;
|
|
358
|
-
}
|
|
359
|
-
catch (err) {
|
|
360
|
-
this.logEvent({ message: 'error', fn: 'setSimGrade', error: err.message || err, stack: err.stack });
|
|
361
|
-
return false;
|
|
362
|
-
}
|
|
363
|
-
});
|
|
364
|
-
}
|
|
365
|
-
setSimWindSpeed(v) {
|
|
366
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
367
|
-
this.logger.logEvent({ message: 'setSimWindSpeed', v });
|
|
368
|
-
try {
|
|
369
|
-
const value = (Math.max(-32.767, Math.min(32.767, v)) + 32.767) * 1000;
|
|
370
|
-
const data = Buffer.alloc(2);
|
|
371
|
-
data.writeInt16LE(Math.round(value), 0);
|
|
372
|
-
const res = yield this.writeWahooFtmsMessage(71, data);
|
|
373
|
-
return res;
|
|
374
|
-
}
|
|
375
|
-
catch (err) {
|
|
376
|
-
this.logEvent({ message: 'error', fn: 'setSimWindSpeed', error: err.message || err, stack: err.stack });
|
|
377
|
-
return false;
|
|
378
|
-
}
|
|
379
|
-
});
|
|
380
|
-
}
|
|
381
|
-
setTargetPower(power) {
|
|
382
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
383
|
-
this.logEvent({ message: 'setTargetPower', power, skip: (this.data.targetPower !== undefined && this.data.targetPower === power) });
|
|
384
|
-
if (this.data.targetPower !== undefined && this.data.targetPower === power)
|
|
385
|
-
return true;
|
|
386
|
-
const hasControl = yield this.requestControl();
|
|
387
|
-
if (!hasControl) {
|
|
388
|
-
this.logEvent({ message: 'setTargetPower failed', reason: 'control is disabled' });
|
|
389
|
-
return false;
|
|
390
|
-
}
|
|
391
|
-
return yield this.setErgMode(power);
|
|
392
|
-
});
|
|
393
|
-
}
|
|
394
|
-
setSlope(slope) {
|
|
395
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
396
|
-
this.logEvent({ message: 'setSlope', slope });
|
|
397
|
-
if (this.prevSlope !== undefined && slope === this.prevSlope)
|
|
398
|
-
return;
|
|
399
|
-
try {
|
|
400
|
-
if (!this.isSimMode) {
|
|
401
|
-
const { weight, crr, cw } = this;
|
|
402
|
-
const hasSimMode = yield this.setSimMode(weight, crr, cw);
|
|
403
|
-
if (!hasSimMode)
|
|
404
|
-
throw new Error('Sim Mode not enabled');
|
|
405
|
-
}
|
|
406
|
-
const res = yield this.setSimGrade(slope);
|
|
407
|
-
this.logEvent({ message: 'setSlope result', res });
|
|
408
|
-
this.prevSlope = slope;
|
|
409
|
-
return res;
|
|
410
|
-
}
|
|
411
|
-
catch (err) {
|
|
412
|
-
this.logEvent({ message: 'setSlope failed', reason: err.message || err });
|
|
413
|
-
this.prevSlope = undefined;
|
|
414
|
-
return false;
|
|
415
|
-
}
|
|
416
|
-
});
|
|
417
|
-
}
|
|
418
|
-
reset() {
|
|
419
|
-
this.data = {};
|
|
420
|
-
this.isSimMode = undefined;
|
|
421
|
-
this.simModeSettings = undefined;
|
|
422
|
-
}
|
|
423
|
-
}
|
|
424
|
-
exports.default = WahooAdvancedFitnessMachineDevice;
|
|
425
|
-
WahooAdvancedFitnessMachineDevice.services = [consts_1.CSP];
|
|
426
|
-
WahooAdvancedFitnessMachineDevice.characteristics = ['2acc', '2ad2', '2ad6', '2ad8', '2ad9', '2ada', consts_1.WAHOO_ADVANCED_TRAINER_CP];
|
|
427
|
-
WahooAdvancedFitnessMachineDevice.detectionPriority = 5;
|
|
428
|
-
ble_interface_1.default.register('WahooAdvancedFitnessMachineDevice', 'wahoo-fm', WahooAdvancedFitnessMachineDevice, WahooAdvancedFitnessMachineDevice.services);
|
|
429
|
-
class WahooAdvancedFmAdapter extends fm_1.FmAdapter {
|
|
430
|
-
constructor(device, protocol) {
|
|
431
|
-
super(device, protocol);
|
|
432
|
-
this.device = device;
|
|
433
|
-
this.ble = protocol.ble;
|
|
434
|
-
this.cyclingMode = this.getDefaultCyclingMode();
|
|
435
|
-
this.logger = new gd_eventlog_1.EventLogger('BLE-WahooFM');
|
|
436
|
-
if (this.device)
|
|
437
|
-
this.device.setLogger(this.logger);
|
|
438
|
-
}
|
|
439
|
-
isSame(device) {
|
|
440
|
-
if (!(device instanceof WahooAdvancedFmAdapter))
|
|
441
|
-
return false;
|
|
442
|
-
const adapter = device;
|
|
443
|
-
return (adapter.getName() === this.getName() && adapter.getProfile() === this.getProfile());
|
|
444
|
-
}
|
|
445
|
-
getProfile() {
|
|
446
|
-
return 'Wahoo Smart Trainer';
|
|
447
|
-
}
|
|
448
|
-
start(props) {
|
|
449
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
450
|
-
this.logger.logEvent({ message: 'wahoo: start requested', profile: this.getProfile(), props, isScanning: this.ble.isScanning() });
|
|
451
|
-
try {
|
|
452
|
-
this.logger.logEvent({ message: 'wahoo: stop previous scan', isScanning: this.ble.isScanning() });
|
|
453
|
-
if (this.ble.isScanning())
|
|
454
|
-
yield this.ble.stopScan();
|
|
455
|
-
this.logger.logEvent({ message: 'wahoo trying device connect' });
|
|
456
|
-
const bleDevice = yield this.ble.connectDevice(this.device);
|
|
457
|
-
this.logger.logEvent({ message: 'wahoo device connected', connected: (bleDevice !== null && bleDevice !== undefined) });
|
|
458
|
-
if (bleDevice) {
|
|
459
|
-
this.device = bleDevice;
|
|
460
|
-
const mode = this.getCyclingMode();
|
|
461
|
-
if (mode && mode.getSetting('bikeType')) {
|
|
462
|
-
const bikeType = mode.getSetting('bikeType').toLowerCase();
|
|
463
|
-
this.device.setCrr(cRR);
|
|
464
|
-
switch (bikeType) {
|
|
465
|
-
case 'race':
|
|
466
|
-
this.device.setCw(cwABike.race);
|
|
467
|
-
break;
|
|
468
|
-
case 'triathlon':
|
|
469
|
-
this.device.setCw(cwABike.triathlon);
|
|
470
|
-
break;
|
|
471
|
-
case 'mountain':
|
|
472
|
-
this.device.setCw(cwABike.mountain);
|
|
473
|
-
break;
|
|
474
|
-
}
|
|
475
|
-
}
|
|
476
|
-
const { user } = props || {};
|
|
477
|
-
const weight = (user && user.weight ? Number(user.weight) :
|
|
478
|
-
yield this.device.setSimMode(weight, this.device.getCrr(), this.device.getCw());
|
|
479
|
-
const startRequest = this.getCyclingMode().getBikeInitRequest();
|
|
480
|
-
yield this.sendUpdate(startRequest);
|
|
481
|
-
bleDevice.on('data', (data) => {
|
|
482
|
-
this.onDeviceData(data);
|
|
483
|
-
});
|
|
484
|
-
return true;
|
|
485
|
-
}
|
|
486
|
-
}
|
|
487
|
-
catch (err) {
|
|
488
|
-
this.logger.logEvent({ message: 'start result: error', error: err.message, profile: this.getProfile() });
|
|
489
|
-
throw new Error(`could not start device, reason:${err.message}`);
|
|
490
|
-
}
|
|
491
|
-
});
|
|
492
|
-
}
|
|
493
|
-
pause() { this.paused = true; return Promise.resolve(true); }
|
|
494
|
-
resume() { this.paused = false; return Promise.resolve(true); }
|
|
495
|
-
}
|
|
496
|
-
exports.WahooAdvancedFmAdapter = WahooAdvancedFmAdapter;
|
|
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
|
+
exports.WahooAdvancedFmAdapter = void 0;
|
|
35
|
+
const ble_interface_1 = __importDefault(require("./ble-interface"));
|
|
36
|
+
const ble_1 = require("./ble");
|
|
37
|
+
const device_1 = require("../device");
|
|
38
|
+
const gd_eventlog_1 = require("gd-eventlog");
|
|
39
|
+
const fm_1 = __importStar(require("./fm"));
|
|
40
|
+
const consts_1 = require("./consts");
|
|
41
|
+
const WAHOO_ADVANCED_FTMS = 'a026ee0b';
|
|
42
|
+
const cwABike = {
|
|
43
|
+
race: 0.35,
|
|
44
|
+
triathlon: 0.29,
|
|
45
|
+
mountain: 0.57
|
|
46
|
+
};
|
|
47
|
+
const cRR = 0.0036;
|
|
48
|
+
const ErgWriteDelay = 2000;
|
|
49
|
+
class WahooAdvancedFitnessMachineDevice extends fm_1.default {
|
|
50
|
+
constructor(props) {
|
|
51
|
+
super(props);
|
|
52
|
+
this.prevCrankData = undefined;
|
|
53
|
+
this.currentCrankData = undefined;
|
|
54
|
+
this.timeOffset = 0;
|
|
55
|
+
this.tsPrevWrite = undefined;
|
|
56
|
+
this.prevSlope = undefined;
|
|
57
|
+
this.isRequestControlBusy = false;
|
|
58
|
+
this.weight = device_1.DEFAULT_BIKE_WEIGHT + device_1.DEFAULT_USER_WEIGHT;
|
|
59
|
+
this.data = {};
|
|
60
|
+
this.wahooCP = consts_1.WAHOO_ADVANCED_TRAINER_CP;
|
|
61
|
+
}
|
|
62
|
+
isMatching(characteristics) {
|
|
63
|
+
if (!characteristics)
|
|
64
|
+
return false;
|
|
65
|
+
const hasWahooCP = characteristics.find(c => (0, ble_1.matches)(c, consts_1.WAHOO_ADVANCED_TRAINER_CP)) !== undefined;
|
|
66
|
+
const hasFTMS = characteristics.find(c => (0, ble_1.matches)(c, consts_1.FTMS_CP)) !== undefined;
|
|
67
|
+
return hasWahooCP && !hasFTMS;
|
|
68
|
+
}
|
|
69
|
+
init() {
|
|
70
|
+
const _super = Object.create(null, {
|
|
71
|
+
initDevice: { get: () => super.initDevice }
|
|
72
|
+
});
|
|
73
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
74
|
+
try {
|
|
75
|
+
yield this.subscribeWriteResponse(this.wahooCP);
|
|
76
|
+
try {
|
|
77
|
+
if (this.wahooCP !== consts_1.WAHOO_ADVANCED_TRAINER_CP_FULL.toLowerCase())
|
|
78
|
+
yield this.subscribeWriteResponse(consts_1.WAHOO_ADVANCED_TRAINER_CP_FULL.toLowerCase());
|
|
79
|
+
}
|
|
80
|
+
catch (err) {
|
|
81
|
+
}
|
|
82
|
+
yield _super.initDevice.call(this);
|
|
83
|
+
return true;
|
|
84
|
+
}
|
|
85
|
+
catch (err) {
|
|
86
|
+
this.logEvent({ message: 'error', fn: 'WahooAdvancedFitnessMachineDevice.init()', error: err.message || err, stack: err.stack });
|
|
87
|
+
return false;
|
|
88
|
+
}
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
setCharacteristicUUIDs(uuids) {
|
|
92
|
+
this.logEvent({ message: 'set uuids', uuids });
|
|
93
|
+
uuids.forEach(c => {
|
|
94
|
+
if ((0, ble_1.matches)(c, consts_1.WAHOO_ADVANCED_TRAINER_CP))
|
|
95
|
+
this.wahooCP = c;
|
|
96
|
+
});
|
|
97
|
+
}
|
|
98
|
+
getProfile() {
|
|
99
|
+
return 'Wahoo Smart Trainer';
|
|
100
|
+
}
|
|
101
|
+
getServiceUUids() {
|
|
102
|
+
return WahooAdvancedFitnessMachineDevice.services;
|
|
103
|
+
}
|
|
104
|
+
isBike() {
|
|
105
|
+
return true;
|
|
106
|
+
}
|
|
107
|
+
isPower() {
|
|
108
|
+
return true;
|
|
109
|
+
}
|
|
110
|
+
isHrm() {
|
|
111
|
+
return this.hasService('180d');
|
|
112
|
+
}
|
|
113
|
+
parseCrankData(crankData) {
|
|
114
|
+
if (!this.prevCrankData)
|
|
115
|
+
this.prevCrankData = { revolutions: 0, time: 0, cntUpdateMissing: -1 };
|
|
116
|
+
const c = this.currentCrankData = crankData;
|
|
117
|
+
const p = this.prevCrankData;
|
|
118
|
+
let rpm = this.data.cadence;
|
|
119
|
+
let hasUpdate = c.time !== p.time;
|
|
120
|
+
if (hasUpdate) {
|
|
121
|
+
let time = c.time - p.time;
|
|
122
|
+
let revs = c.revolutions - p.revolutions;
|
|
123
|
+
if (c.time < p.time) {
|
|
124
|
+
time += 0x10000;
|
|
125
|
+
this.timeOffset += 0x10000;
|
|
126
|
+
}
|
|
127
|
+
if (c.revolutions < p.revolutions)
|
|
128
|
+
revs += 0x10000;
|
|
129
|
+
rpm = 1024 * 60 * revs / time;
|
|
130
|
+
}
|
|
131
|
+
else {
|
|
132
|
+
if (p.cntUpdateMissing < 0 || p.cntUpdateMissing > 2) {
|
|
133
|
+
rpm = 0;
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
const cntUpdateMissing = p.cntUpdateMissing;
|
|
137
|
+
this.prevCrankData = this.currentCrankData;
|
|
138
|
+
if (hasUpdate)
|
|
139
|
+
this.prevCrankData.cntUpdateMissing = 0;
|
|
140
|
+
else
|
|
141
|
+
this.prevCrankData.cntUpdateMissing = cntUpdateMissing + 1;
|
|
142
|
+
return { rpm, time: this.timeOffset + c.time };
|
|
143
|
+
}
|
|
144
|
+
parsePower(_data) {
|
|
145
|
+
const data = Buffer.from(_data);
|
|
146
|
+
try {
|
|
147
|
+
let offset = 4;
|
|
148
|
+
const flags = data.readUInt16LE(0);
|
|
149
|
+
this.data.instantaneousPower = data.readUInt16LE(2);
|
|
150
|
+
if (flags & 0x1)
|
|
151
|
+
data.readUInt8(offset++);
|
|
152
|
+
if (flags & 0x4) {
|
|
153
|
+
data.readUInt16LE(offset);
|
|
154
|
+
offset += 2;
|
|
155
|
+
}
|
|
156
|
+
if (flags & 0x20) {
|
|
157
|
+
const crankData = {
|
|
158
|
+
revolutions: data.readUInt16LE(offset),
|
|
159
|
+
time: data.readUInt16LE(offset + 2)
|
|
160
|
+
};
|
|
161
|
+
const { rpm, time } = this.parseCrankData(crankData);
|
|
162
|
+
this.data.cadence = rpm;
|
|
163
|
+
this.data.time = time;
|
|
164
|
+
offset += 4;
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
catch (err) {
|
|
168
|
+
}
|
|
169
|
+
const { instantaneousPower, cadence, time } = this.data;
|
|
170
|
+
return { instantaneousPower, cadence, time, raw: data.toString('hex') };
|
|
171
|
+
}
|
|
172
|
+
onData(characteristic, data) {
|
|
173
|
+
const hasData = super.onData(characteristic, data);
|
|
174
|
+
if (!hasData)
|
|
175
|
+
return false;
|
|
176
|
+
const uuid = characteristic.toLowerCase();
|
|
177
|
+
let res = undefined;
|
|
178
|
+
switch (uuid) {
|
|
179
|
+
case '2a63':
|
|
180
|
+
res = this.parsePower(data);
|
|
181
|
+
break;
|
|
182
|
+
case '2ad2':
|
|
183
|
+
res = this.parseIndoorBikeData(data);
|
|
184
|
+
break;
|
|
185
|
+
case '2a37':
|
|
186
|
+
res = this.parseHrm(data);
|
|
187
|
+
break;
|
|
188
|
+
case '2ada':
|
|
189
|
+
res = this.parseFitnessMachineStatus(data);
|
|
190
|
+
break;
|
|
191
|
+
default:
|
|
192
|
+
this.logEvent({ message: 'data', uuid, data: data.toString('hex') });
|
|
193
|
+
break;
|
|
194
|
+
}
|
|
195
|
+
if (res) {
|
|
196
|
+
this.emit('data', res);
|
|
197
|
+
return false;
|
|
198
|
+
}
|
|
199
|
+
return true;
|
|
200
|
+
}
|
|
201
|
+
writeWahooFtmsMessage(requestedOpCode, data, props) {
|
|
202
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
203
|
+
try {
|
|
204
|
+
const opcode = Buffer.alloc(1);
|
|
205
|
+
opcode.writeUInt8(requestedOpCode, 0);
|
|
206
|
+
const message = Buffer.concat([opcode, data]);
|
|
207
|
+
this.logEvent({ message: 'wahoo cp:write', data: message.toString('hex') });
|
|
208
|
+
const res = yield this.write(this.wahooCP, message, props);
|
|
209
|
+
const responseData = Buffer.from(res);
|
|
210
|
+
const result = responseData.readUInt8(0);
|
|
211
|
+
return result === 1;
|
|
212
|
+
}
|
|
213
|
+
catch (err) {
|
|
214
|
+
this.logEvent({ message: 'wahoo cp:write failed', opCode: requestedOpCode, reason: err.message });
|
|
215
|
+
return false;
|
|
216
|
+
}
|
|
217
|
+
});
|
|
218
|
+
}
|
|
219
|
+
requestControl() {
|
|
220
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
221
|
+
if (this.hasControl)
|
|
222
|
+
return true;
|
|
223
|
+
this.logEvent({ message: 'requestControl' });
|
|
224
|
+
if (this.isRequestControlBusy)
|
|
225
|
+
return false;
|
|
226
|
+
this.isRequestControlBusy = true;
|
|
227
|
+
try {
|
|
228
|
+
const data = Buffer.alloc(2);
|
|
229
|
+
data.writeUInt8(0xEE, 0);
|
|
230
|
+
data.writeUInt8(0xFC, 1);
|
|
231
|
+
const res = yield this.writeWahooFtmsMessage(32, data, { timeout: 10000 });
|
|
232
|
+
if (res === true) {
|
|
233
|
+
this.hasControl = true;
|
|
234
|
+
}
|
|
235
|
+
else {
|
|
236
|
+
this.logEvent({ message: 'requestControl failed' });
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
catch (err) {
|
|
240
|
+
this.logEvent({ message: 'error', fn: 'requestControl()', error: err.message || err, stack: err.stack });
|
|
241
|
+
}
|
|
242
|
+
this.isRequestControlBusy = false;
|
|
243
|
+
return this.hasControl;
|
|
244
|
+
});
|
|
245
|
+
}
|
|
246
|
+
setPowerAdjusting() {
|
|
247
|
+
this.tsPrevWrite = Date.now();
|
|
248
|
+
}
|
|
249
|
+
isPowerAdjusting() {
|
|
250
|
+
if (this.tsPrevWrite === undefined)
|
|
251
|
+
return false;
|
|
252
|
+
if (this.tsPrevWrite < Date.now() - ErgWriteDelay) {
|
|
253
|
+
this.tsPrevWrite = undefined;
|
|
254
|
+
return false;
|
|
255
|
+
}
|
|
256
|
+
return true;
|
|
257
|
+
}
|
|
258
|
+
setErgMode(power) {
|
|
259
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
260
|
+
this.logger.logEvent({ message: 'setErgMode', power });
|
|
261
|
+
try {
|
|
262
|
+
if (this.isPowerAdjusting())
|
|
263
|
+
return false;
|
|
264
|
+
const data = Buffer.alloc(2);
|
|
265
|
+
data.writeInt16LE(Math.round(power), 0);
|
|
266
|
+
const res = yield this.writeWahooFtmsMessage(66, data);
|
|
267
|
+
if (res === true) {
|
|
268
|
+
this.setPowerAdjusting();
|
|
269
|
+
this.data.targetPower = power;
|
|
270
|
+
this.isSimMode = false;
|
|
271
|
+
this.simModeSettings = undefined;
|
|
272
|
+
}
|
|
273
|
+
return res;
|
|
274
|
+
}
|
|
275
|
+
catch (err) {
|
|
276
|
+
this.logEvent({ message: 'error', fn: 'setErgMode', error: err.message || err, stack: err.stack });
|
|
277
|
+
return false;
|
|
278
|
+
}
|
|
279
|
+
});
|
|
280
|
+
}
|
|
281
|
+
setSimMode(weight, crr, cw) {
|
|
282
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
283
|
+
this.logger.logEvent({ message: 'setSimMode', weight, crr, cw });
|
|
284
|
+
try {
|
|
285
|
+
if (this.isSimMode && this.simModeSettings) {
|
|
286
|
+
if (weight === this.simModeSettings.weight &&
|
|
287
|
+
crr === this.simModeSettings.crr &&
|
|
288
|
+
cw === this.simModeSettings.cw)
|
|
289
|
+
return true;
|
|
290
|
+
}
|
|
291
|
+
const hasControl = yield this.requestControl();
|
|
292
|
+
if (!hasControl) {
|
|
293
|
+
this.logEvent({ message: 'setSimMode failed', reason: 'control is disabled' });
|
|
294
|
+
return false;
|
|
295
|
+
}
|
|
296
|
+
this.weight = weight;
|
|
297
|
+
this.crr = crr;
|
|
298
|
+
this.cw = cw;
|
|
299
|
+
const data = Buffer.alloc(6);
|
|
300
|
+
data.writeInt16LE(Math.round(weight * 100), 0);
|
|
301
|
+
data.writeInt16LE(Math.round(crr * 10000), 2);
|
|
302
|
+
data.writeInt16LE(Math.round(cw * 1000), 4);
|
|
303
|
+
const res = yield this.writeWahooFtmsMessage(67, data);
|
|
304
|
+
this.isSimMode = true;
|
|
305
|
+
this.simModeSettings = { weight, crr, cw };
|
|
306
|
+
return res;
|
|
307
|
+
}
|
|
308
|
+
catch (err) {
|
|
309
|
+
this.logEvent({ message: 'error', fn: 'setSimMode', error: err.message || err, stack: err.stack });
|
|
310
|
+
return false;
|
|
311
|
+
}
|
|
312
|
+
});
|
|
313
|
+
}
|
|
314
|
+
setSimCRR(crr) {
|
|
315
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
316
|
+
this.logger.logEvent({ message: 'setSimCRR', crr });
|
|
317
|
+
try {
|
|
318
|
+
const data = Buffer.alloc(2);
|
|
319
|
+
data.writeInt16LE(Math.round(crr * 10000), 0);
|
|
320
|
+
const res = yield this.writeWahooFtmsMessage(68, data);
|
|
321
|
+
return res;
|
|
322
|
+
}
|
|
323
|
+
catch (err) {
|
|
324
|
+
this.logEvent({ message: 'error', fn: 'setSimCRR', error: err.message || err, stack: err.stack });
|
|
325
|
+
return false;
|
|
326
|
+
}
|
|
327
|
+
});
|
|
328
|
+
}
|
|
329
|
+
setSimWindResistance(cw) {
|
|
330
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
331
|
+
this.logger.logEvent({ message: 'setSimWindResistance', cw });
|
|
332
|
+
try {
|
|
333
|
+
const data = Buffer.alloc(2);
|
|
334
|
+
data.writeInt16LE(Math.round(cw * 1000), 0);
|
|
335
|
+
const res = yield this.writeWahooFtmsMessage(69, data);
|
|
336
|
+
return res;
|
|
337
|
+
}
|
|
338
|
+
catch (err) {
|
|
339
|
+
this.logEvent({ message: 'error', fn: 'setSimWindResistance', error: err.message || err, stack: err.stack });
|
|
340
|
+
return false;
|
|
341
|
+
}
|
|
342
|
+
});
|
|
343
|
+
}
|
|
344
|
+
setSimGrade(slope) {
|
|
345
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
346
|
+
this.logger.logEvent({ message: 'setSimGrade', slope });
|
|
347
|
+
try {
|
|
348
|
+
let s = slope;
|
|
349
|
+
if (s < -100)
|
|
350
|
+
s = -100;
|
|
351
|
+
if (s > 100)
|
|
352
|
+
s = 100;
|
|
353
|
+
const slopeVal = Math.min(Math.round((1 + s / 100) * 65535 / 2.0), 65535);
|
|
354
|
+
const data = Buffer.alloc(2);
|
|
355
|
+
data.writeUInt16LE(slopeVal, 0);
|
|
356
|
+
const res = yield this.writeWahooFtmsMessage(70, data);
|
|
357
|
+
return res;
|
|
358
|
+
}
|
|
359
|
+
catch (err) {
|
|
360
|
+
this.logEvent({ message: 'error', fn: 'setSimGrade', error: err.message || err, stack: err.stack });
|
|
361
|
+
return false;
|
|
362
|
+
}
|
|
363
|
+
});
|
|
364
|
+
}
|
|
365
|
+
setSimWindSpeed(v) {
|
|
366
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
367
|
+
this.logger.logEvent({ message: 'setSimWindSpeed', v });
|
|
368
|
+
try {
|
|
369
|
+
const value = (Math.max(-32.767, Math.min(32.767, v)) + 32.767) * 1000;
|
|
370
|
+
const data = Buffer.alloc(2);
|
|
371
|
+
data.writeInt16LE(Math.round(value), 0);
|
|
372
|
+
const res = yield this.writeWahooFtmsMessage(71, data);
|
|
373
|
+
return res;
|
|
374
|
+
}
|
|
375
|
+
catch (err) {
|
|
376
|
+
this.logEvent({ message: 'error', fn: 'setSimWindSpeed', error: err.message || err, stack: err.stack });
|
|
377
|
+
return false;
|
|
378
|
+
}
|
|
379
|
+
});
|
|
380
|
+
}
|
|
381
|
+
setTargetPower(power) {
|
|
382
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
383
|
+
this.logEvent({ message: 'setTargetPower', power, skip: (this.data.targetPower !== undefined && this.data.targetPower === power) });
|
|
384
|
+
if (this.data.targetPower !== undefined && this.data.targetPower === power)
|
|
385
|
+
return true;
|
|
386
|
+
const hasControl = yield this.requestControl();
|
|
387
|
+
if (!hasControl) {
|
|
388
|
+
this.logEvent({ message: 'setTargetPower failed', reason: 'control is disabled' });
|
|
389
|
+
return false;
|
|
390
|
+
}
|
|
391
|
+
return yield this.setErgMode(power);
|
|
392
|
+
});
|
|
393
|
+
}
|
|
394
|
+
setSlope(slope) {
|
|
395
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
396
|
+
this.logEvent({ message: 'setSlope', slope });
|
|
397
|
+
if (this.prevSlope !== undefined && slope === this.prevSlope)
|
|
398
|
+
return;
|
|
399
|
+
try {
|
|
400
|
+
if (!this.isSimMode) {
|
|
401
|
+
const { weight, crr, cw } = this;
|
|
402
|
+
const hasSimMode = yield this.setSimMode(weight, crr, cw);
|
|
403
|
+
if (!hasSimMode)
|
|
404
|
+
throw new Error('Sim Mode not enabled');
|
|
405
|
+
}
|
|
406
|
+
const res = yield this.setSimGrade(slope);
|
|
407
|
+
this.logEvent({ message: 'setSlope result', res });
|
|
408
|
+
this.prevSlope = slope;
|
|
409
|
+
return res;
|
|
410
|
+
}
|
|
411
|
+
catch (err) {
|
|
412
|
+
this.logEvent({ message: 'setSlope failed', reason: err.message || err });
|
|
413
|
+
this.prevSlope = undefined;
|
|
414
|
+
return false;
|
|
415
|
+
}
|
|
416
|
+
});
|
|
417
|
+
}
|
|
418
|
+
reset() {
|
|
419
|
+
this.data = {};
|
|
420
|
+
this.isSimMode = undefined;
|
|
421
|
+
this.simModeSettings = undefined;
|
|
422
|
+
}
|
|
423
|
+
}
|
|
424
|
+
exports.default = WahooAdvancedFitnessMachineDevice;
|
|
425
|
+
WahooAdvancedFitnessMachineDevice.services = [consts_1.CSP];
|
|
426
|
+
WahooAdvancedFitnessMachineDevice.characteristics = ['2acc', '2ad2', '2ad6', '2ad8', '2ad9', '2ada', consts_1.WAHOO_ADVANCED_TRAINER_CP];
|
|
427
|
+
WahooAdvancedFitnessMachineDevice.detectionPriority = 5;
|
|
428
|
+
ble_interface_1.default.register('WahooAdvancedFitnessMachineDevice', 'wahoo-fm', WahooAdvancedFitnessMachineDevice, WahooAdvancedFitnessMachineDevice.services);
|
|
429
|
+
class WahooAdvancedFmAdapter extends fm_1.FmAdapter {
|
|
430
|
+
constructor(device, protocol) {
|
|
431
|
+
super(device, protocol);
|
|
432
|
+
this.device = device;
|
|
433
|
+
this.ble = protocol.ble;
|
|
434
|
+
this.cyclingMode = this.getDefaultCyclingMode();
|
|
435
|
+
this.logger = new gd_eventlog_1.EventLogger('BLE-WahooFM');
|
|
436
|
+
if (this.device)
|
|
437
|
+
this.device.setLogger(this.logger);
|
|
438
|
+
}
|
|
439
|
+
isSame(device) {
|
|
440
|
+
if (!(device instanceof WahooAdvancedFmAdapter))
|
|
441
|
+
return false;
|
|
442
|
+
const adapter = device;
|
|
443
|
+
return (adapter.getName() === this.getName() && adapter.getProfile() === this.getProfile());
|
|
444
|
+
}
|
|
445
|
+
getProfile() {
|
|
446
|
+
return 'Wahoo Smart Trainer';
|
|
447
|
+
}
|
|
448
|
+
start(props) {
|
|
449
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
450
|
+
this.logger.logEvent({ message: 'wahoo: start requested', profile: this.getProfile(), props, isScanning: this.ble.isScanning() });
|
|
451
|
+
try {
|
|
452
|
+
this.logger.logEvent({ message: 'wahoo: stop previous scan', isScanning: this.ble.isScanning() });
|
|
453
|
+
if (this.ble.isScanning())
|
|
454
|
+
yield this.ble.stopScan();
|
|
455
|
+
this.logger.logEvent({ message: 'wahoo trying device connect' });
|
|
456
|
+
const bleDevice = yield this.ble.connectDevice(this.device);
|
|
457
|
+
this.logger.logEvent({ message: 'wahoo device connected', connected: (bleDevice !== null && bleDevice !== undefined) });
|
|
458
|
+
if (bleDevice) {
|
|
459
|
+
this.device = bleDevice;
|
|
460
|
+
const mode = this.getCyclingMode();
|
|
461
|
+
if (mode && mode.getSetting('bikeType')) {
|
|
462
|
+
const bikeType = mode.getSetting('bikeType').toLowerCase();
|
|
463
|
+
this.device.setCrr(cRR);
|
|
464
|
+
switch (bikeType) {
|
|
465
|
+
case 'race':
|
|
466
|
+
this.device.setCw(cwABike.race);
|
|
467
|
+
break;
|
|
468
|
+
case 'triathlon':
|
|
469
|
+
this.device.setCw(cwABike.triathlon);
|
|
470
|
+
break;
|
|
471
|
+
case 'mountain':
|
|
472
|
+
this.device.setCw(cwABike.mountain);
|
|
473
|
+
break;
|
|
474
|
+
}
|
|
475
|
+
}
|
|
476
|
+
const { user } = props || {};
|
|
477
|
+
const weight = (user && user.weight ? Number(user.weight) : device_1.DEFAULT_USER_WEIGHT) + device_1.DEFAULT_BIKE_WEIGHT;
|
|
478
|
+
yield this.device.setSimMode(weight, this.device.getCrr(), this.device.getCw());
|
|
479
|
+
const startRequest = this.getCyclingMode().getBikeInitRequest();
|
|
480
|
+
yield this.sendUpdate(startRequest);
|
|
481
|
+
bleDevice.on('data', (data) => {
|
|
482
|
+
this.onDeviceData(data);
|
|
483
|
+
});
|
|
484
|
+
return true;
|
|
485
|
+
}
|
|
486
|
+
}
|
|
487
|
+
catch (err) {
|
|
488
|
+
this.logger.logEvent({ message: 'start result: error', error: err.message, profile: this.getProfile() });
|
|
489
|
+
throw new Error(`could not start device, reason:${err.message}`);
|
|
490
|
+
}
|
|
491
|
+
});
|
|
492
|
+
}
|
|
493
|
+
pause() { this.paused = true; return Promise.resolve(true); }
|
|
494
|
+
resume() { this.paused = false; return Promise.resolve(true); }
|
|
495
|
+
}
|
|
496
|
+
exports.WahooAdvancedFmAdapter = WahooAdvancedFmAdapter;
|