loopback-ds-tree-mixin

Create a tree out of a loopback model

Usage no npm install needed!

<script type="module">
  import loopbackDsTreeMixin from 'https://cdn.skypack.dev/loopback-ds-tree-mixin';
</script>

README

loopback-ds-tree-mixin

This module is designed for the Strongloop Loopback framework. It provides a mixin that makes it easy to transform a model into a tree using the Array of Ancestors pattern, more info here

Some notes

  • This module has only been tested to work with MongoDB. Feel free to test it on other Databases and PR if something needs to be added.
  • At the moment only some basic features are covered, like get the tree and create nodes. More features will be added down the road

Installation

npm install loopback-ds-tree-mixin

server.js

In your server/server.js file add the following line before the boot(app, __dirname); line.

...
var app = module.exports = loopback();
...
// Add Readonly Mixin to loopback
require('loopback-ds-tree-mixin')(app);
...

CONFIG

To use with your Models add the mixins attribute to the definition object of your model config.

{
    "name": "Categories",
    "properties": {
        "category": "String",
        "description": "String"

    },
    "mixins": {
        "Tree": {}
    }
}

USAGE

The module returns a promise for each call.

Simple example, get the entire tree

var Category = app.models.Category;
Category.asTree({})
  .then(function (results) {
  console.log(results);
})
  .catch(function (err) {
    console.log(err);
  });

Get all children of a given parent. We can pass a query in the form of an object {id : 1} or {category:'books'} or any valid loopback query. The where property is added to your object. You can also pass a string like 1 which has to be the parent ID or you can pass an object containing the property id (like the entire parent model)

var Category = app.models.Category;
Category.asTree({category : 'Books'})
  .then(function (results) {
  console.log(results);
});
var Category = app.models.Category;
Category.asTree('5616204f1c6a443c256c7a2f')
  .then(function (results) {
  console.log(results);
});
var Category = app.models.Category;
Category.findOne({where : {category : 'Books'}})
.then(function(Parent) {
//note, that we will use the Parent.id in our query to locate children
  Category.asTree(Parent)
    .then(function (results) {
    console.log(results);
  });
});

Some cool options

There are times where you want the children attached to the parent. In this case we can use the withParent option

var Category = app.models.Category;
Category.asTree('5616204f1c6a443c256c7a2f',{withParent:true})
  .then(function (results) {
  //the results will contain the parent
  /*{
    "category" : "Parent category",
    "children" : [....]
  }*/
  console.log(results);
});

In some (not so rare) cases you may need to perform complex operations on the tree, or you may need a quick reference to all nodes. A quick solution is to have the entire tree in a flat form where each element is connected by reference (because javascript) to the appropriate node. To achieve this we use the returnEverything option. We will get back an object containing both the tree and a flat array

var Category = app.models.Category;
Category.asTree('5616204f1c6a443c256c7a2f',{returnEverything:true})
  .then(function (results) {
  //the results will contain the parent
  /*
  returns {tree: treeStructuredArray, flat: aFlatArrayOfTheTree}
  */
  console.log(results);
});

API

The mixin adds a /asTree end point to your model. It takes 2 parameters, the parent query ({"category":"Books"}) and the options object. Only the parent query is required

Adding nodes

To add a node you need to provide the parent and the node.

//Example providing a query object as parent
var Category = app.models.Category;
  Category.addNode({permalink: 'a-category'},
    {
      category : 'A sub category',
      permalink : 'a-sub-category',
      active : true
    })
    .then(function (newItem) {
      console.log(newItem);
    })
    .catch(function (err) {
      console.log('Err', err);
    });

Again, you can provide an object, a string or a loopback model as a parent, just like the asTree() method.

//Example providing a string ID as parent
var Category = app.models.Category;
  Category.addNode('5616204f1c6a443c256c7a2f',
    {
      category : 'A sub category',
      permalink : 'a-sub-category',
      active : true
    })
    .then(function (newItem) {
      console.log(newItem);
    })
    .catch(function (err) {
      console.log('Err', err);
    });
