incyclist-devices 1.4.24 → 1.4.27

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. package/lib/Device.d.ts +1 -1
  2. package/lib/Device.js +2 -2
  3. package/lib/DeviceProtocol.d.ts +3 -1
  4. package/lib/DeviceProtocol.js +2 -0
  5. package/lib/DeviceSupport.d.ts +6 -1
  6. package/lib/DeviceSupport.js +9 -0
  7. package/lib/ant/AntScanner.js +7 -5
  8. package/lib/ant/antfe/AntFEAdapter.js +0 -19
  9. package/lib/ant/anthrm/AntHrmAdapter.js +0 -11
  10. package/lib/ant/antpwr/pwr-adapter.js +0 -11
  11. package/lib/ble/ble-device.d.ts +36 -0
  12. package/lib/ble/ble-device.js +174 -0
  13. package/lib/ble/ble-interface.d.ts +51 -0
  14. package/lib/ble/ble-interface.js +330 -0
  15. package/lib/ble/ble.d.ts +81 -0
  16. package/lib/ble/ble.js +51 -0
  17. package/lib/ble/hrm.d.ts +48 -0
  18. package/lib/ble/hrm.js +128 -0
  19. package/lib/ble/incyclist-protocol.d.ts +31 -0
  20. package/lib/ble/incyclist-protocol.js +106 -0
  21. package/lib/ble/pwr.d.ts +32 -0
  22. package/lib/ble/pwr.js +96 -0
  23. package/lib/calculations.js +2 -3
  24. package/lib/daum/DaumAdapter.d.ts +4 -4
  25. package/lib/daum/DaumAdapter.js +24 -20
  26. package/lib/daum/DaumPowerMeterCyclingMode.d.ts +8 -0
  27. package/lib/daum/DaumPowerMeterCyclingMode.js +21 -0
  28. package/lib/daum/ERGCyclingMode.d.ts +3 -6
  29. package/lib/daum/ERGCyclingMode.js +15 -27
  30. package/lib/daum/classic/DaumClassicAdapter.js +1 -1
  31. package/lib/daum/premium/DaumClassicCyclingMode.d.ts +2 -2
  32. package/lib/daum/premium/DaumClassicCyclingMode.js +2 -2
  33. package/lib/daum/premium/DaumPremiumAdapter.js +1 -1
  34. package/lib/daum/premium/tcpserial.js +0 -1
  35. package/lib/modes/power-base.d.ts +20 -0
  36. package/lib/modes/power-base.js +70 -0
  37. package/lib/modes/power-meter.d.ts +12 -10
  38. package/lib/modes/power-meter.js +30 -46
  39. package/lib/modes/simulator.d.ts +27 -0
  40. package/lib/modes/simulator.js +118 -0
  41. package/lib/simulator/Simulator.js +3 -3
  42. package/package.json +2 -1
