Issues (257)

src/Command/Logs/ShowEventLogCommand.php (1 issue)

1
<?php
2
/**
3
 * This file is part of Part-DB (https://github.com/Part-DB/Part-DB-symfony).
4
 *
5
 * Copyright (C) 2019 - 2022 Jan Böhmer (https://github.com/jbtronics)
6
 *
7
 * This program is free software: you can redistribute it and/or modify
8
 * it under the terms of the GNU Affero General Public License as published
9
 * by the Free Software Foundation, either version 3 of the License, or
10
 * (at your option) any later version.
11
 *
12
 * This program is distributed in the hope that it will be useful,
13
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15
 * GNU Affero General Public License for more details.
16
 *
17
 * You should have received a copy of the GNU Affero General Public License
18
 * along with this program.  If not, see <https://www.gnu.org/licenses/>.
19
 */
20
21
declare(strict_types=1);
22
23
namespace App\Command\Logs;
24
25
use App\Entity\Base\AbstractNamedDBElement;
26
use App\Entity\LogSystem\AbstractLogEntry;
27
use App\Repository\LogEntryRepository;
28
use App\Services\ElementTypeNameGenerator;
29
use App\Services\LogSystem\LogEntryExtraFormatter;
30
use Doctrine\ORM\EntityManagerInterface;
31
use Symfony\Component\Console\Command\Command;
32
use Symfony\Component\Console\Helper\Table;
33
use Symfony\Component\Console\Input\InputInterface;
34
use Symfony\Component\Console\Input\InputOption;
35
use Symfony\Component\Console\Output\OutputInterface;
36
use Symfony\Component\Console\Style\SymfonyStyle;
37
use Symfony\Contracts\Translation\TranslatorInterface;
38
39
class ShowEventLogCommand extends Command
40
{
41
    protected static $defaultName = 'partdb:logs:show|app:show-logs';
42
    protected EntityManagerInterface $entityManager;
43
    protected TranslatorInterface $translator;
44
    protected ElementTypeNameGenerator $elementTypeNameGenerator;
45
    protected LogEntryRepository $repo;
46
    protected LogEntryExtraFormatter $formatter;
47
48
    public function __construct(EntityManagerInterface $entityManager,
49
        TranslatorInterface $translator, ElementTypeNameGenerator $elementTypeNameGenerator, LogEntryExtraFormatter $formatter)
50
    {
51
        $this->entityManager = $entityManager;
52
        $this->translator = $translator;
53
        $this->elementTypeNameGenerator = $elementTypeNameGenerator;
54
        $this->formatter = $formatter;
55
56
        $this->repo = $this->entityManager->getRepository(AbstractLogEntry::class);
57
        parent::__construct();
58
    }
59
60
    public function execute(InputInterface $input, OutputInterface $output): int
61
    {
62
        $io = new SymfonyStyle($input, $output);
63
64
        $onePage = $input->getOption('onePage');
65
66
        $desc = (bool) $input->getOption('oldest_first');
67
        $limit = (int) $input->getOption('count');
68
        $page = (int) $input->getOption('page');
69
        $showExtra = $input->getOption('showExtra');
70
71
        $total_count = $this->repo->count([]);
72
        $max_page = (int) ceil($total_count / $limit);
73
74
        if ($page > $max_page && $max_page > 0) {
75
            $io->error("There is no page ${page}! The maximum page is ${max_page}.");
76
77
            return 1;
78
        }
79
80
        $io->note("There are a total of ${total_count} log entries in the DB.");
81
82
        $continue = true;
83
        while ($continue && $page <= $max_page) {
84
            $this->showPage($output, $desc, $limit, $page, $max_page, $showExtra);
85
86
            if ($onePage) {
87
                return 0;
88
            }
89
90
            $continue = $io->confirm('Do you want to show the next page?');
91
            ++$page;
92
        }
93
94
        return 0;
95
    }
96
97
    protected function configure(): void
98
    {
99
        $this
100
            ->setDescription('List the last event log entries.')
101
            ->addOption('count', 'c', InputOption::VALUE_REQUIRED, 'How many log entries should be shown per page.', 50)
102
            ->addOption('oldest_first', null, InputOption::VALUE_NONE, 'Show older entries first.')
103
            ->addOption('page', 'p', InputOption::VALUE_REQUIRED, 'Which page should be shown?', 1)
104
            ->addOption('onePage', null, InputOption::VALUE_NONE, 'Show only one page (dont ask to go to next).')
105
            ->addOption('showExtra', 'x', InputOption::VALUE_NONE, 'Show a column with the extra data.');
106
    }
107
108
    protected function showPage(OutputInterface $output, bool $desc, int $limit, int $page, int $max_page, bool $showExtra): void
109
    {
110
        $sorting = $desc ? 'ASC' : 'DESC';
111
        $offset = ($page - 1) * $limit;
112
113
        /** @var AbstractLogEntry[] $entries */
114
        $entries = $this->repo->getLogsOrderedByTimestamp($sorting, $limit, $offset);
115
116
        $table = new Table($output);
117
        $table->setHeaderTitle("Page ${page} / ${max_page}");
118
        $headers = ['ID', 'Timestamp', 'Type', 'User', 'Target Type', 'Target'];
119
        if ($showExtra) {
120
            $headers[] = 'Extra data';
121
            $table->setColumnMaxWidth(6, 50);
122
        }
123
        $table->setHeaders($headers);
124
125
        foreach ($entries as $entry) {
126
            $this->addTableRow($table, $entry, $showExtra);
127
        }
128
129
        $table->setColumnMaxWidth(3, 20);
130
        $table->setColumnMaxWidth(5, 30);
131
132
        $table->render();
133
    }
134
135
    protected function addTableRow(Table $table, AbstractLogEntry $entry, bool $showExtra): void
136
    {
137
        $target = $this->repo->getTargetElement($entry);
138
        $target_name = '';
139
        if ($target instanceof AbstractNamedDBElement) {
140
            $target_name = $target->getName().' <info>('.$target->getID().')</info>';
141
        } elseif ($entry->getTargetID()) {
0 ignored issues
show
Bug Best Practice introduced by
The expression $entry->getTargetID() of type integer|null is loosely compared to true; this is ambiguous if the integer can be 0. You might want to explicitly use !== null instead.

In PHP, under loose comparison (like ==, or !=, or switch conditions), values of different types might be equal.

For integer values, zero is a special case, in particular the following results might be unexpected:

0   == false // true
0   == null  // true
123 == false // false
123 == null  // false

// It is often better to use strict comparison
0 === false // false
0 === null  // false
Loading history...
142
            $target_name = '<info>('.$entry->getTargetID().')</info>';
143
        }
144
145
        $target_class = '';
146
        if (null !== $entry->getTargetClass()) {
147
            $target_class = $this->elementTypeNameGenerator->getLocalizedTypeLabel($entry->getTargetClass());
148
        }
149
150
        $row = [
151
            $entry->getID(),
152
            $entry->getTimestamp()->format('Y-m-d H:i:s'),
153
            $entry->getType(),
154
            $entry->getUser()->getFullName(true),
155
            $target_class,
156
            $target_name,
157
        ];
158
159
        if ($showExtra) {
160
            $row[] = $this->formatter->formatConsole($entry);
161
        }
162
163
        $table->addRow($row);
164
    }
165
}
166