* @package calendar * @subpackage tests * @copyright (c) 2020 by Ralf Becker * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License */ namespace EGroupware\calendar; require_once __DIR__.'/../../../api/tests/CalDAVTest.php'; use EGroupware\Api\CalDAVTest; use GuzzleHttp\RequestOptions; use EGroupware\Api\Acl; /** * Class CalDAVsingleDELETE * * This tests check all sorts of DELETE requests by organizer and attendees, with and without (delete) rights on the organizer. * * For CalDAV Synchronizer, which does not distingues between deleting and rejecting events, we only allow the * organizer to delete an event. * * @package EGroupware\calendar * @covers \calendar_groupdav::delete() * @uses \calendar_groupdav::put() * @uses \calendar_groupdav::get() */ class CalDAVsingleDELETE extends CalDAVTest { /** * Users and their ACL for the test * * @var array */ protected $users = [ 'boss' => [], 'secretary' => [ 'rights' => [ 'boss' => Acl::READ|Acl::ADD|Acl::EDIT|Acl::DELETE, ] ], 'other' => [], ]; /** * Create some users incl. ACL */ public function setUp() { parent::setUp(); $this->createUsersACL($this->users, 'calendar'); } /** * Check created users */ public function testPrincipals() { foreach($this->users as $user => &$data) { $response = $this->getClient()->propfind($this->url('/principals/users/'.$user.'/'), [ RequestOptions::HEADERS => [ 'Depth' => 0, ], ]); $this->assertHttpStatus(207, $response); } } const EVENT_BOSS_ATTENDEE_ORGANIZER_URL = '/other/calendar/new-event-boss-attendee-123456789-new.ics'; const EVENT_BOSS_ATTENDEE_URL = '/boss/calendar/new-event-boss-attendee-123456789-new.ics'; const EVENT_BOSS_ATTENDEE_ICAL = <<getClient('other')->put($this->url(self::EVENT_BOSS_ATTENDEE_ORGANIZER_URL), [ RequestOptions::HEADERS => [ 'Content-Type' => 'text/calendar', 'Prefer' => 'return=representation' ], RequestOptions::BODY => self::EVENT_BOSS_ATTENDEE_ICAL, ]); $this->assertHttpStatus([200,201], $response); $this->assertIcal(self::EVENT_BOSS_ATTENDEE_ICAL, $response->getBody()); // secretrary deletes event in boss's calendar $response = $this->getClient('secretary')->delete($this->url(self::EVENT_BOSS_ATTENDEE_URL)); $this->assertHttpStatus(204, $response, 'Secretary delete/rejects for boss'); // use organizer to check event still exists and boss rejected $response = $this->getClient('other')->get($this->url(self::EVENT_BOSS_ATTENDEE_ORGANIZER_URL)); $this->assertHttpStatus(200, $response, 'Check event still exists after DELETE in attendee calendar'); $this->assertIcal(self::EVENT_BOSS_ATTENDEE_ICAL, $response->getBody(), 'Boss should have declined the invitation', ['vEvent' => [['ATTENDEE' => ['mailto:boss@example.org' => ['PARTSTAT' => 'DECLINED', 'RSVP' => 'FALSE']]]]] ); // secretary tries to delete event in organizers calendar $response = $this->getClient('secretary')->delete($this->url(self::EVENT_BOSS_ATTENDEE_ORGANIZER_URL)); $this->assertHttpStatus(403, $response, 'Secretary not allowed to delete for organizer'); // boss deletes/rejects event in his calendar $response = $this->getClient('boss')->delete($this->url(self::EVENT_BOSS_ATTENDEE_URL)); $this->assertHttpStatus(204, $response, 'Boss deletes/rejects in his calendar'); // boss deletes/rejects event in organizers calendar $response = $this->getClient('boss')->delete($this->url(self::EVENT_BOSS_ATTENDEE_ORGANIZER_URL)); $this->assertHttpStatus(204, $response, 'Boss deletes/rejects in organizers calendar'); // use organizer to delete event $response = $this->getClient('other')->delete($this->url(self::EVENT_BOSS_ATTENDEE_ORGANIZER_URL)); $this->assertHttpStatus(204, $response); // use organizer to check event deleted $response = $this->getClient('other')->get($this->url(self::EVENT_BOSS_ATTENDEE_ORGANIZER_URL)); $this->assertHttpStatus(404, $response, "Check event deleted by organizer"); } const EVENT_BOSS_ORGANIZER_URL = '/boss/calendar/new-event-boss-organizer-123456789-new.ics'; const EVENT_BOSS_ORGANIZER_OTHER_URL = '/other/calendar/new-event-boss-organizer-123456789-new.ics'; const EVENT_BOSS_ORGANIZER_ICAL = <<getClient('boss')->put($this->url(self::EVENT_BOSS_ORGANIZER_URL), [ RequestOptions::HEADERS => [ 'Content-Type' => 'text/calendar', 'Prefer' => 'return=representation' ], RequestOptions::BODY => self::EVENT_BOSS_ORGANIZER_ICAL, ]); $this->assertHttpStatus([200,201], $response); $this->assertIcal(self::EVENT_BOSS_ORGANIZER_ICAL, $response->getBody()); // attendee deletes/rejects event in his calendar $response = $this->getClient('other')->delete($this->url(self::EVENT_BOSS_ORGANIZER_OTHER_URL)); $this->assertHttpStatus(204, $response); // secretrary deletes event in boss's calendar $response = $this->getClient('secretary')->delete($this->url(self::EVENT_BOSS_ORGANIZER_URL)); $this->assertHttpStatus(204, $response, 'Secretary deletes for boss'); // use organizer/boss to check event deleted $response = $this->getClient('boss')->get($this->url(self::EVENT_BOSS_ORGANIZER_URL)); $this->assertHttpStatus(404, $response, "Check event deleted by secretary"); } /** * Check organizer (boss) can delete event in his calendar * * @throws \Horde_Icalendar_Exception */ public function testOrganizerDeletes() { // create invitation by boss as organizer $response = $this->getClient('boss')->put($this->url(self::EVENT_BOSS_ORGANIZER_URL), [ RequestOptions::HEADERS => [ 'Content-Type' => 'text/calendar', 'Prefer' => 'return=representation' ], RequestOptions::BODY => self::EVENT_BOSS_ORGANIZER_ICAL, ]); $this->assertHttpStatus([200,201], $response); $this->assertIcal(self::EVENT_BOSS_ORGANIZER_ICAL, $response->getBody()); // organizer deletes event in his calendar $response = $this->getClient('boss')->delete($this->url(self::EVENT_BOSS_ORGANIZER_URL)); $this->assertHttpStatus(204, $response, 'Organizer deletes'); // use organizer/boss to check event deleted $response = $this->getClient('boss')->get($this->url(self::EVENT_BOSS_ORGANIZER_URL)); $this->assertHttpStatus(404, $response, "Check event deleted by organizer"); // use attendee to check event deleted $response = $this->getClient('other')->get($this->url(self::EVENT_BOSS_ORGANIZER_OTHER_URL)); $this->assertHttpStatus(404, $response, "Check event deleted by organizer"); } const EVENT_SECRETARY_ATTENDEE_URL = '/secretary/calendar/new-event-secreatary-attendee-123456789-new.ics'; const EVENT_SECRETARY_ATTENDEE_ORGANIZER_URL = '/boss/calendar/new-event-secreatary-attendee-123456789-new.ics'; const EVENT_SECRETARY_ATTENDEE_ICAL = <<getClient('boss')->put($this->url(self::EVENT_SECRETARY_ATTENDEE_ORGANIZER_URL), [ RequestOptions::HEADERS => [ 'Content-Type' => 'text/calendar', 'Prefer' => 'return=representation' ], RequestOptions::BODY => self::EVENT_SECRETARY_ATTENDEE_ICAL, ]); $this->assertHttpStatus([200,201], $response); $this->assertIcal(self::EVENT_SECRETARY_ATTENDEE_ICAL, $response->getBody()); // secretary deletes in her calendar $response = $this->getClient('secretary')->delete($this->url(self::EVENT_SECRETARY_ATTENDEE_URL)); $this->assertHttpStatus(204, $response, 'Secretary (attendee) deletes'); // use organizer to check it's really deleted $response = $this->getClient('boss')->get($this->url(self::EVENT_SECRETARY_ATTENDEE_ORGANIZER_URL)); $this->assertHttpStatus(404, $response, "Check event deleted by secretary"); } /** * Check secretary as attendee deletes event with CalDAVSynchronizer * * @throws \Horde_Icalendar_Exception */ public function testSecretaryAttendeeDeletesCalDAVSynchronizer() { // create invitation by boss as organizer $response = $this->getClient('boss')->put($this->url(self::EVENT_SECRETARY_ATTENDEE_ORGANIZER_URL), [ RequestOptions::HEADERS => [ 'Content-Type' => 'text/calendar', 'Prefer' => 'return=representation' ], RequestOptions::BODY => self::EVENT_SECRETARY_ATTENDEE_ICAL, ]); $this->assertHttpStatus([200,201], $response); $this->assertIcal(self::EVENT_SECRETARY_ATTENDEE_ICAL, $response->getBody()); // secretary deletes in her calendar with CalDAVSynchronizer $response = $this->getClient('secretary')->delete($this->url(self::EVENT_SECRETARY_ATTENDEE_URL), [RequestOptions::HEADERS => ['User-Agent' => 'CalDAVSynchronizer']]); $this->assertHttpStatus(204, $response, 'Secretary (attendee) deletes/rejects'); // use organizer to check it's NOT deleted, as CalDAVSynchronizer / Outlook does not distinguish between reject and delete $response = $this->getClient('boss')->get($this->url(self::EVENT_SECRETARY_ATTENDEE_ORGANIZER_URL)); $this->assertHttpStatus(200, $response, "Check event NOT deleted by secretary"); $this->assertIcal(self::EVENT_SECRETARY_ATTENDEE_ICAL, $response->getBody(), 'Secretary should have declined the invitation', ['vEvent' => [['ATTENDEE' => ['mailto:secretary@example.org' => ['PARTSTAT' => 'DECLINED', 'RSVP' => 'FALSE']]]]] ); // organizer deletes in his calendar with CalDAVSynchronizer $response = $this->getClient('boss')->delete($this->url(self::EVENT_SECRETARY_ATTENDEE_ORGANIZER_URL), [RequestOptions::HEADERS => ['User-Agent' => 'CalDAVSynchronizer']]); $this->assertHttpStatus(204, $response, 'Organizer deletes'); // use organizer to check it's deleted, as CalDAVSynchronizer / Outlook should still delete for organizer $response = $this->getClient('boss')->get($this->url(self::EVENT_SECRETARY_ATTENDEE_ORGANIZER_URL)); $this->assertHttpStatus(404, $response, "Check event deleted by organizer"); } }