A multi root nested set implementation for DBAL users.
This library provides you write, read and inspection classes for nested sets with multiple root nodes per table. Solely relying on the Doctrine DBAL.
Contrary to other solutions this library has clear boundaries and leaves the software design up to you.
Use composer to install the library
> composer require shopware/dbal-nested-set
You always need a configuration that sets up the basic column names of your implementation:
use ShopwareDbalNestedSetNestedSetConfig;
$config = new NestedSetConfig(
'id', // Primary key column name
'left', // left column name
'right', // right column name
'level' // level column name
);
Then you can use the NestedSetFactory
to create the different classes of the library.
use ShopwareDbalNestedSetNestedSetFactory;
use DoctrineDBALConnection;
$writer = NestedSetFactory::createWriter($dbalConnection, $config);
You may want to create a normalized schema for nested set tables, this can be accomplished through the NestedSetTableFactory
. It will create the base DDL for a tree with indexes. So if you want to add a simple tree with a name column and an autoincrement id it will look like this:
$tableFactory = NestedSetFactory::createTableFactory($connection, $config);
$schema = new DoctrineDBALSchemaSchema();
$table = $tableFactory->createTable(
$schema,
'tree', // table name
'root_id' // nested set root id
);
$table->addColumn('id', 'integer', ['unsigned' => true, 'autoincrement' => true]);
$table->addColumn('name', 'string', ['length' => 255]);
$table->setPrimaryKey(['id']);
$addSql = $schema->toSql($connection->getDatabasePlatform());
Of course this is optional and may be accomplished through any schema configuration tool.
The library provides a NestedSetWriter
class that contains all insert, move and update operations. All operations should be reminiscent of DoctrineDBALConnection::insert()
and DoctrineDBALConnection::update()
and just require plain data.
As an example you can use this to create a tree
$writer = NestedSetFactory::createWriter($dbalConnection, $config);
// create a Root node
$writer->insertRoot('tree', 'root_id', 100, ['name' => 'Clothing']);
// create subnodes
$writer->insertAsFirstChild('tree', 'root_id', 1, ['name' => 'Men']);
$writer->insertAsNextSibling('tree', 'root_id', 2, ['name' => 'Women']);
$writer->insertAsFirstChild('tree', 'root_id', 2, ['name' => 'Suits']);
$writer->insertAsFirstChild('tree', 'root_id', 3, ['name' => 'Dresses']);
$writer->insertAsNextSibling('tree', 'root_id', 5, ['name' => 'Skirts']);
$writer->insertAsNextSibling('tree', 'root_id', 6, ['name' => 'Blouses']);
$writer->insertAsFirstChild('tree', 'root_id', 4, ['name' => 'Jackets']);
$writer->insertAsFirstChild('tree', 'root_id', 4, ['name' => 'Slacks']);
$writer->insertAsFirstChild('tree', 'root_id', 5, ['name' => 'Evening Gowns']);
$writer->insertAsNextSibling('tree', 'root_id', 10, ['name' => 'Sun Dresses']);
And then use the writer to move nodes around
$writer->moveAsNextSibling('tree', 'root_id', 4, 7);
You may want to retrieve information about different nodes. This can be done through the NestedSetTableNodeInspector
.
$inspector = NestedSetFactory::createTableNodeInspector($connection, $config);
$inspector->isLeaf('tree', 'root_id', 9); // true | false
$inspector->isAncestor('tree', 'root_id', 1, 2) // true | false
The NestedSetQueryFactory
helps retrieve a set of nodes from the tree. Since the library has no concept of entities it will only prepare query builders for you ready to add selects, joins and other conditions.
$queryFactory = NestedSetFactory::createQueryFactory($connection, $config);
$data = $queryFactory
->createChildrenQueryBuilder('tree', 't', 'root_id', 2)
->select('*')
->execute()
->fetchAll();
If you want to develop locally you may have to configure the database access through a little shell script:
#!/usr/bin/env bash
export DB_USER='foo'
export DB_PASSWORD='bar'
export DB_HOST='baz'
export DB_NAME='dbal_nested_set'
bin/phpunit