Search code examples
node.jsnestedjavascript-objectshierarchical-dataflatten

Node JS How to build a nested object using parent ids


for my internship, I need to build a nested object using parent IDs I don't want children attribute array. I have an array of object with id and parent id and I use npm flatnest to do it. This works for a one-level hierarchy but the code must be adapted for a multi-hierarchy level. I don't know how to adapt that to multi-hierarchy level.

This is my array of Object

var fn = require("flatnest");

const flat =
[
    { "id": 1, "name": 'Restaurants', 'parentId': 0},
    { "id": 2, "name": 'family restaurant', 'parentId': 1, 'value':'Excellent'},
    { "id": 3, "name": 'Sun restaurant', 'parentId': 1,'value':""},
    { "id": 4, "name": 'Sun restaurant 1', 'parentId': 3, 'value':'Good'},
    { "id": 5, "name": 'Sun restaurant 2', 'parentId': 3, 'value':"bad"},
    { "id": 6, "name": 'Hotels', 'parentId': 0,'value':""},
    { "id": 7, "name": 'Space Hotel', 'parentId': 6,'value':""},
    { "id": 8, "name": 'Sun Hotel', 'parentId': 7,'value':'Nice'},
    { "id": 9, "name": 'Moon Hotel', 'parentId': 7,'value':""},
    { "id": 10, "name": 'Moon Hotel 1', 'parentId': 9, 'value':"Excellent"},
    { "id": 11, "name": 'Moon Hotel 2', 'parentId': 9, 'value':"Worst"},

];

To use nested function of npm flatnest, I have to flat my array of Object (const flat)

My code to flat :

var transform={};

for(var i=0;i<flat.length;i++)
{

  if(typeof flat[parseInt(i)+1] !== 'undefined' )
  {
    if(flat[i].id==flat[i+1].parentId)
    {
      var t = flat[i].name;
      transform[t.concat(".").concat(flat[i+1].name)]=flat[i+1].value;
  }else{
      transform[t.concat(".").concat(flat[i+1].name)]=flat[i+1].value;

    }

}
}

console.log(transform)
var nested = fn.nest(transform)
console.log(nested)

I expect the output of console.log(transform) to be

{ 'Restaurants.family restaurant':'Excellent',
  'Restaurants.Sun restaurant.Sun restaurant 1': 'Good',
  'Restaurants.Sun restaurant.Sun restaurant 2': 'bad',
  'Hotels.Space Hotel.Sun Hotel': 'Nice',
  'Hotels.Space Hotel.Moon Hotel.Moon Hotel 1': 'Excellent',
  'Hotels.Space Hotel.Moon Hotel.Moon Hotel 2' : 'Worst'}

Then by using nested function :

 var nested = fn.nest(transform)
 console.log(nested)

The output must be exactly like that :

 "Restaurants":{
        "family restaurant":"Excellent",
        "Sun restaurant":{
          "Sun restaurant 1":"Good",
          "Sun restaurant 2":"bad"
        }
      },
      "Hotels":{
        "Space Hotel":{
          "Sun Hotel":"Nice",
          "Moon Hotel":{
            "Moon Hotel 1":"Excellent",
            "Moon Hotel 2":"Worst"
          }
        }
      }

}

but the actual output of console.log(transform) is :

{'Restaurants.family restaurant':'Excellent',
 'Restaurant.Sun restaurant':'',
 'Sun restaurant.Sun restaurant 1':'Good',
 'Sun restaurant.Sun restaurant 2':'bad',
 'Sun restaurant.Hotels':'',
 'Hotels.Space Hotel':'',
 'Space Hotel.Sun Hotel':'Nice'
 'Space Hotel.Moon Hotel':'',
 'Moon Hotel.Moon Hotel 1':'Excellent',
 'Moon Hotel.Moon Hotel 2': 'Worst'}

Solution

  • I'm not using flatnest. But the below code works for me. Please check and let me know if it doesn't work for any scenario.

    const flat = [{
        "id": 1,
        "name": 'Restaurants',
        'parentId': 0
      },
      {
        "id": 2,
        "name": 'family restaurant',
        'parentId': 1,
        'value': 'Excellent'
      },
      {
        "id": 3,
        "name": 'Sun restaurant',
        'parentId': 1,
        'value': ""
      },
      {
        "id": 4,
        "name": 'Sun restaurant 1',
        'parentId': 3,
        'value': 'Good'
      },
      {
        "id": 5,
        "name": 'Sun restaurant 2',
        'parentId': 3,
        'value': "bad"
      },
      {
        "id": 6,
        "name": 'Hotels',
        'parentId': 0,
        'value': ""
      },
      {
        "id": 7,
        "name": 'Space Hotel',
        'parentId': 6,
        'value': ""
      },
      {
        "id": 8,
        "name": 'Sun Hotel',
        'parentId': 7,
        'value': 'Nice'
      },
      {
        "id": 9,
        "name": 'Moon Hotel',
        'parentId': 7,
        'value': ""
      },
      {
        "id": 10,
        "name": 'Moon Hotel 1',
        'parentId': 9,
        'value': "Excellent"
      },
      {
        "id": 11,
        "name": 'Moon Hotel 2',
        'parentId': 9,
        'value': "Worst"
      },
    
    ];
    
    const map = new Map();
    const result = flat.reduce((acc, curr) => {
      let val = {}
      if (curr.parentId == 0)
        acc[curr.name] = val;
      else {
        if (map.get(curr.parentId)) {
          if (curr.value != '')
            val = curr.value;
          map.get(curr.parentId)[curr.name] = val;
        }
      }
      map.set(curr.id, val);
      return acc;
    }, {});
    
    console.log(JSON.stringify(result));