mirror of
https://github.com/nextcloud/server.git
synced 2026-06-11 09:42:09 -04:00
Merge pull request #51808 from nextcloud/backport/51600/stable31
[stable31] feat: Limit `ExpireTrash` job to 30 minutes
This commit is contained in:
commit
88c17ebe63
4 changed files with 77 additions and 45 deletions
|
|
@ -12,13 +12,12 @@ use OCA\Files_Trashbin\Helper;
|
|||
use OCA\Files_Trashbin\Trashbin;
|
||||
use OCP\AppFramework\Utility\ITimeFactory;
|
||||
use OCP\BackgroundJob\TimedJob;
|
||||
use OCP\IConfig;
|
||||
use OCP\IUser;
|
||||
use OCP\IAppConfig;
|
||||
use OCP\IUserManager;
|
||||
|
||||
class ExpireTrash extends TimedJob {
|
||||
public function __construct(
|
||||
private IConfig $config,
|
||||
private IAppConfig $appConfig,
|
||||
private IUserManager $userManager,
|
||||
private Expiration $expiration,
|
||||
ITimeFactory $time,
|
||||
|
|
@ -28,12 +27,8 @@ class ExpireTrash extends TimedJob {
|
|||
$this->setInterval(60 * 30);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $argument
|
||||
* @throws \Exception
|
||||
*/
|
||||
protected function run($argument) {
|
||||
$backgroundJob = $this->config->getAppValue('files_trashbin', 'background_job_expire_trash', 'yes');
|
||||
$backgroundJob = $this->appConfig->getValueString('files_trashbin', 'background_job_expire_trash', 'yes');
|
||||
if ($backgroundJob === 'no') {
|
||||
return;
|
||||
}
|
||||
|
|
@ -43,15 +38,28 @@ class ExpireTrash extends TimedJob {
|
|||
return;
|
||||
}
|
||||
|
||||
$this->userManager->callForSeenUsers(function (IUser $user): void {
|
||||
$stopTime = time() + 60 * 30; // Stops after 30 minutes.
|
||||
$offset = $this->appConfig->getValueInt('files_trashbin', 'background_job_expire_trash_offset', 0);
|
||||
$users = $this->userManager->getSeenUsers($offset);
|
||||
|
||||
foreach ($users as $user) {
|
||||
$uid = $user->getUID();
|
||||
if (!$this->setupFS($uid)) {
|
||||
return;
|
||||
continue;
|
||||
}
|
||||
$dirContent = Helper::getTrashFiles('/', $uid, 'mtime');
|
||||
Trashbin::deleteExpiredFiles($dirContent, $uid);
|
||||
});
|
||||
|
||||
$offset++;
|
||||
|
||||
if ($stopTime < time()) {
|
||||
$this->appConfig->setValueInt('files_trashbin', 'background_job_expire_trash_offset', $offset);
|
||||
\OC_Util::tearDownFS();
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
$this->appConfig->setValueInt('files_trashbin', 'background_job_expire_trash_offset', 0);
|
||||
\OC_Util::tearDownFS();
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -11,31 +11,31 @@ use OCA\Files_Trashbin\BackgroundJob\ExpireTrash;
|
|||
use OCA\Files_Trashbin\Expiration;
|
||||
use OCP\AppFramework\Utility\ITimeFactory;
|
||||
use OCP\BackgroundJob\IJobList;
|
||||
use OCP\IConfig;
|
||||
use OCP\IAppConfig;
|
||||
use OCP\IUserManager;
|
||||
use PHPUnit\Framework\MockObject\MockObject;
|
||||
use Test\TestCase;
|
||||
|
||||
class ExpireTrashTest extends TestCase {
|
||||
/** @var IConfig|MockObject */
|
||||
private $config;
|
||||
/** @var IAppConfig&MockObject */
|
||||
private $appConfig;
|
||||
|
||||
/** @var IUserManager|MockObject */
|
||||
/** @var IUserManager&MockObject */
|
||||
private $userManager;
|
||||
|
||||
/** @var Expiration|MockObject */
|
||||
/** @var Expiration&MockObject */
|
||||
private $expiration;
|
||||
|
||||
/** @var IJobList|MockObject */
|
||||
/** @var IJobList&MockObject */
|
||||
private $jobList;
|
||||
|
||||
/** @var ITimeFactory|MockObject */
|
||||
/** @var ITimeFactory&MockObject */
|
||||
private $time;
|
||||
|
||||
protected function setUp(): void {
|
||||
parent::setUp();
|
||||
|
||||
$this->config = $this->createMock(IConfig::class);
|
||||
$this->appConfig = $this->createMock(IAppConfig::class);
|
||||
$this->userManager = $this->createMock(IUserManager::class);
|
||||
$this->expiration = $this->createMock(Expiration::class);
|
||||
$this->jobList = $this->createMock(IJobList::class);
|
||||
|
|
@ -51,22 +51,25 @@ class ExpireTrashTest extends TestCase {
|
|||
}
|
||||
|
||||
public function testConstructAndRun(): void {
|
||||
$this->config->method('getAppValue')
|
||||
$this->appConfig->method('getValueString')
|
||||
->with('files_trashbin', 'background_job_expire_trash', 'yes')
|
||||
->willReturn('yes');
|
||||
$this->appConfig->method('getValueInt')
|
||||
->with('files_trashbin', 'background_job_expire_trash_offset', 0)
|
||||
->willReturn(0);
|
||||
|
||||
$job = new ExpireTrash($this->config, $this->userManager, $this->expiration, $this->time);
|
||||
$job = new ExpireTrash($this->appConfig, $this->userManager, $this->expiration, $this->time);
|
||||
$job->start($this->jobList);
|
||||
}
|
||||
|
||||
public function testBackgroundJobDeactivated(): void {
|
||||
$this->config->method('getAppValue')
|
||||
$this->appConfig->method('getValueString')
|
||||
->with('files_trashbin', 'background_job_expire_trash', 'yes')
|
||||
->willReturn('no');
|
||||
$this->expiration->expects($this->never())
|
||||
->method('getMaxAgeAsTimestamp');
|
||||
|
||||
$job = new ExpireTrash($this->config, $this->userManager, $this->expiration, $this->time);
|
||||
$job = new ExpireTrash($this->appConfig, $this->userManager, $this->expiration, $this->time);
|
||||
$job->start($this->jobList);
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -652,30 +652,14 @@ class Manager extends PublicEmitter implements IUserManager {
|
|||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param \Closure $callback
|
||||
* @psalm-param \Closure(\OCP\IUser):?bool $callback
|
||||
* @since 11.0.0
|
||||
*/
|
||||
public function callForSeenUsers(\Closure $callback) {
|
||||
$limit = 1000;
|
||||
$offset = 0;
|
||||
do {
|
||||
$userIds = $this->getSeenUserIds($limit, $offset);
|
||||
$offset += $limit;
|
||||
foreach ($userIds as $userId) {
|
||||
foreach ($this->backends as $backend) {
|
||||
if ($backend->userExists($userId)) {
|
||||
$user = $this->getUserObject($userId, $backend, false);
|
||||
$return = $callback($user);
|
||||
if ($return === false) {
|
||||
return;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
$users = $this->getSeenUsers();
|
||||
foreach ($users as $user) {
|
||||
$return = $callback($user);
|
||||
if ($return === false) {
|
||||
return;
|
||||
}
|
||||
} while (count($userIds) >= $limit);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
@ -851,4 +835,30 @@ class Manager extends PublicEmitter implements IUserManager {
|
|||
public function getDisplayNameCache(): DisplayNameCache {
|
||||
return $this->displayNameCache;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the list of users sorted by lastLogin, from most recent to least recent
|
||||
*
|
||||
* @param int $offset from which offset to fetch
|
||||
* @return \Iterator<IUser> list of user IDs
|
||||
* @since 30.0.0
|
||||
*/
|
||||
public function getSeenUsers(int $offset = 0): \Iterator {
|
||||
$limit = 1000;
|
||||
|
||||
do {
|
||||
$userIds = $this->getSeenUserIds($limit, $offset);
|
||||
$offset += $limit;
|
||||
|
||||
foreach ($userIds as $userId) {
|
||||
foreach ($this->backends as $backend) {
|
||||
if ($backend->userExists($userId)) {
|
||||
$user = $this->getUserObject($userId, $backend, false);
|
||||
yield $user;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
} while (count($userIds) === $limit);
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -231,4 +231,15 @@ interface IUserManager {
|
|||
* @since 30.0.0
|
||||
*/
|
||||
public function getLastLoggedInUsers(?int $limit = null, int $offset = 0, string $search = ''): array;
|
||||
|
||||
/**
|
||||
* Gets the list of users.
|
||||
* An iterator is returned allowing the caller to stop the iteration at any time.
|
||||
* The offset argument allows the caller to continue the iteration at a specific offset.
|
||||
*
|
||||
* @param int $offset from which offset to fetch
|
||||
* @return \Iterator<IUser> list of IUser object
|
||||
* @since 32.0.0
|
||||
*/
|
||||
public function getSeenUsers(int $offset = 0): \Iterator;
|
||||
}
|
||||
|
|
|
|||
Loading…
Reference in a new issue