3 Commits

Author SHA1 Message Date
loic
5180108598 Added Dashboard tab 2025-07-04 08:19:00 +02:00
jb cr
08cd3eccd6 prepare publishing 2025-04-02 11:18:21 +02:00
jbcr
9dc90bb001 add make by mention (#6) 2025-04-02 11:15:36 +02:00
11 changed files with 152 additions and 3 deletions

View File

@@ -1,3 +1,9 @@
# Version 5.2.0
* Added Dashboard tab
# Version 5.1.1
* Add branding label
# Version 5.1.0
* Added possibility to create one shot job from scheduled job

View File

@@ -13,6 +13,9 @@ use Symfony\Component\HttpKernel\Bundle\Bundle;
class CodeRhapsodieIbexaDataflowBundle extends Bundle
{
public const VERSION = '5.2.0';
public const PRODUCT_NAME = 'ibexadataflow';
protected $name = 'CodeRhapsodieIbexaDataflowBundle';
public function getContainerExtension()

View File

@@ -6,6 +6,7 @@ namespace CodeRhapsodie\IbexaDataflowBundle\Controller;
use CodeRhapsodie\DataflowBundle\Entity\Job;
use CodeRhapsodie\DataflowBundle\Entity\ScheduledDataflow;
use CodeRhapsodie\IbexaDataflowBundle\CodeRhapsodieIbexaDataflowBundle;
use CodeRhapsodie\IbexaDataflowBundle\Form\CreateOneshotType;
use CodeRhapsodie\IbexaDataflowBundle\Form\CreateScheduledType;
use CodeRhapsodie\IbexaDataflowBundle\Form\UpdateScheduledType;
@@ -14,6 +15,7 @@ use CodeRhapsodie\IbexaDataflowBundle\Gateway\JobGateway;
use CodeRhapsodie\IbexaDataflowBundle\Gateway\ScheduledDataflowGateway;
use Doctrine\DBAL\Query\QueryBuilder;
use Ibexa\Contracts\AdminUi\Controller\Controller;
use Ibexa\Contracts\Core\Ibexa;
use Ibexa\Core\MVC\Symfony\Security\Authorization\Attribute;
use Pagerfanta\Doctrine\DBAL\QueryAdapter;
use Pagerfanta\Pagerfanta;
@@ -33,7 +35,18 @@ class DashboardController extends Controller
{
$this->denyAccessUnlessGranted(new Attribute('ibexa_dataflow', 'view'));
return $this->render('@ibexadesign/ibexa_dataflow/Dashboard/main.html.twig');
$data = [
'product' => CodeRhapsodieIbexaDataflowBundle::PRODUCT_NAME,
'version' => CodeRhapsodieIbexaDataflowBundle::VERSION,
'php' => PHP_VERSION,
'ibexa' => Ibexa::VERSION,
];
return $this->render('@ibexadesign/ibexa_dataflow/Dashboard/main.html.twig', [
'link' => 'https://www.code-rhapsodie.fr/product/redirect/'.str_replace('=', '',
base64_encode(json_encode($data))
),
]);
}
public function repeating(Request $request): Response
@@ -117,7 +130,7 @@ class DashboardController extends Controller
{
$pager = new Pagerfanta(
new ExceptionJSONDecoderAdapter(
new QueryAdapter($query, fn($queryBuilder) => $queryBuilder->select('COUNT(DISTINCT id) AS total_results')
new QueryAdapter($query, fn ($queryBuilder) => $queryBuilder->select('COUNT(DISTINCT id) AS total_results')
->resetQueryPart('orderBy')
->setMaxResults(1))
)
@@ -127,4 +140,13 @@ class DashboardController extends Controller
return $pager;
}
public function dashboard(): Response
{
$this->denyAccessUnlessGranted(new Attribute('ibexa_dataflow', 'view'));
return $this->render('@ibexadesign/ibexa_dataflow/Dashboard/dashboard.html.twig', [
'jobs' => $this->jobGateway->getListPendindOrRunning(),
]);
}
}

View File

@@ -54,5 +54,15 @@ final class JobGateway
{
$this->jobRepository->save($job);
}
public function getListPendindOrRunning(): array
{
$qb = $this->jobRepository->createQueryBuilder('w');
return $qb->andWhere($qb->expr()->in('w.status', ':status'))
->setParameter('status', implode(',',[Job::STATUS_RUNNING, Job::STATUS_PENDING, Job::STATUS_QUEUED]))
->orderBy('w.requested_date', 'ASC')
->execute()
->fetchAllAssociative();
}
}
class_alias(JobGateway::class, 'CodeRhapsodie\EzDataflowBundle\Gateway\JobGateway');

View File

@@ -129,6 +129,14 @@ services:
arguments:
$jobRepository: '@CodeRhapsodie\DataflowBundle\Repository\JobRepository'
CodeRhapsodie\IbexaDataflowBundle\Tab\DashboardTab:
parent: Ibexa\Contracts\AdminUi\Tab\AbstractTab
public: false
arguments:
$httpKernelRuntime: '@twig.runtime.httpkernel'
tags:
- { name: ibexa.admin_ui.tab, group: coderhapsodie-ibexa_dataflow }
CodeRhapsodie\IbexaDataflowBundle\Tab\RepeatingTab:
parent: Ibexa\Contracts\AdminUi\Tab\AbstractTab
public: false

View File

@@ -83,3 +83,7 @@ coderhapsodie.dataflow.update.next: 'Next execution'
coderhapsodie.ibexa_dataflow.workflow.edit.success: 'Dataflow schedule successfully updated.'
coderhapsodie.ibexa_dataflow.workflow.edit.error: 'An error occurred during the dataflow schedule update: "%message%".'
coderhapsodie.ibexa_dataflow.notfound: 'Requested data is not found'
coderhapsodie.ibexa_dataflow.powered_by: 'Powered by'
coderhapsodie.ibexa_dataflow.made_by: 'Made by'
coderhapsodie.ibexa_dataflow.dashboard: Dashboard
coderhapsodie.ibexa_dataflow.dashboard.title: Running or waiting tasks

View File

@@ -81,3 +81,7 @@ coderhapsodie.dataflow.update.next: 'Prochaine exécution'
coderhapsodie.ibexa_dataflow.workflow.edit.success: 'La programmation du dataflow a été mise à jour avec succès.'
coderhapsodie.ibexa_dataflow.workflow.edit.error: 'Une erreur est survenue lors de la modification de la programmation du dataflow : "%message%".'
coderhapsodie.ibexa_dataflow.notfound: 'Les données demandées sont introuvables'
coderhapsodie.ibexa_dataflow.powered_by: 'Propulsé par'
coderhapsodie.ibexa_dataflow.made_by: 'Fabriqué par'
coderhapsodie.ibexa_dataflow.dashboard: Dashboard
coderhapsodie.ibexa_dataflow.dashboard.title: Tâches en cours ou en attente

View File

@@ -0,0 +1,5 @@
{%- block content -%}
{% include '@ibexadesign/ibexa_dataflow/parts/tab/dashboard_list.html.twig' with {
identifier: 'ibexa_dataflow_schedule_dashboard_results',
} %}
{%- endblock -%}

File diff suppressed because one or more lines are too long

View File

@@ -0,0 +1,42 @@
{% set id = identifier|default('ibexa_dataflow_history_results') %}
{% import '@ibexadesign/ibexa_dataflow/macros.twig' as macros %}
<div id="loading_{{ id }}" class="text-center" hidden>
<svg class="ez-icon ez-icon--extra-large">
<use xmlns:xlink="http://www.w3.org/1999/xlink"
xlink:href="{{ ibexa_icon_path('spinner') }}"></use>
</svg>
</div>
<div id="{{ id }}" data-loader="loading_{{ id }}" class="history-details-aware">
{% set body_rows = [] %}
{% for job in jobs %}
{% set body_row_cols = [] %}
{% set body_row_cols = body_row_cols|merge([
{content: job.label},
{content: date(job.requested_date)|ibexa_short_datetime},
{content: job.start_time ? date(job.start_time)|ibexa_short_datetime : '—'},
{content: macros.translateStatus(job.status)},
]) %}
{% set body_rows = body_rows|merge([{ cols: body_row_cols }]) %}
{% endfor %}
{% embed '@ibexadesign/ui/component/table/table.html.twig' with {
headline: 'coderhapsodie.ibexa_dataflow.dashboard.title'|trans,
head_cols: [
{ content: 'coderhapsodie.ibexa_dataflow.history.list.name'|trans },
{ content: 'coderhapsodie.ibexa_dataflow.history.list.request'|trans },
{ content: 'coderhapsodie.ibexa_dataflow.history.list.start'|trans },
{ content: 'coderhapsodie.ibexa_dataflow.history.list.status'|trans },
{ },
],
body_rows,
empty_table_info_text: 'coderhapsodie.ibexa_dataflow.history.list.empty'|trans,
} %}
{% endembed %}
</div>

45
src/Tab/DashboardTab.php Normal file
View File

@@ -0,0 +1,45 @@
<?php
declare(strict_types=1);
namespace CodeRhapsodie\IbexaDataflowBundle\Tab;
use CodeRhapsodie\IbexaDataflowBundle\Controller\DashboardController;
use Ibexa\Contracts\AdminUi\Tab\AbstractControllerBasedTab;
use Ibexa\Contracts\AdminUi\Tab\OrderedTabInterface;
use Symfony\Component\HttpKernel\Controller\ControllerReference;
class DashboardTab extends AbstractControllerBasedTab implements OrderedTabInterface
{
/**
* {@inheritdoc}
*/
public function getControllerReference(array $parameters): ControllerReference
{
return new ControllerReference(DashboardController::class.'::dashboard');
}
/**
* {@inheritdoc}
*/
public function getOrder(): int
{
return 0;
}
/**
* {@inheritdoc}
*/
public function getIdentifier(): string
{
return 'code-rhapsodie-ibexa_dataflow-dashboard';
}
/**
* {@inheritdoc}
*/
public function getName(): string
{
return $this->translator->trans('coderhapsodie.ibexa_dataflow.dashboard');
}
}