Skip loading attached objects for files when we know the file is visible
Summary: Depends on D19222. Ref T13106. We currently execute an edge query (and possibly an object query) when loading builtin files, but this is never necessary because we know these files are always visible. Instead, skip this logic for builtin files and profile image files; these files have global visibility and will never get a different policy result because of file attachment information. (In theory, we could additionally skip this for files with the most open visibility policy or some other trivially visible policy like the user's PHID, but we do actually care about the attachment data some of the time.) Test Plan: Saw queries drop from 151 to 145 on local test page. Checked file attachment data in Files, saw it still working correctly. Maniphest Tasks: T13106 Differential Revision: https://secure.phabricator.com/D19223
This commit is contained in:
		| @@ -152,46 +152,74 @@ final class PhabricatorFileQuery | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   protected function loadPage() { |   protected function loadPage() { | ||||||
|     $files = $this->loadStandardPage(new PhabricatorFile()); |     $files = $this->loadStandardPage($this->newResultObject()); | ||||||
|  |  | ||||||
|     if (!$files) { |     if (!$files) { | ||||||
|       return $files; |       return $files; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     // Figure out which files we need to load attached objects for. In most | ||||||
|  |     // cases, we need to load attached objects to perform policy checks for | ||||||
|  |     // files. | ||||||
|  |  | ||||||
|  |     // However, in some special cases where we know files will always be | ||||||
|  |     // visible, we skip this. See T8478 and T13106. | ||||||
|  |     $need_objects = array(); | ||||||
|  |     foreach ($files as $file) { | ||||||
|  |       $always_visible = false; | ||||||
|  |  | ||||||
|  |       if ($file->getIsProfileImage()) { | ||||||
|  |         $always_visible = true; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if ($file->isBuiltin()) { | ||||||
|  |         $always_visible = true; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if ($always_visible) { | ||||||
|  |         // We just treat these files as though they aren't attached to | ||||||
|  |         // anything. This saves a query in common cases when we're loading | ||||||
|  |         // profile images or builtins. We could be slightly more nuanced | ||||||
|  |         // about this and distinguish between "not attached to anything" and | ||||||
|  |         // "might be attached but policy checks don't need to care". | ||||||
|  |         $file->attachObjectPHIDs(array()); | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       $need_objects[] = $file; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     $viewer = $this->getViewer(); |     $viewer = $this->getViewer(); | ||||||
|     $is_omnipotent = $viewer->isOmnipotent(); |     $is_omnipotent = $viewer->isOmnipotent(); | ||||||
|  |  | ||||||
|     // We need to load attached objects to perform policy checks for files. |     // If we have any files left which do need objects, load the edges now. | ||||||
|     // First, load the edges. |  | ||||||
|  |  | ||||||
|     $edge_type = PhabricatorFileHasObjectEdgeType::EDGECONST; |  | ||||||
|     $file_phids = mpull($files, 'getPHID'); |  | ||||||
|     $edges = id(new PhabricatorEdgeQuery()) |  | ||||||
|       ->withSourcePHIDs($file_phids) |  | ||||||
|       ->withEdgeTypes(array($edge_type)) |  | ||||||
|       ->execute(); |  | ||||||
|  |  | ||||||
|     $object_phids = array(); |     $object_phids = array(); | ||||||
|     foreach ($files as $file) { |     if ($need_objects) { | ||||||
|       $phids = array_keys($edges[$file->getPHID()][$edge_type]); |       $edge_type = PhabricatorFileHasObjectEdgeType::EDGECONST; | ||||||
|       $file->attachObjectPHIDs($phids); |       $file_phids = mpull($need_objects, 'getPHID'); | ||||||
|  |  | ||||||
|       if ($file->getIsProfileImage()) { |       $edges = id(new PhabricatorEdgeQuery()) | ||||||
|         // If this is a profile image, don't bother loading related files. |         ->withSourcePHIDs($file_phids) | ||||||
|         // It will always be visible, and we can get into trouble if we try |         ->withEdgeTypes(array($edge_type)) | ||||||
|         // to load objects and end up stuck in a cycle. See T8478. |         ->execute(); | ||||||
|         continue; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if ($is_omnipotent) { |       foreach ($need_objects as $file) { | ||||||
|         // If the viewer is omnipotent, we don't need to load the associated |         $phids = array_keys($edges[$file->getPHID()][$edge_type]); | ||||||
|         // objects either since they can certainly see the object. Skipping |         $file->attachObjectPHIDs($phids); | ||||||
|         // this can improve performance and prevent cycles. |  | ||||||
|         continue; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       foreach ($phids as $phid) { |         if ($is_omnipotent) { | ||||||
|         $object_phids[$phid] = true; |           // If the viewer is omnipotent, we don't need to load the associated | ||||||
|  |           // objects either since the viewer can certainly see the object. | ||||||
|  |           // Skipping this can improve performance and prevent cycles. This | ||||||
|  |           // could possibly become part of the profile/builtin code above which | ||||||
|  |           // short circuits attacment policy checks in cases where we know them | ||||||
|  |           // to be unnecessary. | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         foreach ($phids as $phid) { | ||||||
|  |           $object_phids[$phid] = true; | ||||||
|  |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -203,7 +231,7 @@ final class PhabricatorFileQuery | |||||||
|  |  | ||||||
|     $xforms = id(new PhabricatorTransformedFile())->loadAllWhere( |     $xforms = id(new PhabricatorTransformedFile())->loadAllWhere( | ||||||
|       'transformedPHID IN (%Ls)', |       'transformedPHID IN (%Ls)', | ||||||
|       $file_phids); |       mpull($files, 'getPHID')); | ||||||
|     $xform_phids = mpull($xforms, 'getOriginalPHID', 'getTransformedPHID'); |     $xform_phids = mpull($xforms, 'getOriginalPHID', 'getTransformedPHID'); | ||||||
|     foreach ($xform_phids as $derived_phid => $original_phid) { |     foreach ($xform_phids as $derived_phid => $original_phid) { | ||||||
|       $object_phids[$original_phid] = true; |       $object_phids[$original_phid] = true; | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 epriestley
					epriestley