eltharin/autoqb

automaticquerybuilder Bundle for symfony

Installs: 12

Dependents: 0

Suggesters: 0

Security: 0

Stars: 0

Watchers: 1

Forks: 0

Open Issues: 0

Type:symfony-bundle

V1.0.1 2025-03-03 15:04 UTC

This package is auto-updated.

Last update: 2025-03-31 15:08:28 UTC


README

Latest Stable Version Total Downloads Latest Unstable Version License

Installation

  • Require the bundle with composer:
composer require eltharin/autoqb

What is AutomaticQueryBuilder Bundle?

This bundle will create automatic custom queryBuilder (DQL) in repository.

Joins can be piloted by arguments without create one function each time joins are not the same.

How It Works ?

First add AutoQbRepository trait in your repository.

This add getDQL() function and change findAll() :

use Eltharin\AutomaticQueryBuilderBundle\Repository\AutoQbRepository;

class MyEntityRepository extends ServiceEntityRepository
{
    use AutoQbRepository;
    
    ....
}

Now you can use it.

Inline method

In a service you can call your repository and ask link directly in getDQL function :

    $dql = $myRepository->getDQL();

Considering $myRepository is ClassTest1 Entity Repository, DQL query will be :

    SELECT classtest1 
    FROM EltharinAutoQBTests\Entity\ClassTest1 classtest1

alias is tableName by default, or autoQBAlias property in repository or you can ether set in DQLOptions

you can pass a DqlOptions object to getDQL to ask relation :

    $dql = $myRepository->getDQL((new DqlOptions)->with('classtest1__classtest2'));

Considering ClassTest1 as a relation to entity ClassTest2, DQL query will be :

    SELECT classtest1, classtest1__classtest2 
    FROM EltharinAutoQBTests\Entity\ClassTest1 classtest1 
    LEFT JOIN classtest1.classtest2 classtest1__classtest2

you can ask a relation from relation without request middle, that's automatic :

    $dql = $myRepository->getDQL((new DqlOptions)->with('classtest1__classtest2__classtest3'));

Considering ClassTest1 as a relation to entity ClassTest2, DQL query will be :

    SELECT classtest1, classtest1__classtest2, classtest1__classtest2__classtest3 
    FROM EltharinAutoQBTests\Entity\ClassTest1 classtest1 
    LEFT JOIN classtest1.classtest2 classtest1__classtest2 
    LEFT JOIN classtest1__classtest2.classtest3 classtest1__classtest2__classtest3

you find it's not enought automatic ? OK let's go for attributes

By properties attributes

You can also add attributes to your relations to make it more automatic, note you can have many attributes, the first compatible with your query will be taken :

We can add a relation to ClassTest4 in our entity, and add a AutoQBField attribute, with autolink at true, relation will be always taken.

    #[ORM\ManyToOne()]
    #[ORM\JoinColumn(nullable: false)]
    #[AutoQbField(autoLink: true)]
    private ?ClassTest4 $classtest4 = null;

now if we execute getDQL :

    $dql = $myRepository->getDQL();

Considering $myRepository is ClassTest1 Entity Repository, DQL query will be :

    SELECT classtest1, classtest1__classtest4 
    FROM EltharinAutoQBTests\Entity\ClassTest1 classtest1
    LEFT JOIN classtest1.classtest4 classtest1__classtest4

You can change the jointype :

    #[ORM\ManyToOne()]
    #[ORM\JoinColumn(nullable: false)]
    #[AutoQbField(autoLink: true, joinType: 'inner')]
    private ?ClassTest4 $classtest4 = null;

now if we execute getDQL :

    $dql = $myRepository->getDQL();

Considering $myRepository is ClassTest1 Entity Repository, DQL query will be :

    SELECT classtest1, classtest1__classtest4
    FROM EltharinAutoQBTests\Entity\ClassTest1 classtest1
    INNER JOIN classtest1.classtest4 classtest1__classtest4

Automatic is good, but sometimes we want a relation only if root,

