Passed
Push — master ( 90909a...268acd )
by Morris
14:11 queued 11s
created

ResetRenderedTexts::getAvatarsToDelete()   A

Complexity

Conditions 3
Paths 3

Size

Total Lines 6
Code Lines 4

Duplication

Lines 0
Ratio 0 %

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 3
eloc 4
c 1
b 0
f 0
nc 3
nop 0
dl 0
loc 6
rs 10
1
<?php
2
3
declare(strict_types=1);
4
5
/**
6
 * @copyright Copyright (c) 2021, Daniel Calviño Sánchez <[email protected]>
7
 *
8
 * @author Daniel Calviño Sánchez <[email protected]>
9
 *
10
 * @license GNU AGPL version 3 or any later version
11
 *
12
 * This program is free software: you can redistribute it and/or modify
13
 * it under the terms of the GNU Affero General Public License as
14
 * published by the Free Software Foundation, either version 3 of the
15
 * License, or (at your option) any later version.
16
 *
17
 * This program is distributed in the hope that it will be useful,
18
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
20
 * GNU Affero General Public License for more details.
21
 *
22
 * You should have received a copy of the GNU Affero General Public License
23
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
24
 *
25
 */
26
27
namespace OC\Core\Command\Preview;
28
29
use OC\Preview\Storage\Root;
30
use OCP\DB\QueryBuilder\IQueryBuilder;
31
use OCP\Files\IMimeTypeLoader;
32
use OCP\Files\NotFoundException;
33
use OCP\Files\NotPermittedException;
34
use OCP\IAvatarManager;
35
use OCP\IDBConnection;
36
use OCP\IUserManager;
37
use Symfony\Component\Console\Command\Command;
38
use Symfony\Component\Console\Input\InputInterface;
39
use Symfony\Component\Console\Input\InputOption;
40
use Symfony\Component\Console\Output\OutputInterface;
41
42
class ResetRenderedTexts extends Command {
43
44
	/** @var IDBConnection */
45
	protected $connection;
46
47
	/** @var IUserManager */
48
	protected $userManager;
49
50
	/** @var IAvatarManager */
51
	protected $avatarManager;
52
53
	/** @var Root */
54
	private $previewFolder;
55
56
	/** @var IMimeTypeLoader */
57
	private $mimeTypeLoader;
58
59
	public function __construct(IDBConnection $connection,
60
								IUserManager $userManager,
61
								IAvatarManager $avatarManager,
62
								Root $previewFolder,
63
								IMimeTypeLoader $mimeTypeLoader) {
64
		parent::__construct();
65
66
		$this->connection = $connection;
67
		$this->userManager = $userManager;
68
		$this->avatarManager = $avatarManager;
69
		$this->previewFolder = $previewFolder;
70
		$this->mimeTypeLoader = $mimeTypeLoader;
71
	}
72
73
	protected function configure() {
74
		$this
75
			->setName('preview:reset-rendered-texts')
76
			->setDescription('Deletes all generated avatars and previews of text and md files')
77
			->addOption('dry', 'd', InputOption::VALUE_NONE, 'Dry mode - will not delete any files - in combination with the verbose mode one could check the operations.');
78
	}
79
80
	protected function execute(InputInterface $input, OutputInterface $output): int {
81
		$dryMode = $input->getOption('dry');
82
83
		if ($dryMode) {
84
			$output->writeln('INFO: The command is run in dry mode and will not modify anything.');
85
			$output->writeln('');
86
		}
87
88
		$this->deleteAvatars($output, $dryMode);
89
		$this->deletePreviews($output, $dryMode);
90
91
		return 0;
92
	}
93
94
	private function deleteAvatars(OutputInterface $output, bool $dryMode): void {
95
		$avatarsToDeleteCount = 0;
96
97
		foreach ($this->getAvatarsToDelete() as [$userId, $avatar]) {
98
			$output->writeln('Deleting avatar for ' . $userId, OutputInterface::VERBOSITY_VERBOSE);
99
100
			$avatarsToDeleteCount++;
101
102
			if ($dryMode) {
103
				continue;
104
			}
105
106
			try {
107
				$avatar->remove();
108
			} catch (NotFoundException $e) {
109
				// continue
110
			} catch (NotPermittedException $e) {
111
				// continue
112
			}
113
		}
114
115
		$output->writeln('Deleted ' . $avatarsToDeleteCount . ' avatars');
116
		$output->writeln('');
117
	}
118
119
	private function getAvatarsToDelete(): \Iterator {
120
		foreach ($this->userManager->search('') as $user) {
121
			$avatar = $this->avatarManager->getAvatar($user->getUID());
122
123
			if (!$avatar->isCustomAvatar()) {
124
				yield [$user->getUID(), $avatar];
125
			}
126
		}
127
	}
128
129
	private function deletePreviews(OutputInterface $output, bool $dryMode): void {
130
		$previewsToDeleteCount = 0;
131
132
		foreach ($this->getPreviewsToDelete() as ['name' => $previewFileId, 'path' => $filePath]) {
133
			$output->writeln('Deleting previews for ' . $filePath, OutputInterface::VERBOSITY_VERBOSE);
134
135
			$previewsToDeleteCount++;
136
137
			if ($dryMode) {
138
				continue;
139
			}
140
141
			try {
142
				$preview = $this->previewFolder->getFolder((string)$previewFileId);
143
				$preview->delete();
144
			} catch (NotFoundException $e) {
145
				// continue
146
			} catch (NotPermittedException $e) {
147
				// continue
148
			}
149
		}
150
151
		$output->writeln('Deleted ' . $previewsToDeleteCount . ' previews');
152
	}
153
154
	// Copy pasted and adjusted from
155
	// "lib/private/Preview/BackgroundCleanupJob.php".
156
	private function getPreviewsToDelete(): \Iterator {
157
		$qb = $this->connection->getQueryBuilder();
158
		$qb->select('path', 'mimetype')
159
			->from('filecache')
160
			->where($qb->expr()->eq('fileid', $qb->createNamedParameter($this->previewFolder->getId())));
161
		$cursor = $qb->execute();
162
		$data = $cursor->fetch();
163
		$cursor->closeCursor();
164
165
		if ($data === null) {
166
			return [];
167
		}
168
169
		/*
170
		 * This lovely like is the result of the way the new previews are stored
171
		 * We take the md5 of the name (fileid) and split the first 7 chars. That way
172
		 * there are not a gazillion files in the root of the preview appdata.
173
		 */
174
		$like = $this->connection->escapeLikeParameter($data['path']) . '/_/_/_/_/_/_/_/%';
175
176
		$qb = $this->connection->getQueryBuilder();
177
		$qb->select('a.name', 'b.path')
178
			->from('filecache', 'a')
179
			->leftJoin('a', 'filecache', 'b', $qb->expr()->eq(
180
				$qb->expr()->castColumn('a.name', IQueryBuilder::PARAM_INT), 'b.fileid'
181
			))
182
			->where(
183
				$qb->expr()->andX(
184
					$qb->expr()->like('a.path', $qb->createNamedParameter($like)),
185
					$qb->expr()->eq('a.mimetype', $qb->createNamedParameter($this->mimeTypeLoader->getId('httpd/unix-directory'))),
186
					$qb->expr()->orX(
187
						$qb->expr()->eq('b.mimetype', $qb->createNamedParameter($this->mimeTypeLoader->getId('text/plain'))),
188
						$qb->expr()->eq('b.mimetype', $qb->createNamedParameter($this->mimeTypeLoader->getId('text/markdown'))),
189
						$qb->expr()->eq('b.mimetype', $qb->createNamedParameter($this->mimeTypeLoader->getId('text/x-markdown')))
190
					)
191
				)
192
			);
193
194
		$cursor = $qb->execute();
195
196
		while ($row = $cursor->fetch()) {
197
			yield $row;
198
		}
199
200
		$cursor->closeCursor();
201
	}
202
}
203