@@ -0,0 +1,70 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ const CyclingMode_1 = require("../CyclingMode");
7
+ const Device_1 = require("../Device");
8
+ const calculations_1 = __importDefault(require("../calculations"));
9
+ const gd_eventlog_1 = require("gd-eventlog");
10
+ class PowerBasedCyclingModeBase extends CyclingMode_1.CyclingModeBase {
11
+ constructor(adapter, props) {
12
+ super(adapter, props);
13
+ this.prevUpdateTS = 0;
14
+ this.data = { speed: 0, power: 0, distanceInternal: 0, pedalRpm: 0, isPedalling: false, heartrate: 0 };
15
+ }
16
+ initLogger(defaultLogName) {
17
+ const a = this.adapter;
18
+ this.logger = (a && a.getLogger) ? a.getLogger() : undefined;
19
+ if (!this.logger)
20
+ this.logger = new gd_eventlog_1.EventLogger(defaultLogName);
21
+ }
22
+ getWeight() {
23
+ const a = this.adapter;
24
+ const m = (a && a.getWeight && a.getWeight()) ? a.getWeight() : Device_1.DEFAULT_BIKE_WEIGHT + Device_1.DEFAULT_USER_WEIGHT;
25
+ return m;
26
+ }
27
+ getTimeSinceLastUpdate() {
28
+ const ts = Date.now();
29
+ const duration = this.prevUpdateTS === 0 ? 0 : ((ts - this.prevUpdateTS) / 1000);
30
+ this.prevUpdateTS = ts;
31
+ return duration;
32
+ }
33
+ calculateSpeedAndDistance(power, slope, m, t, props = {}) {
34
+ const prevData = this.data || {};
35
+ const vPrev = (prevData.speed || 0) / 3.6;
36
+ const EkinPrev = 1 / 2 * m * vPrev * vPrev;
37
+ let powerToMaintainSpeed = calculations_1.default.calculatePower(m, vPrev, slope, props);
38
+ const powerDelta = powerToMaintainSpeed - power;
39
+ const Ekin = EkinPrev - powerDelta * t;
40
+ if (Ekin > 0) {
41
+ const v = Math.sqrt(2 * Ekin / m);
42
+ const speed = v * 3.6;
43
+ const distance = v * t;
44
+ this.data.speed = speed;
45
+ return { speed, distance };
46
+ }
47
+ else {
48
+ const v = vPrev * 0.5;
49
+ const speed = v * 3.6;
50
+ const distance = v * t;
51
+ this.data.speed = speed;
52
+ return { speed, distance };
53
+ }
54
+ }
55
+ calculatePowerAndDistance(speed, slope, m, t, props = {}) {
56
+ const prevData = this.data || {};
57
+ const vPrev = (prevData.speed || 0) / 3.6;
58
+ const EkinPrev = 1 / 2 * m * vPrev * vPrev;
59
+ const vTarget = (speed || 0) / 3.6;
60
+ const Ekin = 1 / 2 * m * vTarget * vTarget;
61
+ const powerDelta = t !== 0 ? (EkinPrev - Ekin) / t : 0;
62
+ const powerToMaintainSpeed = calculations_1.default.calculatePower(m, vPrev, slope, props);
63
+ const power = powerToMaintainSpeed - powerDelta;
64
+ const v = speed / 3.6;
65
+ const distance = v * t;
66
+ this.data.power = power;
67
+ return { power, distance };
68
+ }
69
+ }
70
+ exports.default = PowerBasedCyclingModeBase;
@@ -1,18 +1,20 @@
1
- import { EventLogger } from 'gd-eventlog';
2
- import CyclingMode, { CyclingModeProperty, IncyclistBikeData, Settings, UpdateRequest, CyclingModeBase } from '../CyclingMode';
1
+ import CyclingMode, { CyclingModeProperty, IncyclistBikeData, Settings, UpdateRequest } from '../CyclingMode';
2
+ import PowerBasedCyclingModeBase from './power-base';
3
3
  import { DeviceAdapter } from '../Device';
4
- export default class PowerMeterCyclingMode extends CyclingModeBase implements CyclingMode {
5
- logger: EventLogger;
6
- data: IncyclistBikeData;
7
- prevRequest: UpdateRequest;
8
- prevUpdateTS: number;
9
- hasBikeUpdate: boolean;
4
+ export declare const config: {
5
+ name: string;
6
+ description: string;
7
+ properties: any[];
8
+ };
9
+ export default class PowerMeterCyclingMode extends PowerBasedCyclingModeBase implements CyclingMode {
10
10
  constructor(adapter: DeviceAdapter, props?: Settings);
11
11
  getName(): string;
12
12
  getDescription(): string;
13
13
  getProperties(): CyclingModeProperty[];
14
14
  getProperty(name: string): CyclingModeProperty;
15
15
  getBikeInitRequest(): UpdateRequest;
16
- sendBikeUpdate(request: UpdateRequest): UpdateRequest;
17
- updateData(data: IncyclistBikeData): IncyclistBikeData;
16
+ sendBikeUpdate(request?: UpdateRequest): UpdateRequest;
17
+ updateData(bikeData: IncyclistBikeData, props?: {
18
+ log: boolean;
19
+ }): IncyclistBikeData;
18
20
  }
@@ -3,84 +3,68 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
3
3
  return (mod && mod.__esModule) ? mod : { "default": mod };
4
4
  };
5
5
  Object.defineProperty(exports, "__esModule", { value: true });