For example ClassTest1 have a manyToOne relation to ClassTest2 whitch have a manyToOne relation with ClassTest3

When we query ClassTest1 with ClassTest2 relation, we don't want have ClassTest3 automaticly but if we query ClassTest2 we want ever.

in ClassTest2 we add a AutoQbField attribute on ClassTest3 relation, add autolink and whenAlias arguments :

    #[ORM\ManyToOne()]
    #[ORM\JoinColumn(nullable: false)]
    #[AutoQbField(whenAlias: 'classtest2', autoLink: true)]
    private ?ClassTest3 $classtest3 = null;
    $dql = $class1Repository->getDQL((new DqlOptions)->with('classtest1__classtest2'));

will render DQL :

    SELECT classtest1, classtest1__classtest2
    FROM EltharinAutoQBTests\Entity\ClassTest1 classtest1
    LEFT JOIN classtest1.classtest2 classtest1__classtest2

and

    $dql = $class2Repository->getDQL();

will render DQL :

    SELECT classtest2, classtest2__classtest3
    FROM EltharinAutoQBTests\Entity\ClassTest2 classtest2
    LEFT JOIN classtest2.classtest3 classtest2__classtest3

now imagine you have a property in ClassTest2 named "visible". when you make a query on ClassTest1 you want have only classTest2 visible but in some circonstances, you want have all.

Remember, you can have many AutoQbField attributes :

    #[ORM\ManyToOne()]
    #[ORM\JoinColumn(nullable: false)]
    #[AutoQbField(relationAlias: 'classtest2All')]
    #[AutoQbField(conditionType: 'WITH', condition: '##alias##.visible = 1')]
    private ?ClassTest2 $classtest2 = null;

note we have set the relationAlias argument in first.

now :

    $dql = $class1Repository->getDQL((new DqlOptions)->with('classtest1__classtest2'));

will render DQL :

    SELECT classtest1, classtest1__classtest2 
    FROM EltharinAutoQBTests\Entity\ClassTest1 classtest1
    LEFT JOIN classtest1.classtest2 classtest1__classtest2 
        WITH classtest1__classtest2.visible = 1

and if we query the alias :

    $dql = $class1Repository->getDQL((new DqlOptions)->with('classtest1__classtest2All'));

will render DQL :

    SELECT classtest1, classtest1__classtest2All 
    FROM EltharinAutoQBTests\Entity\ClassTest1 classtest1
    LEFT JOIN classtest1.classtest2 classtest1__classtest2All

you can also ask indexBy relation :

    #[ORM\ManyToOne()]
    #[ORM\JoinColumn(nullable: false)]
    #[AutoQbField(indexBy: 'idForIndexBy')]
    private ?ClassTest3 $classtest3 = null;

will render DQL :

    SELECT mySuperAlias8, mySuperAlias8__classtest3 
    FROM EltharinAutoQBTests\Entity\ClassTest8 mySuperAlias8 
    LEFT JOIN mySuperAlias8.classtest3 mySuperAlias8__classtest3 
        INDEX BY mySuperAlias8__classtest3.idForIndexBy,

Now, IF you want more, you can make your own function in repository and call it :

    #[ORM\ManyToOne()]
    #[ORM\JoinColumn(nullable: false)]
    #[AutoQbField(callback: 'myCallbackInRepository')]
    private ?ClassTest4 $classtest4 = null;
    public function myCallbackInRepository(QueryBuilder $qb, AssociationMapping $assoc, AutoQbField $propertyAttribute, string $alias, string $subAlias, $from = [], $count = 0, ?array $with = [])
    {
        $qb->andWhere($alias . '.property = :something')->setParameter('something', 'something');
    }
    SELECT mySuperAlias8, mySuperAlias8__classtest4 
    FROM EltharinAutoQBTests\Entity\ClassTest8 mySuperAlias8 
    LEFT JOIN mySuperAlias8.classtest4 mySuperAlias8__classtest4 
    WHERE mySuperAlias8.property = :something