1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-25 22:18:19 +01:00

Implement an "only if the rule did not match last time" policy for Herald rules

Summary: Depends on D18927. Ref T13048. This implements a new policy which allows Herald rules to fire on some kinds of state changes.

Test Plan:
Wrote and tested rules with the new policy:

{F5394971}

{F5394972}

Also wrote and tested rules with the old policies:

{F5394973}

Reviewers: amckinley

Reviewed By: amckinley

Maniphest Tasks: T13048

Differential Revision: https://secure.phabricator.com/D18930
This commit is contained in:
epriestley 2018-01-25 08:03:29 -08:00
parent 204d1de683
commit a34b6bdd06
4 changed files with 102 additions and 24 deletions

View file

@ -774,6 +774,7 @@ abstract class HeraldAdapter extends Phobject {
if (!$this->isSingleEventAdapter()) { if (!$this->isSingleEventAdapter()) {
$options[] = HeraldRule::REPEAT_FIRST; $options[] = HeraldRule::REPEAT_FIRST;
$options[] = HeraldRule::REPEAT_CHANGE;
} }
return $options; return $options;
@ -897,12 +898,15 @@ abstract class HeraldAdapter extends Phobject {
)); ));
} }
if ($rule->isRepeatEvery()) { if ($rule->isRepeatFirst()) {
$action_text = $action_text = pht(
pht('Take these actions every time this rule matches:'); 'Take these actions the first time this rule matches:');
} else if ($rule->isRepeatOnChange()) {
$action_text = pht(
'Take these actions if this rule did not match the last time:');
} else { } else {
$action_text = $action_text = pht(
pht('Take these actions the first time this rule matches:'); 'Take these actions every time this rule matches:');
} }
$action_title = phutil_tag( $action_title = phutil_tag(

View file

@ -218,7 +218,7 @@ final class HeraldRuleController extends HeraldController {
), ),
pht('New Action'))) pht('New Action')))
->setDescription(pht( ->setDescription(pht(
'Take these actions %s this rule matches:', 'Take these actions %s',
$repetition_selector)) $repetition_selector))
->setContent(javelin_tag( ->setContent(javelin_tag(
'table', 'table',

View file

@ -14,6 +14,7 @@ final class HeraldEngine extends Phobject {
private $forbiddenFields = array(); private $forbiddenFields = array();
private $forbiddenActions = array(); private $forbiddenActions = array();
private $skipEffects = array();
public function setDryRun($dry_run) { public function setDryRun($dry_run) {
$this->dryRun = $dry_run; $this->dryRun = $dry_run;
@ -171,13 +172,31 @@ final class HeraldEngine extends Phobject {
return; return;
} }
$rules = mpull($rules, null, 'getID'); // Update the "applied" state table. How this table works depends on the
$applied_ids = array(); // repetition policy for the rule.
//
// REPEAT_EVERY: We delete existing rows for the rule, then write nothing.
// This policy doesn't use any state.
//
// REPEAT_FIRST: We keep existing rows, then write additional rows for
// rules which fired. This policy accumulates state over the life of the
// object.
//
// REPEAT_CHANGE: We delete existing rows, then write all the rows which
// matched. This policy only uses the state from the previous run.
// Mark all the rules that have had their effects applied as having been $rules = mpull($rules, null, 'getID');
// executed for the current object.
$rule_ids = mpull($xscripts, 'getRuleID'); $rule_ids = mpull($xscripts, 'getRuleID');
$delete_ids = array();
foreach ($rules as $rule_id => $rule) {
if ($rule->isRepeatFirst()) {
continue;
}
$delete_ids[] = $rule_id;
}
$applied_ids = array();
foreach ($rule_ids as $rule_id) { foreach ($rule_ids as $rule_id) {
if (!$rule_id) { if (!$rule_id) {
// Some apply transcripts are purely informational and not associated // Some apply transcripts are purely informational and not associated
@ -190,13 +209,30 @@ final class HeraldEngine extends Phobject {
continue; continue;
} }
if ($rule->isRepeatFirst()) { if ($rule->isRepeatFirst() || $rule->isRepeatOnChange()) {
$applied_ids[] = $rule_id; $applied_ids[] = $rule_id;
} }
} }
if ($applied_ids) { // Also include "only if this rule did not match the last time" rules
// which matched but were skipped in the "applied" list.
foreach ($this->skipEffects as $rule_id => $ignored) {
$applied_ids[] = $rule_id;
}
if ($delete_ids || $applied_ids) {
$conn_w = id(new HeraldRule())->establishConnection('w'); $conn_w = id(new HeraldRule())->establishConnection('w');
if ($delete_ids) {
queryfx(
$conn_w,
'DELETE FROM %T WHERE phid = %s AND ruleID IN (%Ld)',
HeraldRule::TABLE_RULE_APPLIED,
$adapter->getPHID(),
$delete_ids);
}
if ($applied_ids) {
$sql = array(); $sql = array();
foreach ($applied_ids as $id) { foreach ($applied_ids as $id) {
$sql[] = qsprintf( $sql[] = qsprintf(
@ -212,6 +248,7 @@ final class HeraldEngine extends Phobject {
implode(', ', $sql)); implode(', ', $sql));
} }
} }
}
public function getTranscript() { public function getTranscript() {
$this->transcript->save(); $this->transcript->save();
@ -311,6 +348,30 @@ final class HeraldEngine extends Phobject {
} }
} }
// If this rule matched, and is set to run "if it did not match the last
// time", and we matched the last time, we're going to return a match in
// the transcript but set a flag so we don't actually apply any effects.
// We need the rule to match so that storage gets updated properly. If we
// just pretend the rule didn't match it won't cause any effects (which
// is correct), but it also won't set the "it matched" flag in storage,
// so the next run after this one would incorrectly trigger again.
$is_dry_run = $this->getDryRun();
if ($result && !$is_dry_run) {
$is_on_change = $rule->isRepeatOnChange();
if ($is_on_change) {
$did_apply = $rule->getRuleApplied($object->getPHID());
if ($did_apply) {
$reason = pht(
'This rule matched, but did not take any actions because it '.
'is configured to act only if it did not match the last time.');
$this->skipEffects[$rule->getID()] = true;
}
}
}
$this->newRuleTranscript($rule) $this->newRuleTranscript($rule)
->setResult($result) ->setResult($result)
->setReason($reason); ->setReason($reason);
@ -363,6 +424,11 @@ final class HeraldEngine extends Phobject {
HeraldRule $rule, HeraldRule $rule,
HeraldAdapter $object) { HeraldAdapter $object) {
$rule_id = $rule->getID();
if (isset($this->skipEffects[$rule_id])) {
return array();
}
$effects = array(); $effects = array();
foreach ($rule->getActions() as $action) { foreach ($rule->getActions() as $action) {
$effect = id(new HeraldEffect()) $effect = id(new HeraldEffect())

View file

@ -32,6 +32,7 @@ final class HeraldRule extends HeraldDAO
const REPEAT_EVERY = 'every'; const REPEAT_EVERY = 'every';
const REPEAT_FIRST = 'first'; const REPEAT_FIRST = 'first';
const REPEAT_CHANGE = 'change';
protected function getConfiguration() { protected function getConfiguration() {
return array( return array(
@ -282,6 +283,10 @@ final class HeraldRule extends HeraldDAO
return ($this->getRepetitionPolicyStringConstant() === self::REPEAT_FIRST); return ($this->getRepetitionPolicyStringConstant() === self::REPEAT_FIRST);
} }
public function isRepeatOnChange() {
return ($this->getRepetitionPolicyStringConstant() === self::REPEAT_CHANGE);
}
public static function getRepetitionPolicySelectOptionMap() { public static function getRepetitionPolicySelectOptionMap() {
$map = self::getRepetitionPolicyMap(); $map = self::getRepetitionPolicyMap();
return ipull($map, 'select'); return ipull($map, 'select');
@ -290,10 +295,13 @@ final class HeraldRule extends HeraldDAO
private static function getRepetitionPolicyMap() { private static function getRepetitionPolicyMap() {
return array( return array(
self::REPEAT_EVERY => array( self::REPEAT_EVERY => array(
'select' => pht('every time'), 'select' => pht('every time this rule matches:'),
), ),
self::REPEAT_FIRST => array( self::REPEAT_FIRST => array(
'select' => pht('only the first time'), 'select' => pht('only the first time this rule matches:'),
),
self::REPEAT_CHANGE => array(
'select' => pht('if this rule did not match the last time:'),
), ),
); );
} }