incyclist-devices 1.4.23 → 1.4.26

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 (50) hide show
  1. package/lib/CyclingMode.js +1 -0
  2. package/lib/Device.js +1 -0
  3. package/lib/DeviceProtocol.d.ts +1 -0
  4. package/lib/DeviceProtocol.js +2 -0
  5. package/lib/DeviceSupport.js +21 -8
  6. package/lib/ant/AntAdapter.js +1 -0
  7. package/lib/ant/AntScanner.js +30 -13
  8. package/lib/ant/antfe/AntFEAdapter.js +7 -26
  9. package/lib/ant/anthrm/AntHrmAdapter.js +1 -12
  10. package/lib/ant/antpwr/AntPWRAdapter.d.ts +24 -0
  11. package/lib/ant/antpwr/AntPWRAdapter.js +252 -0
  12. package/lib/ant/antpwr/pwr-adapter.d.ts +48 -0
  13. package/lib/ant/antpwr/pwr-adapter.js +248 -0
  14. package/lib/ant/utils.js +3 -1
  15. package/lib/calculations.js +3 -3
  16. package/lib/daum/DaumAdapter.d.ts +2 -2
  17. package/lib/daum/DaumAdapter.js +43 -27
  18. package/lib/daum/DaumPowerMeterCyclingMode.d.ts +8 -0
  19. package/lib/daum/DaumPowerMeterCyclingMode.js +21 -0
  20. package/lib/daum/ERGCyclingMode.d.ts +3 -6
  21. package/lib/daum/ERGCyclingMode.js +15 -27
  22. package/lib/daum/SmartTrainerCyclingMode.js +1 -0
  23. package/lib/daum/classic/DaumClassicAdapter.js +2 -2
  24. package/lib/daum/classic/DaumClassicProtocol.js +19 -7
  25. package/lib/daum/classic/bike.js +26 -26
  26. package/lib/daum/classic/utils.js +1 -0
  27. package/lib/daum/constants.js +1 -0
  28. package/lib/daum/premium/DaumClassicCyclingMode.d.ts +2 -2
  29. package/lib/daum/premium/DaumClassicCyclingMode.js +2 -2
  30. package/lib/daum/premium/DaumPremiumAdapter.js +2 -2
  31. package/lib/daum/premium/DaumPremiumProtocol.js +19 -7
  32. package/lib/daum/premium/bike.js +18 -17
  33. package/lib/daum/premium/tcpserial.js +0 -1
  34. package/lib/daum/premium/utils.js +1 -0
  35. package/lib/kettler/comms.js +2 -1
  36. package/lib/kettler/ergo-racer/adapter.js +22 -10
  37. package/lib/kettler/ergo-racer/modes/power-meter.d.ts +2 -2
  38. package/lib/kettler/ergo-racer/modes/power-meter.js +12 -4
  39. package/lib/kettler/ergo-racer/protocol.js +19 -7
  40. package/lib/modes/power-base.d.ts +20 -0
  41. package/lib/modes/power-base.js +70 -0
  42. package/lib/modes/power-meter.d.ts +20 -0
  43. package/lib/modes/power-meter.js +71 -0
  44. package/lib/modes/simulator.d.ts +27 -0
  45. package/lib/modes/simulator.js +118 -0
  46. package/lib/simulator/Simulator.js +23 -10
  47. package/lib/types/route.js +1 -0
  48. package/lib/types/user.js +1 -0
  49. package/lib/utils.js +3 -1
  50. package/package.json +2 -1
