Witaj Gościu! ( Zaloguj | Rejestruj )

Forum PHP.pl

> Zaawansowane filtrowanie Symfony + Doctrine
Pyton_000
post 19.04.2021, 18:55:31
Post #1





Grupa: Zarejestrowani
Postów: 8 068
Pomógł: 1414
Dołączył: 26.10.2005

Ostrzeżenie: (0%)
-----


A napiszę coś może ciekawszego niż pytania o BOM, czy headers already sent smile.gif

Opis tego co chciałbym uzyskać opiszę w skali micro. Musi to mieć przełożenie na skalę macro.

Mamy sobie stronę z raportem.
Raport dotyczy użytkowników (bazowa tabela)
Muszę teraz dodać do tego filtrowanie - prościzna
Jest sobie N filtrów po których mogę filtrować userów. Na cele demonstracyjne przyjmiemy:
- filtrowanie po statusie użytkowania (tabela users)
- filtrowanie po zamówieniach np. użytkownicy mający zamówienia w statusie pedding (tabela orders)
- filtrowanie po produktach z zamówieć np: użytkownicy którzy zamówili zestaw X (tabela order_positions)
- filtrowanie po adresach użytkowników (tabela user_adresses)

Jak widać do każdego niemal filtra potrzebny jest LEFT JOIN z tabelką + odpowiedni WHERE
Muszę zbudować w miarę przyjazne rozwiązanie które pozwoli mi zdefiniować Criteria filtrów które będą miały w sobie JOIN + CONDITIONS.

Problemy które widzę to:
- Kolejność JOIN ma znaczenie (chyba oczowiste)
- Każde Criteria musi zawierać pełny zestaw danych typu JOIN i WHERE
- Podczas aplikowania wielu Criterias JOIN muszą być odfiltrowane i wrzucone w odpowiedniej kolejności (jak określić wagę...)
- i pewnie wiele innych

Generalnie idealne rozwiązanie byłoby zrobić

Kod
new CriteriaCollection([
   new OrderStatusCriteria(...),
   new OrderProductSthCriteria(...),
])->applyOn($qb)


To taka rozkmina.
Ma ktoś jakieś koncepcje albo rozwiązania? Zapraszam doo dyskusji smile.gif

PS. Inną metodą byłoby zbudowanie Grafu gdzie Vertex jest tabelą a Edge jest relacją między Vertexami. Przy robieniu A - C wybirać wszystko i skleić w kolejności w jakiej algo wyszuka ścieżkę (nie musi to być A - B - C ale może być A - D - E - C)
Go to the top of the page
+Quote Post
 
Start new topic
Odpowiedzi
rad11
post 20.04.2021, 15:56:29
Post #2





Grupa: Zarejestrowani
Postów: 1 270
Pomógł: 184
Dołączył: 7.10.2012
Skąd: Warszawa

Ostrzeżenie: (0%)
-----


A może utworzyć odpowiednie traity z metodami ktore przyjmą query buildera i będą do niego doklejać odpowiednie filtry oczywiście taka metoda by zwracała voida jakby nie było np podanego filtra. Potem wystarczyło by w odpowiedniej kolejności uruchomić metody tych traitow.

Tak jeszcze mysle ze te metody w traitach mogły by sprawdzac czy istnieją jakieś joiny i jeśli istnieją to nie dodawać kolejnych

Ten post edytował rad11 20.04.2021, 16:09:36
Go to the top of the page
+Quote Post
Pyton_000
post 20.04.2021, 18:39:13
Post #3





Grupa: Zarejestrowani
Postów: 8 068
Pomógł: 1414
Dołączył: 26.10.2005

Ostrzeżenie: (0%)
-----


@rad11
Chciałbym uniknąć Trait bo byłoby ich N w raporcie, a aktualnie i tak są tam metody które sprawdzają `if(filterBy_X) then $qb->where(...)` a joiny są wpakowane na sztywno.

@kayman
Tak, to już jest potworek biggrin.gif Działa i ma się dobrze, ale sposób w jaki to zostało napisane powoduje spore problemy ze zrozumieniem, dodawaniem itd.

@viking
Nie dokładnie to samo ale Doctrine ma https://www.doctrine-project.org/projects/d...ce/filters.html


Wpadł mi jeszcze do głowy pomysł ad moich Contexts.
Mogę zbudować sobie ContextCollection ze wszystkimi dostępnymi Contextami. Potem odbierając request odpalę sobie CotextBuilder i wyciągnę wszystkie contexty które będą odpowiadały paramsom.
W Context mogę sprawdzić czy QueryBuilder ma już wpisany alias do join i jeśli nie ma takowego to dodać, a jeśli już ma to nie a na końcu dodać tylko condition.

Taki draft mojej myśli:
  1. <?php
  2.  
  3. class ContextInterface {
  4. public function apply(QueryBuilder $qb);
  5. public function queryParam(): string;
  6. }
  7.  
  8. class UsernameFilterContext extends ContextInterface {
  9.  
  10. private static USERNAME_ALIAS = 'up';
  11.  
  12. public function apply(QueryBuilder $qb, mixed $bind) {
  13. $aliases = $qb->getAliases();
  14. if(!(\in_array(self::USERNAME_ALIAS, $aliases))) {
  15. $qb->join(UserProfile::class, self::USERNAME, 'WITH', 'u.id = ' . self::USERNAME_ALIAS .'.user_id');
  16. }
  17. $qb->where(self::USERNAME_ALIAS .'.username = ?1')
  18. ->bindParam(1, $bind);
  19. }
  20.  
  21. public function queryParam(): string {
  22. request 'username';
  23. }
  24. }
  25.  
  26. class ContextCollection {
  27. private array $contextCollection = [];
  28.  
  29. public function addContext(ContextCollection $context) {
  30. $this->contextCollection[$context->queryParam()] = $context;
  31. }
  32. }
  33.  
  34. class ContextBuilder {
  35. public function __construct(ServerRequestInterface $request, ConextCollection $contextCollection) {
  36. $this->contextCollection = $contextCollection;
  37. $this->request = $request;
  38. }
  39.  
  40. public function build(QueryBuilder $qb) {
  41. foreach($this->request->getQuery() as $query) {
  42. if(\in_array($query, $this->contextCollection)) {
  43. $this->contextCollection[$query]->apply($qb);
  44. }
  45. }
  46. }
  47. }


Ten post edytował Pyton_000 20.04.2021, 18:39:35
Go to the top of the page
+Quote Post

Posty w temacie


Reply to this topicStart new topic
1 Użytkowników czyta ten temat (1 Gości i 0 Anonimowych użytkowników)
0 Zarejestrowanych:

 



RSS Wersja Lo-Fi Aktualny czas: 13.06.2024 - 19:32