mirror of
https://github.com/EGroupware/egroupware.git
synced 2025-01-27 16:29:22 +01:00
basic managed attachment support, tested with iCal from OS X mountain lion
This commit is contained in:
parent
0546a15913
commit
edd4cc49ca
@ -791,28 +791,7 @@ class calendar_ical extends calendar_boupdate
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case 'ATTACH':
|
case 'ATTACH':
|
||||||
static $url_prefix;
|
groupdav::add_attach('calendar', $event['id'], $attributes, $parameters);
|
||||||
if (!isset($url_prefix))
|
|
||||||
{
|
|
||||||
$url_prefix = '';
|
|
||||||
if ($GLOBALS['egw_info']['server']['webserver_url'][0] == '/')
|
|
||||||
{
|
|
||||||
$url_prefix = ($_SERVER['HTTPS'] ? 'https' : 'http').'://'.$_SERVER['HTTP_HOST'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
foreach(egw_vfs::find(egw_link::vfs_path('calendar', $event['id'], '', true), array(
|
|
||||||
'type' => 'F',
|
|
||||||
'need_mime' => true,
|
|
||||||
), true) as $path => $stat)
|
|
||||||
{
|
|
||||||
$attributes['ATTACH'][] = $url_prefix.egw::link(egw_vfs::download_url($path));
|
|
||||||
$parameters['ATTACH'][] = array(
|
|
||||||
'MANAGED-ID' => groupdav::path2managed_id($path),
|
|
||||||
'FMTTYP' => $stat['mime'],
|
|
||||||
'SIZE' => $stat['size'],
|
|
||||||
'FILENAME' => egw_vfs::basename($path),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@ -1779,6 +1758,12 @@ class calendar_ical extends calendar_boupdate
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// handle ATTACH attribute for managed attachments
|
||||||
|
if ($updated_id)
|
||||||
|
{
|
||||||
|
groupdav::handle_attach('calendar', $updated_id, $event['attach'], $event['attach-delete-by-put']);
|
||||||
|
}
|
||||||
|
|
||||||
if ($this->log)
|
if ($this->log)
|
||||||
{
|
{
|
||||||
$event_info['stored_event'] = $this->read($event_info['stored_event']['id']);
|
$event_info['stored_event'] = $this->read($event_info['stored_event']['id']);
|
||||||
@ -3011,6 +2996,11 @@ class calendar_ical extends calendar_boupdate
|
|||||||
|
|
||||||
if ($this->calendarOwner) $event['owner'] = $this->calendarOwner;
|
if ($this->calendarOwner) $event['owner'] = $this->calendarOwner;
|
||||||
|
|
||||||
|
// parsing ATTACH attributes for managed attachments
|
||||||
|
$attr = $component->getAttribute('X-EGROUPWARE-ATTACH-INCLUDED');
|
||||||
|
$event['attach-delete-by-put'] = !is_a($attr, PEAR_Error) && $attr === 'TRUE';
|
||||||
|
$event['attach'] = $component->getAllAttributes('ATTACH');
|
||||||
|
|
||||||
if ($this->log)
|
if ($this->log)
|
||||||
{
|
{
|
||||||
error_log(__FILE__.'['.__LINE__.'] '.__METHOD__."()\n" .
|
error_log(__FILE__.'['.__LINE__.'] '.__METHOD__."()\n" .
|
||||||
|
@ -1233,8 +1233,6 @@ class groupdav extends HTTP_WebDAV_Server
|
|||||||
* @todo support for rid parameter
|
* @todo support for rid parameter
|
||||||
* @todo managed-id does NOT change on update
|
* @todo managed-id does NOT change on update
|
||||||
* @todo updates of attachments through vfs need to call $handler->update_tags($id) too
|
* @todo updates of attachments through vfs need to call $handler->update_tags($id) too
|
||||||
* @todo stripping attachments added via PUT direct and make them managed ones (urls are NOT yet supported too)
|
|
||||||
* @todo update of attachments via PUT on calendar resource (not sure if I want delete), allows to re-user managed-ids ...
|
|
||||||
*/
|
*/
|
||||||
protected function managed_attachements(&$options, $id, groupdav_handler $handler, $action)
|
protected function managed_attachements(&$options, $id, groupdav_handler $handler, $action)
|
||||||
{
|
{
|
||||||
@ -1258,36 +1256,16 @@ class groupdav extends HTTP_WebDAV_Server
|
|||||||
substr($this->_SERVER['HTTP_CONTENT_DISPOSITION'], 0, 10) === 'attachment' &&
|
substr($this->_SERVER['HTTP_CONTENT_DISPOSITION'], 0, 10) === 'attachment' &&
|
||||||
preg_match('/filename="?([^";]+)/', $this->_SERVER['HTTP_CONTENT_DISPOSITION'], $matches))
|
preg_match('/filename="?([^";]+)/', $this->_SERVER['HTTP_CONTENT_DISPOSITION'], $matches))
|
||||||
{
|
{
|
||||||
$filename = $matches[1];
|
$filename = egw_vfs::basename($matches[1]);
|
||||||
$parts = explode('.', $filename);
|
|
||||||
$ext = '.'.array_pop($parts);
|
|
||||||
$filename = implode('.', $parts);
|
|
||||||
}
|
}
|
||||||
else
|
if (!($to = self::fopen_attachment($handler->app, $handler->get_id($entry), $filename, $this->_SERVER['CONTENT_TYPE'], $path)) ||
|
||||||
{
|
|
||||||
$filename = 'attachment';
|
|
||||||
if (isset($options['content_type']) && ($ext = mime_magic::mime2ext($options['content_type'])))
|
|
||||||
{
|
|
||||||
$ext = '.'.$ext;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
$ext = '';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for($i = 1; $i < 100; ++$i)
|
|
||||||
{
|
|
||||||
$path = egw_link::vfs_path($handler->app, $handler->get_id($entry), $filename.($i > 1 ? '-'.$i : '').$ext, true);
|
|
||||||
if (!egw_vfs::stat($path)) break;
|
|
||||||
}
|
|
||||||
if (!($to = egw_vfs::fopen($path, 'w')) ||
|
|
||||||
isset($options['stream']) && ($copied=stream_copy_to_stream($options['stream'], $to)) === false ||
|
isset($options['stream']) && ($copied=stream_copy_to_stream($options['stream'], $to)) === false ||
|
||||||
isset($options['content']) && ($copied=fwrite($to, $options['content'])) === false)
|
isset($options['content']) && ($copied=fwrite($to, $options['content'])) === false)
|
||||||
{
|
{
|
||||||
return '403 Forbidden';
|
return '403 Forbidden';
|
||||||
}
|
}
|
||||||
fclose($to);
|
fclose($to);
|
||||||
error_log(__METHOD__."() content-type=$options[content_type], filename=$filename, ext=$ext: $path created $copied bytes copied");
|
error_log(__METHOD__."() content-type=$options[content_type], filename=$filename: $path created $copied bytes copied");
|
||||||
$ret = '201 Created';
|
$ret = '201 Created';
|
||||||
header(self::MANAGED_ID_HEADER.': '.self::path2managed_id($path));
|
header(self::MANAGED_ID_HEADER.': '.self::path2managed_id($path));
|
||||||
break;
|
break;
|
||||||
@ -1333,6 +1311,153 @@ class groupdav extends HTTP_WebDAV_Server
|
|||||||
return $ret;
|
return $ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handle ATTACH attribute on importing iCals
|
||||||
|
*
|
||||||
|
* - turn inline attachments into managed attachments
|
||||||
|
* - delete NOT included attachments, $delete_via_put is true
|
||||||
|
* @todo: store URLs not from our managed attachments
|
||||||
|
*
|
||||||
|
* @param string $app eg. 'calendar'
|
||||||
|
* @param int|string $id
|
||||||
|
* @param array $attach array of array with values for keys 'name', 'params', 'value'
|
||||||
|
* @param boolean $delete_via_put
|
||||||
|
*/
|
||||||
|
public static function handle_attach($app, $id, $attach, $delete_via_put=false)
|
||||||
|
{
|
||||||
|
error_log(__METHOD__."('$app', $id, attach=".array2string($attach).", delete_via_put=".array2string($delete_via_put).')');
|
||||||
|
|
||||||
|
if (!egw_link::file_access($app, $id, EGW_ACL_EDIT))
|
||||||
|
{
|
||||||
|
error_log(__METHOD__."('$app', $id, ...) no rights to update attachments");
|
||||||
|
return; // no rights --> nothing to do
|
||||||
|
}
|
||||||
|
if (!is_array($attach)) $attach = array(); // could be PEAR_Error if not set
|
||||||
|
|
||||||
|
if ($delete_via_put)
|
||||||
|
{
|
||||||
|
foreach(egw_vfs::find(egw_link::vfs_path($app, $id, '', true), array('type' => 'F')) as $path)
|
||||||
|
{
|
||||||
|
$found = false;
|
||||||
|
foreach($attach as $key => $attr)
|
||||||
|
{
|
||||||
|
if ($attr['params']['MANAGED-ID'] === self::path2managed_id($path))
|
||||||
|
{
|
||||||
|
$found = true;
|
||||||
|
unset($attach[$key]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!$found)
|
||||||
|
{
|
||||||
|
$ok = egw_vfs::unlink($path);
|
||||||
|
error_log(__METHOD__."('$app', $id, ...) egw_vfs::unlink('$path') returned ".array2string($ok));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// turn inline attachments into managed ones
|
||||||
|
foreach($attach as $key => $attr)
|
||||||
|
{
|
||||||
|
if ($attr['params']['VALUE'] === 'BINARY')
|
||||||
|
{
|
||||||
|
if (!($to = self::fopen_attachment($app, $id, $filename=$attr['params']['FILENAME'], $attr['params']['FMTTYPE'], $path)) ||
|
||||||
|
($copied=fwrite($to, $attr['value'])) === false)
|
||||||
|
{
|
||||||
|
error_log(__METHOD__."('$app', $id, ...) failed to add attachment ".array2string($attr).") ");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
fclose($to);
|
||||||
|
error_log(__METHOD__."('$app', $id, ...)) content-type={$attr['params']['FMTTYPE']}, filename=$filename: $path created $copied bytes copied");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
error_log(__METHOD__."('$app', $id, ...) unsupported URI attachment ".array2string($attr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Open attachment for writing
|
||||||
|
*
|
||||||
|
* @param string $app
|
||||||
|
* @param int|string $id
|
||||||
|
* @param string $filename defaults to 'attachment'
|
||||||
|
* @param string $mime=null mime-type to generate extension
|
||||||
|
* @param string &$path=null on return path opened
|
||||||
|
* @return resource
|
||||||
|
*/
|
||||||
|
protected static function fopen_attachment($app, $id, $filename, $mime=null, &$path=null)
|
||||||
|
{
|
||||||
|
$filename = empty($filename) ? 'attachment' : egw_vfs::basename($filename);
|
||||||
|
|
||||||
|
if (strpos($mime, ';')) list($mime) = explode(';', $mime); // in case it contains eg. charset info
|
||||||
|
|
||||||
|
$ext = !empty($mime) ? mime_magic::mime2ext($mime) : '';
|
||||||
|
|
||||||
|
if (!$ext || substr($filename, -strlen($ext)-1) == '.'.$ext ||
|
||||||
|
preg_match('/\.([^.]+)$/', $filename, $matches) && mime_magic::ext2mime($matches[1]) == $mime)
|
||||||
|
{
|
||||||
|
$parts = explode('.', $filename);
|
||||||
|
$ext = '.'.array_pop($parts);
|
||||||
|
$filename = implode('.', $parts);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$ext = '.'.$ext;
|
||||||
|
}
|
||||||
|
for($i = 1; $i < 100; ++$i)
|
||||||
|
{
|
||||||
|
$path = egw_link::vfs_path($app, $id, $filename.($i > 1 ? '-'.$i : '').$ext, true);
|
||||||
|
if (!egw_vfs::stat($path)) break;
|
||||||
|
}
|
||||||
|
if ($i >= 100) return null;
|
||||||
|
|
||||||
|
if (!egw_vfs::file_exists($dir = egw_vfs::dirname($path)) && !egw_vfs::mkdir($dir))
|
||||||
|
{
|
||||||
|
error_log(__METHOD__."('$app', $id, ...) failed to create entry dir $dir!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return egw_vfs::fopen($path, 'w');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add ATTACH attribute(s) for iCal
|
||||||
|
*
|
||||||
|
* @param string $app eg. 'calendar'
|
||||||
|
* @param int|string $id
|
||||||
|
* @param array &$attributes
|
||||||
|
* @param array &$parameters
|
||||||
|
*/
|
||||||
|
public static function add_attach($app, $id, array &$attributes, array &$parameters)
|
||||||
|
{
|
||||||
|
static $url_prefix;
|
||||||
|
if (!isset($url_prefix))
|
||||||
|
{
|
||||||
|
$url_prefix = '';
|
||||||
|
if ($GLOBALS['egw_info']['server']['webserver_url'][0] == '/')
|
||||||
|
{
|
||||||
|
$url_prefix = ($_SERVER['HTTPS'] ? 'https' : 'http').'://'.$_SERVER['HTTP_HOST'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
foreach(egw_vfs::find(egw_link::vfs_path($app, $id, '', true), array(
|
||||||
|
'type' => 'F',
|
||||||
|
'need_mime' => true,
|
||||||
|
), true) as $path => $stat)
|
||||||
|
{
|
||||||
|
$attributes['ATTACH'][] = $url_prefix.egw::link(egw_vfs::download_url($path));
|
||||||
|
$parameters['ATTACH'][] = array(
|
||||||
|
'MANAGED-ID' => groupdav::path2managed_id($path),
|
||||||
|
'FMTTYP' => $stat['mime'],
|
||||||
|
'SIZE' => $stat['size'],
|
||||||
|
'FILENAME' => egw_vfs::basename($path),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
// if we have attachments, set X-attribute to enable deleting them by put
|
||||||
|
// (works around events synced before without ATTACH attributes)
|
||||||
|
if ($attributes['ATTACH']) $attributes['X-EGROUPWARE-ATTACH-INCLUDED'] = 'TRUE';
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return managed-id of a vfs-path
|
* Return managed-id of a vfs-path
|
||||||
*
|
*
|
||||||
|
Loading…
Reference in New Issue
Block a user