//Example providing a loopback model as parent
var Category = app.models.Category;
Category.findOne({where : {category : 'Books'}})
.then(function(Parent) {
  Category.addNode(Parent,
    {
      category : 'A sub category',
      permalink : 'a-sub-category',
      active : true
    })
    .then(function (newItem) {
      console.log(newItem);
    })
});

API

The mixin adds a /addNode end point to your model. It takes 2 parameters, the parent query ({"category":"Books"}) and the new node object. Both parameters are required

Deleting nodes

You can safely delete a node via the deleteNode method. By safely we mean that you can be sure that the children will be handled accordingly. If a node has X amount of children, and you just delete it via the loopback API, then all the children still point to that parent. By using the deleteNode method, you ensure that these children will become orphaned so that you can take care of them later on as you please. You can also delete the parent along with the children in one call by adding the withChildren option. Just as above, you can locate the node you want to delete either as an ID string, a loopback query or a loopback model.

//Default operation. all children will become orphaned
//set withChildren to true to delete the children along with the parent
var Category = app.models.Category;
Category.deleteNode({category : 'Books'},{withChildren:false})
.then(function(result) {
    //True if all went well
})
.catch(function(err){

});

API

The mixin adds a /deleteNode end point to your model. It takes 2 parameters, the node query ({"category":"Books"}) and the options object. Only the first parameter is required

Moving nodes

Moving a node consists of the following actions :

  • Find the node
  • Find the parent
  • Make the node a child of the parent
  • Update all (if any) of the nodes children to the new path
var Category = app.models.Category;
//move category sci-fi under the category books
Category.moveNode({permalink: 'sci-fi'},{permalink: 'Books'})
.then(function(result) {
    //get the sci-fi category back with the path updated
})
.catch(function(err){

});

API

The mixin adds a /moveNode end point to your model. It takes 2 parameters, the node query ({"category":"Books"}) and the parent query object (same as node). Both parameters are required

Saving a json tree

In some cases you might need to update the entire tree based on a json representation. Maybe your front-end sends the json of the tree after the user has finished re-ordering it. Assuming that the tree is formed more or less as it is in your model, and that the children nodes live in a children array, you can pass this json and the mixin will update your DB accordingly.

Notes

  • ONLY parent - ancestors are updated. Updating other properties is risky when considering that every model is different.
  • On a front-end we usually show just a part of the tree, root nodes are hidden from the user in most cases. If that is true, then you need to pass the option prependRoot to the method (as usually, can be a query, stringID or model). If you don't do this, then the tree will break
  • This is feature is not 100% solid. It worked on many test cases but as most front-end plugins transform the JSON they produce, there might be several cases where this fails.
var Category = app.models.Category;
//req.body.tree is the JSON representation
//req.body.prependRoot is the root i want to append
Category.saveJsonTree(req.body.tree,{prependRoot : req.body.prependRoot || false})
      .then(function (result) {
        res.send(result);
      })
      .catch(function (err) {

      });

API

The mixin adds a /saveJsonTree end point to your model. It takes 2 parameters, the JSON tree (valid Array tree) and the options object. Only the first parameter is required.

Events

The mixin emits events on the loopback Event bus for certain operations.

  • lbTree.add.success when a node was added. Returns the new node
  • lbTree.move.before Just before start the moving operation. Return the previous parent, the new parent and the node.
  • lbTree.move.after After the moving operation. Return the previous parent, the new parent and the node.
  • lbTree.move.childrenFound During the move when we have found all the children. Return the children.
  • lbTree.move.parent During the move we have the parent. Returns the parent.
  • lbTree.move.newPath The new path after the move ended. Returns the node.
  • lbTree.delete Node deleted. Returns {success : true/false}
  • lbTree.saveJsTree When a JSTree operations is done. Returns the tree
var Category = app.models.Category;

Category.on('lbTree.move.childrenFound',function(children){
    console.log("childrenFound",children);
});
Category.on('lbTree.move.parent',function(children){
    console.log("parent",children);
});

TESTING

Install dev dependencies:

npm i -D

Run tests:

npm test