server/apps/dav/lib/Provisioning/Apple/AppleProvisioningPlugin.php

270 lines
6.7 KiB
PHP
Raw Normal View History

<?php
/**
* @copyright 2018, Georg Ehrke <oc.list@georgehrke.com>
*
* @author Georg Ehrke <oc.list@georgehrke.com>
*
* @license GNU AGPL version 3 or any later version
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
namespace OCA\DAV\Provisioning\Apple;
use OCA\Theming\ThemingDefaults;
use OCP\IL10N;
use OCP\IRequest;
use OCP\IURLGenerator;
use OCP\IUserSession;
use Sabre\DAV\Server;
use Sabre\DAV\ServerPlugin;
use Sabre\DAV\UUIDUtil;
use Sabre\HTTP\RequestInterface;
use Sabre\HTTP\ResponseInterface;
class AppleProvisioningPlugin extends ServerPlugin {
/**
* @var Server
*/
protected $server;
/**
* @var IURLGenerator
*/
protected $urlGenerator;
/**
* @var IUserSession
*/
protected $userSession;
/**
* @var ThemingDefaults
*/
protected $themingDefaults;
/**
* @var IRequest
*/
protected $request;
/**
* @var IL10N
*/
protected $l10n;
/**
* @var \closure
*/
protected $uuidClosure;
/**
* AppleProvisioningPlugin constructor.
*
* @param IUserSession $userSession
* @param IURLGenerator $urlGenerator
* @param ThemingDefaults $themingDefaults
* @param IRequest $request
* @param IL10N $l10n
* @param \closure $uuidClosure
*/
public function __construct(IUserSession $userSession, IURLGenerator $urlGenerator,
ThemingDefaults $themingDefaults, IRequest $request,
IL10N $l10n, \closure $uuidClosure) {
$this->userSession = $userSession;
$this->urlGenerator = $urlGenerator;
$this->themingDefaults = $themingDefaults;
$this->request = $request;
$this->l10n = $l10n;
$this->uuidClosure = $uuidClosure;
}
/**
* @param Server $server
*/
public function initialize(Server $server) {
$this->server = $server;
$this->server->on('method:GET', [$this, 'httpGet'], 90);
}
/**
* @param RequestInterface $request
* @param ResponseInterface $response
* @return boolean
*/
public function httpGet(RequestInterface $request, ResponseInterface $response):bool {
if ($request->getPath() !== 'provisioning/' . AppleProvisioningNode::FILENAME) {
return true;
}
$user = $this->userSession->getUser();
if (!$user) {
return true;
}
$serverProtocol = $this->request->getServerProtocol();
$useSSL = ($serverProtocol === 'https');
if (!$useSSL) {
$response->setStatus(200);
$response->setHeader('Content-Type', 'text/plain; charset=utf-8');
$response->setBody($this->l10n->t('Your %s needs to be configured to use HTTPS in order to use CalDAV and CardDAV with iOS/macOS.', [$this->themingDefaults->getName()]));
return false;
}
$absoluteURL = $request->getAbsoluteUrl();
$parsedUrl = parse_url($absoluteURL);
if (isset($parsedUrl['port'])) {
$serverPort = (int) $parsedUrl['port'];
} else {
$serverPort = 443;
}
$server_url = $parsedUrl['host'];
$description = $this->themingDefaults->getName();
$userId = $user->getUID();
$reverseDomain = implode('.', array_reverse(explode('.', $parsedUrl['host'])));
$caldavUUID = call_user_func($this->uuidClosure);
$carddavUUID = call_user_func($this->uuidClosure);
$profileUUID = call_user_func($this->uuidClosure);
$caldavIdentifier = $reverseDomain . '.' . $caldavUUID;
$carddavIdentifier = $reverseDomain . '.' . $carddavUUID;
$profileIdentifier = $reverseDomain . '.' . $profileUUID;
$caldavDescription = $this->l10n->t('Configures a CalDAV account');
$caldavDisplayname = $description . ' CalDAV';
$carddavDescription = $this->l10n->t('Configures a CardDAV account');
$carddavDisplayname = $description . ' CardDAV';
$filename = $userId . '-' . AppleProvisioningNode::FILENAME;
$xmlSkeleton = $this->getTemplate();
$body = vsprintf($xmlSkeleton, array_map(function($v) {
return \htmlspecialchars($v, ENT_XML1, 'UTF-8');
}, [
$description,
$server_url,
$userId,
$serverPort,
$caldavDescription,
$caldavDisplayname,
$caldavIdentifier,
$caldavUUID,
$description,
$server_url,
$userId,
$serverPort,
$carddavDescription,
$carddavDisplayname,
$carddavIdentifier,
$carddavUUID,
$description,
$profileIdentifier,
$profileUUID
]
));
$response->setStatus(200);
$response->setHeader('Content-Disposition', 'attachment; filename="' . $filename . '"');
$response->setHeader('Content-Type', 'application/xml; charset=utf-8');
$response->setBody($body);
return false;
}
/**
* @return string
*/
private function getTemplate():string {
return <<<EOF
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>PayloadContent</key>
<array>
<dict>
<key>CalDAVAccountDescription</key>
<string>%s</string>
<key>CalDAVHostName</key>
<string>%s</string>
<key>CalDAVUsername</key>
<string>%s</string>
<key>CalDAVUseSSL</key>
<true/>
<key>CalDAVPort</key>
<integer>%s</integer>
<key>PayloadDescription</key>
<string>%s</string>
<key>PayloadDisplayName</key>
<string>%s</string>
<key>PayloadIdentifier</key>
<string>%s</string>
<key>PayloadType</key>
<string>com.apple.caldav.account</string>
<key>PayloadUUID</key>
<string>%s</string>
<key>PayloadVersion</key>
<integer>1</integer>
</dict>
<dict>
<key>CardDAVAccountDescription</key>
<string>%s</string>
<key>CardDAVHostName</key>
<string>%s</string>
<key>CardDAVUsername</key>
<string>%s</string>
<key>CardDAVUseSSL</key>
<true/>
<key>CardDAVPort</key>
<integer>%s</integer>
<key>PayloadDescription</key>
<string>%s</string>
<key>PayloadDisplayName</key>
<string>%s</string>
<key>PayloadIdentifier</key>
<string>%s</string>
<key>PayloadType</key>
<string>com.apple.carddav.account</string>
<key>PayloadUUID</key>
<string>%s</string>
<key>PayloadVersion</key>
<integer>1</integer>
</dict>
</array>
<key>PayloadDisplayName</key>
<string>%s</string>
<key>PayloadIdentifier</key>
<string>%s</string>
<key>PayloadRemovalDisallowed</key>
<false/>
<key>PayloadType</key>
<string>Configuration</string>
<key>PayloadUUID</key>
<string>%s</string>
<key>PayloadVersion</key>
<integer>1</integer>
</dict>
</plist>
EOF;
}
}