AclTest.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502
  1. <?php
  2. namespace Symfony\Tests\Component\Security\Acl\Domain;
  3. use Symfony\Component\Security\Acl\Domain\UserSecurityIdentity;
  4. use Symfony\Component\Security\Acl\Domain\RoleSecurityIdentity;
  5. use Symfony\Component\Security\Acl\Domain\PermissionGrantingStrategy;
  6. use Symfony\Component\Security\Acl\Domain\ObjectIdentity;
  7. use Symfony\Component\Security\Acl\Domain\Acl;
  8. class AclTest extends \PHPUnit_Framework_TestCase
  9. {
  10. public function testConstructor()
  11. {
  12. $acl = new Acl(1, $oid = new ObjectIdentity('foo', 'foo'), $permissionStrategy = new PermissionGrantingStrategy(), array(), true);
  13. $this->assertSame(1, $acl->getId());
  14. $this->assertSame($oid, $acl->getObjectIdentity());
  15. $this->assertNull($acl->getParentAcl());
  16. $this->assertTrue($acl->isEntriesInheriting());
  17. }
  18. /**
  19. * @expectedException \OutOfBoundsException
  20. * @dataProvider getDeleteAceTests
  21. */
  22. public function testDeleteAceThrowsExceptionOnInvalidIndex($type)
  23. {
  24. $acl = $this->getAcl();
  25. $acl->{'delete'.$type.'Ace'}(0);
  26. }
  27. /**
  28. * @dataProvider getDeleteAceTests
  29. */
  30. public function testDeleteAce($type)
  31. {
  32. $acl = $this->getAcl();
  33. $acl->{'insert'.$type.'Ace'}(new RoleSecurityIdentity('foo'), 1);
  34. $acl->{'insert'.$type.'Ace'}(new RoleSecurityIdentity('foo'), 2, 1);
  35. $acl->{'insert'.$type.'Ace'}(new RoleSecurityIdentity('foo'), 3, 2);
  36. $listener = $this->getListener(array(
  37. $type.'Aces', 'aceOrder', 'aceOrder', $type.'Aces',
  38. ));
  39. $acl->addPropertyChangedListener($listener);
  40. $this->assertEquals(3, count($acl->{'get'.$type.'Aces'}()));
  41. $acl->{'delete'.$type.'Ace'}(0);
  42. $this->assertEquals(2, count($aces = $acl->{'get'.$type.'Aces'}()));
  43. $this->assertEquals(2, $aces[0]->getMask());
  44. $this->assertEquals(3, $aces[1]->getMask());
  45. $acl->{'delete'.$type.'Ace'}(1);
  46. $this->assertEquals(1, count($aces = $acl->{'get'.$type.'Aces'}()));
  47. $this->assertEquals(2, $aces[0]->getMask());
  48. }
  49. public function getDeleteAceTests()
  50. {
  51. return array(
  52. array('class'),
  53. array('object'),
  54. );
  55. }
  56. /**
  57. * @expectedException \OutOfBoundsException
  58. * @dataProvider getDeleteFieldAceTests
  59. */
  60. public function testDeleteFieldAceThrowsExceptionOnInvalidIndex($type)
  61. {
  62. $acl = $this->getAcl();
  63. $acl->{'delete'.$type.'Ace'}('foo', 0);
  64. }
  65. /**
  66. * @dataProvider getDeleteFieldAceTests
  67. */
  68. public function testDeleteFieldAce($type)
  69. {
  70. $acl = $this->getAcl();
  71. $acl->{'insert'.$type.'Ace'}('foo', new RoleSecurityIdentity('foo'), 1, 0);
  72. $acl->{'insert'.$type.'Ace'}('foo', new RoleSecurityIdentity('foo'), 2, 1);
  73. $acl->{'insert'.$type.'Ace'}('foo', new RoleSecurityIdentity('foo'), 3, 2);
  74. $listener = $this->getListener(array(
  75. $type.'Aces', 'aceOrder', 'aceOrder', $type.'Aces',
  76. ));
  77. $acl->addPropertyChangedListener($listener);
  78. $this->assertEquals(3, count($acl->{'get'.$type.'Aces'}('foo')));
  79. $acl->{'delete'.$type.'Ace'}(0, 'foo');
  80. $this->assertEquals(2, count($aces = $acl->{'get'.$type.'Aces'}('foo')));
  81. $this->assertEquals(2, $aces[0]->getMask());
  82. $this->assertEquals(3, $aces[1]->getMask());
  83. $acl->{'delete'.$type.'Ace'}(1, 'foo');
  84. $this->assertEquals(1, count($aces = $acl->{'get'.$type.'Aces'}('foo')));
  85. $this->assertEquals(2, $aces[0]->getMask());
  86. }
  87. public function getDeleteFieldAceTests()
  88. {
  89. return array(
  90. array('classField'),
  91. array('objectField'),
  92. );
  93. }
  94. /**
  95. * @dataProvider getInsertAceTests
  96. */
  97. public function testInsertAce($property, $method)
  98. {
  99. $acl = $this->getAcl();
  100. $listener = $this->getListener(array(
  101. $property, 'aceOrder', $property, 'aceOrder', $property
  102. ));
  103. $acl->addPropertyChangedListener($listener);
  104. $sid = new RoleSecurityIdentity('foo');
  105. $acl->$method($sid, 1);
  106. $acl->$method($sid, 2);
  107. $acl->$method($sid, 3, 1, false);
  108. $this->assertEquals(3, count($aces = $acl->{'get'.$property}()));
  109. $this->assertEquals(2, $aces[0]->getMask());
  110. $this->assertEquals(3, $aces[1]->getMask());
  111. $this->assertEquals(1, $aces[2]->getMask());
  112. }
  113. /**
  114. * @expectedException \OutOfBoundsException
  115. * @dataProvider getInsertAceTests
  116. */
  117. public function testInsertClassAceThrowsExceptionOnInvalidIndex($property, $method)
  118. {
  119. $acl = $this->getAcl();
  120. $acl->$method(new RoleSecurityIdentity('foo'), 1, 1);
  121. }
  122. public function getInsertAceTests()
  123. {
  124. return array(
  125. array('classAces', 'insertClassAce'),
  126. array('objectAces', 'insertObjectAce'),
  127. );
  128. }
  129. /**
  130. * @dataProvider getInsertFieldAceTests
  131. */
  132. public function testInsertClassFieldAce($property, $method)
  133. {
  134. $acl = $this->getAcl();
  135. $listener = $this->getListener(array(
  136. $property, $property, 'aceOrder', $property,
  137. 'aceOrder', 'aceOrder', $property,
  138. ));
  139. $acl->addPropertyChangedListener($listener);
  140. $sid = new RoleSecurityIdentity('foo');
  141. $acl->$method('foo', $sid, 1);
  142. $acl->$method('foo2', $sid, 1);
  143. $acl->$method('foo', $sid, 3);
  144. $acl->$method('foo', $sid, 2);
  145. $this->assertEquals(3, count($aces = $acl->{'get'.$property}('foo')));
  146. $this->assertEquals(1, count($acl->{'get'.$property}('foo2')));
  147. $this->assertEquals(2, $aces[0]->getMask());
  148. $this->assertEquals(3, $aces[1]->getMask());
  149. $this->assertEquals(1, $aces[2]->getMask());
  150. }
  151. /**
  152. * @expectedException \OutOfBoundsException
  153. * @dataProvider getInsertFieldAceTests
  154. */
  155. public function testInsertClassFieldAceThrowsExceptionOnInvalidIndex($property, $method)
  156. {
  157. $acl = $this->getAcl();
  158. $acl->$method('foo', new RoleSecurityIdentity('foo'), 1, 1);
  159. }
  160. public function getInsertFieldAceTests()
  161. {
  162. return array(
  163. array('classFieldAces', 'insertClassFieldAce'),
  164. array('objectFieldAces', 'insertObjectFieldAce'),
  165. );
  166. }
  167. public function testIsFieldGranted()
  168. {
  169. $sids = array(new RoleSecurityIdentity('ROLE_FOO'), new RoleSecurityIdentity('ROLE_IDDQD'));
  170. $masks = array(1, 2, 4);
  171. $strategy = $this->getMock('Symfony\Component\Security\Acl\Model\PermissionGrantingStrategyInterface');
  172. $acl = new Acl(1, new ObjectIdentity(1, 'foo'), $strategy, array(), true);
  173. $strategy
  174. ->expects($this->once())
  175. ->method('isFieldGranted')
  176. ->with($this->equalTo($acl), $this->equalTo('foo'), $this->equalTo($masks), $this->equalTo($sids), $this->isTrue())
  177. ->will($this->returnValue(true))
  178. ;
  179. $this->assertTrue($acl->isFieldGranted('foo', $masks, $sids, true));
  180. }
  181. public function testIsGranted()
  182. {
  183. $sids = array(new RoleSecurityIdentity('ROLE_FOO'), new RoleSecurityIdentity('ROLE_IDDQD'));
  184. $masks = array(1, 2, 4);
  185. $strategy = $this->getMock('Symfony\Component\Security\Acl\Model\PermissionGrantingStrategyInterface');
  186. $acl = new Acl(1, new ObjectIdentity(1, 'foo'), $strategy, array(), true);
  187. $strategy
  188. ->expects($this->once())
  189. ->method('isGranted')
  190. ->with($this->equalTo($acl), $this->equalTo($masks), $this->equalTo($sids), $this->isTrue())
  191. ->will($this->returnValue(true))
  192. ;
  193. $this->assertTrue($acl->isGranted($masks, $sids, true));
  194. }
  195. public function testSetGetParentAcl()
  196. {
  197. $acl = $this->getAcl();
  198. $parentAcl = $this->getAcl();
  199. $listener = $this->getListener(array('parentAcl'));
  200. $acl->addPropertyChangedListener($listener);
  201. $this->assertNull($acl->getParentAcl());
  202. $acl->setParentAcl($parentAcl);
  203. $this->assertSame($parentAcl, $acl->getParentAcl());
  204. }
  205. public function testSetIsEntriesInheriting()
  206. {
  207. $acl = $this->getAcl();
  208. $listener = $this->getListener(array('entriesInheriting'));
  209. $acl->addPropertyChangedListener($listener);
  210. $this->assertTrue($acl->isEntriesInheriting());
  211. $acl->setEntriesInheriting(false);
  212. $this->assertFalse($acl->isEntriesInheriting());
  213. }
  214. public function testIsSidLoadedWhenAllSidsAreLoaded()
  215. {
  216. $acl = $this->getAcl();
  217. $this->assertTrue($acl->isSidLoaded(new UserSecurityIdentity('foo', 'Foo')));
  218. $this->assertTrue($acl->isSidLoaded(new RoleSecurityIdentity('ROLE_FOO', 'Foo')));
  219. }
  220. public function testIsSidLoaded()
  221. {
  222. $acl = new Acl(1, new ObjectIdentity('1', 'foo'), new PermissionGrantingStrategy(), array(new UserSecurityIdentity('foo', 'Foo'), new UserSecurityIdentity('johannes', 'Bar')), true);
  223. $this->assertTrue($acl->isSidLoaded(new UserSecurityIdentity('foo', 'Foo')));
  224. $this->assertTrue($acl->isSidLoaded(new UserSecurityIdentity('johannes', 'Bar')));
  225. $this->assertTrue($acl->isSidLoaded(array(
  226. new UserSecurityIdentity('foo', 'Foo'),
  227. new UserSecurityIdentity('johannes', 'Bar'),
  228. )));
  229. $this->assertFalse($acl->isSidLoaded(new RoleSecurityIdentity('ROLE_FOO')));
  230. $this->assertFalse($acl->isSidLoaded(new UserSecurityIdentity('schmittjoh@gmail.com', 'Moo')));
  231. $this->assertFalse($acl->isSidLoaded(array(
  232. new UserSecurityIdentity('foo', 'Foo'),
  233. new UserSecurityIdentity('johannes', 'Bar'),
  234. new RoleSecurityIdentity('ROLE_FOO'),
  235. )));
  236. }
  237. /**
  238. * @dataProvider getUpdateAceTests
  239. * @expectedException \OutOfBoundsException
  240. */
  241. public function testUpdateAceThrowsOutOfBoundsExceptionOnInvalidIndex($type)
  242. {
  243. $acl = $this->getAcl();
  244. $acl->{'update'.$type}(0, 1);
  245. }
  246. /**
  247. * @dataProvider getUpdateAceTests
  248. */
  249. public function testUpdateAce($type)
  250. {
  251. $acl = $this->getAcl();
  252. $acl->{'insert'.$type}(new RoleSecurityIdentity('foo'), 1);
  253. $listener = $this->getListener(array(
  254. 'mask', 'mask', 'strategy',
  255. ));
  256. $acl->addPropertyChangedListener($listener);
  257. $aces = $acl->{'get'.$type.'s'}();
  258. $ace = reset($aces);
  259. $this->assertEquals(1, $ace->getMask());
  260. $this->assertEquals('all', $ace->getStrategy());
  261. $acl->{'update'.$type}(0, 3);
  262. $this->assertEquals(3, $ace->getMask());
  263. $this->assertEquals('all', $ace->getStrategy());
  264. $acl->{'update'.$type}(0, 1, 'foo');
  265. $this->assertEquals(1, $ace->getMask());
  266. $this->assertEquals('foo', $ace->getStrategy());
  267. }
  268. public function getUpdateAceTests()
  269. {
  270. return array(
  271. array('classAce'),
  272. array('objectAce'),
  273. );
  274. }
  275. /**
  276. * @dataProvider getUpdateFieldAceTests
  277. * @expectedException \OutOfBoundsException
  278. */
  279. public function testUpdateFieldAceThrowsExceptionOnInvalidIndex($type)
  280. {
  281. $acl = $this->getAcl();
  282. $acl->{'update'.$type}(0, 'foo', 1);
  283. }
  284. /**
  285. * @dataProvider getUpdateFieldAceTests
  286. */
  287. public function testUpdateFieldAce($type)
  288. {
  289. $acl = $this->getAcl();
  290. $acl->{'insert'.$type}('foo', new UserSecurityIdentity('foo', 'Foo'), 1);
  291. $listener = $this->getListener(array(
  292. 'mask', 'mask', 'strategy'
  293. ));
  294. $acl->addPropertyChangedListener($listener);
  295. $aces = $acl->{'get'.$type.'s'}('foo');
  296. $ace = reset($aces);
  297. $this->assertEquals(1, $ace->getMask());
  298. $this->assertEquals('all', $ace->getStrategy());
  299. $acl->{'update'.$type}(0, 'foo', 3);
  300. $this->assertEquals(3, $ace->getMask());
  301. $this->assertEquals('all', $ace->getStrategy());
  302. $acl->{'update'.$type}(0, 'foo', 1, 'foo');
  303. $this->assertEquals(1, $ace->getMask());
  304. $this->assertEquals('foo', $ace->getStrategy());
  305. }
  306. public function getUpdateFieldAceTests()
  307. {
  308. return array(
  309. array('classFieldAce'),
  310. array('objectFieldAce'),
  311. );
  312. }
  313. /**
  314. * @dataProvider getUpdateAuditingTests
  315. * @expectedException \OutOfBoundsException
  316. */
  317. public function testUpdateAuditingThrowsExceptionOnInvalidIndex($type)
  318. {
  319. $acl = $this->getAcl();
  320. $acl->{'update'.$type.'Auditing'}(0, true, false);
  321. }
  322. /**
  323. * @dataProvider getUpdateAuditingTests
  324. */
  325. public function testUpdateAuditing($type)
  326. {
  327. $acl = $this->getAcl();
  328. $acl->{'insert'.$type.'Ace'}(new RoleSecurityIdentity('foo'), 1);
  329. $listener = $this->getListener(array(
  330. 'auditFailure', 'auditSuccess', 'auditFailure',
  331. ));
  332. $acl->addPropertyChangedListener($listener);
  333. $aces = $acl->{'get'.$type.'Aces'}();
  334. $ace = reset($aces);
  335. $this->assertFalse($ace->isAuditSuccess());
  336. $this->assertFalse($ace->isAuditFailure());
  337. $acl->{'update'.$type.'Auditing'}(0, false, true);
  338. $this->assertFalse($ace->isAuditSuccess());
  339. $this->assertTrue($ace->isAuditFailure());
  340. $acl->{'update'.$type.'Auditing'}(0, true, false);
  341. $this->assertTrue($ace->isAuditSuccess());
  342. $this->assertFalse($ace->isAuditFailure());
  343. }
  344. public function getUpdateAuditingTests()
  345. {
  346. return array(
  347. array('class'),
  348. array('object'),
  349. );
  350. }
  351. /**
  352. * @expectedException \InvalidArgumentException
  353. * @dataProvider getUpdateFieldAuditingTests
  354. */
  355. public function testUpdateFieldAuditingthrowsExceptionOnInvalidField($type)
  356. {
  357. $acl = $this->getAcl();
  358. $acl->{'update'.$type.'Auditing'}(0, 'foo', true, true);
  359. }
  360. /**
  361. * @expectedException \OutOfBoundsException
  362. * @dataProvider getUpdateFieldAuditingTests
  363. */
  364. public function testUpdateFieldAuditingThrowsExceptionOnInvalidIndex($type)
  365. {
  366. $acl = $this->getAcl();
  367. $acl->{'insert'.$type.'Ace'}('foo', new RoleSecurityIdentity('foo'), 1);
  368. $acl->{'update'.$type.'Auditing'}(1, 'foo', true, false);
  369. }
  370. /**
  371. * @dataProvider getUpdateFieldAuditingTests
  372. */
  373. public function testUpdateFieldAuditing($type)
  374. {
  375. $acl = $this->getAcl();
  376. $acl->{'insert'.$type.'Ace'}('foo', new RoleSecurityIdentity('foo'), 1);
  377. $listener = $this->getListener(array(
  378. 'auditSuccess', 'auditSuccess', 'auditFailure',
  379. ));
  380. $acl->addPropertyChangedListener($listener);
  381. $aces = $acl->{'get'.$type.'Aces'}('foo');
  382. $ace = reset($aces);
  383. $this->assertFalse($ace->isAuditSuccess());
  384. $this->assertFalse($ace->isAuditFailure());
  385. $acl->{'update'.$type.'Auditing'}(0, 'foo', true, false);
  386. $this->assertTrue($ace->isAuditSuccess());
  387. $this->assertFalse($ace->isAuditFailure());
  388. $acl->{'update'.$type.'Auditing'}(0, 'foo', false, true);
  389. $this->assertFalse($ace->isAuditSuccess());
  390. $this->assertTrue($ace->isAuditFailure());
  391. }
  392. public function getUpdateFieldAuditingTests()
  393. {
  394. return array(
  395. array('classField'),
  396. array('objectField'),
  397. );
  398. }
  399. protected function getListener($expectedChanges)
  400. {
  401. $aceProperties = array('aceOrder', 'mask', 'strategy', 'auditSuccess', 'auditFailure');
  402. $listener = $this->getMock('Doctrine\Common\PropertyChangedListener');
  403. foreach ($expectedChanges as $index => $property) {
  404. if (in_array($property, $aceProperties)) {
  405. $class = 'Symfony\Component\Security\Acl\Domain\Entry';
  406. } else {
  407. $class = 'Symfony\Component\Security\Acl\Domain\Acl';
  408. }
  409. $listener
  410. ->expects($this->at($index))
  411. ->method('propertyChanged')
  412. ->with($this->isInstanceOf($class), $this->equalTo($property))
  413. ;
  414. }
  415. return $listener;
  416. }
  417. protected function getAcl()
  418. {
  419. return new Acl(1, new ObjectIdentity(1, 'foo'), new PermissionGrantingStrategy(), array(), true);
  420. }
  421. }