Issues (51)

tests/Acl/AclTest.php (1 issue)

Labels
Severity
1
<?php
2
3
declare(strict_types=1);
4
5
namespace EcodevTests\Felix\Acl;
6
7
use Ecodev\Felix\Acl\Acl;
8
use Ecodev\Felix\Acl\Assertion\IsMyself;
9
use Ecodev\Felix\Acl\MultipleRoles;
10
use Ecodev\Felix\Model\CurrentUser;
11
use EcodevTests\Felix\Blog\Model\User;
12
use Laminas\Permissions\Acl\Assertion\AssertionInterface;
13
use Laminas\Permissions\Acl\Resource\ResourceInterface;
14
use Laminas\Permissions\Acl\Role\RoleInterface;
15
use PHPUnit\Framework\TestCase;
16
17
final class AclTest extends TestCase
18
{
19
    protected function tearDown(): void
20
    {
21
        CurrentUser::set(null);
22
    }
23
24
    public function testIsCurrentUserAllowed(): void
25
    {
26
        $acl = new class() extends Acl {
27
            public function __construct()
28
            {
29
                parent::__construct();
30
                $user = $this->createModelResource(User::class);
31
                $this->addRole('anonymous');
32
                $this->addRole('member');
33
                $this->allow('member', [$user], ['update'], new IsMyself());
34
            }
35
        };
36
37
        $user = new User();
38
39
        $owner = new User();
40
        $owner->setName('sarah');
41
        CurrentUser::set($owner);
42
        $user->setOwner($owner);
43
44
        CurrentUser::set(null);
45
        self::assertFalse($acl->isCurrentUserAllowed($user, 'update'), 'anonymous cannot update');
46
        self::assertSame('Non-logged user with role anonymous is not allowed on resource "User#null" with privilege "update"', $acl->getLastDenialMessage());
47
48
        CurrentUser::set($owner);
49
        self::assertFalse($acl->isCurrentUserAllowed($user, 'update'), 'student cannot update even if owner');
50
        self::assertSame('User "sarah" with role member is not allowed on resource "User#null" with privilege "update" because it is not himself', $acl->getLastDenialMessage());
51
52
        $other = new User();
53
        $other->setName('john');
54
        CurrentUser::set($other);
55
        self::assertFalse($acl->isCurrentUserAllowed($user, 'update'), 'other user cannot update');
56
        self::assertSame('User "john" with role member is not allowed on resource "User#null" with privilege "update" because it is not himself', $acl->getLastDenialMessage());
57
58
        // Test again the first case to assert that reject reason does not leak from one assertion to the next
59
        CurrentUser::set(null);
60
        self::assertFalse($acl->isCurrentUserAllowed($user, 'update'), 'anonymous cannot update');
61
        self::assertSame('Non-logged user with role anonymous is not allowed on resource "User#null" with privilege "update"', $acl->getLastDenialMessage());
62
    }
63
64
    public function testMultipleReasons(): void
65
    {
66
        $acl = new class($this->createRejectAssertion()) extends Acl {
67
            public function __construct(
68
                private AssertionInterface $reject,
69
            ) {
70
                parent::__construct();
71
                $user = $this->createModelResource(User::class);
72
                $this->addRole('anonymous');
73
                $this->addRole('member', 'anonymous');
74
                $this->allow('anonymous', [$user], ['update'], $this->reject);
75
                $this->allow('member', [$user], ['update'], new IsMyself());
76
            }
77
        };
78
79
        $user = new User();
80
        $user->setName('sarah');
81
        CurrentUser::set($user);
82
83
        self::assertFalse($acl->isCurrentUserAllowed(new User(), 'update'), 'student cannot update even if user');
84
        $expected = <<<STRING
85
            User "sarah" with role member is not allowed on resource "User#null" with privilege "update" because:
86
87
            - it is not himself
88
            - mocked reason
89
            STRING;
90
        self::assertSame($expected, $acl->getLastDenialMessage());
91
    }
92
93
    public function testMultipleReasonsNotDuplicated(): void
94
    {
95
        $acl = new class($this->createRejectAssertion()) extends Acl {
96
            public function __construct(
97
                private AssertionInterface $reject,
98
            ) {
99
                parent::__construct();
100
                $user = $this->createModelResource(User::class);
101
                $this->addRole('anonymous');
102
                $this->addRole('member', 'anonymous');
103
                $this->allow('anonymous', [$user], ['update'], $this->reject);
104
                $this->allow('member', [$user], ['update'], $this->reject);
105
            }
106
        };
107
108
        $user = new User();
109
        $user->setName('sarah');
110
        CurrentUser::set($user);
111
112
        self::assertFalse($acl->isCurrentUserAllowed(new User(), 'update'), 'student cannot update even if user');
113
        $expected = <<<STRING
114
            User "sarah" with role member is not allowed on resource "User#null" with privilege "update" because mocked reason
115
            STRING;
116
        self::assertSame($expected, $acl->getLastDenialMessage());
117
    }
118
119
    public function testResourceCanBeStringToo(): void
120
    {
121
        $acl = new Acl();
122
        $acl->addRole('anonymous');
123
        $acl->addResource('my-resource');
124
        $acl->allow('anonymous', 'my-resource', ['update'], $this->createRejectAssertion());
125
126
        self::assertFalse($acl->isCurrentUserAllowed('my-resource', 'update'), 'student cannot update even if user');
127
        self::assertSame('Non-logged user with role anonymous is not allowed on resource "my-resource" with privilege "update" because mocked reason', $acl->getLastDenialMessage());
128
    }
129
130
    public function testMultipleRoles(): void
131
    {
132
        $acl = new class() extends Acl {
133
            public function __construct()
134
            {
135
                parent::__construct();
136
                $user = $this->createModelResource(User::class);
137
                $this->addRole('reader');
138
                $this->addRole('writer');
139
                $this->allow('writer', [$user], ['update']);
140
            }
141
        };
142
143
        CurrentUser::set(new User(new MultipleRoles()));
144
        self::assertFalse($acl->isCurrentUserAllowed(new User(), 'update'));
145
        self::assertSame('User "" with role [] is not allowed on resource "User#null" with privilege "update"', $acl->getLastDenialMessage());
146
147
        CurrentUser::set(new User(new MultipleRoles(['reader'])));
148
        self::assertFalse($acl->isCurrentUserAllowed(new User(), 'update'));
149
        self::assertSame('User "" with role [reader] is not allowed on resource "User#null" with privilege "update"', $acl->getLastDenialMessage());
150
151
        CurrentUser::set(new User(new MultipleRoles(['reader', 'writer'])));
152
        self::assertTrue($acl->isCurrentUserAllowed(new User(), 'update'));
153
        self::assertNull($acl->getLastDenialMessage());
154
155
        self::assertFalse($acl->isAllowed(new MultipleRoles(), User::class, 'update'));
156
        self::assertFalse($acl->isAllowed(new MultipleRoles(['reader']), User::class, 'update'));
157
        self::assertTrue($acl->isAllowed(new MultipleRoles(['reader', 'writer']), User::class, 'update'));
158
    }
159
160
    public function testIsTranslatable(): void
161
    {
162
        $acl = new Acl();
163
        $acl->addRole('my-role');
164
        $acl->addResource('my-resource');
165
        $acl->allow('my-role', 'my-resource', 'my-privilege');
166
167
        $acl->setTranslations(['my-resource' => 'translated-resource'], ['my-privilege' => 'translated-privilege']);
168
169
        self::assertSame([
170
            [
171
                'resource' => 'translated-resource',
172
                'privileges' => [
173
                    [
174
                        'privilege' => 'translated-privilege',
175
                        'allowed' => true,
176
                        'allowIf' => [],
177
                        'denyIf' => [],
178
                    ],
179
                ],
180
            ],
181
        ], $acl->show('my-role'));
182
183
        self::assertSame([
184
            [
185
                'resource' => 'my-resource',
186
                'privileges' => [
187
                    [
188
                        'privilege' => 'my-privilege',
189
                        'allowed' => true,
190
                        'allowIf' => [],
191
                        'denyIf' => [],
192
                    ],
193
                ],
194
            ],
195
        ], $acl->show('my-role', false));
196
    }
197
198
    public function testGetPrivilegesByResource(): void
199
    {
200
        $acl = new Acl();
201
        $acl->addRole('my-role');
202
        $acl->addResource('my-resource');
203
        $acl->allow('my-role', 'my-resource', 'my-privilege');
204
205
        self::assertSame([
206
            'my-resource' => [
207
                'my-privilege',
208
            ],
209
        ], $acl->getPrivilegesByResource());
210
    }
211
212
    public function testIncompleteTranslationWillThrowException(): void
213
    {
214
        $acl = new Acl();
215
        $acl->addRole('my-role');
216
        $acl->addResource('my-resource');
217
        $acl->allow('my-role', 'my-resource', 'my-privilege');
218
219
        $acl->setTranslations(['my-resource' => 'translated-resource'], ['typo-here' => 'translated-privilege']);
220
221
        $this->expectExceptionMessage('Was not marked as translatable: my-privilege');
222
        $acl->show('my-role');
223
    }
224
225
    private function createRejectAssertion(): AssertionInterface
226
    {
227
        return new class() implements AssertionInterface {
228
            /**
229
             * @param Acl $acl
230
             */
231
            public function assert(\Laminas\Permissions\Acl\Acl $acl, ?RoleInterface $role = null, ?ResourceInterface $resource = null, mixed $privilege = null)
232
            {
233
                return $acl->reject('mocked reason');
0 ignored issues
show
The method reject() does not exist on Laminas\Permissions\Acl\Acl. It seems like you code against a sub-type of Laminas\Permissions\Acl\Acl such as Ecodev\Felix\Acl\Acl. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

233
                return $acl->/** @scrutinizer ignore-call */ reject('mocked reason');
Loading history...
234
            }
235
        };
236
    }
237
}
238