123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621 |
- <?php
- /**
- * SPDX-FileCopyrightText: 2020 Nextcloud GmbH and Nextcloud contributors
- * SPDX-FileCopyrightText: 2016 ownCloud, Inc.
- * SPDX-License-Identifier: AGPL-3.0-only
- */
- namespace OCA\DAV\Tests\unit\CalDAV;
- use OCA\DAV\CalDAV\BirthdayService;
- use OCA\DAV\CalDAV\CalDavBackend;
- use OCA\DAV\CalDAV\Calendar;
- use OCP\IConfig;
- use OCP\IL10N;
- use PHPUnit\Framework\MockObject\MockObject;
- use Psr\Log\LoggerInterface;
- use Sabre\DAV\PropPatch;
- use Sabre\VObject\Reader;
- use Test\TestCase;
- class CalendarTest extends TestCase {
- /** @var IL10N */
- protected $l10n;
- /** @var IConfig */
- protected $config;
- /** @var MockObject|LoggerInterface */
- protected $logger;
- protected function setUp(): void {
- parent::setUp();
- $this->l10n = $this->getMockBuilder(IL10N::class)
- ->disableOriginalConstructor()->getMock();
- $this->config = $this->createMock(IConfig::class);
- $this->logger = $this->createMock(LoggerInterface::class);
- $this->l10n
- ->expects($this->any())
- ->method('t')
- ->willReturnCallback(function ($text, $parameters = []) {
- return vsprintf($text, $parameters);
- });
- }
- public function testDelete(): void {
- /** @var MockObject | CalDavBackend $backend */
- $backend = $this->getMockBuilder(CalDavBackend::class)->disableOriginalConstructor()->getMock();
- $backend->expects($this->once())->method('updateShares');
- $backend->expects($this->any())->method('getShares')->willReturn([
- ['href' => 'principal:user2']
- ]);
- $calendarInfo = [
- '{http://owncloud.org/ns}owner-principal' => 'user1',
- 'principaluri' => 'user2',
- 'id' => 666,
- 'uri' => 'cal',
- ];
- $c = new Calendar($backend, $calendarInfo, $this->l10n, $this->config, $this->logger);
- $c->delete();
- }
- public function testDeleteFromGroup(): void {
- /** @var MockObject | CalDavBackend $backend */
- $backend = $this->getMockBuilder(CalDavBackend::class)->disableOriginalConstructor()->getMock();
- $backend->expects($this->once())->method('updateShares');
- $backend->expects($this->any())->method('getShares')->willReturn([
- ['href' => 'principal:group2']
- ]);
- $calendarInfo = [
- '{http://owncloud.org/ns}owner-principal' => 'user1',
- 'principaluri' => 'user2',
- 'id' => 666,
- 'uri' => 'cal',
- ];
- $c = new Calendar($backend, $calendarInfo, $this->l10n, $this->config, $this->logger);
- $c->delete();
- }
- public function testDeleteOwn(): void {
- /** @var MockObject | CalDavBackend $backend */
- $backend = $this->createMock(CalDavBackend::class);
- $backend->expects($this->never())->method('updateShares');
- $backend->expects($this->never())->method('getShares');
- $this->config->expects($this->never())->method('setUserValue');
- $backend->expects($this->once())->method('deleteCalendar')
- ->with(666);
- $calendarInfo = [
- '{http://owncloud.org/ns}owner-principal' => 'user1',
- 'principaluri' => 'user1',
- 'id' => 666,
- 'uri' => 'cal',
- ];
- $c = new Calendar($backend, $calendarInfo, $this->l10n, $this->config, $this->logger);
- $c->delete();
- }
- public function testDeleteBirthdayCalendar(): void {
- /** @var MockObject | CalDavBackend $backend */
- $backend = $this->createMock(CalDavBackend::class);
- $backend->expects($this->once())->method('deleteCalendar')
- ->with(666);
- $this->config->expects($this->once())
- ->method('setUserValue')
- ->with('user1', 'dav', 'generateBirthdayCalendar', 'no');
- $calendarInfo = [
- '{http://owncloud.org/ns}owner-principal' => 'principals/users/user1',
- 'principaluri' => 'principals/users/user1',
- 'id' => 666,
- 'uri' => 'contact_birthdays',
- ];
- $c = new Calendar($backend, $calendarInfo, $this->l10n, $this->config, $this->logger);
- $c->delete();
- }
- public function dataPropPatch() {
- return [
- ['user1', 'user2', [], true],
- ['user1', 'user2', [
- '{http://owncloud.org/ns}calendar-enabled' => true,
- ], true],
- ['user1', 'user2', [
- '{DAV:}displayname' => true,
- ], true],
- ['user1', 'user2', [
- '{DAV:}displayname' => true,
- '{http://owncloud.org/ns}calendar-enabled' => true,
- ], true],
- ['user1', 'user1', [], false],
- ['user1', 'user1', [
- '{http://owncloud.org/ns}calendar-enabled' => true,
- ], false],
- ['user1', 'user1', [
- '{DAV:}displayname' => true,
- ], false],
- ['user1', 'user1', [
- '{DAV:}displayname' => true,
- '{http://owncloud.org/ns}calendar-enabled' => true,
- ], false],
- ];
- }
- /**
- * @dataProvider dataPropPatch
- */
- public function testPropPatch($ownerPrincipal, $principalUri, $mutations, $shared): void {
- /** @var MockObject | CalDavBackend $backend */
- $backend = $this->getMockBuilder(CalDavBackend::class)->disableOriginalConstructor()->getMock();
- $calendarInfo = [
- '{http://owncloud.org/ns}owner-principal' => $ownerPrincipal,
- 'principaluri' => $principalUri,
- 'id' => 666,
- 'uri' => 'default'
- ];
- $c = new Calendar($backend, $calendarInfo, $this->l10n, $this->config, $this->logger);
- $propPatch = new PropPatch($mutations);
- if (!$shared) {
- $backend->expects($this->once())
- ->method('updateCalendar')
- ->with(666, $propPatch);
- }
- $c->propPatch($propPatch);
- $this->addToAssertionCount(1);
- }
- /**
- * @dataProvider providesReadOnlyInfo
- */
- public function testAcl($expectsWrite, $readOnlyValue, $hasOwnerSet, $uri = 'default'): void {
- /** @var MockObject | CalDavBackend $backend */
- $backend = $this->getMockBuilder(CalDavBackend::class)->disableOriginalConstructor()->getMock();
- $backend->expects($this->any())->method('applyShareAcl')->willReturnArgument(1);
- $calendarInfo = [
- 'principaluri' => 'user2',
- 'id' => 666,
- 'uri' => $uri
- ];
- if (!is_null($readOnlyValue)) {
- $calendarInfo['{http://owncloud.org/ns}read-only'] = $readOnlyValue;
- }
- if ($hasOwnerSet) {
- $calendarInfo['{http://owncloud.org/ns}owner-principal'] = 'user1';
- }
- $c = new Calendar($backend, $calendarInfo, $this->l10n, $this->config, $this->logger);
- $acl = $c->getACL();
- $childAcl = $c->getChildACL();
- $expectedAcl = [[
- 'privilege' => '{DAV:}read',
- 'principal' => $hasOwnerSet ? 'user1' : 'user2',
- 'protected' => true
- ], [
- 'privilege' => '{DAV:}read',
- 'principal' => ($hasOwnerSet ? 'user1' : 'user2') . '/calendar-proxy-write',
- 'protected' => true,
- ], [
- 'privilege' => '{DAV:}read',
- 'principal' => ($hasOwnerSet ? 'user1' : 'user2') . '/calendar-proxy-read',
- 'protected' => true,
- ]];
- if ($uri === BirthdayService::BIRTHDAY_CALENDAR_URI) {
- $expectedAcl[] = [
- 'privilege' => '{DAV:}write-properties',
- 'principal' => $hasOwnerSet ? 'user1' : 'user2',
- 'protected' => true
- ];
- $expectedAcl[] = [
- 'privilege' => '{DAV:}write-properties',
- 'principal' => ($hasOwnerSet ? 'user1' : 'user2') . '/calendar-proxy-write',
- 'protected' => true
- ];
- } else {
- $expectedAcl[] = [
- 'privilege' => '{DAV:}write',
- 'principal' => $hasOwnerSet ? 'user1' : 'user2',
- 'protected' => true
- ];
- $expectedAcl[] = [
- 'privilege' => '{DAV:}write',
- 'principal' => ($hasOwnerSet ? 'user1' : 'user2') . '/calendar-proxy-write',
- 'protected' => true
- ];
- }
- $expectedAcl[] = [
- 'privilege' => '{DAV:}write-properties',
- 'principal' => ($hasOwnerSet ? 'user1' : 'user2') . '/calendar-proxy-read',
- 'protected' => true
- ];
- if ($hasOwnerSet) {
- $expectedAcl[] = [
- 'privilege' => '{DAV:}read',
- 'principal' => 'user2',
- 'protected' => true
- ];
- if ($expectsWrite) {
- $expectedAcl[] = [
- 'privilege' => '{DAV:}write',
- 'principal' => 'user2',
- 'protected' => true
- ];
- } else {
- $expectedAcl[] = [
- 'privilege' => '{DAV:}write-properties',
- 'principal' => 'user2',
- 'protected' => true
- ];
- }
- }
- $this->assertEquals($expectedAcl, $acl);
- $this->assertEquals($expectedAcl, $childAcl);
- }
- public function providesReadOnlyInfo() {
- return [
- 'read-only property not set' => [true, null, true],
- 'read-only property is false' => [true, false, true],
- 'read-only property is true' => [false, true, true],
- 'read-only property not set and no owner' => [true, null, false],
- 'read-only property is false and no owner' => [true, false, false],
- 'read-only property is true and no owner' => [false, true, false],
- 'birthday calendar' => [false, false, false, BirthdayService::BIRTHDAY_CALENDAR_URI]
- ];
- }
- /**
- * @dataProvider providesConfidentialClassificationData
- * @param int $expectedChildren
- * @param bool $isShared
- */
- public function testPrivateClassification($expectedChildren, $isShared): void {
- $calObject0 = ['uri' => 'event-0', 'classification' => CalDavBackend::CLASSIFICATION_PUBLIC];
- $calObject1 = ['uri' => 'event-1', 'classification' => CalDavBackend::CLASSIFICATION_CONFIDENTIAL];
- $calObject2 = ['uri' => 'event-2', 'classification' => CalDavBackend::CLASSIFICATION_PRIVATE];
- /** @var MockObject | CalDavBackend $backend */
- $backend = $this->getMockBuilder(CalDavBackend::class)->disableOriginalConstructor()->getMock();
- $backend->expects($this->any())->method('getCalendarObjects')->willReturn([
- $calObject0, $calObject1, $calObject2
- ]);
- $backend->expects($this->any())->method('getMultipleCalendarObjects')
- ->with(666, ['event-0', 'event-1', 'event-2'])
- ->willReturn([
- $calObject0, $calObject1, $calObject2
- ]);
- $backend->expects($this->any())->method('getCalendarObject')
- ->willReturn($calObject2)->with(666, 'event-2');
- $backend->expects($this->any())->method('applyShareAcl')->willReturnArgument(1);
- $calendarInfo = [
- 'principaluri' => 'user2',
- 'id' => 666,
- 'uri' => 'cal',
- ];
- if ($isShared) {
- $calendarInfo['{http://owncloud.org/ns}owner-principal'] = 'user1';
- }
- $c = new Calendar($backend, $calendarInfo, $this->l10n, $this->config, $this->logger);
- $children = $c->getChildren();
- $this->assertEquals($expectedChildren, count($children));
- $children = $c->getMultipleChildren(['event-0', 'event-1', 'event-2']);
- $this->assertEquals($expectedChildren, count($children));
- $this->assertEquals(!$isShared, $c->childExists('event-2'));
- }
- /**
- * @dataProvider providesConfidentialClassificationData
- * @param int $expectedChildren
- * @param bool $isShared
- */
- public function testConfidentialClassification($expectedChildren, $isShared): void {
- $start = '20160609';
- $end = '20160610';
- $calData = <<<EOD
- BEGIN:VCALENDAR
- PRODID:-//ownCloud calendar v1.2.2
- BEGIN:VEVENT
- CREATED:20160602T133732
- DTSTAMP:20160602T133732
- LAST-MODIFIED:20160602T133732
- UID:wej2z68l9h
- SUMMARY:Test Event
- LOCATION:Somewhere ...
- ATTENDEE;ROLE=REQ-PARTICIPANT;PARTSTAT=NEEDS-ACTION;CUTYPE=INDIVIDUAL;CN=de
- epdiver:MAILTO:thomas.mueller@tmit.eu
- ORGANIZER;CN=deepdiver:MAILTO:thomas.mueller@tmit.eu
- DESCRIPTION:maybe ....
- DTSTART;TZID=Europe/Berlin;VALUE=DATE:$start
- DTEND;TZID=Europe/Berlin;VALUE=DATE:$end
- RRULE:FREQ=DAILY
- BEGIN:VALARM
- ACTION:AUDIO
- TRIGGER:-PT15M
- END:VALARM
- END:VEVENT
- BEGIN:VTIMEZONE
- TZID:Europe/Berlin
- BEGIN:DAYLIGHT
- DTSTART:19810329T020000
- RRULE:FREQ=YEARLY;BYMONTH=3;BYDAY=-1SU
- TZNAME:MESZ
- TZOFFSETFROM:+0100
- TZOFFSETTO:+0200
- END:DAYLIGHT
- BEGIN:STANDARD
- DTSTART:19961027T030000
- RRULE:FREQ=YEARLY;BYMONTH=10;BYDAY=-1SU
- TZNAME:MEZ
- TZOFFSETFROM:+0200
- TZOFFSETTO:+0100
- END:STANDARD
- END:VTIMEZONE
- END:VCALENDAR
- EOD;
- $calObject0 = ['uri' => 'event-0', 'classification' => CalDavBackend::CLASSIFICATION_PUBLIC];
- $calObject1 = ['uri' => 'event-1', 'classification' => CalDavBackend::CLASSIFICATION_CONFIDENTIAL, 'calendardata' => $calData];
- $calObject2 = ['uri' => 'event-2', 'classification' => CalDavBackend::CLASSIFICATION_PRIVATE];
- /** @var MockObject | CalDavBackend $backend */
- $backend = $this->getMockBuilder(CalDavBackend::class)->disableOriginalConstructor()->getMock();
- $backend->expects($this->any())->method('getCalendarObjects')->willReturn([
- $calObject0, $calObject1, $calObject2
- ]);
- $backend->expects($this->any())->method('getMultipleCalendarObjects')
- ->with(666, ['event-0', 'event-1', 'event-2'])
- ->willReturn([
- $calObject0, $calObject1, $calObject2
- ]);
- $backend->expects($this->any())->method('getCalendarObject')
- ->willReturn($calObject1)->with(666, 'event-1');
- $backend->expects($this->any())->method('applyShareAcl')->willReturnArgument(1);
- $calendarInfo = [
- '{http://owncloud.org/ns}owner-principal' => $isShared ? 'user1' : 'user2',
- 'principaluri' => 'user2',
- 'id' => 666,
- 'uri' => 'cal',
- ];
- $c = new Calendar($backend, $calendarInfo, $this->l10n, $this->config, $this->logger);
- $this->assertEquals(count($c->getChildren()), $expectedChildren);
- // test private event
- $privateEvent = $c->getChild('event-1');
- $calData = $privateEvent->get();
- $event = Reader::read($calData);
- $this->assertEquals($start, $event->VEVENT->DTSTART->getValue());
- $this->assertEquals($end, $event->VEVENT->DTEND->getValue());
- if ($isShared) {
- $this->assertEquals('Busy', $event->VEVENT->SUMMARY->getValue());
- $this->assertArrayNotHasKey('ATTENDEE', $event->VEVENT);
- $this->assertArrayNotHasKey('LOCATION', $event->VEVENT);
- $this->assertArrayNotHasKey('DESCRIPTION', $event->VEVENT);
- $this->assertArrayNotHasKey('ORGANIZER', $event->VEVENT);
- } else {
- $this->assertEquals('Test Event', $event->VEVENT->SUMMARY->getValue());
- }
- // Test l10n
- $l10n = $this->createMock(IL10N::class);
- if ($isShared) {
- $l10n->expects($this->once())
- ->method('t')
- ->with('Busy')
- ->willReturn("Translated busy");
- } else {
- $l10n->expects($this->never())
- ->method('t');
- }
- $c = new Calendar($backend, $calendarInfo, $l10n, $this->config, $this->logger);
- $calData = $c->getChild('event-1')->get();
- $event = Reader::read($calData);
- if ($isShared) {
- $this->assertEquals('Translated busy', $event->VEVENT->SUMMARY->getValue());
- } else {
- $this->assertEquals('Test Event', $event->VEVENT->SUMMARY->getValue());
- }
- }
- public function providesConfidentialClassificationData() {
- return [
- [3, false],
- [2, true]
- ];
- }
- public function testRemoveVAlarms() {
- $publicObjectData = <<<EOD
- BEGIN:VCALENDAR
- VERSION:2.0
- PRODID:-//Nextcloud calendar v1.5.6
- CALSCALE:GREGORIAN
- BEGIN:VEVENT
- CREATED:20171022T125130
- DTSTAMP:20171022T125130
- LAST-MODIFIED:20171022T125130
- UID:PPL24TH8UGOWE94XET87ER
- SUMMARY:Foo bar blub
- CLASS:PUBLIC
- STATUS:CONFIRMED
- DTSTART;VALUE=DATE:20171024
- DTEND;VALUE=DATE:20171025
- BEGIN:VALARM
- ACTION:AUDIO
- TRIGGER:-PT15M
- END:VALARM
- END:VEVENT
- END:VCALENDAR
- EOD;
- $confidentialObjectData = <<<EOD
- BEGIN:VCALENDAR
- VERSION:2.0
- PRODID:-//Nextcloud calendar v1.5.6
- CALSCALE:GREGORIAN
- BEGIN:VEVENT
- CREATED:20171022T125130
- DTSTAMP:20171022T125130
- LAST-MODIFIED:20171022T125130
- UID:PPL24TH8UGOWE94XET87ER
- SUMMARY:Foo bar blub
- CLASS:CONFIDENTIAL
- STATUS:CONFIRMED
- DTSTART;VALUE=DATE:20171024
- DTEND;VALUE=DATE:20171025
- BEGIN:VALARM
- ACTION:AUDIO
- TRIGGER:-PT15M
- END:VALARM
- END:VEVENT
- END:VCALENDAR
- EOD;
- $publicObjectDataWithoutVAlarm = <<<EOD
- BEGIN:VCALENDAR
- VERSION:2.0
- PRODID:-//Nextcloud calendar v1.5.6
- CALSCALE:GREGORIAN
- BEGIN:VEVENT
- CREATED:20171022T125130
- DTSTAMP:20171022T125130
- LAST-MODIFIED:20171022T125130
- UID:PPL24TH8UGOWE94XET87ER
- SUMMARY:Foo bar blub
- CLASS:PUBLIC
- STATUS:CONFIRMED
- DTSTART;VALUE=DATE:20171024
- DTEND;VALUE=DATE:20171025
- END:VEVENT
- END:VCALENDAR
- EOD;
- $confidentialObjectCleaned = <<<EOD
- BEGIN:VCALENDAR
- VERSION:2.0
- PRODID:-//Nextcloud calendar v1.5.6
- CALSCALE:GREGORIAN
- BEGIN:VEVENT
- CREATED:20171022T125130
- UID:PPL24TH8UGOWE94XET87ER
- SUMMARY:Busy
- CLASS:CONFIDENTIAL
- DTSTART;VALUE=DATE:20171024
- DTEND;VALUE=DATE:20171025
- END:VEVENT
- END:VCALENDAR
- EOD;
- $publicObject = ['uri' => 'event-0',
- 'classification' => CalDavBackend::CLASSIFICATION_PUBLIC,
- 'calendardata' => $publicObjectData];
- $confidentialObject = ['uri' => 'event-1',
- 'classification' => CalDavBackend::CLASSIFICATION_CONFIDENTIAL,
- 'calendardata' => $confidentialObjectData];
- /** @var MockObject | CalDavBackend $backend */
- $backend = $this->createMock(CalDavBackend::class);
- $backend->expects($this->any())
- ->method('getCalendarObjects')
- ->willReturn([$publicObject, $confidentialObject]);
- $backend->expects($this->any())
- ->method('getMultipleCalendarObjects')
- ->with(666, ['event-0', 'event-1'])
- ->willReturn([$publicObject, $confidentialObject]);
- $backend->expects($this->any())
- ->method('getCalendarObject')
- ->willReturnCallback(function ($cId, $uri) use ($publicObject, $confidentialObject) {
- switch ($uri) {
- case 'event-0':
- return $publicObject;
- case 'event-1':
- return $confidentialObject;
- default:
- throw new \Exception('unexpected uri');
- }
- });
- $backend->expects($this->any())
- ->method('applyShareAcl')
- ->willReturnArgument(1);
- $calendarInfoOwner = [
- '{http://owncloud.org/ns}owner-principal' => 'user1',
- 'principaluri' => 'user1',
- 'id' => 666,
- 'uri' => 'cal',
- ];
- $calendarInfoSharedRW = [
- '{http://owncloud.org/ns}owner-principal' => 'user1',
- 'principaluri' => 'user2',
- 'id' => 666,
- 'uri' => 'cal',
- ];
- $calendarInfoSharedRO = [
- '{http://owncloud.org/ns}owner-principal' => 'user1',
- '{http://owncloud.org/ns}read-only' => true,
- 'principaluri' => 'user2',
- 'id' => 666,
- 'uri' => 'cal',
- ];
- $ownerCalendar = new Calendar($backend, $calendarInfoOwner, $this->l10n, $this->config, $this->logger);
- $rwCalendar = new Calendar($backend, $calendarInfoSharedRW, $this->l10n, $this->config, $this->logger);
- $roCalendar = new Calendar($backend, $calendarInfoSharedRO, $this->l10n, $this->config, $this->logger);
- $this->assertCount(2, $ownerCalendar->getChildren());
- $this->assertCount(2, $rwCalendar->getChildren());
- $this->assertCount(2, $roCalendar->getChildren());
- // calendar data shall not be altered for the owner
- $this->assertEquals($ownerCalendar->getChild('event-0')->get(), $publicObjectData);
- $this->assertEquals($ownerCalendar->getChild('event-1')->get(), $confidentialObjectData);
- // valarms shall not be removed for read-write shares
- $this->assertEquals(
- $this->fixLinebreak($rwCalendar->getChild('event-0')->get()),
- $this->fixLinebreak($publicObjectData));
- $this->assertEquals(
- $this->fixLinebreak($rwCalendar->getChild('event-1')->get()),
- $this->fixLinebreak($confidentialObjectCleaned));
- // valarms shall be removed for read-only shares
- $this->assertEquals(
- $this->fixLinebreak($roCalendar->getChild('event-0')->get()),
- $this->fixLinebreak($publicObjectDataWithoutVAlarm));
- $this->assertEquals(
- $this->fixLinebreak($roCalendar->getChild('event-1')->get()),
- $this->fixLinebreak($confidentialObjectCleaned));
- }
- private function fixLinebreak($str) {
- return preg_replace('~(*BSR_ANYCRLF)\R~', "\r\n", $str);
- }
- }
|