6
- const gd_eventlog_1 = require("gd-eventlog");
7
- const CyclingMode_1 = require("../CyclingMode");
8
- const Device_1 = require("../Device");
9
- const calculations_1 = __importDefault(require("../calculations"));
10
- const config = {
6
+ const power_base_1 = __importDefault(require("./power-base"));
7
+ exports.config = {
11
8
  name: 'PowerMeter',
12
9
  description: 'Power and cadence are taken from device. Speed is calculated from power and current slope\nThis mode will not respect maximum power and/or workout limits',
13
10
  properties: []
14
11
  };
15
- class PowerMeterCyclingMode extends CyclingMode_1.CyclingModeBase {
12
+ class PowerMeterCyclingMode extends power_base_1.default {
16
13
  constructor(adapter, props) {
17
14
  super(adapter, props);
18
- this.prevUpdateTS = 0;
19
- this.hasBikeUpdate = false;
20
- const a = adapter;
21
- this.logger = (a && a.getLogger) ? a.getLogger() : undefined;
22
- if (!this.logger)
23
- this.logger = new gd_eventlog_1.EventLogger('PowerMeter');
15
+ this.initLogger('PowerMeterMode');
16
+ this.data = { speed: 0, slope: 0, power: 0, distanceInternal: 0, pedalRpm: 0, isPedalling: false, heartrate: 0 };
24
17
  }
25
18
  getName() {
26
- return config.name;
19
+ return exports.config.name;
27
20
  }
28
21
  getDescription() {
29
- return config.description;
22
+ return exports.config.description;
30
23
  }
31
24
  getProperties() {
32
- return config.properties;
25
+ return exports.config.properties;
33
26
  }
34
27
  getProperty(name) {
35
- return config.properties.find(p => p.name === name);
28
+ return exports.config.properties.find(p => p.name === name);
36
29
  }
37
30
  getBikeInitRequest() {
38
31
  return {};
39
32
  }
40
- sendBikeUpdate(request) {
41
- if (request.slope)
33
+ sendBikeUpdate(request = {}) {
34
+ const prevData = this.data || {};
35
+ const prevSlope = prevData.slope || 0;
36
+ if (request.slope && request.slope !== prevSlope) {
42
37
  this.data.slope = request.slope;
43
- this.logger.logEvent({ message: "processing update request", request, prev: this.prevRequest });
44
- this.prevRequest = {};
38
+ this.updateData(this.data, { log: false });
39
+ }
40
+ this.logger.logEvent({ message: "processing update request", request });
45
41
  return {};
46
42
  }
47
- updateData(data) {
43
+ updateData(bikeData, props = { log: true }) {
48
44
  try {
49
45
  const prevData = this.data || {};
50
- const prevRequest = this.prevRequest || {};
51
- const bikeData = JSON.parse(JSON.stringify(data));
52
- let power = data.power || 0;
53
- let speed = data.speed || 0;
54
- let slope = (prevData.slope !== undefined ? prevData.slope : prevRequest.slope || 0);
55
- let distanceInternal = prevData.distanceInternal || 0;
46
+ const data = JSON.parse(JSON.stringify(bikeData));
47
+ let power = bikeData.power || 0;
48
+ const slope = prevData.slope || 0;
49
+ const distanceInternal = prevData.distanceInternal || 0;
56
50
  if (!bikeData.pedalRpm || bikeData.isPedalling === false) {
57
- speed = 0;
58
51
  power = 0;
59
52
  }
60
- let ts = Date.now();
61
- const a = this.adapter;
62
- const m = a.getWeight ? a.getWeight() : Device_1.DEFAULT_BIKE_WEIGHT + Device_1.DEFAULT_USER_WEIGHT;
63
- let duration = this.prevUpdateTS === 0 ? 0 : ((ts - this.prevUpdateTS) / 1000);
64
- const vPrev = (prevData.speed || 0) / 3.6;
65
- const EkinPrev = 1 / 2 * m * vPrev * vPrev;
66
- let powerRequired = calculations_1.default.calculatePower(m, vPrev, prevData.slope || 0);
67
- const powerDelta = powerRequired - power;
68
- const Ekin = EkinPrev - powerDelta * duration;
69
- const v = Math.sqrt(2 * Ekin / m);
70
- speed = v * 3.6;
71
- distanceInternal += Math.round(v * duration);
72
- data.speed = parseFloat(speed.toFixed(1));
53
+ const m = this.getWeight();
54
+ let t = this.getTimeSinceLastUpdate();
55
+ const { speed, distance } = this.calculateSpeedAndDistance(power, slope, m, t);
56
+ data.speed = speed;
73
57
  data.power = Math.round(power);
74
- data.distanceInternal = Math.round(distanceInternal);
58
+ data.distanceInternal = Math.round(distanceInternal + distance);
75
59
  data.slope = slope;
76
- this.logger.logEvent({ message: "updateData result", data, bikeData, prevRequest: {}, prevSpeed: prevData.speed });
77
- this.data = JSON.parse(JSON.stringify(data));
78
- this.prevUpdateTS = ts;
60
+ if (props.log)
61
+ this.logger.logEvent({ message: "updateData result", data, bikeData, prevSpeed: prevData.speed });
62
+ this.data = data;
79
63
  }
80
64
  catch (err) {
81
65
  this.logger.logEvent({ message: 'error', fn: 'updateData()', error: err.message || err });
82
66
  }
83
- return data;
67
+ return this.data;
84
68
  }
85
69
  }
86
70
  exports.default = PowerMeterCyclingMode;
@@ -0,0 +1,27 @@
1
+ import { EventLogger } from "gd-eventlog";
2
+ import CyclingMode, { CyclingModeBase, CyclingModeProperty, IncyclistBikeData, UpdateRequest } from "../CyclingMode";
3
+ import { Simulator } from "../simulator/Simulator";
4
+ export declare type ERGEvent = {
5
+ rpmUpdated?: boolean;
6
+ gearUpdated?: boolean;
7
+ starting?: boolean;
8
+ tsStart?: number;
9
+ };
10
+ export default class SimulatorCyclingMode extends CyclingModeBase implements CyclingMode {
11
+ logger: EventLogger;
12
+ data: IncyclistBikeData;
13
+ prevRequest: UpdateRequest;
14
+ prevUpdateTS: number;
15
+ hasBikeUpdate: boolean;
16
+ chain: number[];
17
+ cassette: number[];
18
+ event: ERGEvent;
19
+ constructor(adapter: Simulator, props?: any);
20
+ getName(): string;
21
+ getDescription(): string;
22
+ getProperties(): CyclingModeProperty[];
23
+ getProperty(name: string): CyclingModeProperty;
24
+ getBikeInitRequest(): UpdateRequest;
25
+ sendBikeUpdate(request: UpdateRequest): UpdateRequest;
26
+ updateData(bikeData: IncyclistBikeData): IncyclistBikeData;
27
+ }
@@ -0,0 +1,118 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ const gd_eventlog_1 = require("gd-eventlog");
7
+ const CyclingMode_1 = require("../CyclingMode");
8
+ const calculations_1 = __importDefault(require("../calculations"));
9
+ const config = {
10
+ name: "Simulator",
11
+ description: "Simulates a ride with constant speed or power output",
12
+ properties: [
13
+ { key: 'mode', name: 'Simulation Type', description: '', type: CyclingMode_1.CyclingModeProperyType.SingleSelect, options: ['Speed', 'Power'], default: 'Power' },
14
+ { key: 'delay', name: 'Start Delay', description: 'Delay (in s) at start of training', type: CyclingMode_1.CyclingModeProperyType.Integer, default: 2, min: 0, max: 30 },
15
+ { key: 'power', name: 'Power', description: 'Power (in W) at start of training', condition: (s) => !s.mode || s.mode === 'Power', type: CyclingMode_1.CyclingModeProperyType.Integer, default: 150, min: 25, max: 800 },
16
+ { key: 'speed', name: 'Speed', description: 'Speed (in km/h) at start of training', condition: (s) => s.mode === 'Speed', type: CyclingMode_1.CyclingModeProperyType.Integer, default: 30, min: 5, max: 50 },
17
+ ]
18
+ };
19
+ class SimulatorCyclingMode extends CyclingMode_1.CyclingModeBase {
20
+ constructor(adapter, props) {
21
+ super(adapter, props);
22
+ this.prevUpdateTS = 0;
23
+ this.hasBikeUpdate = false;
24
+ this.event = {};
25
+ this.logger = adapter.logger || new gd_eventlog_1.EventLogger('SIMMode');
26
+ this.data = {};
27
+ this.logger.logEvent({ message: 'constructor', props });
28
+ }
29
+ getName() {
30
+ return config.name;
31
+ }
32
+ getDescription() {
33
+ return config.description;
34
+ }
35
+ getProperties() {
36
+ return config.properties;
37
+ }
38
+ getProperty(name) {
39
+ return config.properties.find(p => p.name === name);
40
+ }
41
+ getBikeInitRequest() {
42
+ return {};
43
+ }
44
+ sendBikeUpdate(request) {
45
+ this.logger.logEvent({ message: 'bike update request', request });
46
+ const r = request || { refresh: true };
47
+ if (r.refresh) {
48
+ if (Object.keys(r).length === 1)
49
+ return this.prevRequest || {};
50
+ delete r.refresh;
51
+ }
52
+ if (request.slope !== undefined) {
53
+ if (!this.data)
54
+ this.data = {};
55
+ this.data.slope = request.slope;
56
+ }
57
+ this.prevRequest = request ? JSON.parse(JSON.stringify(request)) : {};
58
+ return r;
59
+ }
60
+ updateData(bikeData) {
61
+ const prevData = JSON.parse(JSON.stringify(this.data || {}));
62
+ const prevSpeed = prevData.speed;
63
+ const prevRequest = this.prevRequest || {};
64
+ const data = this.data || {};
65
+ const mode = this.getSetting('mode');
66
+ delete this.event.gearUpdated;
67
+ delete this.event.rpmUpdated;
68
+ try {
69
+ let rpm = 90;
70
+ let power = (mode === 'Power' || !mode) ? Number(this.getSetting('power')) : bikeData.power || 0;
71
+ let slope = (prevData.slope !== undefined ? prevData.slope : prevRequest.slope || 0);
72
+ let speed = mode === 'Speed' ? Number(this.getSetting('speed')) : bikeData.speed || 0;
73
+ let m = 75;
74
+ let distanceInternal = prevData.distanceInternal || 0;
75
+ let ts = Date.now();
76
+ let duration = this.prevUpdateTS === 0 ? 0 : ((ts - this.prevUpdateTS) / 1000);
77
+ if (mode === 'Power' || !mode) {
78
+ speed = calculations_1.default.calculateSpeed(m, power, slope, { bikeType: 'race' });
79
+ }
80
+ else if (mode === 'Speed') {
81
+ power = calculations_1.default.calculatePower(m, speed / 3.6, slope, { bikeType: 'race' });
82
+ }
83
+ if (prevRequest.targetPower) {
84
+ power = prevRequest.targetPower;
85
+ speed = calculations_1.default.calculateSpeed(m, power, slope, { bikeType: 'race' });
86
+ }
87
+ if (prevRequest.maxPower && power > prevRequest.maxPower) {
88
+ power = prevRequest.maxPower;
89
+ speed = calculations_1.default.calculateSpeed(m, power, slope, { bikeType: 'race' });
90
+ }
91
+ else if (prevRequest.minPower && power < prevRequest.minPower) {
92
+ power = prevRequest.minPower;
93
+ speed = calculations_1.default.calculateSpeed(m, power, slope, { bikeType: 'race' });
94
+ }
95
+ let v = speed / 3.6;
96
+ distanceInternal += Math.round(v * duration);
97
+ data.speed = parseFloat(speed.toFixed(1));
98
+ data.power = Math.round(power);
99
+ data.distanceInternal = distanceInternal;
100
+ data.slope = slope;
101
+ data.pedalRpm = rpm;
102
+ if (data.time !== undefined)
103
+ data.time += duration;
104
+ else
105
+ data.time = 0;
106
+ data.heartrate = bikeData.heartrate;
107
+ data.isPedalling = true;
108
+ this.prevUpdateTS = ts;
109
+ }
110
+ catch (err) {
111
+ this.logger.logEvent({ message: 'error', fn: 'updateData()', error: err.message || err });
112
+ }
113
+ this.logger.logEvent({ message: "updateData result", data, bikeData, prevRequest, prevSpeed });
114
+ this.data = data;
115
+ return data;
116
+ }
117
+ }
118
+ exports.default = SimulatorCyclingMode;
@@ -23,7 +23,7 @@ const DeviceProtocol_1 = __importStar(require("../DeviceProtocol"));
23
23
  const DeviceRegistry_1 = __importDefault(require("../DeviceRegistry"));
24
24
  const Device_1 = __importDefault(require("../Device"));
25
25
  const gd_eventlog_1 = require("gd-eventlog");
26
- const simulator_mode_1 = __importDefault(require("./simulator-mode"));
26
+ const simulator_1 = __importDefault(require("../modes/simulator"));
27
27
  const DEFAULT_SETTINGS = { name: 'Simulator', port: '', isBot: false };
28
28
  class Simulator extends Device_1.default {
29
29
  constructor(protocol, props = DEFAULT_SETTINGS) {
@@ -59,11 +59,11 @@ class Simulator extends Device_1.default {
59
59
  }
60
60
  getSupportedCyclingModes() {
61
61
  const supported = [];
62
- supported.push(simulator_mode_1.default);
62
+ supported.push(simulator_1.default);
63
63
  return supported;
64
64
  }
65
65
  getDefaultCyclingMode() {
66
- return new simulator_mode_1.default(this);
66
+ return new simulator_1.default(this);
67
67
  }
68
68
  getCyclingMode() {
69
69
  if (!this.cyclingMode)
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "incyclist-devices",
3
- "version": "1.4.24",
3
+ "version": "1.4.27",
4
4
  "dependencies": {
5
5
  "@serialport/parser-byte-length": "^9.0.1",
6
6
  "@serialport/parser-delimiter": "^9.0.1",
@@ -22,6 +22,7 @@
22
22
  "typescript": "^4.5.2"
23
23
  },
24
24
  "scripts": {
25
+ "lint": "eslint . --ext .ts",
25
26
  "build": "tsc",
26
27
  "test": "jest --coverage",
27
28
  "dev": "nodemon --watch src -e ts,js --exec npm run build"