mirror of
https://github.com/nextcloud/server.git
synced 2026-05-25 02:34:12 -04:00
Merge pull request #59970 from nextcloud/backport/59693/stable27
Some checks failed
Cypress / init (push) Has been cancelled
Lint php / php-lint (push) Has been cancelled
Node tests / versions (push) Has been cancelled
Node / node (push) Has been cancelled
Psalm static code analysis / static-code-analysis (push) Has been cancelled
Psalm static code analysis / static-code-analysis-security (push) Has been cancelled
Psalm static code analysis / static-code-analysis-ocp (push) Has been cancelled
Cypress / runner 1 (push) Has been cancelled
Cypress / runner 2 (push) Has been cancelled
Cypress / runner component (push) Has been cancelled
Cypress / cypress-summary (push) Has been cancelled
Lint php / php-lint-summary (push) Has been cancelled
Node tests / test (push) Has been cancelled
Node tests / jsunit (push) Has been cancelled
Node tests / handlebars (push) Has been cancelled
Some checks failed
Cypress / init (push) Has been cancelled
Lint php / php-lint (push) Has been cancelled
Node tests / versions (push) Has been cancelled
Node / node (push) Has been cancelled
Psalm static code analysis / static-code-analysis (push) Has been cancelled
Psalm static code analysis / static-code-analysis-security (push) Has been cancelled
Psalm static code analysis / static-code-analysis-ocp (push) Has been cancelled
Cypress / runner 1 (push) Has been cancelled
Cypress / runner 2 (push) Has been cancelled
Cypress / runner component (push) Has been cancelled
Cypress / cypress-summary (push) Has been cancelled
Lint php / php-lint-summary (push) Has been cancelled
Node tests / test (push) Has been cancelled
Node tests / jsunit (push) Has been cancelled
Node tests / handlebars (push) Has been cancelled
This commit is contained in:
commit
420596d1d4
7 changed files with 265 additions and 14 deletions
|
|
@ -66,6 +66,8 @@ return array(
|
|||
'OCA\\DAV\\CalDAV\\Outbox' => $baseDir . '/../lib/CalDAV/Outbox.php',
|
||||
'OCA\\DAV\\CalDAV\\Plugin' => $baseDir . '/../lib/CalDAV/Plugin.php',
|
||||
'OCA\\DAV\\CalDAV\\Principal\\Collection' => $baseDir . '/../lib/CalDAV/Principal/Collection.php',
|
||||
'OCA\\DAV\\CalDAV\\Principal\\ProxyRead' => $baseDir . '/../lib/CalDAV/Principal/ProxyRead.php',
|
||||
'OCA\\DAV\\CalDAV\\Principal\\ProxyWrite' => $baseDir . '/../lib/CalDAV/Principal/ProxyWrite.php',
|
||||
'OCA\\DAV\\CalDAV\\Principal\\User' => $baseDir . '/../lib/CalDAV/Principal/User.php',
|
||||
'OCA\\DAV\\CalDAV\\Proxy\\Proxy' => $baseDir . '/../lib/CalDAV/Proxy/Proxy.php',
|
||||
'OCA\\DAV\\CalDAV\\Proxy\\ProxyMapper' => $baseDir . '/../lib/CalDAV/Proxy/ProxyMapper.php',
|
||||
|
|
|
|||
|
|
@ -81,6 +81,8 @@ class ComposerStaticInitDAV
|
|||
'OCA\\DAV\\CalDAV\\Outbox' => __DIR__ . '/..' . '/../lib/CalDAV/Outbox.php',
|
||||
'OCA\\DAV\\CalDAV\\Plugin' => __DIR__ . '/..' . '/../lib/CalDAV/Plugin.php',
|
||||
'OCA\\DAV\\CalDAV\\Principal\\Collection' => __DIR__ . '/..' . '/../lib/CalDAV/Principal/Collection.php',
|
||||
'OCA\\DAV\\CalDAV\\Principal\\ProxyRead' => __DIR__ . '/..' . '/../lib/CalDAV/Principal/ProxyRead.php',
|
||||
'OCA\\DAV\\CalDAV\\Principal\\ProxyWrite' => __DIR__ . '/..' . '/../lib/CalDAV/Principal/ProxyWrite.php',
|
||||
'OCA\\DAV\\CalDAV\\Principal\\User' => __DIR__ . '/..' . '/../lib/CalDAV/Principal/User.php',
|
||||
'OCA\\DAV\\CalDAV\\Proxy\\Proxy' => __DIR__ . '/..' . '/../lib/CalDAV/Proxy/Proxy.php',
|
||||
'OCA\\DAV\\CalDAV\\Proxy\\ProxyMapper' => __DIR__ . '/..' . '/../lib/CalDAV/Proxy/ProxyMapper.php',
|
||||
|
|
|
|||
23
apps/dav/lib/CalDAV/Principal/ProxyRead.php
Normal file
23
apps/dav/lib/CalDAV/Principal/ProxyRead.php
Normal file
|
|
@ -0,0 +1,23 @@
|
|||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
/**
|
||||
* SPDX-FileCopyrightText: 2026 Nextcloud GmbH and Nextcloud contributors
|
||||
* SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
*/
|
||||
|
||||
namespace OCA\DAV\CalDAV\Principal;
|
||||
|
||||
use Sabre\DAVACL;
|
||||
|
||||
class ProxyRead extends \Sabre\CalDAV\Principal\ProxyRead implements DAVACL\IACL {
|
||||
use DAVACL\ACLTrait;
|
||||
|
||||
/**
|
||||
* @inheritDoc
|
||||
*/
|
||||
public function getOwner() {
|
||||
return $this->principalInfo['uri'];
|
||||
}
|
||||
}
|
||||
23
apps/dav/lib/CalDAV/Principal/ProxyWrite.php
Normal file
23
apps/dav/lib/CalDAV/Principal/ProxyWrite.php
Normal file
|
|
@ -0,0 +1,23 @@
|
|||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
/**
|
||||
* SPDX-FileCopyrightText: 2026 Nextcloud GmbH and Nextcloud contributors
|
||||
* SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
*/
|
||||
|
||||
namespace OCA\DAV\CalDAV\Principal;
|
||||
|
||||
use Sabre\DAVACL;
|
||||
|
||||
class ProxyWrite extends \Sabre\CalDAV\Principal\ProxyWrite implements DAVACL\IACL {
|
||||
use DAVACL\ACLTrait;
|
||||
|
||||
/**
|
||||
* @inheritDoc
|
||||
*/
|
||||
public function getOwner() {
|
||||
return $this->principalInfo['uri'];
|
||||
}
|
||||
}
|
||||
|
|
@ -51,4 +51,44 @@ class User extends \Sabre\CalDAV\Principal\User {
|
|||
];
|
||||
return $acl;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a specific child node, referenced by its name.
|
||||
*
|
||||
* @param string $name
|
||||
*
|
||||
* @return \Sabre\DAV\INode
|
||||
*/
|
||||
public function getChild($name) {
|
||||
$principal = $this->principalBackend->getPrincipalByPath($this->getPrincipalURL() . '/' . $name);
|
||||
if (!$principal) {
|
||||
throw new \Sabre\DAV\Exception\NotFound("Node with name $name was not found");
|
||||
}
|
||||
if ($name === 'calendar-proxy-read') {
|
||||
return new ProxyRead($this->principalBackend, $this->principalProperties);
|
||||
}
|
||||
|
||||
if ($name === 'calendar-proxy-write') {
|
||||
return new ProxyWrite($this->principalBackend, $this->principalProperties);
|
||||
}
|
||||
|
||||
throw new \Sabre\DAV\Exception\NotFound("Node with name $name was not found");
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an array with all the child nodes.
|
||||
*
|
||||
* @return \Sabre\DAV\INode[]
|
||||
*/
|
||||
public function getChildren() {
|
||||
$r = [];
|
||||
if ($this->principalBackend->getPrincipalByPath($this->getPrincipalURL() . '/calendar-proxy-read')) {
|
||||
$r[] = new ProxyRead($this->principalBackend, $this->principalProperties);
|
||||
}
|
||||
if ($this->principalBackend->getPrincipalByPath($this->getPrincipalURL() . '/calendar-proxy-write')) {
|
||||
$r[] = new ProxyWrite($this->principalBackend, $this->principalProperties);
|
||||
}
|
||||
|
||||
return $r;
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -61,21 +61,39 @@ class CalDavContext implements \Behat\Behat\Context\Context {
|
|||
|
||||
/** @AfterScenario */
|
||||
public function afterScenario() {
|
||||
$davUrl = $this->baseUrl. '/remote.php/dav/calendars/admin/MyCalendar';
|
||||
try {
|
||||
$this->client->delete(
|
||||
$davUrl,
|
||||
[
|
||||
'auth' => [
|
||||
'admin',
|
||||
'admin',
|
||||
],
|
||||
'headers' => [
|
||||
'X-NC-CalDAV-No-Trashbin' => '1',
|
||||
foreach (['MyCalendar', 'MyCalendar2'] as $calendarName) {
|
||||
try {
|
||||
$this->client->delete(
|
||||
$this->baseUrl . '/remote.php/dav/calendars/admin/' . $calendarName,
|
||||
[
|
||||
'auth' => ['admin', 'admin'],
|
||||
'headers' => ['X-NC-CalDAV-No-Trashbin' => '1'],
|
||||
]
|
||||
]
|
||||
);
|
||||
} catch (\GuzzleHttp\Exception\ClientException $e) {
|
||||
);
|
||||
} catch (\GuzzleHttp\Exception\ClientException $e) {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/** @AfterScenario @caldav-delegation */
|
||||
public function afterDelegationScenario() {
|
||||
foreach (['calendar-proxy-read', 'calendar-proxy-write'] as $proxyType) {
|
||||
try {
|
||||
$propPatch = new \Sabre\DAV\Xml\Request\PropPatch();
|
||||
$propPatch->properties = ['{DAV:}group-member-set' => new \Sabre\DAV\Xml\Property\Href([])];
|
||||
$xml = new \Sabre\Xml\Service();
|
||||
$body = $xml->write('{DAV:}propertyupdate', $propPatch, '/');
|
||||
$this->client->request(
|
||||
'PROPPATCH',
|
||||
$this->baseUrl . '/remote.php/dav/principals/users/admin/' . $proxyType,
|
||||
[
|
||||
'headers' => ['Content-Type' => 'application/xml; charset=UTF-8'],
|
||||
'body' => $body,
|
||||
'auth' => ['admin', 'admin'],
|
||||
]
|
||||
);
|
||||
} catch (\GuzzleHttp\Exception\ClientException $e) {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
@ -105,6 +123,80 @@ class CalDavContext implements \Behat\Behat\Context\Context {
|
|||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @Then The CalDAV response should contain a property :key
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function theCaldavResponseShouldContainAProperty(string $key): void {
|
||||
/** @var \Sabre\DAV\Xml\Response\MultiStatus $multiStatus */
|
||||
$multiStatus = $this->responseXml['value'];
|
||||
$responses = $multiStatus->getResponses()[0]->getResponseProperties();
|
||||
if (!isset($responses[200])) {
|
||||
throw new \Exception(
|
||||
sprintf(
|
||||
'Expected code 200 got [%s]',
|
||||
implode(',', array_keys($responses)),
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
$props = $responses[200];
|
||||
if (!array_key_exists($key, $props)) {
|
||||
throw new \Exception(
|
||||
sprintf(
|
||||
'Expected property %s in %s',
|
||||
$key,
|
||||
json_encode($props, JSON_PRETTY_PRINT),
|
||||
)
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @Then The CalDAV response should contain an href :href
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function theCaldavResponseShouldContainAnHref(string $href): void {
|
||||
/** @var \Sabre\DAV\Xml\Response\MultiStatus $multiStatus */
|
||||
$multiStatus = $this->responseXml['value'];
|
||||
foreach ($multiStatus->getResponses() as $response) {
|
||||
if ($response->getHref() === $href) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
throw new \Exception(
|
||||
sprintf(
|
||||
'Expected href %s not found in response',
|
||||
$href,
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* @Then The CalDAV response should be multi status
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function theCaldavResponseShouldBeMultiStatus(): void {
|
||||
if ($this->response->getStatusCode() !== 207) {
|
||||
throw new \Exception(
|
||||
sprintf(
|
||||
'Expected code 207 got %s',
|
||||
$this->response->getStatusCode()
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
$body = $this->response->getBody()->getContents();
|
||||
if ($body && substr($body, 0, 1) === '<') {
|
||||
$reader = new Sabre\Xml\Reader();
|
||||
$reader->xml($body);
|
||||
$reader->elementMap['{DAV:}multistatus'] = \Sabre\DAV\Xml\Response\MultiStatus::class;
|
||||
$reader->elementMap['{DAV:}response'] = \Sabre\DAV\Xml\Element\Response::class;
|
||||
$reader->elementMap['{urn:ietf:params:xml:ns:caldav}schedule-default-calendar-URL'] = \Sabre\DAV\Xml\Property\Href::class;
|
||||
$this->responseXml = $reader->parse();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @Then The CalDAV HTTP status code should be :code
|
||||
* @param int $code
|
||||
|
|
@ -258,4 +350,43 @@ class CalDavContext implements \Behat\Behat\Context\Context {
|
|||
$this->response = $e->getResponse();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @Given :user updates property :key to href :value of principal :principal on the endpoint :endpoint
|
||||
*/
|
||||
public function updatesHrefPropertyOfPrincipal(
|
||||
string $user,
|
||||
string $key,
|
||||
string $value,
|
||||
string $principal,
|
||||
string $endpoint,
|
||||
): void {
|
||||
$davUrl = $this->baseUrl . $endpoint . $principal;
|
||||
$password = ($user === 'admin') ? 'admin' : '123456';
|
||||
|
||||
$propPatch = new \Sabre\DAV\Xml\Request\PropPatch();
|
||||
$propPatch->properties = [$key => new \Sabre\DAV\Xml\Property\Href($value)];
|
||||
|
||||
$xml = new \Sabre\Xml\Service();
|
||||
$body = $xml->write('{DAV:}propertyupdate', $propPatch, '/');
|
||||
|
||||
try {
|
||||
$this->response = $this->client->request(
|
||||
'PROPPATCH',
|
||||
$davUrl,
|
||||
[
|
||||
'headers' => [
|
||||
'Content-Type' => 'application/xml; charset=UTF-8',
|
||||
],
|
||||
'body' => $body,
|
||||
'auth' => [
|
||||
$user,
|
||||
$password,
|
||||
],
|
||||
]
|
||||
);
|
||||
} catch (\GuzzleHttp\Exception\ClientException $e) {
|
||||
$this->response = $e->getResponse();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
|||
30
build/integration/features/caldav-delegation.feature
Normal file
30
build/integration/features/caldav-delegation.feature
Normal file
|
|
@ -0,0 +1,30 @@
|
|||
# SPDX-FileCopyrightText: 2026 Nextcloud GmbH and Nextcloud contributors
|
||||
# SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
Feature: calendar delegation
|
||||
Calendar delegation grants another user/principal control of a calendar account,
|
||||
including all calendars the delegator can access.
|
||||
|
||||
@caldav-delegation
|
||||
Scenario: admin grants user0 read access to her calendar account
|
||||
Given user "admin" exists
|
||||
And user "user0" exists
|
||||
When "admin" updates property "{DAV:}group-member-set" to href "/remote.php/dav/principals/users/user0" of principal "users/admin/calendar-proxy-read" on the endpoint "/remote.php/dav/principals/"
|
||||
Then The CalDAV response should be multi status
|
||||
And The CalDAV response should contain an href "/remote.php/dav/principals/users/admin/calendar-proxy-read"
|
||||
And The CalDAV response should contain a property "{DAV:}group-member-set"
|
||||
|
||||
@caldav-delegation
|
||||
Scenario: admin grants write access to her calendar account
|
||||
Given user "admin" exists
|
||||
And user "user0" exists
|
||||
When "admin" updates property "{DAV:}group-member-set" to href "/remote.php/dav/principals/users/user0" of principal "users/admin/calendar-proxy-write" on the endpoint "/remote.php/dav/principals/"
|
||||
Then The CalDAV response should be multi status
|
||||
And The CalDAV response should contain an href "/remote.php/dav/principals/users/admin/calendar-proxy-write"
|
||||
And The CalDAV response should contain a property "{DAV:}group-member-set"
|
||||
|
||||
Scenario: Admin cannot grant User1 access to User0's calendar account
|
||||
Given user "admin" exists
|
||||
And user "user0" exists
|
||||
And user "user1" exists
|
||||
When "admin" updates property "{DAV:}group-member-set" to href "/remote.php/dav/principals/users/user1" of principal "users/user0/calendar-proxy-write" on the endpoint "/remote.php/dav/principals/"
|
||||
Then The CalDAV HTTP status code should be "404"
|
||||
Loading…
Reference in a new issue