Skip to content

Commit 66b78c1

Browse files
committed
Allow restricting reports by name & author
Co-uuthored-by: Jonada Hoxha <[email protected]>
1 parent 73dae07 commit 66b78c1

File tree

6 files changed

+93
-6
lines changed

6 files changed

+93
-6
lines changed

application/controllers/ReportController.php

+4-2
Original file line numberDiff line numberDiff line change
@@ -34,9 +34,11 @@ public function init()
3434

3535
$report = Model\Report::on($this->getDb())
3636
->with(['timeframe'])
37-
->filter(Filter::equal('id', $reportId))
38-
->first();
37+
->filter(Filter::equal('id', $reportId));
3938

39+
$this->applyRestrictions($report);
40+
41+
$report = $report->first();
4042
if ($report === null) {
4143
$this->httpNotFound($this->translate('Report not found'));
4244
}

application/controllers/ReportsController.php

+6-4
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,8 @@ public function indexAction()
4343
$reports = Report::on($this->getDb())
4444
->withColumns(['report.timeframe.name']);
4545

46+
$this->applyRestrictions($reports);
47+
4648
$sortControl = $this->createSortControl(
4749
$reports,
4850
[
@@ -64,16 +66,16 @@ public function indexAction()
6466
Html::tag('td', null, $report->timeframe->name),
6567
Html::tag('td', null, $report->ctime->format('Y-m-d H:i')),
6668
Html::tag('td', null, $report->mtime->format('Y-m-d H:i')),
67-
Html::tag('td', ['class' => 'icon-col'], [
68-
new Link(
69+
! $this->hasPermission('reporting/reports')
70+
? null
71+
: Html::tag('td', ['class' => 'icon-col'], new Link(
6972
new Icon('edit'),
7073
Url::fromPath('reporting/report/edit', ['id' => $report->id]),
7174
[
7275
'data-icinga-modal' => true,
7376
'data-no-icinga-ajax' => true
7477
]
75-
)
76-
])
78+
))
7779
]);
7880
}
7981

configuration.php

+5
Original file line numberDiff line numberDiff line change
@@ -50,4 +50,9 @@
5050
'reporting/timeframes',
5151
$this->translate('Allow managing timeframes')
5252
);
53+
54+
$this->provideRestriction(
55+
'reporting/filter/reports',
56+
$this->translate('Restrict access to the reports that match the provided filter')
57+
);
5358
}

doc/03-Configuration.md

+13
Original file line numberDiff line numberDiff line change
@@ -31,3 +31,16 @@ reporting/reports | Reports (create, edit, delete)
3131
reporting/schedules | Schedules (create, edit, delete)
3232
reporting/templates | Templates (create, edit, delete)
3333
reporting/timeframes | Timeframes (create, edit, delete)
34+
35+
## Restrictions
36+
37+
Icinga Reporting currently provides a single restriction that can be used to limit users to a specific set of reports,
38+
while having the `reporting/reports` permission.
39+
40+
> **Note:**
41+
>
42+
> Filters from multiple roles will expand the available access.
43+
44+
| Name | Description |
45+
|--------------------------|---------------------------------------------------------------|
46+
| reporting/filter/reports | Restrict access to the reports that match the provided filter |

library/Reporting/Common/Auth.php

+63
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
<?php
2+
3+
/* Icinga Reporting | (c) 2023 Icinga GmbH | GPLv2 */
4+
5+
namespace Icinga\Module\Reporting\Common;
6+
7+
use Icinga\Authentication\Auth as IcingaAuth;
8+
use Icinga\Exception\ConfigurationError;
9+
use ipl\Orm\Query;
10+
use ipl\Stdlib\Filter;
11+
use ipl\Web\Filter\QueryString;
12+
13+
trait Auth
14+
{
15+
/**
16+
* Apply restrictions of this module
17+
*
18+
* @param Query $query
19+
*/
20+
protected function applyRestrictions(Query $query): void
21+
{
22+
$restrictions = IcingaAuth::getInstance()->getRestrictions('reporting/filter/reports');
23+
24+
$queryFilter = Filter::any();
25+
foreach ($restrictions as $restriction) {
26+
$queryFilter->add($this->parseRestriction($restriction, 'reporting/filter/reports'));
27+
}
28+
29+
$query->filter($queryFilter);
30+
}
31+
32+
/**
33+
* Parse the query string of the given restriction
34+
*
35+
* @param string $queryString
36+
* @param string $restriction
37+
*
38+
* @return Filter\Rule
39+
*/
40+
protected function parseRestriction(string $queryString, string $restriction): Filter\Rule
41+
{
42+
return QueryString::fromString($queryString)
43+
->on(
44+
QueryString::ON_CONDITION,
45+
function (Filter\Condition $condition) use ($restriction, $queryString) {
46+
$allowedColumns = ['report.name', 'report.author'];
47+
if (in_array($condition->getColumn(), $allowedColumns, true)) {
48+
return;
49+
}
50+
51+
throw new ConfigurationError(
52+
t(
53+
'Cannot apply restriction %s using the filter %s.'
54+
. ' You can only use the following columns: %s'
55+
),
56+
$restriction,
57+
$queryString,
58+
implode(', ', $allowedColumns)
59+
);
60+
}
61+
)->parse();
62+
}
63+
}

library/Reporting/Web/Controller.php

+2
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,10 @@
44

55
namespace Icinga\Module\Reporting\Web;
66

7+
use Icinga\Module\Reporting\Common\Auth;
78
use ipl\Web\Compat\CompatController;
89

910
class Controller extends CompatController
1011
{
12+
use Auth;
1113
}

0 commit comments

Comments
 (0)