Api - When doing monthly share cleanup, check to see if share target is still valid

- removed some duplicated code
This commit is contained in:
nathangray 2019-07-18 15:25:28 -06:00
parent 4b8d4dd5e3
commit 58c53efd49
3 changed files with 59 additions and 128 deletions

View File

@ -119,4 +119,13 @@ class Sharing extends \EGroupware\Api\Sharing
echo lang('EPL Only'); echo lang('EPL Only');
} }
/**
* Check that a share path still exists (and is readable)
*/
protected static function check_path($share)
{
list($app, $id) = explode('::', $share['share_path']);
return (boolean) \EGroupware\Api\Link::title($app, $id);
}
} }

View File

@ -370,7 +370,6 @@ class Sharing
public function is_entry($share = false) public function is_entry($share = false)
{ {
if(!$share) $share = $this->share;
list($app, $id) = explode('::', $share['share_path']); list($app, $id) = explode('::', $share['share_path']);
return $share && $share['share_path'] && return $share && $share['share_path'] &&
$app && $id && !in_array($app, array('filemanager', 'vfs')) ;//&& array_key_exists($app, $GLOBALS['egw_info']['apps']); $app && $id && !in_array($app, array('filemanager', 'vfs')) ;//&& array_key_exists($app, $GLOBALS['egw_info']['apps']);
@ -378,7 +377,7 @@ class Sharing
public function need_session() public function need_session()
{ {
return $this->use_filemanager() || $this->is_entry(); return $this->use_filemanager() || static::is_entry($this->session);
} }
/** /**
@ -463,7 +462,7 @@ class Sharing
return $ui->editor($this->share['share_path']); return $ui->editor($this->share['share_path']);
} }
// use pure WebDAV for everything but GET requests to directories // use pure WebDAV for everything but GET requests to directories
else if (!$this->use_filemanager() && !$this->is_entry()) else if (!$this->use_filemanager() && !static::is_entry($this->share))
{ {
// send a content-disposition header, so browser knows how to name downloaded file // send a content-disposition header, so browser knows how to name downloaded file
if (!Vfs::is_dir($this->share['share_root'])) if (!Vfs::is_dir($this->share['share_root']))
@ -591,11 +590,12 @@ class Sharing
// if not already installed, install periodic cleanup of shares // if not already installed, install periodic cleanup of shares
$async = new Asyncservice(); $async = new Asyncservice();
if (!($job = $async->read(self::ASYNC_JOB_ID)) || $job[self::ASYNC_JOB_ID]['method'] === 'egw_sharing::tmp_cleanup') $method = 'EGroupware\\Api\\Sharing::tmp_cleanup';
if (!($job = $async->read(self::ASYNC_JOB_ID)) || $job[self::ASYNC_JOB_ID]['method'] !== $method)
{ {
if ($job) $async->delete(self::ASYNC_JOB_ID); // update not working old class-name if ($job) $async->delete(self::ASYNC_JOB_ID); // update not working old class-name
$async->set_timer(array('day' => 28), self::ASYNC_JOB_ID, 'EGroupware\\Api\\Vfs\\Sharing::tmp_cleanup',null); $async->set_timer(array('day' => 28), self::ASYNC_JOB_ID, $method ,null);
} }
return $share; return $share;
@ -668,36 +668,10 @@ class Sharing
if (is_scalar($keys)) $keys = array('share_id' => $keys); if (is_scalar($keys)) $keys = array('share_id' => $keys);
// get all temp. files, to be able to delete them
$tmp_paths = array();
foreach(self::$db->select(self::TABLE, 'share_path', array(
"share_path LIKE '/home/%/.tmp/%'")+$keys, __LINE__, __FILE__, false) as $row)
{
$tmp_paths[] = $row['share_path'];
}
// delete specified shares // delete specified shares
self::$db->delete(self::TABLE, $keys, __LINE__, __FILE__); self::$db->delete(self::TABLE, $keys, __LINE__, __FILE__);
$deleted = self::$db->affected_rows(); $deleted = self::$db->affected_rows();
// check if temp. files are used elsewhere
if ($tmp_paths)
{
foreach(self::$db->select(self::TABLE, 'share_path,COUNT(*) AS cnt', array(
'share_path' => $tmp_paths,
), __LINE__, __FILE__, false, 'GROUP BY share_path') as $row)
{
if (($key = array_search($row['share_path'], $tmp_paths)))
{
unset($tmp_paths[$key]);
}
}
// if not delete them
foreach($tmp_paths as $path)
{
Vfs::remove($path);
}
}
return $deleted; return $deleted;
} }
@ -748,7 +722,7 @@ class Sharing
else else
{ {
$share_ids = array(); $share_ids = array();
foreach(self::$db->selec(self::TABLE, 'share_id', array( foreach(self::$db->select(self::TABLE, 'share_id', array(
'share_path' => $row['share_path'], 'share_path' => $row['share_path'],
), __LINE__, __FILE__) as $id) ), __LINE__, __FILE__) as $id)
{ {
@ -757,7 +731,8 @@ class Sharing
} }
if ($share_ids) if ($share_ids)
{ {
self::$db->delete(self::TABLE, array('share_id' => $share_ids), __LINE__, __FILE__); $class = self::get_share_class($row);
$class::delete($share_ids);
} }
} }
@ -765,10 +740,13 @@ class Sharing
if (class_exists('EGroupware\\Collabora\\Wopi')) if (class_exists('EGroupware\\Collabora\\Wopi'))
{ {
self::$db->delete(self::TABLE, array( self::$db->delete(self::TABLE, array(
'share_expires < '.self::$db->quote(Api\DateTime::to(self::WOPI_KEEP, 'Y-m-d')), 'share_expires < '.self::$db->quote(DateTime::to(self::WOPI_KEEP, 'Y-m-d')),
'share_writable IN ('.Wopi::WOPI_WRITABLE.','.Wopi::WOPI_READONLY.')', 'share_writable IN ('.\EGroupware\Collabora\Wopi::WOPI_WRITABLE.','.\EGroupware\Collabora\Wopi::WOPI_READONLY.')',
), __LINE__, __FILE__); ), __LINE__, __FILE__);
} }
// Now check the remaining shares
static::cleanup_missing_paths();
} }
catch (\Exception $e) { catch (\Exception $e) {
_egw_log_exception($e); _egw_log_exception($e);
@ -776,6 +754,39 @@ class Sharing
Vfs::$is_root = false; Vfs::$is_root = false;
} }
/**
* Check share paths and if the path is no longer there / valid, remove the share
*/
public static function cleanup_missing_paths()
{
if (!isset(self::$db)) self::$db = $GLOBALS['egw']->db;
foreach(self::$db->select(self::TABLE, array(
'share_id','share_path'
), array(), __LINE__, __FILE__, false) as $share)
{
$class = self::get_share_class($share);
if(!$class::check_path($share))
{
$class::delete($share);
}
}
}
/**
* Check that the share path is still valid, and if not, delete it.
* This should be overridden.
*
* @param Array share
*
* @return boolean Is the share still valid
*/
protected static function check_path($share)
{
return true;
}
/** /**
* Generate link from share or share-token * Generate link from share or share-token
* *

View File

@ -289,26 +289,6 @@ class Sharing extends \EGroupware\Api\Sharing
} }
} }
/**
* Api\Storage\Base instance for egw_sharing table
*
* @var Api\Storage\Base
*/
protected static $so;
/**
* Get a so_sql instance initialised for shares
*/
public static function so()
{
if (!isset(self::$so))
{
self::$so = new Api\Storage\Base('phpgwapi', self::TABLE, null, '', true);
self::$so->set_times('string');
}
return self::$so;
}
/** /**
* Delete specified shares and unlink temp. files * Delete specified shares and unlink temp. files
* *
@ -329,9 +309,7 @@ class Sharing extends \EGroupware\Api\Sharing
$tmp_paths[] = $row['share_path']; $tmp_paths[] = $row['share_path'];
} }
// delete specified shares $deleted = parent::delete($keys);
self::$db->delete(self::TABLE, $keys, __LINE__, __FILE__);
$deleted = self::$db->affected_rows();
// check if temp. files are used elsewhere // check if temp. files are used elsewhere
if ($tmp_paths) if ($tmp_paths)
@ -355,78 +333,11 @@ class Sharing extends \EGroupware\Api\Sharing
} }
/** /**
* Home long to keep temp. files: 100 day * Check that a share path still exists (and is readable)
*/ */
const TMP_KEEP = 8640000; protected static function check_path($share)
/**
* How long to keep automatic created Wopi shares
*/
const WOPI_KEEP = '-3month';
/**.
* Periodic (monthly) cleanup of temporary sharing files (download link)
*
* Exlicit expireds shares are delete, as ones created over 100 days ago and last accessed over 100 days ago.
*/
public static function tmp_cleanup()
{ {
if (!isset(self::$db)) self::$db = $GLOBALS['egw']->db; return file_exists($share['share_path']);
Vfs::$is_root = true;
try {
$cols = array(
'share_path',
'MAX(share_expires) AS share_expires',
'MAX(share_created) AS share_created',
'MAX(share_last_accessed) AS share_last_accessed',
);
if (($group_concat = self::$db->group_concat('share_id'))) $cols[] = $group_concat.' AS share_id';
// remove expired tmp-files unconditionally
$having = 'HAVING MAX(share_expires) < '.self::$db->quote(self::$db->to_timestamp(time())).' OR '.
// remove without expiration date, when created over 100 days ago AND
'MAX(share_expires) IS NULL AND MAX(share_created) < '.self::$db->quote(self::$db->to_timestamp(time()-self::TMP_KEEP)). ' AND '.
// (last accessed over 100 days ago OR never)
'(MAX(share_last_accessed) IS NULL OR MAX(share_last_accessed) < '.self::$db->quote(self::$db->to_timestamp(time()-self::TMP_KEEP)).')';
foreach(self::$db->select(self::TABLE, $cols, array(
"share_path LIKE '/home/%/.tmp/%'",
), __LINE__, __FILE__, false, 'GROUP BY share_path '.$having) as $row)
{
Vfs::remove($row['share_path']);
if ($group_concat)
{
$share_ids = $row['share_id'] ? explode(',', $row['share_id']) : array();
}
else
{
$share_ids = array();
foreach(self::$db->selec(self::TABLE, 'share_id', array(
'share_path' => $row['share_path'],
), __LINE__, __FILE__) as $id)
{
$share_ids[] = $id['share_id'];
}
}
if ($share_ids)
{
self::$db->delete(self::TABLE, array('share_id' => $share_ids), __LINE__, __FILE__);
}
}
// delete automatic created and expired Collabora shares older then 3 month
if (class_exists('EGroupware\\Collabora\\Wopi'))
{
self::$db->delete(self::TABLE, array(
'share_expires < '.self::$db->quote(Api\DateTime::to(self::WOPI_KEEP, 'Y-m-d')),
'share_writable IN ('.Wopi::WOPI_WRITABLE.','.Wopi::WOPI_READONLY.')',
), __LINE__, __FILE__);
}
}
catch (\Exception $e) {
_egw_log_exception($e);
}
Vfs::$is_root = false;
} }
/** /**