@@ -0,0 +1,20 @@
1
+ import { IncyclistBikeData, Settings, CyclingModeBase } from '../CyclingMode';
2
+ import { DeviceAdapter } from '../Device';
3
+ import { EventLogger } from 'gd-eventlog';
4
+ export default class PowerBasedCyclingModeBase extends CyclingModeBase {
5
+ data: IncyclistBikeData;
6
+ prevUpdateTS: number;
7
+ logger: EventLogger;
8
+ constructor(adapter: DeviceAdapter, props?: Settings);
9
+ initLogger(defaultLogName: any): void;
10
+ getWeight(): any;
11
+ getTimeSinceLastUpdate(): number;
12
+ calculateSpeedAndDistance(power: number, slope: number, m: number, t: number, props?: {}): {
13
+ speed: number;
14
+ distance: number;
15
+ };
16
+ calculatePowerAndDistance(speed: number, slope: number, m: number, t: number, props?: {}): {
17
+ power: number;
18
+ distance: number;
19
+ };
20
+ }
@@ -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;
@@ -0,0 +1,20 @@
1
+ import CyclingMode, { CyclingModeProperty, IncyclistBikeData, Settings, UpdateRequest } from '../CyclingMode';
2
+ import PowerBasedCyclingModeBase from './power-base';
3
+ import { DeviceAdapter } from '../Device';
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
+ constructor(adapter: DeviceAdapter, props?: Settings);
11
+ getName(): string;
12
+ getDescription(): string;
13
+ getProperties(): CyclingModeProperty[];
14
+ getProperty(name: string): CyclingModeProperty;
15
+ getBikeInitRequest(): UpdateRequest;
16
+ sendBikeUpdate(request?: UpdateRequest): UpdateRequest;
17
+ updateData(bikeData: IncyclistBikeData, props?: {
18
+ log: boolean;
19
+ }): IncyclistBikeData;
20
+ }
@@ -0,0 +1,71 @@
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
+ exports.config = void 0;
7
+ const power_base_1 = __importDefault(require("./power-base"));
8
+ exports.config = {
9
+ name: 'PowerMeter',
10
+ 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',
11
+ properties: []
12
+ };
13
+ class PowerMeterCyclingMode extends power_base_1.default {
14
+ constructor(adapter, props) {
15
+ super(adapter, props);
16
+ this.initLogger('PowerMeterMode');
17
+ this.data = { speed: 0, slope: 0, power: 0, distanceInternal: 0, pedalRpm: 0, isPedalling: false, heartrate: 0 };
18
+ }
19
+ getName() {
20
+ return exports.config.name;
21
+ }
22
+ getDescription() {
23
+ return exports.config.description;
24
+ }
25
+ getProperties() {
26
+ return exports.config.properties;
27
+ }
28
+ getProperty(name) {
29
+ return exports.config.properties.find(p => p.name === name);
30
+ }
31
+ getBikeInitRequest() {
32
+ return {};
33
+ }
34
+ sendBikeUpdate(request = {}) {
35
+ const prevData = this.data || {};
36
+ const prevSlope = prevData.slope || 0;
37
+ if (request.slope && request.slope !== prevSlope) {
38
+ this.data.slope = request.slope;
39
+ this.updateData(this.data, { log: false });
40
+ }
41
+ this.logger.logEvent({ message: "processing update request", request });
42
+ return {};
43
+ }
44
+ updateData(bikeData, props = { log: true }) {
45
+ try {
46
+ const prevData = this.data || {};
47
+ const data = JSON.parse(JSON.stringify(bikeData));
48
+ let power = bikeData.power || 0;
49
+ const slope = prevData.slope || 0;
50
+ const distanceInternal = prevData.distanceInternal || 0;
51
+ if (!bikeData.pedalRpm || bikeData.isPedalling === false) {
52
+ power = 0;
53
+ }
54
+ const m = this.getWeight();
55
+ let t = this.getTimeSinceLastUpdate();
56
+ const { speed, distance } = this.calculateSpeedAndDistance(power, slope, m, t);
57
+ data.speed = speed;
58
+ data.power = Math.round(power);
59
+ data.distanceInternal = Math.round(distanceInternal + distance);
60
+ data.slope = slope;
61
+ if (props.log)
62
+ this.logger.logEvent({ message: "updateData result", data, bikeData, prevSpeed: prevData.speed });
63
+ this.data = data;
64
+ }
65
+ catch (err) {
66
+ this.logger.logEvent({ message: 'error', fn: 'updateData()', error: err.message || err });
67
+ }
68
+ return this.data;
69
+ }
70
+ }
71
+ 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;
@@ -1,4 +1,23 @@
1
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
+ };
2
21
  var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
3
22
  function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
4
23
  return new (P || (P = Promise))(function (resolve, reject) {
@@ -8,22 +27,16 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
8
27
  step((generator = generator.apply(thisArg, _arguments || [])).next());
9
28
  });
10
29
  };
11
- var __importStar = (this && this.__importStar) || function (mod) {
12
- if (mod && mod.__esModule) return mod;
13
- var result = {};
14
- if (mod != null) for (var k in mod) if (Object.hasOwnProperty.call(mod, k)) result[k] = mod[k];
15
- result["default"] = mod;
16
- return result;
17
- };
18
30
  var __importDefault = (this && this.__importDefault) || function (mod) {
19
31
  return (mod && mod.__esModule) ? mod : { "default": mod };
20
32
  };
21
33
  Object.defineProperty(exports, "__esModule", { value: true });
34
+ exports.Simulator = void 0;
22
35
  const DeviceProtocol_1 = __importStar(require("../DeviceProtocol"));
23
36
  const DeviceRegistry_1 = __importDefault(require("../DeviceRegistry"));
24
37
  const Device_1 = __importDefault(require("../Device"));
25
38
  const gd_eventlog_1 = require("gd-eventlog");
26
- const simulator_mode_1 = __importDefault(require("./simulator-mode"));
39
+ const simulator_1 = __importDefault(require("../modes/simulator"));
27
40
  const DEFAULT_SETTINGS = { name: 'Simulator', port: '', isBot: false };
28
41
  class Simulator extends Device_1.default {
29
42
  constructor(protocol, props = DEFAULT_SETTINGS) {
@@ -59,11 +72,11 @@ class Simulator extends Device_1.default {
59
72
  }
60
73
  getSupportedCyclingModes() {
61
74
  const supported = [];
62
- supported.push(simulator_mode_1.default);
75
+ supported.push(simulator_1.default);
63
76
  return supported;
64
77
  }
65
78
  getDefaultCyclingMode() {
66
- return new simulator_mode_1.default(this);
79
+ return new simulator_1.default(this);
67
80
  }
68
81
  getCyclingMode() {
69
82
  if (!this.cyclingMode)
@@ -1,5 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.RouteType = void 0;
3
4
  var RouteType;
4
5
  (function (RouteType) {
5
6
  RouteType["FREE_RIDE"] = "free ride";
package/lib/types/user.js CHANGED
@@ -1,5 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Gender = void 0;
3
4
  var Gender;
4
5
  (function (Gender) {
5
6
  Gender["MALE"] = "M";
package/lib/utils.js CHANGED
@@ -9,9 +9,11 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
9
9
  });
10
10
  };
11
11
  Object.defineProperty(exports, "__esModule", { value: true });
12
- exports.sleep = (ms) => {
12
+ exports.Queue = exports.hexstr = exports.intVal = exports.floatVal = exports.runWithRetries = exports.sleep = void 0;
13
+ const sleep = (ms) => {
13
14
  return new Promise(resolve => setTimeout(resolve, ms));
14
15
  };
16
+ exports.sleep = sleep;
15
17
  function runWithRetries(fn, maxRetries, timeBetween) {
16
18
  return new Promise((resolve, reject) => {
17
19
  let retries = 0;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "incyclist-devices",
3
- "version": "1.4.23",
3
+ "version": "1.4.26",
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"