3 final class PhabricatorFileQuery
4 extends PhabricatorCursorPagedPolicyAwareQuery
{
9 private $explicitUploads;
11 private $dateCreatedAfter;
12 private $dateCreatedBefore;
13 private $contentHashes;
19 private $needTransforms;
22 private $storageEngines;
24 public function withIDs(array $ids) {
29 public function withPHIDs(array $phids) {
30 $this->phids
= $phids;
34 public function withAuthorPHIDs(array $phids) {
35 $this->authorPHIDs
= $phids;
39 public function withDateCreatedBefore($date_created_before) {
40 $this->dateCreatedBefore
= $date_created_before;
44 public function withDateCreatedAfter($date_created_after) {
45 $this->dateCreatedAfter
= $date_created_after;
49 public function withContentHashes(array $content_hashes) {
50 $this->contentHashes
= $content_hashes;
54 public function withBuiltinKeys(array $keys) {
55 $this->builtinKeys
= $keys;
59 public function withIsBuiltin($is_builtin) {
60 $this->isBuiltin
= $is_builtin;
65 * Select files which are transformations of some other file. For example,
66 * you can use this query to find previously generated thumbnails of an image
69 * As a parameter, provide a list of transformation specifications. Each
70 * specification is a dictionary with the keys `originalPHID` and `transform`.
71 * The `originalPHID` is the PHID of the original file (the file which was
72 * transformed) and the `transform` is the name of the transform to query
73 * for. If you pass `true` as the `transform`, all transformations of the
74 * file will be selected.
80 * 'originalPHID' => 'PHID-FILE-aaaa',
81 * 'transform' => 'sepia',
84 * 'originalPHID' => 'PHID-FILE-bbbb',
85 * 'transform' => true,
89 * This selects the `"sepia"` transformation of the file with PHID
90 * `PHID-FILE-aaaa` and all transformations of the file with PHID
93 * @param list<dict> List of transform specifications, described above.
96 public function withTransforms(array $specs) {
97 foreach ($specs as $spec) {
98 if (!is_array($spec) ||
99 empty($spec['originalPHID']) ||
100 empty($spec['transform'])) {
103 "Transform specification must be a dictionary with keys ".
110 $this->transforms
= $specs;
114 public function withLengthBetween($min, $max) {
115 $this->minLength
= $min;
116 $this->maxLength
= $max;
120 public function withNames(array $names) {
121 $this->names
= $names;
125 public function withIsPartial($partial) {
126 $this->isPartial
= $partial;
130 public function withIsDeleted($deleted) {
131 $this->isDeleted
= $deleted;
135 public function withNameNgrams($ngrams) {
136 return $this->withNgramsConstraint(
137 id(new PhabricatorFileNameNgrams()),
141 public function withStorageEngines(array $engines) {
142 $this->storageEngines
= $engines;
146 public function showOnlyExplicitUploads($explicit_uploads) {
147 $this->explicitUploads
= $explicit_uploads;
151 public function needTransforms(array $transforms) {
152 $this->needTransforms
= $transforms;
156 public function newResultObject() {
157 return new PhabricatorFile();
160 protected function loadPage() {
161 $files = $this->loadStandardPage($this->newResultObject());
167 // Figure out which files we need to load attached objects for. In most
168 // cases, we need to load attached objects to perform policy checks for
171 // However, in some special cases where we know files will always be
172 // visible, we skip this. See T8478 and T13106.
173 $need_objects = array();
174 $need_xforms = array();
175 foreach ($files as $file) {
176 $always_visible = false;
178 if ($file->getIsProfileImage()) {
179 $always_visible = true;
182 if ($file->isBuiltin()) {
183 $always_visible = true;
186 if ($always_visible) {
187 // We just treat these files as though they aren't attached to
188 // anything. This saves a query in common cases when we're loading
189 // profile images or builtins. We could be slightly more nuanced
190 // about this and distinguish between "not attached to anything" and
191 // "might be attached but policy checks don't need to care".
192 $file->attachObjectPHIDs(array());
196 $need_objects[] = $file;
197 $need_xforms[] = $file;
200 $viewer = $this->getViewer();
201 $is_omnipotent = $viewer->isOmnipotent();
203 // If we have any files left which do need objects, load the edges now.
204 $object_phids = array();
206 $edge_type = PhabricatorFileHasObjectEdgeType
::EDGECONST
;
207 $file_phids = mpull($need_objects, 'getPHID');
209 $edges = id(new PhabricatorEdgeQuery())
210 ->withSourcePHIDs($file_phids)
211 ->withEdgeTypes(array($edge_type))
214 foreach ($need_objects as $file) {
215 $phids = array_keys($edges[$file->getPHID()][$edge_type]);
216 $file->attachObjectPHIDs($phids);
218 if ($is_omnipotent) {
219 // If the viewer is omnipotent, we don't need to load the associated
220 // objects either since the viewer can certainly see the object.
221 // Skipping this can improve performance and prevent cycles. This
222 // could possibly become part of the profile/builtin code above which
223 // short circuits attacment policy checks in cases where we know them
224 // to be unnecessary.
228 foreach ($phids as $phid) {
229 $object_phids[$phid] = true;
234 // If this file is a transform of another file, load that file too. If you
235 // can see the original file, you can see the thumbnail.
237 // TODO: It might be nice to put this directly on PhabricatorFile and
238 // remove the PhabricatorTransformedFile table, which would be a little
242 $xforms = id(new PhabricatorTransformedFile())->loadAllWhere(
243 'transformedPHID IN (%Ls)',
244 mpull($need_xforms, 'getPHID'));
245 $xform_phids = mpull($xforms, 'getOriginalPHID', 'getTransformedPHID');
246 foreach ($xform_phids as $derived_phid => $original_phid) {
247 $object_phids[$original_phid] = true;
250 $xform_phids = array();
253 $object_phids = array_keys($object_phids);
255 // Now, load the objects.
259 // NOTE: We're explicitly turning policy exceptions off, since the rule
260 // here is "you can see the file if you can see ANY associated object".
261 // Without this explicit flag, we'll incorrectly throw unless you can
262 // see ALL associated objects.
264 $objects = id(new PhabricatorObjectQuery())
265 ->setParentQuery($this)
266 ->setViewer($this->getViewer())
267 ->withPHIDs($object_phids)
268 ->setRaisePolicyExceptions(false)
270 $objects = mpull($objects, null, 'getPHID');
273 foreach ($files as $file) {
274 $file_objects = array_select_keys($objects, $file->getObjectPHIDs());
275 $file->attachObjects($file_objects);
278 foreach ($files as $key => $file) {
279 $original_phid = idx($xform_phids, $file->getPHID());
280 if ($original_phid == PhabricatorPHIDConstants
::PHID_VOID
) {
281 // This is a special case for builtin files, which are handled
284 } else if ($original_phid) {
285 $original = idx($objects, $original_phid);
287 // If the viewer can't see the original file, also prevent them from
288 // seeing the transformed file.
289 $this->didRejectResult($file);
296 $file->attachOriginalFile($original);
302 protected function didFilterPage(array $files) {
303 $xform_keys = $this->needTransforms
;
304 if ($xform_keys !== null) {
305 $xforms = id(new PhabricatorTransformedFile())->loadAllWhere(
306 'originalPHID IN (%Ls) AND transform IN (%Ls)',
307 mpull($files, 'getPHID'),
311 $xfiles = id(new PhabricatorFile())->loadAllWhere(
313 mpull($xforms, 'getTransformedPHID'));
314 $xfiles = mpull($xfiles, null, 'getPHID');
317 $xform_map = array();
318 foreach ($xforms as $xform) {
319 $xfile = idx($xfiles, $xform->getTransformedPHID());
323 $original_phid = $xform->getOriginalPHID();
324 $xform_key = $xform->getTransform();
325 $xform_map[$original_phid][$xform_key] = $xfile;
328 $default_xforms = array_fill_keys($xform_keys, null);
330 foreach ($files as $file) {
331 $file_xforms = idx($xform_map, $file->getPHID(), array());
332 $file_xforms +
= $default_xforms;
333 $file->attachTransforms($file_xforms);
340 protected function buildJoinClauseParts(AphrontDatabaseConnection
$conn) {
341 $joins = parent
::buildJoinClauseParts($conn);
343 if ($this->transforms
) {
346 'JOIN %T t ON t.transformedPHID = f.phid',
347 id(new PhabricatorTransformedFile())->getTableName());
353 protected function buildWhereClauseParts(AphrontDatabaseConnection
$conn) {
354 $where = parent
::buildWhereClauseParts($conn);
356 if ($this->ids
!== null) {
363 if ($this->phids
!== null) {
370 if ($this->authorPHIDs
!== null) {
373 'f.authorPHID IN (%Ls)',
377 if ($this->explicitUploads
!== null) {
380 'f.isExplicitUpload = %d',
381 (int)$this->explicitUploads
);
384 if ($this->transforms
!== null) {
386 foreach ($this->transforms
as $transform) {
387 if ($transform['transform'] === true) {
388 $clauses[] = qsprintf(
390 '(t.originalPHID = %s)',
391 $transform['originalPHID']);
393 $clauses[] = qsprintf(
395 '(t.originalPHID = %s AND t.transform = %s)',
396 $transform['originalPHID'],
397 $transform['transform']);
400 $where[] = qsprintf($conn, '%LO', $clauses);
403 if ($this->dateCreatedAfter
!== null) {
406 'f.dateCreated >= %d',
407 $this->dateCreatedAfter
);
410 if ($this->dateCreatedBefore
!== null) {
413 'f.dateCreated <= %d',
414 $this->dateCreatedBefore
);
417 if ($this->contentHashes
!== null) {
420 'f.contentHash IN (%Ls)',
421 $this->contentHashes
);
424 if ($this->minLength
!== null) {
431 if ($this->maxLength
!== null) {
438 if ($this->names
!== null) {
445 if ($this->isPartial
!== null) {
449 (int)$this->isPartial
);
452 if ($this->isDeleted
!== null) {
456 (int)$this->isDeleted
);
459 if ($this->builtinKeys
!== null) {
462 'builtinKey IN (%Ls)',
466 if ($this->isBuiltin
!== null) {
467 if ($this->isBuiltin
) {
470 'builtinKey IS NOT NULL');
474 'builtinKey IS NULL');
478 if ($this->storageEngines
!== null) {
481 'storageEngine IN (%Ls)',
482 $this->storageEngines
);
488 protected function getPrimaryTableAlias() {
492 public function getQueryApplicationClass() {
493 return 'PhabricatorFilesApplication';