mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-11 07:11:04 +01:00
Make query engines "overheat" instead of stalling when filtering too many results
Summary: Ref T11773. This is an initial first step toward a more complete solution, but should make the worst case much less bad: prior to this change, the worst case was "30 second exeuction timeout". After this patch, the worst case is "no results + explanatory message", which is strictly better. Test Plan: Made all feed stories fail policy checks, loaded home page. - Before adding overheating: 9,600 queries / 20 seconds - After adding overheating: 376 queries / 800ms Reviewers: chad Reviewed By: chad Maniphest Tasks: T11773 Differential Revision: https://secure.phabricator.com/D16735
This commit is contained in:
parent
a00e867de0
commit
a3253f78ce
2 changed files with 54 additions and 0 deletions
|
@ -238,6 +238,8 @@ final class PhabricatorApplicationSearchController
|
||||||
$nux_view = null;
|
$nux_view = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$is_overheated = $query->getIsOverheated();
|
||||||
|
|
||||||
if ($nux_view) {
|
if ($nux_view) {
|
||||||
$box->appendChild($nux_view);
|
$box->appendChild($nux_view);
|
||||||
} else {
|
} else {
|
||||||
|
@ -265,6 +267,10 @@ final class PhabricatorApplicationSearchController
|
||||||
$box->appendChild($list->getContent());
|
$box->appendChild($list->getContent());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($is_overheated) {
|
||||||
|
$box->appendChild($this->newOverheatedView($objects));
|
||||||
|
}
|
||||||
|
|
||||||
$result_header = $list->getHeader();
|
$result_header = $list->getHeader();
|
||||||
if ($result_header) {
|
if ($result_header) {
|
||||||
$box->setHeader($result_header);
|
$box->setHeader($result_header);
|
||||||
|
@ -525,4 +531,27 @@ final class PhabricatorApplicationSearchController
|
||||||
->setDropdownMenu($action_list);
|
->setDropdownMenu($action_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function newOverheatedView(array $results) {
|
||||||
|
if ($results) {
|
||||||
|
$message = pht(
|
||||||
|
'Most objects matching your query are not visible to you, so '.
|
||||||
|
'filtering results is taking a long time. Only some results are '.
|
||||||
|
'shown. Refine your query to find results more quickly.');
|
||||||
|
} else {
|
||||||
|
$message = pht(
|
||||||
|
'Most objects matching your query are not visible to you, so '.
|
||||||
|
'filtering results is taking a long time. Refine your query to '.
|
||||||
|
'find results more quickly.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return id(new PHUIInfoView())
|
||||||
|
->setSeverity(PHUIInfoView::SEVERITY_WARNING)
|
||||||
|
->setFlush(true)
|
||||||
|
->setTitle(pht('Query Overheated'))
|
||||||
|
->setErrors(
|
||||||
|
array(
|
||||||
|
$message,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,6 +44,7 @@ abstract class PhabricatorPolicyAwareQuery extends PhabricatorOffsetPagedQuery {
|
||||||
* and `null` (inherit from parent query, with no exceptions by default).
|
* and `null` (inherit from parent query, with no exceptions by default).
|
||||||
*/
|
*/
|
||||||
private $raisePolicyExceptions;
|
private $raisePolicyExceptions;
|
||||||
|
private $isOverheated;
|
||||||
|
|
||||||
|
|
||||||
/* -( Query Configuration )------------------------------------------------ */
|
/* -( Query Configuration )------------------------------------------------ */
|
||||||
|
@ -215,6 +216,14 @@ abstract class PhabricatorPolicyAwareQuery extends PhabricatorOffsetPagedQuery {
|
||||||
|
|
||||||
$this->willExecute();
|
$this->willExecute();
|
||||||
|
|
||||||
|
// If we examine and filter significantly more objects than the query
|
||||||
|
// limit, we stop early. This prevents us from looping through a huge
|
||||||
|
// number of records when the viewer can see few or none of them. See
|
||||||
|
// T11773 for some discussion.
|
||||||
|
$this->isOverheated = false;
|
||||||
|
$overheat_limit = $limit * 10;
|
||||||
|
$total_seen = 0;
|
||||||
|
|
||||||
do {
|
do {
|
||||||
if ($need) {
|
if ($need) {
|
||||||
$this->rawResultLimit = min($need - $count, 1024);
|
$this->rawResultLimit = min($need - $count, 1024);
|
||||||
|
@ -232,6 +241,8 @@ abstract class PhabricatorPolicyAwareQuery extends PhabricatorOffsetPagedQuery {
|
||||||
$page = array();
|
$page = array();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$total_seen += count($page);
|
||||||
|
|
||||||
if ($page) {
|
if ($page) {
|
||||||
$maybe_visible = $this->willFilterPage($page);
|
$maybe_visible = $this->willFilterPage($page);
|
||||||
if ($maybe_visible) {
|
if ($maybe_visible) {
|
||||||
|
@ -302,6 +313,11 @@ abstract class PhabricatorPolicyAwareQuery extends PhabricatorOffsetPagedQuery {
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->nextPage($page);
|
$this->nextPage($page);
|
||||||
|
|
||||||
|
if ($overheat_limit && ($total_seen >= $overheat_limit)) {
|
||||||
|
$this->isOverheated = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
} while (true);
|
} while (true);
|
||||||
|
|
||||||
$results = $this->didLoadResults($results);
|
$results = $this->didLoadResults($results);
|
||||||
|
@ -369,6 +385,15 @@ abstract class PhabricatorPolicyAwareQuery extends PhabricatorOffsetPagedQuery {
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public function getIsOverheated() {
|
||||||
|
if ($this->isOverheated === null) {
|
||||||
|
throw new PhutilInvalidStateException('execute');
|
||||||
|
}
|
||||||
|
return $this->isOverheated;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return a map of all object PHIDs which were loaded in the query but
|
* Return a map of all object PHIDs which were loaded in the query but
|
||||||
* filtered out by policy constraints. This allows a caller to distinguish
|
* filtered out by policy constraints. This allows a caller to distinguish
|
||||||
|
|
Loading…
Reference in a new issue