I have an object similar to this one:
JavaScript
x
23
23
1
const obj = {
2
operator: 'AND',
3
attributes: [
4
{
5
attribute: 'attr1',
6
value: '123'
7
},
8
{
9
attribute: 'attr2',
10
value: '234'
11
},
12
{
13
operator: 'OR',
14
attributes: [
15
{
16
attribute: 'attr3',
17
value: 'xxx'
18
}
19
]
20
}
21
]
22
}
23
And I want to construct a tree from it:
JavaScript
1
60
60
1
{
2
"id": "9aabbb89-cdef-4012-b456-7178a79bf800",
3
"type": "group",
4
"path": [
5
"9aabbb89-cdef-4012-b456-7178a79bf800"
6
],
7
"children1": {
8
"a89aaa88-0123-4456-b89a-b178a79c0aab": {
9
"type": "rule",
10
"id": "a89aaa88-0123-4456-b89a-b178a79c0aab",
11
"properties": {
12
"attribute": "attr1",
13
"value": "123"
14
},
15
"path": [
16
"9aabbb89-cdef-4012-b456-7178a79bf800",
17
"a89aaa88-0123-4456-b89a-b178a79c0aab"
18
]
19
},
20
"a9babbb8-cdef-4012-b456-7178a79c35ab": {
21
"type": "rule",
22
"id": "a9babbb8-cdef-4012-b456-7178a79c35ab",
23
"properties": {
24
"attribute": "attr2",
25
"value": "234"
26
},
27
"path": [
28
"9aabbb89-cdef-4012-b456-7178a79bf800",
29
"a9babbb8-cdef-4012-b456-7178a79c35ab"
30
]
31
},
32
"b9aaabb9-0123-4456-b89a-b178a79cc5c2": {
33
"type": "group",
34
"id": "b9aaabb9-0123-4456-b89a-b178a79cc5c2",
35
"properties": {
36
"conjunction": "OR"
37
},
38
"path": [
39
"9aabbb89-cdef-4012-b456-7178a79bf800",
40
"b9aaabb9-0123-4456-b89a-b178a79cc5c2"
41
],
42
"children1": {
43
"9baa88b9-cdef-4012-b456-7178a79cc5c3": {
44
"type": "rule",
45
"id": "9baa88b9-cdef-4012-b456-7178a79cc5c3",
46
"properties": {
47
"attribute": "attr3",
48
"value": "xxx"
49
},
50
"path": [
51
"9aabbb89-cdef-4012-b456-7178a79bf800",
52
"b9aaabb9-0123-4456-b89a-b178a79cc5c2",
53
"9baa88b9-cdef-4012-b456-7178a79cc5c3"
54
]
55
}
56
}
57
}
58
}
59
}
60
I write this code to create the tree:
JavaScript
1
29
29
1
function buildTree(obj) {
2
if (!obj || !Object.keys(obj).length) return {};
3
4
if ( 'attribute' in obj) {
5
const id = uuid4();
6
return {[id]: {
7
id,
8
type: 'rule',
9
properties: {
10
attribute: obj.attribute,
11
value: obj.value,
12
13
},
14
}
15
}
16
}
17
18
if (obj.operator === 'AND' || obj.operator === 'OR') {
19
return { id: uuid4(),
20
type: 'group',
21
properties: {
22
conjunction: obj.operator
23
},
24
children1: obj.operands.map(buildTree),
25
}
26
27
}
28
}
29
However, I’m stucked how to keep the track of the parent ids of a children and save them in a path
attribute. How can I fix that?
Advertisement
Answer
You have a recursive function, so just pass the path
down through the calls (starting with an empty array) and append to it as you go:
JavaScript
1
79
79
1
function buildTree(obj, path = []) {
2
if (!obj || !Object.keys(obj).length) return {};
3
const id = uuid4()
4
const newPath = [path, id];
5
if ('attribute' in obj) {
6
return {
7
[id]: {
8
id,
9
type: 'rule',
10
path: newPath,
11
properties: {
12
attribute: obj.attribute,
13
value: obj.value,
14
15
},
16
}
17
}
18
}
19
20
if (obj.operator === 'AND' || obj.operator === 'OR') {
21
22
return {
23
id: id,
24
type: 'group',
25
path: newPath,
26
properties: {
27
conjunction: obj.operator
28
},
29
children1: obj.attributes.map(x => buildTree(x, newPath)),
30
}
31
32
}
33
}
34
35
36
const obj = {
37
operator: 'AND',
38
attributes: [{
39
attribute: 'attr1',
40
value: '123'
41
},
42
{
43
attribute: 'attr2',
44
value: '234'
45
},
46
{
47
operator: 'OR',
48
attributes: [{
49
attribute: 'attr3',
50
value: 'xxx'
51
}]
52
}
53
]
54
}
55
56
function uuid4() {
57
let array = new Uint8Array(16)
58
crypto.getRandomValues(array)
59
60
// Manipulate the 9th byte
61
array[8] &= 0b00111111 // Clear the first two bits
62
array[8] |= 0b10000000 // Set the first two bits to 10
63
64
// Manipulate the 7th byte
65
array[6] &= 0b00001111 // Clear the first four bits
66
array[6] |= 0b01000000 // Set the first four bits to 0100
67
68
const pattern = "XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX"
69
let idx = 0
70
71
return pattern.replace(
72
/XX/g,
73
() => array[idx++].toString(16).padStart(2, "0"), // padStart ensures a leading zero, if needed
74
)
75
}
76
77
78
79
console.log(buildTree(obj));