mirror of
https://github.com/EGroupware/egroupware.git
synced 2025-01-03 04:29:28 +01:00
WIP REST API for mail
currently we can launch (interactive) compose windows, if user is online ToDo: - send mails for a user - authentication as arbitrary user with an API token
This commit is contained in:
parent
26027796b3
commit
dfef4ce0c5
@ -50,6 +50,39 @@ egw.extend('links', egw.MODULE_GLOBAL, function()
|
|||||||
*/
|
*/
|
||||||
let title_uid = null;
|
let title_uid = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Encode query parameters
|
||||||
|
*
|
||||||
|
* @param object|array|string _values
|
||||||
|
* @param string? _prefix
|
||||||
|
* @param array? _query
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
function urlencode(_values, _prefix, _query)
|
||||||
|
{
|
||||||
|
if (typeof _query === 'undefined') _query = [];
|
||||||
|
if (Array.isArray(_values))
|
||||||
|
{
|
||||||
|
if (!_prefix) throw "array of value needs a prefix";
|
||||||
|
for(const value of _values)
|
||||||
|
{
|
||||||
|
_query.push(_prefix+'[]='+encodeURIComponent(value));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (_values && typeof _values === 'object')
|
||||||
|
{
|
||||||
|
for(const name in _values)
|
||||||
|
{
|
||||||
|
urlencode(_values[name], _prefix ? _prefix+'['+name+']' : name, _query);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_query.push(_prefix+'='+encodeURIComponent(_values || ''));
|
||||||
|
}
|
||||||
|
return _query;
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
/**
|
/**
|
||||||
* Check if $app is in the registry and has an entry for $name
|
* Check if $app is in the registry and has an entry for $name
|
||||||
@ -339,25 +372,7 @@ egw.extend('links', egw.MODULE_GLOBAL, function()
|
|||||||
}
|
}
|
||||||
|
|
||||||
// if there are vars, we add them urlencoded to the url
|
// if there are vars, we add them urlencoded to the url
|
||||||
let query = [];
|
return Object.keys(vars).length ? _url+'?'+urlencode(vars).join('&') : _url;
|
||||||
|
|
||||||
for(let name in vars)
|
|
||||||
{
|
|
||||||
let val = vars[name] || ''; // fix error for eg. null, which is an object!
|
|
||||||
if (typeof val == 'object')
|
|
||||||
{
|
|
||||||
for(let i=0; i < val.length; ++i)
|
|
||||||
{
|
|
||||||
query.push(name+'[]='+encodeURIComponent(val[i]));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
query.push(name+'='+encodeURIComponent(val));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return query.length ? _url+'?'+query.join('&') : _url;
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -277,6 +277,20 @@ class CalDAV extends HTTP_WebDAV_Server
|
|||||||
$this->dav_powered_by = str_replace('EGroupware','EGroupware '.$GLOBALS['egw_info']['server']['versions']['phpgwapi'],
|
$this->dav_powered_by = str_replace('EGroupware','EGroupware '.$GLOBALS['egw_info']['server']['versions']['phpgwapi'],
|
||||||
$this->dav_powered_by);
|
$this->dav_powered_by);
|
||||||
|
|
||||||
|
// detected available additional APIs from applications
|
||||||
|
$this->root += Cache::getInstance(__CLASS__, 'user-'.$GLOBALS['egw_info']['user']['account_id'], static function()
|
||||||
|
{
|
||||||
|
$apis = [];
|
||||||
|
foreach($GLOBALS['egw_info']['user']['apps'] as $app => $data)
|
||||||
|
{
|
||||||
|
if (class_exists('EGroupware\\'.ucfirst($app).'\\ApiHandler'))
|
||||||
|
{
|
||||||
|
$apis[$app] = [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $apis;
|
||||||
|
}, [], 86400);
|
||||||
|
|
||||||
parent::__construct();
|
parent::__construct();
|
||||||
// hack to allow to use query parameters in WebDAV, which HTTP_WebDAV_Server interprets as part of the path
|
// hack to allow to use query parameters in WebDAV, which HTTP_WebDAV_Server interprets as part of the path
|
||||||
list($this->_SERVER['REQUEST_URI']) = explode('?',$this->_SERVER['REQUEST_URI']);
|
list($this->_SERVER['REQUEST_URI']) = explode('?',$this->_SERVER['REQUEST_URI']);
|
||||||
|
@ -414,8 +414,12 @@ abstract class Handler
|
|||||||
|
|
||||||
if (!array_key_exists($app,$handler_cache))
|
if (!array_key_exists($app,$handler_cache))
|
||||||
{
|
{
|
||||||
$class = $app.'_groupdav';
|
if (!class_exists($class='EGroupware\\'.ucfirst($app).'\\ApiHandler') &&
|
||||||
if (!class_exists($class) && !class_exists($class = __NAMESPACE__.'\\'.ucfirst($app))) return null;
|
!class_exists($class=$app.'_groupdav') &&
|
||||||
|
!class_exists($class=__NAMESPACE__.'\\'.ucfirst($app)))
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
$handler_cache[$app] = new $class($app, $groupdav);
|
$handler_cache[$app] = new $class($app, $groupdav);
|
||||||
}
|
}
|
||||||
|
@ -1570,26 +1570,35 @@ class Session
|
|||||||
// if there are vars, we add them urlencoded to the url
|
// if there are vars, we add them urlencoded to the url
|
||||||
if (count($vars))
|
if (count($vars))
|
||||||
{
|
{
|
||||||
$query = array();
|
$ret_url .= '?' . implode('&', self::urlencode($vars));
|
||||||
foreach($vars as $key => $value)
|
|
||||||
{
|
|
||||||
if (is_array($value))
|
|
||||||
{
|
|
||||||
foreach($value as $val)
|
|
||||||
{
|
|
||||||
$query[] = $key.'[]='.urlencode($val);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
$query[] = $key.'='.urlencode($value ?? '');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$ret_url .= '?' . implode('&',$query);
|
|
||||||
}
|
}
|
||||||
return $ret_url;
|
return $ret_url;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Recursively encode GET parameters
|
||||||
|
*
|
||||||
|
* @param array|string $values
|
||||||
|
* @param string $prefix
|
||||||
|
* @param array& $query
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
protected static function urlencode($values, string $prefix='', array &$query=[])
|
||||||
|
{
|
||||||
|
if (is_array($values))
|
||||||
|
{
|
||||||
|
foreach($values as $name => $value)
|
||||||
|
{
|
||||||
|
self::urlencode($value, $prefix ? $prefix.'['.(is_int($name) ? '' : $name).']' : $name, $query);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$query[] = $prefix.'='.urlencode($values);
|
||||||
|
}
|
||||||
|
return $query;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Regexp to validate IPv4 and IPv6
|
* Regexp to validate IPv4 and IPv6
|
||||||
*/
|
*/
|
||||||
|
124
doc/REST-CalDAV-CardDAV/Mail.md
Normal file
124
doc/REST-CalDAV-CardDAV/Mail.md
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
# EGroupware REST API for Mail
|
||||||
|
|
||||||
|
> Currently only sending mail or launching interactive compose windows
|
||||||
|
|
||||||
|
Implemented requests (relative to https://example.org/egroupware/groupdav.php)
|
||||||
|
|
||||||
|
- ```GET /mail``` get different mail accounts available to user
|
||||||
|
<details>
|
||||||
|
<summary>Example: Querying available identities / signatures</summary>
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -i https://example.org/egroupware/mail --user <user> -H 'Accept: application/json'
|
||||||
|
HTTP/1.1 200 OK
|
||||||
|
Content-Type: application/json; charset=utf-8
|
||||||
|
|
||||||
|
{
|
||||||
|
"responses": {
|
||||||
|
"/ralf/mail/1": "Ralf Becker boulder.egroupware.org <ralf@boulder.egroupware.org>",
|
||||||
|
"/ralf/mail/52": "Ralf Becker <sysop@testbox.egroupware.org>",
|
||||||
|
"/ralf/mail/85": "Ralf Becker <RalfBeckerKL@gmail.com>"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
</details>
|
||||||
|
|
||||||
|
- ```POST /mail[/<id>]``` send mail for default or given account <id>
|
||||||
|
<details>
|
||||||
|
<summary>Example: Sending mail</summary>
|
||||||
|
|
||||||
|
The content of the POST request is a JSON encoded object with following attributes
|
||||||
|
- ```to```: array of strings with (RFC882) email addresses like ```["info@egroupware.org", "Ralf Becker <rb@egroupware.org"]```
|
||||||
|
- ```cc```: array of strings with (RFC882) email addresses (optional)
|
||||||
|
- ```bcc```: array of strings with (RFC882) email addresses (optional)
|
||||||
|
- ```replyTo```: string with (RFC822) email address (optional)
|
||||||
|
- ```subject```: string with subject
|
||||||
|
- ```body```: string plain text body (optional)
|
||||||
|
- ```bodyHtml```: string with html body (optional)
|
||||||
|
- ```attachments```: array of strings returned from uploaded attachments (see below) or VFS path ```["/mail/attachment/<token>", "/home/<user>/<filename>", ...]```
|
||||||
|
- ```attachmentType```: one of the following strings (optional, default "attach")
|
||||||
|
- "attach" send as attachment
|
||||||
|
- "link" send as sharing link
|
||||||
|
- "share_ro" send a readonly share using the current file content (VFS only)
|
||||||
|
- "share_rw" send as writable share (VFS and EPL only)
|
||||||
|
- ```shareExpiration```: "yyyy-mm-dd", default not accessed in 100 days (EPL only)
|
||||||
|
- ```sharePassword```: string with password required to access share, default none (EPL only)
|
||||||
|
- ```folder```: folder to store send mail, default Sent folder
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -i https://example.org/egroupware/mail --user <user> \
|
||||||
|
-X POST -H 'Content-Type: application/json' \
|
||||||
|
--content `{"to":["info@egroupware.org"],"subject":"Testmail","body":"This is a test :)\n\nRegards"}`
|
||||||
|
HTTP/1.1 204 No Content
|
||||||
|
```
|
||||||
|
If you are not authenticated you will get:
|
||||||
|
```
|
||||||
|
HTTP/1.1 401 Unauthorized
|
||||||
|
WWW-Authenticate: Basic realm="EGroupware CalDAV/CardDAV/GroupDAV server"
|
||||||
|
X-WebDAV-Status: 401 Unauthorized
|
||||||
|
```
|
||||||
|
If there is an error sending the mail you will get:
|
||||||
|
```
|
||||||
|
HTTP/1.1 500 Internal Server Error
|
||||||
|
Content-Type: application/json
|
||||||
|
Content-Length: ...
|
||||||
|
|
||||||
|
{"error": 123,"message":"SMTP Server not reachable"}
|
||||||
|
```
|
||||||
|
</details>
|
||||||
|
|
||||||
|
- ```POST /mail[/<id>]/compose``` launch compose window
|
||||||
|
<details>
|
||||||
|
<summary>Example: Opening a compose window</summary>
|
||||||
|
|
||||||
|
Parameters are identical to send mail request above, thought there are additional responses:
|
||||||
|
- compose window successful opened
|
||||||
|
```
|
||||||
|
HTTP/1.1 200 OK
|
||||||
|
Content-Type: application/json
|
||||||
|
|
||||||
|
{
|
||||||
|
"status": 200,
|
||||||
|
"message": "Request to open compose window sent",
|
||||||
|
"extra": {
|
||||||
|
"preset": {
|
||||||
|
"to": [
|
||||||
|
"Birgit Becker <bb@egroupware.org"
|
||||||
|
],
|
||||||
|
"cc": [
|
||||||
|
"info@egroupware.org"
|
||||||
|
],
|
||||||
|
"subject": "Testmail",
|
||||||
|
"body": "<pre>This is a test :)\n\nRegards</pre>",
|
||||||
|
"mimeType": "html",
|
||||||
|
"identity": "52"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
- user is not online, therefore compose window can NOT be opened
|
||||||
|
```
|
||||||
|
404 Not found
|
||||||
|
Content-Type: application/json
|
||||||
|
|
||||||
|
{
|
||||||
|
"error": 404,
|
||||||
|
"message": "User 'ralf' (#5) is NOT online"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
</details>
|
||||||
|
|
||||||
|
- ```POST /mail/attachments/<filename>``` upload mail attachments
|
||||||
|
<details>
|
||||||
|
<summary>Example: Uploading an attachment to be used for sending or composing mail</summary>
|
||||||
|
|
||||||
|
The content of the POST request is the attachment, a Location header in the response gives you a URL
|
||||||
|
to use in further requests, instead of the attachment.
|
||||||
|
|
||||||
|
```
|
||||||
|
curl -i https://example.org/egroupware/mail/attachment/<filename> --user <user> \
|
||||||
|
--data-binary @<file> -H 'Content-Type: <content-type-of-file>'
|
||||||
|
HTTP/1.1 204 No Content
|
||||||
|
Location: https://example.org/egroupware/mail/attachment/<token>
|
||||||
|
```
|
||||||
|
</details>
|
@ -31,6 +31,7 @@ One can use the following URLs relative (!) to https://example.org/egroupware/gr
|
|||||||
- ```/infolog/``` infologs of current user
|
- ```/infolog/``` infologs of current user
|
||||||
- ```/(resources|locations)/<resource-name>/calendar``` calendar of a resource/location, if user has rights to view
|
- ```/(resources|locations)/<resource-name>/calendar``` calendar of a resource/location, if user has rights to view
|
||||||
- ```/<current-username>/(resource|location)-<resource-name>``` shared calendar from a resource/location
|
- ```/<current-username>/(resource|location)-<resource-name>``` shared calendar from a resource/location
|
||||||
|
- ```/mail/``` only REST API, currently only send EMail or launch interactive compose windows
|
||||||
|
|
||||||
Shared addressbooks or calendars are only shown in the users home-set, if he subscribed to it via his CalDAV preferences!
|
Shared addressbooks or calendars are only shown in the users home-set, if he subscribed to it via his CalDAV preferences!
|
||||||
|
|
||||||
@ -544,4 +545,4 @@ use ```<domain-name>:<name>``` like in JsCalendar
|
|||||||
- [ ] InfoLog
|
- [ ] InfoLog
|
||||||
- [ ] Calendar
|
- [ ] Calendar
|
||||||
- [ ] relatedTo / links
|
- [ ] relatedTo / links
|
||||||
- [ ] storing not native supported attributes eg. localization
|
- [ ] storing not native supported attributes eg. localization
|
@ -68,11 +68,11 @@ class mail_compose
|
|||||||
var $composeID;
|
var $composeID;
|
||||||
var $sessionData;
|
var $sessionData;
|
||||||
|
|
||||||
function __construct()
|
function __construct(int $_acc_id=null)
|
||||||
{
|
{
|
||||||
$this->displayCharset = Api\Translation::charset();
|
$this->displayCharset = Api\Translation::charset();
|
||||||
|
|
||||||
$profileID = (int)$GLOBALS['egw_info']['user']['preferences']['mail']['ActiveProfileID'];
|
$profileID = $_acc_id ?: (int)$GLOBALS['egw_info']['user']['preferences']['mail']['ActiveProfileID'];
|
||||||
$this->mail_bo = Mail::getInstance(true,$profileID);
|
$this->mail_bo = Mail::getInstance(true,$profileID);
|
||||||
$GLOBALS['egw_info']['user']['preferences']['mail']['ActiveProfileID'] = $this->mail_bo->profileID;
|
$GLOBALS['egw_info']['user']['preferences']['mail']['ActiveProfileID'] = $this->mail_bo->profileID;
|
||||||
|
|
||||||
@ -397,14 +397,7 @@ class mail_compose
|
|||||||
// and we want to avoid that
|
// and we want to avoid that
|
||||||
$isFirstLoad = !($actionToProcess=='composeasnew');//true;
|
$isFirstLoad = !($actionToProcess=='composeasnew');//true;
|
||||||
$this->composeID = $_content['composeID'] = $this->generateComposeID();
|
$this->composeID = $_content['composeID'] = $this->generateComposeID();
|
||||||
if (!is_array($_content))
|
$_content = $this->setDefaults($_content+['mailidentity' => $_REQUEST['preset']['identity'] ?? null]);
|
||||||
{
|
|
||||||
$_content = $this->setDefaults();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
$_content = $this->setDefaults($_content);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// VFS Selector was used
|
// VFS Selector was used
|
||||||
if (!empty($_content['selectFromVFSForCompose']))
|
if (!empty($_content['selectFromVFSForCompose']))
|
||||||
|
297
mail/src/ApiHandler.php
Normal file
297
mail/src/ApiHandler.php
Normal file
@ -0,0 +1,297 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* EGroupware Mail: REST API
|
||||||
|
*
|
||||||
|
* @link https://www.egroupware.org
|
||||||
|
* @package mail
|
||||||
|
* @author Ralf Becker <rb@egroupware.org>
|
||||||
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace EGroupware\Mail;
|
||||||
|
|
||||||
|
use EGroupware\Api;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* REST API for mail
|
||||||
|
*/
|
||||||
|
class ApiHandler extends Api\CalDAV\Handler
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Constructor
|
||||||
|
*
|
||||||
|
* @param string $app 'calendar', 'addressbook' or 'infolog'
|
||||||
|
* @param Api\CalDAV $caldav calling class
|
||||||
|
*/
|
||||||
|
function __construct($app, Api\CalDAV $caldav)
|
||||||
|
{
|
||||||
|
parent::__construct($app, $caldav);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Options for json_encode of responses
|
||||||
|
*/
|
||||||
|
const JSON_RESPONSE_OPTIONS = JSON_PRETTY_PRINT|JSON_UNESCAPED_UNICODE|JSON_UNESCAPED_SLASHES;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handle post request for mail (send or compose mail and upload attachments)
|
||||||
|
*
|
||||||
|
* @param array &$options
|
||||||
|
* @param int $id
|
||||||
|
* @param int $user =null account_id of owner, default null
|
||||||
|
* @return mixed boolean true on success, false on failure or string with http status (eg. '404 Not Found')
|
||||||
|
*/
|
||||||
|
function post(&$options,$id,$user=null)
|
||||||
|
{
|
||||||
|
if ($this->debug) error_log(__METHOD__."($id, $user)".print_r($options,true));
|
||||||
|
$path = $options['path'];
|
||||||
|
if (empty($user))
|
||||||
|
{
|
||||||
|
$user = $GLOBALS['egw_info']['user']['account_id'];
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$prefix = '/'.Api\Accounts::id2name($user);
|
||||||
|
if (str_starts_with($path, $prefix)) $path = substr($path, strlen($prefix));
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (str_starts_with($path, '/mail/attachments/'))
|
||||||
|
{
|
||||||
|
$attachment_path = tempnam($GLOBALS['egw_info']['server']['temp_dir'], 'attach--'.
|
||||||
|
(str_replace('/', '-', substr($options['path'], 18)) ?: 'no-name').'--');
|
||||||
|
if (file_put_contents($attachment_path, $options['content']))
|
||||||
|
{
|
||||||
|
header('Location: '.($location = '/mail/attachments/'.substr(basename($attachment_path), 8)));
|
||||||
|
echo json_encode([
|
||||||
|
'status' => 200,
|
||||||
|
'message' => 'Attachment stored',
|
||||||
|
'location' => $location,
|
||||||
|
], self::JSON_RESPONSE_OPTIONS);
|
||||||
|
return '200 Ok';
|
||||||
|
}
|
||||||
|
throw new \Exception('Error storing attachment');
|
||||||
|
}
|
||||||
|
elseif (preg_match('#^/mail/((\d+):(\d+)/)?(compose)?#', $path, $matches))
|
||||||
|
{
|
||||||
|
$acc_id = $matches[2] ?? null;
|
||||||
|
$ident_id = $matches[3] ?? null;
|
||||||
|
$do_compose = (bool)($matches[4] ?? false);
|
||||||
|
if (!($data = json_decode($options['content'], true)))
|
||||||
|
{
|
||||||
|
throw new \Exception('Error decoding JSON: '.json_last_error_msg(), 422);
|
||||||
|
}
|
||||||
|
// ToDo: check required attributes
|
||||||
|
|
||||||
|
// for compose we need to construct a URL and push it to the client (or give an error if the client is not online)
|
||||||
|
if ($do_compose)
|
||||||
|
{
|
||||||
|
if (!Api\Json\Push::isOnline($user))
|
||||||
|
{
|
||||||
|
$account_lid = Api\Accounts::id2name($user);
|
||||||
|
throw new \Exception("User '$account_lid' (#$user) is NOT online", 404);
|
||||||
|
}
|
||||||
|
$extra = [
|
||||||
|
//'menuaction' => 'mail.mail_compose.compose',
|
||||||
|
'preset' => array_filter(array_intersect_key($data, array_flip(['to', 'cc', 'bcc', 'subject']))+[
|
||||||
|
'body' => $data['bodyHtml'] ?? null ?: $data['body'] ?? '',
|
||||||
|
'mimeType' => !empty($data['bodyHtml']) ? 'html' : 'plain',
|
||||||
|
'identity' => $ident_id,
|
||||||
|
]+self::prepareAttachments($data['attachments'] ?? [], $data['attachmentType'] ?? 'attach')),
|
||||||
|
];
|
||||||
|
$push = new Api\Json\Push($user);
|
||||||
|
//$push->call('egw.open_link', $link, '_blank', '640x1024');
|
||||||
|
$push->call('egw.open', '', 'mail', 'add', $extra, '_blank', 'mail');
|
||||||
|
header('Content-Type: application/json');
|
||||||
|
echo json_encode([
|
||||||
|
'status' => 200,
|
||||||
|
'message' => 'Request to open compose window sent',
|
||||||
|
'extra' => $extra,
|
||||||
|
], self::JSON_RESPONSE_OPTIONS);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
$compose = new mail_compose($acc_id);
|
||||||
|
$compose->compose([
|
||||||
|
'mailaccount' => $acc_id.':'.$ident_id,
|
||||||
|
'mail_plaintext' => $data['body'] ?? null,
|
||||||
|
'mail_htmltext' => $data['bodyHtml'] ?? null,
|
||||||
|
'mimeType' => !empty($data['bodyHtml']) ? 'html' : 'plain',
|
||||||
|
'file' => array_map(__CLASS__.'::uploadsForAttachments', $data['attachments'] ?? []),
|
||||||
|
]+array_diff_key($data+array_flip(['attachments', 'body', 'bodyHtml'])));
|
||||||
|
}
|
||||||
|
|
||||||
|
header('Content-Type: application/json');
|
||||||
|
echo $options['content'];
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
catch (\Throwable $e) {
|
||||||
|
_egw_log_exception($e);
|
||||||
|
header('Content-Type: application/json');
|
||||||
|
echo json_encode([
|
||||||
|
'error' => $code = $e->getCode() ?: 500,
|
||||||
|
'message' => $e->getMessage(),
|
||||||
|
]+(empty($GLOBALS['egw_info']['server']['exception_show_trace']) ? [] : [
|
||||||
|
'trace' => array_map(static function($trace)
|
||||||
|
{
|
||||||
|
$trace['file'] = str_replace(EGW_SERVER_ROOT.'/', '', $trace['file']);
|
||||||
|
return $trace;
|
||||||
|
}, $e->getTrace())
|
||||||
|
]), self::JSON_RESPONSE_OPTIONS);
|
||||||
|
return (400 <= $code && $code < 600 ? $code : 500).' '.$e->getMessage();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert an attachment name into an upload array for mail_compose::compose
|
||||||
|
*
|
||||||
|
* @param string[]? $attachments either "/mail/attachments/<token>" / file in temp_dir or VFS path
|
||||||
|
* @param string? $attachmentType "attach" (default), "link", "share_ro", "share_rw"
|
||||||
|
* @return array with values for keys "file", "name" and "filemode"
|
||||||
|
* @throws Exception if file not found or unreadable
|
||||||
|
*/
|
||||||
|
protected static function prepareAttachments(array $attachments, string $attachmentType=null)
|
||||||
|
{
|
||||||
|
$ret = [];
|
||||||
|
foreach($attachments as $attachment)
|
||||||
|
{
|
||||||
|
if (preg_match('#^/mail/attachments/(([^/]+)--[^/.-]{6,})$#', $attachment, $matches))
|
||||||
|
{
|
||||||
|
if (!file_exists($path=$GLOBALS['egw_info']['server']['temp_dir'].'/attach--'.$matches[1]))
|
||||||
|
{
|
||||||
|
throw new \Exception("Attachment $attachment NOT found", 400);
|
||||||
|
}
|
||||||
|
$ret['file'][] = $path;
|
||||||
|
$ret['name'][] = $matches[2];
|
||||||
|
/*return [
|
||||||
|
'name' => $matches[2],
|
||||||
|
'type' => Api\Vfs::mime_content_type($path),
|
||||||
|
'file' => $path,
|
||||||
|
'size' => filesize($path),
|
||||||
|
];*/
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (!Api\Vfs::is_readable($attachment))
|
||||||
|
{
|
||||||
|
throw new \Exception("Attachment $attachment NOT found", 400);
|
||||||
|
}
|
||||||
|
$ret['file'][] = Api\Vfs::PREFIX.$attachment;
|
||||||
|
$ret['name'][] = Api\Vfs::basename($attachment);
|
||||||
|
/*return [
|
||||||
|
'name' => Api\Vfs::basename($attachment),
|
||||||
|
'type' => Api\Vfs::mime_content_type($attachment),
|
||||||
|
'file' => Api\Vfs::PREFIX.$attachment,
|
||||||
|
'size' => filesize(Api\Vfs::PREFIX.$attachment),
|
||||||
|
];*/
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ($ret)
|
||||||
|
{
|
||||||
|
$ret['filemode'] = $attachmentType ?? 'attach';
|
||||||
|
if (!in_array($ret['filemode'], $valid=['attach', 'link', 'share_ro', 'share_rw']))
|
||||||
|
{
|
||||||
|
throw new \Exception("Invalid value '$ret[filemode]' for attachmentType, must be one of: '".implode("', '", $valid)."'", 422);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handle propfind request for an application folder
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @param array &$options
|
||||||
|
* @param array &$files
|
||||||
|
* @param int $user account_id
|
||||||
|
* @return mixed boolean true on success, false on failure or string with http status (eg. '404 Not Found')
|
||||||
|
*/
|
||||||
|
function propfind($path,&$options,&$files,$user)
|
||||||
|
{
|
||||||
|
if ($path === '/mail/' || $user && $path === '/'.Api\Accounts::id2name($user).'/mail/')
|
||||||
|
{
|
||||||
|
foreach(Api\Mail\Account::search($user ?? true,false) as $acc_id => $account)
|
||||||
|
{
|
||||||
|
// do NOT add SMTP only accounts as identities
|
||||||
|
if (!$account->is_imap(false)) continue;
|
||||||
|
|
||||||
|
foreach($account->identities($acc_id) as $ident_id => $identity)
|
||||||
|
{
|
||||||
|
$files['files'][] = [
|
||||||
|
'path' => $path.$ident_id,
|
||||||
|
'props' => ['name' => ['val' => $identity]],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return '501 Not Implemented';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handle get request for an applications entry
|
||||||
|
*
|
||||||
|
* @param array &$options
|
||||||
|
* @param int $id
|
||||||
|
* @param int $user =null account_id
|
||||||
|
* @return mixed boolean true on success, false on failure or string with http status (eg. '404 Not Found')
|
||||||
|
*/
|
||||||
|
function get(&$options,$id,$user=null)
|
||||||
|
{
|
||||||
|
header('Content-Type: application/json');
|
||||||
|
echo json_encode($all=iterator_to_array(Api\Mail\Account::identities([], true, 'name',
|
||||||
|
$user ?: $GLOBALS['egw_info']['user']['account_id'])), JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE);
|
||||||
|
return true;
|
||||||
|
return '501 Not Implemented';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handle get request for an applications entry
|
||||||
|
*
|
||||||
|
* @param array &$options
|
||||||
|
* @param int $id
|
||||||
|
* @param int $user =null account_id of owner, default null
|
||||||
|
* @return mixed boolean true on success, false on failure or string with http status (eg. '404 Not Found')
|
||||||
|
*/
|
||||||
|
function put(&$options,$id,$user=null)
|
||||||
|
{
|
||||||
|
return '501 Not Implemented';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handle get request for an applications entry
|
||||||
|
*
|
||||||
|
* @param array &$options
|
||||||
|
* @param int $id
|
||||||
|
* @param int $user account_id of collection owner
|
||||||
|
* @return mixed boolean true on success, false on failure or string with http status (eg. '404 Not Found')
|
||||||
|
*/
|
||||||
|
function delete(&$options,$id,$user)
|
||||||
|
{
|
||||||
|
return '501 Not Implemented';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read an entry
|
||||||
|
*
|
||||||
|
* @param string|int $id
|
||||||
|
* @param string $path =null implementation can use it, used in call from _common_get_put_delete
|
||||||
|
* @return array|boolean array with entry, false if no read rights, null if $id does not exist
|
||||||
|
*/
|
||||||
|
function read($id /*,$path=null*/)
|
||||||
|
{
|
||||||
|
return '501 Not Implemented';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if user has the necessary rights on an entry
|
||||||
|
*
|
||||||
|
* @param int $acl Api\Acl::READ, Api\Acl::EDIT or Api\Acl::DELETE
|
||||||
|
* @param array|int $entry entry-array or id
|
||||||
|
* @return boolean null if entry does not exist, false if no access, true if access permitted
|
||||||
|
*/
|
||||||
|
function check_access($acl,$entry)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user