kahu-signalk 0.0.9 → 0.0.10

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 (2) hide show
  1. package/package.json +1 -1
  2. package/plugin/index.js +60 -19
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "kahu-signalk",
3
- "version": "0.0.9",
3
+ "version": "0.0.10",
4
4
  "description": "Contribute AIS and ARPA targets from your vessel to crowdsourcing for marine safety!",
5
5
  "keywords": [
6
6
  "signalk-node-server-plugin",
package/plugin/index.js CHANGED
@@ -10,7 +10,50 @@ while (packageDir !== path.dirname(packageDir)
10
10
  packageDir = path.dirname(packageDir);
11
11
  }
12
12
 
13
- const nmeaRattmRegex = /\$RATTM,(\d{2}),([\d\.\-]+),([\d\.\-]+),([^,]*),([\d\.\-]+),([\d\.\-]+),([^,]*),([^,]*),([^,]*),([^,]*),([^,]*),([^,]*),(..*)\*([A-Fa-f0-9]{2})\s*/;
13
+ const parseOptionalFloat = (value) => {
14
+ if (value === undefined || value === null || value === '') return undefined;
15
+ const parsed = parseFloat(value);
16
+ return Number.isNaN(parsed) ? undefined : parsed;
17
+ };
18
+
19
+ const parseRattmSentence = (sentence) => {
20
+ const [payload] = sentence.split('*');
21
+ if (!payload) return null;
22
+
23
+ const parts = payload.trim().split(',');
24
+ if (parts[0] !== '$RATTM') return null;
25
+ if (parts.length < 16) return null;
26
+
27
+ const target_id = parseInt(parts[1], 10);
28
+ const target_distance = parseOptionalFloat(parts[2]);
29
+ const target_bearing = parseOptionalFloat(parts[3]);
30
+ const target_bearing_unit = parts[4] || '';
31
+ const target_speed = parseOptionalFloat(parts[5]);
32
+ const target_course = parseOptionalFloat(parts[6]);
33
+ const target_course_unit = parts[7] || '';
34
+ const target_distance_unit = parts[10] || '';
35
+ const target_name = parts[12] || '';
36
+ const target_status = parts[13] || '';
37
+
38
+ if (!Number.isFinite(target_id) ||
39
+ !Number.isFinite(target_distance) ||
40
+ !Number.isFinite(target_bearing)) {
41
+ return null;
42
+ }
43
+
44
+ return {
45
+ target_id,
46
+ target_distance,
47
+ target_bearing,
48
+ target_bearing_unit,
49
+ target_speed,
50
+ target_course,
51
+ target_course_unit,
52
+ target_distance_unit,
53
+ target_name,
54
+ target_status,
55
+ };
56
+ };
14
57
 
15
58
  const deg2rad = (degrees) => degrees * (Math.PI / 180);
16
59
 
@@ -48,21 +91,24 @@ module.exports = (app) => {
48
91
  parser: ({ id, sentence, parts, tags }, session) => {
49
92
  if (sentence.startsWith("$RATTL")) {
50
93
  } else if (sentence.startsWith("$RATTM")) {
51
- const match = nmeaRattmRegex.exec(sentence);
52
-
53
- if (!match) {
94
+ const parsed = parseRattmSentence(sentence);
95
+ if (!parsed) {
54
96
  console.error("Failed to parse RATTM NMEA sentence: [", sentence, "]");
55
97
  return;
56
98
  }
57
- if (match.length - 1 != 14) {
58
- console.log("Only parsed ", (match.length - 1), " fields of RATTM NMEA sentence: [", sentence, "]");
59
- return;
60
- }
61
-
62
- const target_id = parseInt(match[1]);
63
- const target_distance = parseFloat(match[2]);
64
- const target_bearing = parseFloat(match[3]);
65
- const target_bearing_unit = match[4];
99
+
100
+ const {
101
+ target_id,
102
+ target_distance,
103
+ target_bearing,
104
+ target_bearing_unit,
105
+ target_speed,
106
+ target_course,
107
+ target_course_unit,
108
+ target_distance_unit,
109
+ target_name,
110
+ target_status,
111
+ } = parsed;
66
112
 
67
113
  if (target_bearing_unit === 'R') {
68
114
  console.warn("Relative bearings not yet supported, skipping RATTM sentence");
@@ -81,16 +127,11 @@ module.exports = (app) => {
81
127
  distance: target_distance,
82
128
  bearing: target_bearing,
83
129
  bearing_unit: target_bearing_unit,
84
- distance_unit: match[10],
130
+ distance_unit: target_distance_unit,
85
131
  }
86
132
 
87
- const target_speed = parseFloat(match[5]);
88
- const target_course = parseFloat(match[6]);
89
- const target_course_unit = match[7];
90
133
  // target_distance_closes_point_of_approac: parseInt(match[8]),
91
134
  // target_time_closes_point_of_approac: parseInt(match[9]),
92
- const target_name = match[11];
93
- const target_status = match[12];
94
135
 
95
136
  const now = new Date();
96
137
  if (now - plugin.route_updates[target_id] > 60000) {