@hello.nrfcloud.com/proto-map 5.6.2 → 6.0.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/README.md +0 -16
- package/api/Context.ts +1 -1
- package/api/{ResourceUpdate.spec.ts → ObjectUpdate.spec.ts} +5 -5
- package/api/{ResourceUpdate.ts → ObjectUpdate.ts} +4 -4
- package/dist/api/Context.js +1 -1
- package/dist/api/DeviceId.js +7 -7
- package/dist/api/Devices.js +11 -11
- package/dist/api/History.js +10 -10
- package/dist/api/History.spec.js +17 -17
- package/dist/api/LwM2M.js +11 -11
- package/dist/api/ObjectUpdate.js +17 -0
- package/dist/api/ObjectUpdate.spec.js +26 -0
- package/dist/api/ShareDeviceOwnershipConfirmed.js +4 -4
- package/dist/api/ShareDeviceRequest.js +4 -4
- package/dist/api/Timestamp.js +3 -3
- package/dist/api/Timestamp.spec.js +9 -9
- package/dist/api/index.js +8 -8
- package/dist/generator/addDocBlock.js +2 -2
- package/dist/generator/generateLwM2MDefinitions.js +27 -27
- package/dist/generator/generateLwM2MDefinitions.spec.js +32 -32
- package/dist/generator/generateLwm2mTimestampResources.js +7 -7
- package/dist/generator/generateModels.js +26 -111
- package/dist/generator/generateType.js +21 -21
- package/dist/generator/generateValidator.js +29 -29
- package/dist/generator/generateValidators.js +9 -9
- package/dist/generator/lwm2m.js +23 -23
- package/dist/generator/models.js +14 -96
- package/dist/generator/printNode.js +2 -2
- package/dist/generator/tokenizeName.js +2 -2
- package/dist/generator/tokenizeName.spec.js +20 -20
- package/dist/generator/types.js +37 -37
- package/dist/lwm2m/LWM2MObjectDefinition.js +30 -30
- package/dist/lwm2m/check-lwm2m-rules.js +34 -34
- package/dist/lwm2m/definitions.js +4 -4
- package/dist/lwm2m/fromXML2JSON.js +4 -4
- package/dist/lwm2m/instanceTs.js +2 -2
- package/dist/lwm2m/instanceTs.spec.js +10 -10
- package/dist/lwm2m/isRegisteredLwM2MObject.js +2 -2
- package/dist/lwm2m/isRegisteredLwM2MObject.spec.js +12 -12
- package/dist/lwm2m/parseRangeEnumeration.js +1 -1
- package/dist/lwm2m/parseRangeEnumeration.spec.js +11 -11
- package/dist/lwm2m/unwrapNestedArray.js +2 -2
- package/dist/lwm2m/unwrapNestedArray.spec.js +155 -155
- package/dist/lwm2m/validate.js +1 -1
- package/dist/lwm2m/validation.js +16 -16
- package/dist/markdown/getCodeBlock.js +3 -29
- package/dist/markdown/parseREADME.js +5 -5
- package/dist/models/check-model-rules.js +16 -106
- package/dist/models/models.js +6 -115
- package/dist/models/types.js +0 -12
- package/dist/senml/SenMLSchema.js +16 -16
- package/dist/senml/SenMLSchema.spec.js +11 -11
- package/dist/senml/hasValue.js +2 -2
- package/dist/senml/hasValue.spec.js +4 -4
- package/dist/senml/lwm2mToSenML.js +8 -8
- package/dist/senml/lwm2mToSenML.spec.js +42 -42
- package/dist/senml/parseResourceId.js +1 -1
- package/dist/senml/parseResourceId.spec.js +6 -6
- package/dist/senml/senMLtoLwM2M.js +14 -14
- package/dist/senml/senMLtoLwM2M.spec.js +53 -53
- package/dist/senml/validateSenML.js +2 -2
- package/dist/senml/validateSenML.spec.js +11 -11
- package/models/README.md +0 -4
- package/models/check-model-rules.ts +1 -90
- package/models/models.ts +1 -8
- package/models/types.ts +0 -16
- package/package.json +5 -10
- package/dist/api/ResourceUpdate.js +0 -17
- package/dist/api/ResourceUpdate.spec.js +0 -26
- package/dist/generator/isDir.js +0 -163
- package/dist/generator/isDir.spec.js +0 -212
- package/dist/markdown/getFrontMatter.js +0 -15
- package/dist/models/asset_tracker_v2+AWS/examples/examples.spec.js +0 -489
- package/models/PCA20035+solar/README.md +0 -10
- package/models/PCA20035+solar/transforms/airQuality.md +0 -48
- package/models/PCA20035+solar/transforms/battery.md +0 -46
- package/models/PCA20035+solar/transforms/button.md +0 -45
- package/models/PCA20035+solar/transforms/deviceInfo.md +0 -72
- package/models/PCA20035+solar/transforms/gain.md +0 -45
- package/models/PCA20035+solar/transforms/geolocationFromGroundfix.md +0 -67
- package/models/PCA20035+solar/transforms/geolocationFromMessage.md +0 -80
- package/models/PCA20035+solar/transforms/humidity.md +0 -43
- package/models/PCA20035+solar/transforms/networkInfo.md +0 -84
- package/models/PCA20035+solar/transforms/pressure.md +0 -43
- package/models/PCA20035+solar/transforms/temperature.md +0 -43
- package/models/asset_tracker_v2+AWS/README.md +0 -6
- package/models/asset_tracker_v2+AWS/examples/examples.spec.ts +0 -229
- package/models/asset_tracker_v2+AWS/examples/shadow/example-1.json +0 -24
- package/models/asset_tracker_v2+AWS/examples/shadow/example-2.json +0 -30
- package/models/asset_tracker_v2+AWS/examples/shadow/example-3.json +0 -37
- package/models/asset_tracker_v2+AWS/examples/shadow/example-4.json +0 -48
- package/models/asset_tracker_v2+AWS/examples/shadow/example-5.json +0 -43
- package/models/asset_tracker_v2+AWS/transforms/GNSS.md +0 -66
- package/models/asset_tracker_v2+AWS/transforms/battery-voltage.md +0 -50
- package/models/asset_tracker_v2+AWS/transforms/device-info.md +0 -61
- package/models/asset_tracker_v2+AWS/transforms/env.md +0 -69
- package/models/asset_tracker_v2+AWS/transforms/fuel-gauge.md +0 -62
- package/models/asset_tracker_v2+AWS/transforms/roam.md +0 -100
- package/models/asset_tracker_v2+AWS/transforms/solar.md +0 -58
|
@@ -1,46 +1,46 @@
|
|
|
1
|
-
import { it, describe } from
|
|
2
|
-
import { LwM2MObjectID } from
|
|
3
|
-
import assert from
|
|
4
|
-
import { lwm2mToSenML } from
|
|
5
|
-
void describe(
|
|
6
|
-
void it(
|
|
1
|
+
import { it, describe } from 'node:test';
|
|
2
|
+
import { LwM2MObjectID } from '../lwm2m/LwM2MObjectID.js';
|
|
3
|
+
import assert from 'node:assert/strict';
|
|
4
|
+
import { lwm2mToSenML } from './lwm2mToSenML.js';
|
|
5
|
+
void describe('lwm2mToSenML()', function() {
|
|
6
|
+
void it('should convert LwM2M to SenML', function() {
|
|
7
7
|
var location = {
|
|
8
8
|
ObjectID: LwM2MObjectID.Geolocation_14201,
|
|
9
|
-
ObjectVersion:
|
|
9
|
+
ObjectVersion: '1.0',
|
|
10
10
|
Resources: {
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
11
|
+
'0': 62.469414,
|
|
12
|
+
'1': 6.151946,
|
|
13
|
+
'6': 'Fixed',
|
|
14
|
+
'3': 1,
|
|
15
|
+
'99': new Date(1710147413003)
|
|
16
16
|
}
|
|
17
17
|
};
|
|
18
18
|
var level1 = {
|
|
19
19
|
ObjectID: LwM2MObjectID.SeaWaterLevel_14230,
|
|
20
|
-
ObjectVersion:
|
|
20
|
+
ObjectVersion: '1.0',
|
|
21
21
|
Resources: {
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
22
|
+
'0': 84.3,
|
|
23
|
+
'1': 'AES',
|
|
24
|
+
'99': new Date(1710140400000)
|
|
25
25
|
}
|
|
26
26
|
};
|
|
27
27
|
var level2 = {
|
|
28
28
|
ObjectID: LwM2MObjectID.SeaWaterLevel_14230,
|
|
29
|
-
ObjectVersion:
|
|
29
|
+
ObjectVersion: '1.0',
|
|
30
30
|
Resources: {
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
31
|
+
'0': 140.4,
|
|
32
|
+
'1': 'AES',
|
|
33
|
+
'99': new Date(1710144000000)
|
|
34
34
|
}
|
|
35
35
|
};
|
|
36
36
|
var level3 = {
|
|
37
37
|
ObjectID: LwM2MObjectID.SeaWaterLevel_14230,
|
|
38
|
-
ObjectVersion:
|
|
38
|
+
ObjectVersion: '1.0',
|
|
39
39
|
ObjectInstanceID: 1,
|
|
40
40
|
Resources: {
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
41
|
+
'0': 140.7,
|
|
42
|
+
'1': 'AES',
|
|
43
|
+
'99': new Date(1710144001000)
|
|
44
44
|
}
|
|
45
45
|
};
|
|
46
46
|
var lwm2m = [
|
|
@@ -51,52 +51,52 @@ void describe("lwm2mToSenML()", function() {
|
|
|
51
51
|
];
|
|
52
52
|
var expected = [
|
|
53
53
|
{
|
|
54
|
-
bn:
|
|
55
|
-
n:
|
|
54
|
+
bn: '14201/0/',
|
|
55
|
+
n: '0',
|
|
56
56
|
v: 62.469414,
|
|
57
57
|
bt: 1710147413003
|
|
58
58
|
},
|
|
59
59
|
{
|
|
60
|
-
n:
|
|
60
|
+
n: '1',
|
|
61
61
|
v: 6.151946
|
|
62
62
|
},
|
|
63
63
|
{
|
|
64
|
-
n:
|
|
64
|
+
n: '3',
|
|
65
65
|
v: 1
|
|
66
66
|
},
|
|
67
67
|
{
|
|
68
|
-
n:
|
|
69
|
-
vs:
|
|
68
|
+
n: '6',
|
|
69
|
+
vs: 'Fixed'
|
|
70
70
|
},
|
|
71
71
|
{
|
|
72
|
-
bn:
|
|
73
|
-
n:
|
|
72
|
+
bn: '14230/0/',
|
|
73
|
+
n: '0',
|
|
74
74
|
v: 84.3,
|
|
75
75
|
bt: 1710140400000
|
|
76
76
|
},
|
|
77
77
|
{
|
|
78
|
-
n:
|
|
79
|
-
vs:
|
|
78
|
+
n: '1',
|
|
79
|
+
vs: 'AES'
|
|
80
80
|
},
|
|
81
81
|
{
|
|
82
|
-
bn:
|
|
83
|
-
n:
|
|
82
|
+
bn: '14230/0/',
|
|
83
|
+
n: '0',
|
|
84
84
|
v: 140.4,
|
|
85
85
|
bt: 1710144000000
|
|
86
86
|
},
|
|
87
87
|
{
|
|
88
|
-
n:
|
|
89
|
-
vs:
|
|
88
|
+
n: '1',
|
|
89
|
+
vs: 'AES'
|
|
90
90
|
},
|
|
91
91
|
{
|
|
92
|
-
bn:
|
|
93
|
-
n:
|
|
92
|
+
bn: '14230/1/',
|
|
93
|
+
n: '0',
|
|
94
94
|
v: 140.7,
|
|
95
95
|
bt: 1710144001000
|
|
96
96
|
},
|
|
97
97
|
{
|
|
98
|
-
n:
|
|
99
|
-
vs:
|
|
98
|
+
n: '1',
|
|
99
|
+
vs: 'AES'
|
|
100
100
|
}
|
|
101
101
|
];
|
|
102
102
|
assert.deepEqual(lwm2mToSenML(lwm2m), expected);
|
|
@@ -46,7 +46,7 @@ function _unsupported_iterable_to_array(o, minLen) {
|
|
|
46
46
|
}
|
|
47
47
|
export var parseResourceId = function(resourceId) {
|
|
48
48
|
if (!/^\d+\/\d+\/\d+\/\d+$/.test(resourceId)) return null;
|
|
49
|
-
var _resourceId_split_map = _sliced_to_array(resourceId.split(
|
|
49
|
+
var _resourceId_split_map = _sliced_to_array(resourceId.split('/').map(function(s) {
|
|
50
50
|
return parseInt(s, 10);
|
|
51
51
|
}), 4), ObjectID = _resourceId_split_map[0], ObjectInstanceID = _resourceId_split_map[1], ResourceID = _resourceId_split_map[2], ResourceInstanceId = _resourceId_split_map[3];
|
|
52
52
|
return {
|
|
@@ -1,9 +1,9 @@
|
|
|
1
|
-
import assert from
|
|
2
|
-
import { describe, it } from
|
|
3
|
-
import { parseResourceId } from
|
|
4
|
-
void describe(
|
|
5
|
-
void it(
|
|
6
|
-
return assert.deepEqual(parseResourceId(
|
|
1
|
+
import assert from 'node:assert';
|
|
2
|
+
import { describe, it } from 'node:test';
|
|
3
|
+
import { parseResourceId } from './parseResourceId.js';
|
|
4
|
+
void describe('parseResourceId()', function() {
|
|
5
|
+
void it('should parse an LwM2M resource ID', function() {
|
|
6
|
+
return assert.deepEqual(parseResourceId('14201/1/2/0'), {
|
|
7
7
|
ObjectID: 14201,
|
|
8
8
|
ObjectInstanceID: 1,
|
|
9
9
|
ResourceID: 2,
|
|
@@ -50,9 +50,9 @@ function _object_spread_props(target, source) {
|
|
|
50
50
|
}
|
|
51
51
|
return target;
|
|
52
52
|
}
|
|
53
|
-
import { timestampResources } from
|
|
54
|
-
import { parseResourceId } from
|
|
55
|
-
import { hasValue } from
|
|
53
|
+
import { timestampResources } from '../lwm2m/timestampResources.js';
|
|
54
|
+
import { parseResourceId } from './parseResourceId.js';
|
|
55
|
+
import { hasValue } from './hasValue.js';
|
|
56
56
|
var isInfoForDifferentInstance = function(currentObject, resourceID, currentBaseTime) {
|
|
57
57
|
var _currentObject_Resources_tsRes, _currentObject_Resources;
|
|
58
58
|
if (currentObject === undefined) return true;
|
|
@@ -65,16 +65,16 @@ var isInfoForDifferentInstance = function(currentObject, resourceID, currentBase
|
|
|
65
65
|
return false;
|
|
66
66
|
};
|
|
67
67
|
var getValue = function(measurement) {
|
|
68
|
-
if (
|
|
69
|
-
if (
|
|
70
|
-
if (
|
|
71
|
-
if (
|
|
72
|
-
if (
|
|
73
|
-
if (
|
|
68
|
+
if ('bv' in measurement && !('v' in measurement)) return measurement.bv;
|
|
69
|
+
if ('bv' in measurement && 'v' in measurement) return measurement.bv + measurement.v;
|
|
70
|
+
if ('v' in measurement) return measurement.v;
|
|
71
|
+
if ('vs' in measurement) return measurement.vs;
|
|
72
|
+
if ('vb' in measurement) return measurement.vb;
|
|
73
|
+
if ('vd' in measurement) return measurement.vd;
|
|
74
74
|
return undefined;
|
|
75
75
|
};
|
|
76
76
|
export var senMLtoLwM2M = function(senML) {
|
|
77
|
-
var currentBaseName =
|
|
77
|
+
var currentBaseName = '';
|
|
78
78
|
var currentBaseTime = undefined;
|
|
79
79
|
var currentObject = undefined;
|
|
80
80
|
var items = [];
|
|
@@ -82,11 +82,11 @@ export var senMLtoLwM2M = function(senML) {
|
|
|
82
82
|
try {
|
|
83
83
|
for(var _iterator = senML[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true){
|
|
84
84
|
var item = _step.value;
|
|
85
|
-
if (
|
|
86
|
-
if (
|
|
85
|
+
if ('bn' in item && item.bn !== undefined) currentBaseName = item.bn;
|
|
86
|
+
if ('bt' in item && item.bt !== undefined) currentBaseTime = item.bt;
|
|
87
87
|
if (!hasValue(item)) continue;
|
|
88
88
|
var _item_n;
|
|
89
|
-
var itemResourceId = "".concat(currentBaseName).concat((_item_n = item.n) !== null && _item_n !== void 0 ? _item_n :
|
|
89
|
+
var itemResourceId = "".concat(currentBaseName).concat((_item_n = item.n) !== null && _item_n !== void 0 ? _item_n : '', "/0");
|
|
90
90
|
var resourceId = parseResourceId(itemResourceId);
|
|
91
91
|
if (resourceId === null) throw new Error("Invalid resource ID: ".concat(itemResourceId));
|
|
92
92
|
if (currentObject === undefined || currentBaseTime !== undefined && isInfoForDifferentInstance(currentObject, resourceId, currentBaseTime)) {
|
|
@@ -99,7 +99,7 @@ export var senMLtoLwM2M = function(senML) {
|
|
|
99
99
|
Resources: _define_property({}, tsRes, new Date(currentBaseTime))
|
|
100
100
|
};
|
|
101
101
|
if (resourceId.ObjectInstanceID !== 0) currentObject.ObjectInstanceID = resourceId.ObjectInstanceID;
|
|
102
|
-
if (
|
|
102
|
+
if ('blv' in item) currentObject.ObjectVersion = item.blv;
|
|
103
103
|
}
|
|
104
104
|
if ((currentObject === null || currentObject === void 0 ? void 0 : currentObject.Resources) === undefined) continue;
|
|
105
105
|
var value = getValue(item);
|
|
@@ -1,34 +1,34 @@
|
|
|
1
|
-
import assert from
|
|
2
|
-
import { describe, it } from
|
|
3
|
-
import { senMLtoLwM2M } from
|
|
4
|
-
void describe(
|
|
5
|
-
void it(
|
|
1
|
+
import assert from 'node:assert/strict';
|
|
2
|
+
import { describe, it } from 'node:test';
|
|
3
|
+
import { senMLtoLwM2M } from './senMLtoLwM2M.js';
|
|
4
|
+
void describe('senMLtoLwM2M()', function() {
|
|
5
|
+
void it('should resolve a senML message into objects', function() {
|
|
6
6
|
var input = [
|
|
7
7
|
{
|
|
8
|
-
bn:
|
|
9
|
-
blv:
|
|
8
|
+
bn: '14201/1/',
|
|
9
|
+
blv: '1.1',
|
|
10
10
|
bt: 1698155694999,
|
|
11
|
-
n:
|
|
11
|
+
n: '0',
|
|
12
12
|
v: 33.98755678796222
|
|
13
13
|
},
|
|
14
14
|
{
|
|
15
|
-
n:
|
|
15
|
+
n: '1',
|
|
16
16
|
v: -84.506132079174634
|
|
17
17
|
},
|
|
18
18
|
{
|
|
19
|
-
n:
|
|
19
|
+
n: '2',
|
|
20
20
|
v: 295.468994140625
|
|
21
21
|
},
|
|
22
22
|
{
|
|
23
|
-
n:
|
|
23
|
+
n: '3',
|
|
24
24
|
v: 17.74077033996582
|
|
25
25
|
},
|
|
26
26
|
{
|
|
27
|
-
n:
|
|
27
|
+
n: '4',
|
|
28
28
|
v: 26.376304626464844
|
|
29
29
|
},
|
|
30
30
|
{
|
|
31
|
-
n:
|
|
31
|
+
n: '5',
|
|
32
32
|
v: 359.1545715332
|
|
33
33
|
}
|
|
34
34
|
];
|
|
@@ -36,7 +36,7 @@ void describe("senMLtoLwM2M()", function() {
|
|
|
36
36
|
{
|
|
37
37
|
ObjectID: 14201,
|
|
38
38
|
ObjectInstanceID: 1,
|
|
39
|
-
ObjectVersion:
|
|
39
|
+
ObjectVersion: '1.1',
|
|
40
40
|
Resources: {
|
|
41
41
|
0: 33.98755678796222,
|
|
42
42
|
1: -84.506132079174634,
|
|
@@ -50,96 +50,96 @@ void describe("senMLtoLwM2M()", function() {
|
|
|
50
50
|
];
|
|
51
51
|
assert.deepEqual(senMLtoLwM2M(input), expected);
|
|
52
52
|
});
|
|
53
|
-
void it(
|
|
53
|
+
void it('should drop empty resources', function() {
|
|
54
54
|
var input = [
|
|
55
55
|
{
|
|
56
|
-
bn:
|
|
57
|
-
n:
|
|
58
|
-
vs:
|
|
56
|
+
bn: '14203/0/',
|
|
57
|
+
n: '0',
|
|
58
|
+
vs: 'LTE-M',
|
|
59
59
|
bt: 1676369307222
|
|
60
60
|
},
|
|
61
61
|
{
|
|
62
|
-
n:
|
|
62
|
+
n: '1',
|
|
63
63
|
v: 20
|
|
64
64
|
},
|
|
65
65
|
{
|
|
66
|
-
n:
|
|
66
|
+
n: '2',
|
|
67
67
|
v: -79
|
|
68
68
|
},
|
|
69
69
|
{
|
|
70
|
-
n:
|
|
70
|
+
n: '3',
|
|
71
71
|
v: 6
|
|
72
72
|
},
|
|
73
73
|
{
|
|
74
|
-
n:
|
|
74
|
+
n: '4',
|
|
75
75
|
v: 56879116
|
|
76
76
|
},
|
|
77
77
|
{
|
|
78
|
-
n:
|
|
78
|
+
n: '5',
|
|
79
79
|
v: 24001
|
|
80
80
|
},
|
|
81
81
|
{
|
|
82
|
-
n:
|
|
83
|
-
vs:
|
|
82
|
+
n: '6',
|
|
83
|
+
vs: '10.160.243.113'
|
|
84
84
|
},
|
|
85
85
|
{
|
|
86
|
-
n:
|
|
86
|
+
n: '11'
|
|
87
87
|
}
|
|
88
88
|
];
|
|
89
89
|
var expected = [
|
|
90
90
|
{
|
|
91
91
|
ObjectID: 14203,
|
|
92
92
|
Resources: {
|
|
93
|
-
0:
|
|
93
|
+
0: 'LTE-M',
|
|
94
94
|
1: 20,
|
|
95
95
|
2: -79,
|
|
96
96
|
3: 6,
|
|
97
97
|
4: 56879116,
|
|
98
98
|
5: 24001,
|
|
99
|
-
6:
|
|
99
|
+
6: '10.160.243.113',
|
|
100
100
|
99: new Date(1676369307222)
|
|
101
101
|
}
|
|
102
102
|
}
|
|
103
103
|
];
|
|
104
104
|
assert.deepEqual(senMLtoLwM2M(input), expected);
|
|
105
105
|
});
|
|
106
|
-
void it(
|
|
106
|
+
void it('should ignore repeated base properties', function() {
|
|
107
107
|
var input = [
|
|
108
108
|
{
|
|
109
|
-
bn:
|
|
110
|
-
n:
|
|
111
|
-
vs:
|
|
109
|
+
bn: '14203/0/',
|
|
110
|
+
n: '0',
|
|
111
|
+
vs: 'LTE-M',
|
|
112
112
|
bt: 1699049665511
|
|
113
113
|
},
|
|
114
114
|
{
|
|
115
|
-
n:
|
|
115
|
+
n: '1',
|
|
116
116
|
v: 20
|
|
117
117
|
},
|
|
118
118
|
{
|
|
119
|
-
bn:
|
|
120
|
-
n:
|
|
119
|
+
bn: '14203/0/',
|
|
120
|
+
n: '2',
|
|
121
121
|
v: -89,
|
|
122
122
|
bt: 1699049665511
|
|
123
123
|
},
|
|
124
124
|
{
|
|
125
|
-
n:
|
|
125
|
+
n: '3',
|
|
126
126
|
v: 2305
|
|
127
127
|
},
|
|
128
128
|
{
|
|
129
|
-
n:
|
|
129
|
+
n: '4',
|
|
130
130
|
v: 34784790
|
|
131
131
|
},
|
|
132
132
|
{
|
|
133
|
-
n:
|
|
133
|
+
n: '5',
|
|
134
134
|
v: 24202
|
|
135
135
|
},
|
|
136
136
|
{
|
|
137
|
-
n:
|
|
138
|
-
vs:
|
|
137
|
+
n: '6',
|
|
138
|
+
vs: '100.81.95.75'
|
|
139
139
|
},
|
|
140
140
|
{
|
|
141
|
-
bn:
|
|
142
|
-
n:
|
|
141
|
+
bn: '14203/0/',
|
|
142
|
+
n: '11',
|
|
143
143
|
v: 7,
|
|
144
144
|
bt: 1699049665511
|
|
145
145
|
}
|
|
@@ -148,13 +148,13 @@ void describe("senMLtoLwM2M()", function() {
|
|
|
148
148
|
{
|
|
149
149
|
ObjectID: 14203,
|
|
150
150
|
Resources: {
|
|
151
|
-
0:
|
|
151
|
+
0: 'LTE-M',
|
|
152
152
|
1: 20,
|
|
153
153
|
2: -89,
|
|
154
154
|
3: 2305,
|
|
155
155
|
4: 34784790,
|
|
156
156
|
5: 24202,
|
|
157
|
-
6:
|
|
157
|
+
6: '100.81.95.75',
|
|
158
158
|
11: 7,
|
|
159
159
|
99: new Date(1699049665511)
|
|
160
160
|
}
|
|
@@ -162,29 +162,29 @@ void describe("senMLtoLwM2M()", function() {
|
|
|
162
162
|
];
|
|
163
163
|
assert.deepEqual(senMLtoLwM2M(input), expected);
|
|
164
164
|
});
|
|
165
|
-
void it(
|
|
165
|
+
void it('should handle multiple measurements for the same resource', function() {
|
|
166
166
|
var input = [
|
|
167
167
|
{
|
|
168
|
-
bn:
|
|
169
|
-
n:
|
|
168
|
+
bn: '14205/0/',
|
|
169
|
+
n: '0',
|
|
170
170
|
v: 21,
|
|
171
171
|
bt: 1699049600000
|
|
172
172
|
},
|
|
173
173
|
{
|
|
174
|
-
bn:
|
|
175
|
-
n:
|
|
174
|
+
bn: '14205/1/',
|
|
175
|
+
n: '0',
|
|
176
176
|
v: 31,
|
|
177
177
|
bt: 1699049600000
|
|
178
178
|
},
|
|
179
179
|
{
|
|
180
|
-
bn:
|
|
181
|
-
n:
|
|
180
|
+
bn: '14205/0/',
|
|
181
|
+
n: '0',
|
|
182
182
|
v: 22,
|
|
183
183
|
bt: 1699049700000
|
|
184
184
|
},
|
|
185
185
|
{
|
|
186
|
-
bn:
|
|
187
|
-
n:
|
|
186
|
+
bn: '14205/1/',
|
|
187
|
+
n: '0',
|
|
188
188
|
v: 32,
|
|
189
189
|
bt: 1699049700000
|
|
190
190
|
}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import { SenML } from
|
|
2
|
-
import { validate } from
|
|
1
|
+
import { SenML } from '../senml/SenMLSchema.js';
|
|
2
|
+
import { validate } from '../validate.js';
|
|
3
3
|
var validator = validate(SenML);
|
|
4
4
|
export var validateSenML = function(maybeSenML) {
|
|
5
5
|
return validator(maybeSenML);
|
|
@@ -1,26 +1,26 @@
|
|
|
1
|
-
import { describe, it } from
|
|
2
|
-
import assert from
|
|
3
|
-
import { validateSenML } from
|
|
4
|
-
void describe(
|
|
5
|
-
void it(
|
|
1
|
+
import { describe, it } from 'node:test';
|
|
2
|
+
import assert from 'node:assert/strict';
|
|
3
|
+
import { validateSenML } from './validateSenML.js';
|
|
4
|
+
void describe('validateSenML()', function() {
|
|
5
|
+
void it('should validate', function() {
|
|
6
6
|
var senMl = [
|
|
7
7
|
{
|
|
8
|
-
bn:
|
|
9
|
-
blv:
|
|
10
|
-
n:
|
|
8
|
+
bn: '14202/1/',
|
|
9
|
+
blv: '1.1',
|
|
10
|
+
n: '0',
|
|
11
11
|
v: 99,
|
|
12
12
|
bt: 1699049685992
|
|
13
13
|
},
|
|
14
14
|
{
|
|
15
|
-
n:
|
|
15
|
+
n: '1',
|
|
16
16
|
v: 4.179
|
|
17
17
|
},
|
|
18
18
|
{
|
|
19
|
-
n:
|
|
19
|
+
n: '2',
|
|
20
20
|
v: 0
|
|
21
21
|
},
|
|
22
22
|
{
|
|
23
|
-
n:
|
|
23
|
+
n: '3',
|
|
24
24
|
v: 25.7
|
|
25
25
|
}
|
|
26
26
|
];
|
package/models/README.md
CHANGED
|
@@ -4,7 +4,3 @@ Models are defined in the subdirectories of this folder.
|
|
|
4
4
|
|
|
5
5
|
A model definition consists of a `README.md` that provides a human-friendly
|
|
6
6
|
description of the model.
|
|
7
|
-
|
|
8
|
-
Optionally, if the device does not send senML directly,
|
|
9
|
-
[transforms can be defined](../README.md#model-transform-definitions) that
|
|
10
|
-
convert the payload sent by the model to senML.
|
|
@@ -1,15 +1,8 @@
|
|
|
1
1
|
import chalk from 'chalk'
|
|
2
|
-
import jsonata from 'jsonata'
|
|
3
2
|
import assert from 'node:assert/strict'
|
|
4
3
|
import { readFile, readdir, stat } from 'node:fs/promises'
|
|
5
4
|
import path from 'node:path'
|
|
6
|
-
import {
|
|
7
|
-
import { senMLtoLwM2M } from '../senml/senMLtoLwM2M.js'
|
|
8
|
-
import { getCodeBlock } from '../markdown/getCodeBlock.js'
|
|
9
|
-
import { getFrontMatter } from '../markdown/getFrontMatter.js'
|
|
10
|
-
import { validateSenML } from '../senml/validateSenML.js'
|
|
11
|
-
import { isRegisteredLwM2MObject } from '../lwm2m/isRegisteredLwM2MObject.js'
|
|
12
|
-
import { hasValue } from '../senml/hasValue.js'
|
|
5
|
+
import { ModelIDRegExp } from './types.js'
|
|
13
6
|
import { parseREADME } from 'markdown/parseREADME.js'
|
|
14
7
|
|
|
15
8
|
console.log(chalk.gray('Models rules check'))
|
|
@@ -40,86 +33,4 @@ for (const model of await readdir(modelsDir)) {
|
|
|
40
33
|
throw new Error(`README is not valid for ${model}!`)
|
|
41
34
|
}
|
|
42
35
|
console.log(chalk.green('✔'), chalk.gray(`README.md is valid`))
|
|
43
|
-
|
|
44
|
-
// Validate jsonata expressions
|
|
45
|
-
let hasTransforms = false
|
|
46
|
-
const transformsFolder = path.join(modelDir, 'transforms')
|
|
47
|
-
try {
|
|
48
|
-
await stat(transformsFolder)
|
|
49
|
-
hasTransforms = true
|
|
50
|
-
console.log(' ', chalk.gray('Transforms:'))
|
|
51
|
-
} catch {
|
|
52
|
-
console.log(' ', chalk.gray('No transforms found.'))
|
|
53
|
-
}
|
|
54
|
-
if (hasTransforms) {
|
|
55
|
-
for (const transform of (await readdir(transformsFolder)).filter((f) =>
|
|
56
|
-
f.endsWith('.md'),
|
|
57
|
-
)) {
|
|
58
|
-
console.log(' ', chalk.white('·'), chalk.white.bold(transform))
|
|
59
|
-
const markdown = await readFile(
|
|
60
|
-
path.join(modelDir, 'transforms', transform),
|
|
61
|
-
'utf-8',
|
|
62
|
-
)
|
|
63
|
-
|
|
64
|
-
// Validate front-matter
|
|
65
|
-
const type = getFrontMatter(markdown, FrontMatter).type
|
|
66
|
-
console.log(' ', chalk.green('✔'), chalk.gray(`Type ${type} is valid`))
|
|
67
|
-
const findBlock = getCodeBlock(markdown)
|
|
68
|
-
const matchExpression = findBlock('jsonata', 'Match Expression')
|
|
69
|
-
const transformExpression = findBlock('jsonata', 'Transform Expression')
|
|
70
|
-
const inputExample = JSON.parse(findBlock('json', 'Input Example'))
|
|
71
|
-
const resultExample = JSON.parse(findBlock('json', 'Result Example'))
|
|
72
|
-
|
|
73
|
-
const selectResult = await jsonata(matchExpression).evaluate(inputExample)
|
|
74
|
-
if (selectResult !== true) {
|
|
75
|
-
throw new Error(
|
|
76
|
-
`The select expression did not evaluate to true with the given example.`,
|
|
77
|
-
)
|
|
78
|
-
}
|
|
79
|
-
console.log(
|
|
80
|
-
' ',
|
|
81
|
-
chalk.green('✔'),
|
|
82
|
-
chalk.gray('Select expression evaluated to true for the example input'),
|
|
83
|
-
)
|
|
84
|
-
|
|
85
|
-
const transformResult = await jsonata(
|
|
86
|
-
// For testing purposes this function call result is hardcoded
|
|
87
|
-
transformExpression.replace('$millis()', '1699999999999'),
|
|
88
|
-
).evaluate(inputExample)
|
|
89
|
-
|
|
90
|
-
const maybeValidSenML = validateSenML(transformResult.filter(hasValue))
|
|
91
|
-
if ('errors' in maybeValidSenML) {
|
|
92
|
-
console.error(maybeValidSenML.errors)
|
|
93
|
-
throw new Error('The JSONata expression must produce valid SenML')
|
|
94
|
-
}
|
|
95
|
-
|
|
96
|
-
assert.deepEqual(maybeValidSenML.value, resultExample)
|
|
97
|
-
console.log(
|
|
98
|
-
' ',
|
|
99
|
-
chalk.green('✔'),
|
|
100
|
-
chalk.gray('Transformation result is valid SenML'),
|
|
101
|
-
)
|
|
102
|
-
|
|
103
|
-
assert.deepEqual(maybeValidSenML.value, resultExample)
|
|
104
|
-
console.log(
|
|
105
|
-
' ',
|
|
106
|
-
chalk.green('✔'),
|
|
107
|
-
chalk.gray('The transformation result matches the example'),
|
|
108
|
-
)
|
|
109
|
-
|
|
110
|
-
// Validate
|
|
111
|
-
for (const object of senMLtoLwM2M(maybeValidSenML.value)) {
|
|
112
|
-
if (!isRegisteredLwM2MObject(object, console.error)) {
|
|
113
|
-
throw new Error(
|
|
114
|
-
'The LwM2M object must follow LwM2M schema definition',
|
|
115
|
-
)
|
|
116
|
-
}
|
|
117
|
-
console.log(
|
|
118
|
-
' ',
|
|
119
|
-
chalk.green('✔'),
|
|
120
|
-
chalk.gray('SenML object is valid LwM2M'),
|
|
121
|
-
)
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
36
|
}
|