mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-18 18:51:12 +01:00
herald: add the ability to execute a rule the first time only
Summary: - added a new config class for representing the kind of repetition a rule has (once, every time, first time only) - added an email action to herald rules for differential to allow someone to get an email but only the first one - changed the herald rule ui to allow a user to pick the amount of repetition Test Plan: created a test rule and ran it over and over Reviewed By: epriestley Reviewers: epriestley, tuomaspelkonen CC: aran, epriestley, gc3 Revert Plan: Tags: - begin *PUBLIC* platform impact section - Bugzilla: # - end platform impact - Differential Revision: 357
This commit is contained in:
parent
c0210d6817
commit
22c1b38655
12 changed files with 238 additions and 24 deletions
7
resources/sql/patches/041.heraldrepetition.sql
Normal file
7
resources/sql/patches/041.heraldrepetition.sql
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
CREATE TABLE phabricator_herald.herald_ruleapplied (
|
||||||
|
ruleID int unsigned not null,
|
||||||
|
phid varchar(64) binary not null,
|
||||||
|
PRIMARY KEY(ruleID, phid)
|
||||||
|
) ENGINE=InnoDB;
|
||||||
|
|
||||||
|
ALTER TABLE phabricator_herald.herald_rule add repetitionPolicy int unsigned;
|
|
@ -237,6 +237,7 @@ phutil_register_library_map(array(
|
||||||
'HeraldObjectAdapter' => 'applications/herald/adapter/base',
|
'HeraldObjectAdapter' => 'applications/herald/adapter/base',
|
||||||
'HeraldObjectTranscript' => 'applications/herald/storage/transcript/object',
|
'HeraldObjectTranscript' => 'applications/herald/storage/transcript/object',
|
||||||
'HeraldRecursiveConditionsException' => 'applications/herald/engine/engine/exception',
|
'HeraldRecursiveConditionsException' => 'applications/herald/engine/engine/exception',
|
||||||
|
'HeraldRepetitionPolicyConfig' => 'applications/herald/config/repetitionpolicy',
|
||||||
'HeraldRule' => 'applications/herald/storage/rule',
|
'HeraldRule' => 'applications/herald/storage/rule',
|
||||||
'HeraldRuleController' => 'applications/herald/controller/rule',
|
'HeraldRuleController' => 'applications/herald/controller/rule',
|
||||||
'HeraldRuleTranscript' => 'applications/herald/storage/transcript/rule',
|
'HeraldRuleTranscript' => 'applications/herald/storage/transcript/rule',
|
||||||
|
|
|
@ -289,13 +289,19 @@ class DifferentialRevisionEditor {
|
||||||
array_keys($add['ccs']),
|
array_keys($add['ccs']),
|
||||||
$this->actorPHID);
|
$this->actorPHID);
|
||||||
|
|
||||||
// Add the author to the relevant set of users so they get a copy of the
|
// Add the author and users included from Herald rules to the relevant set
|
||||||
// email.
|
// of users so they get a copy of the email.
|
||||||
if (!$this->silentUpdate) {
|
if (!$this->silentUpdate) {
|
||||||
if ($is_new) {
|
if ($is_new) {
|
||||||
$add['rev'][$this->getActorPHID()] = true;
|
$add['rev'][$this->getActorPHID()] = true;
|
||||||
|
if ($diff) {
|
||||||
|
$add['rev'] += $adapter->getEmailPHIDsAddedByHerald();
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
$stable['rev'][$this->getActorPHID()] = true;
|
$stable['rev'][$this->getActorPHID()] = true;
|
||||||
|
if ($diff) {
|
||||||
|
$stable['rev'] += $adapter->getEmailPHIDsAddedByHerald();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -356,13 +362,6 @@ class DifferentialRevisionEditor {
|
||||||
// TODO
|
// TODO
|
||||||
// $revision->saveTransaction();
|
// $revision->saveTransaction();
|
||||||
|
|
||||||
$event = array(
|
|
||||||
'revision_id' => $revision->getID(),
|
|
||||||
'PHID' => $revision->getPHID(),
|
|
||||||
'action' => $is_new ? 'create' : 'update',
|
|
||||||
'actor' => $this->getActorPHID(),
|
|
||||||
);
|
|
||||||
|
|
||||||
// TODO: Move this into a worker task thing.
|
// TODO: Move this into a worker task thing.
|
||||||
PhabricatorSearchDifferentialIndexer::indexRevision($revision);
|
PhabricatorSearchDifferentialIndexer::indexRevision($revision);
|
||||||
|
|
||||||
|
|
|
@ -27,6 +27,7 @@ class HeraldDifferentialRevisionAdapter extends HeraldObjectAdapter {
|
||||||
|
|
||||||
protected $newCCs = array();
|
protected $newCCs = array();
|
||||||
protected $remCCs = array();
|
protected $remCCs = array();
|
||||||
|
protected $emailPHIDs = array();
|
||||||
|
|
||||||
protected $repository;
|
protected $repository;
|
||||||
protected $affectedPackages;
|
protected $affectedPackages;
|
||||||
|
@ -64,6 +65,10 @@ class HeraldDifferentialRevisionAdapter extends HeraldObjectAdapter {
|
||||||
return $this->remCCs;
|
return $this->remCCs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getEmailPHIDsAddedByHerald() {
|
||||||
|
return $this->emailPHIDs;
|
||||||
|
}
|
||||||
|
|
||||||
public function getPHID() {
|
public function getPHID() {
|
||||||
return $this->revision->getPHID();
|
return $this->revision->getPHID();
|
||||||
}
|
}
|
||||||
|
@ -247,17 +252,23 @@ class HeraldDifferentialRevisionAdapter extends HeraldObjectAdapter {
|
||||||
true,
|
true,
|
||||||
'OK, did nothing.');
|
'OK, did nothing.');
|
||||||
break;
|
break;
|
||||||
|
case HeraldActionConfig::ACTION_EMAIL:
|
||||||
case HeraldActionConfig::ACTION_ADD_CC:
|
case HeraldActionConfig::ACTION_ADD_CC:
|
||||||
|
$op = ($action == HeraldActionConfig::ACTION_EMAIL) ? 'email' : 'CC';
|
||||||
$base_target = $effect->getTarget();
|
$base_target = $effect->getTarget();
|
||||||
$forbidden = array();
|
$forbidden = array();
|
||||||
foreach ($base_target as $key => $fbid) {
|
foreach ($base_target as $key => $fbid) {
|
||||||
if (isset($forbidden_ccs[$fbid])) {
|
if (isset($forbidden_ccs[$fbid])) {
|
||||||
$forbidden[] = $fbid;
|
$forbidden[] = $fbid;
|
||||||
unset($base_target[$key]);
|
unset($base_target[$key]);
|
||||||
|
} else {
|
||||||
|
if ($action == HeraldActionConfig::ACTION_EMAIL) {
|
||||||
|
$this->emailPHIDs[$fbid] = true;
|
||||||
} else {
|
} else {
|
||||||
$this->newCCs[$fbid] = true;
|
$this->newCCs[$fbid] = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if ($forbidden) {
|
if ($forbidden) {
|
||||||
$failed = clone $effect;
|
$failed = clone $effect;
|
||||||
|
@ -267,17 +278,18 @@ class HeraldDifferentialRevisionAdapter extends HeraldObjectAdapter {
|
||||||
$result[] = new HeraldApplyTranscript(
|
$result[] = new HeraldApplyTranscript(
|
||||||
$effect,
|
$effect,
|
||||||
true,
|
true,
|
||||||
'Added these addresses to CC list. Others could not be added.');
|
'Added these addresses to '.$op.' list. '.
|
||||||
|
'Others could not be added.');
|
||||||
}
|
}
|
||||||
$result[] = new HeraldApplyTranscript(
|
$result[] = new HeraldApplyTranscript(
|
||||||
$failed,
|
$failed,
|
||||||
false,
|
false,
|
||||||
'CC forbidden, these addresses have unsubscribed.');
|
$op.' forbidden, these addresses have unsubscribed.');
|
||||||
} else {
|
} else {
|
||||||
$result[] = new HeraldApplyTranscript(
|
$result[] = new HeraldApplyTranscript(
|
||||||
$effect,
|
$effect,
|
||||||
true,
|
true,
|
||||||
'Added addresses to CC list.');
|
'Added addresses to '.$op.' list.');
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case HeraldActionConfig::ACTION_REMOVE_CC:
|
case HeraldActionConfig::ACTION_REMOVE_CC:
|
||||||
|
|
|
@ -41,6 +41,7 @@ class HeraldActionConfig {
|
||||||
array(
|
array(
|
||||||
self::ACTION_ADD_CC,
|
self::ACTION_ADD_CC,
|
||||||
self::ACTION_REMOVE_CC,
|
self::ACTION_REMOVE_CC,
|
||||||
|
self::ACTION_EMAIL,
|
||||||
self::ACTION_NOTHING,
|
self::ACTION_NOTHING,
|
||||||
));
|
));
|
||||||
case HeraldContentTypeConfig::CONTENT_TYPE_COMMIT:
|
case HeraldContentTypeConfig::CONTENT_TYPE_COMMIT:
|
||||||
|
|
|
@ -0,0 +1,64 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Copyright 2011 Facebook, Inc.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
class HeraldRepetitionPolicyConfig {
|
||||||
|
const FIRST = 'first'; // only execute the first time (no repeating)
|
||||||
|
const EVERY = 'every'; // repeat every time
|
||||||
|
|
||||||
|
private static $policyIntMap = array(
|
||||||
|
self::FIRST => 0,
|
||||||
|
self::EVERY => 1,
|
||||||
|
);
|
||||||
|
|
||||||
|
private static $policyMap = array(
|
||||||
|
self::FIRST => 'only the first time',
|
||||||
|
self::EVERY => 'every time',
|
||||||
|
);
|
||||||
|
|
||||||
|
public static function getMap() {
|
||||||
|
return self::$policyMap;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getMapForContentType($type) {
|
||||||
|
switch ($type) {
|
||||||
|
case HeraldContentTypeConfig::CONTENT_TYPE_DIFFERENTIAL:
|
||||||
|
return array_select_keys(
|
||||||
|
self::$policyMap,
|
||||||
|
array(
|
||||||
|
self::EVERY,
|
||||||
|
self::FIRST,
|
||||||
|
));
|
||||||
|
|
||||||
|
case HeraldContentTypeConfig::CONTENT_TYPE_COMMIT:
|
||||||
|
case HeraldContentTypeConfig::CONTENT_TYPE_MERGE:
|
||||||
|
case HeraldContentTypeConfig::CONTENT_TYPE_OWNERS:
|
||||||
|
return array();
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw new Exception("Unknown content type '{$type}'.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function toInt($str) {
|
||||||
|
return idx(self::$policyIntMap, $str, self::$policyIntMap[self::EVERY]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function toString($int) {
|
||||||
|
return idx(array_flip(self::$policyIntMap), $int, self::EVERY);
|
||||||
|
}
|
||||||
|
}
|
14
src/applications/herald/config/repetitionpolicy/__init__.php
Normal file
14
src/applications/herald/config/repetitionpolicy/__init__.php
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* This file is automatically generated. Lint this module to rebuild it.
|
||||||
|
* @generated
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
phutil_require_module('phabricator', 'applications/herald/config/contenttype');
|
||||||
|
|
||||||
|
phutil_require_module('phutil', 'utils');
|
||||||
|
|
||||||
|
|
||||||
|
phutil_require_source('HeraldRepetitionPolicyConfig.php');
|
|
@ -74,6 +74,11 @@ class HeraldRuleController extends HeraldController {
|
||||||
$rule->setName($request->getStr('name'));
|
$rule->setName($request->getStr('name'));
|
||||||
$rule->setMustMatchAll(($request->getStr('must_match') == 'all'));
|
$rule->setMustMatchAll(($request->getStr('must_match') == 'all'));
|
||||||
|
|
||||||
|
$repetition_policy_param = $request->getStr('repetition_policy');
|
||||||
|
$rule->setRepetitionPolicy(
|
||||||
|
HeraldRepetitionPolicyConfig::toInt($repetition_policy_param)
|
||||||
|
);
|
||||||
|
|
||||||
if (!strlen($rule->getName())) {
|
if (!strlen($rule->getName())) {
|
||||||
$e_name = "Required";
|
$e_name = "Required";
|
||||||
$errors[] = "Rule must have a name.";
|
$errors[] = "Rule must have a name.";
|
||||||
|
@ -250,6 +255,45 @@ class HeraldRuleController extends HeraldController {
|
||||||
$action = '/herald/rule/'.$rule->getID().'/';
|
$action = '/herald/rule/'.$rule->getID().'/';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Make the selector for choosing how often this rule should be repeated
|
||||||
|
$repetition_selector = "";
|
||||||
|
$repetition_policy = HeraldRepetitionPolicyConfig::toString(
|
||||||
|
$rule->getRepetitionPolicy()
|
||||||
|
);
|
||||||
|
$repetition_options = HeraldRepetitionPolicyConfig::getMapForContentType(
|
||||||
|
$rule->getContentType()
|
||||||
|
);
|
||||||
|
|
||||||
|
if (empty($repetition_options)) {
|
||||||
|
// default option is 'every time'
|
||||||
|
$repetition_selector = idx(
|
||||||
|
HeraldRepetitionPolicyConfig::getMap(),
|
||||||
|
HeraldRepetitionPolicyConfig::EVERY
|
||||||
|
);
|
||||||
|
} else if (count($repetition_options) == 1) {
|
||||||
|
// if there's only 1 option, just pick it for the user
|
||||||
|
$repetition_selector = reset($repetition_options);
|
||||||
|
} else {
|
||||||
|
// give the user all the options for this rule type
|
||||||
|
$tags = array();
|
||||||
|
|
||||||
|
foreach ($repetition_options as $name => $option) {
|
||||||
|
$tags[] = phutil_render_tag(
|
||||||
|
'option',
|
||||||
|
array (
|
||||||
|
'selected' => ($repetition_policy == $name) ? 'selected' : null,
|
||||||
|
'value' => $name,
|
||||||
|
),
|
||||||
|
phutil_escape_html($option)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$repetition_selector =
|
||||||
|
'<select name="repetition_policy">'.
|
||||||
|
implode("\n", $tags).
|
||||||
|
'</select>';
|
||||||
|
}
|
||||||
|
|
||||||
require_celerity_resource('herald-css');
|
require_celerity_resource('herald-css');
|
||||||
|
|
||||||
$type_name = $content_type_map[$rule->getContentType()];
|
$type_name = $content_type_map[$rule->getContentType()];
|
||||||
|
@ -320,7 +364,9 @@ class HeraldRuleController extends HeraldController {
|
||||||
),
|
),
|
||||||
'Create New Action').
|
'Create New Action').
|
||||||
'</div>'.
|
'</div>'.
|
||||||
'<p>Take these actions:</p>'.
|
'<p>'.
|
||||||
|
'Take these actions '.$repetition_selector.' this rule matches:'.
|
||||||
|
'</p>'.
|
||||||
'<div style="clear: both;"></div>'.
|
'<div style="clear: both;"></div>'.
|
||||||
javelin_render_tag(
|
javelin_render_tag(
|
||||||
'table',
|
'table',
|
||||||
|
|
|
@ -12,6 +12,7 @@ phutil_require_module('phabricator', 'applications/herald/config/action');
|
||||||
phutil_require_module('phabricator', 'applications/herald/config/condition');
|
phutil_require_module('phabricator', 'applications/herald/config/condition');
|
||||||
phutil_require_module('phabricator', 'applications/herald/config/contenttype');
|
phutil_require_module('phabricator', 'applications/herald/config/contenttype');
|
||||||
phutil_require_module('phabricator', 'applications/herald/config/field');
|
phutil_require_module('phabricator', 'applications/herald/config/field');
|
||||||
|
phutil_require_module('phabricator', 'applications/herald/config/repetitionpolicy');
|
||||||
phutil_require_module('phabricator', 'applications/herald/config/valuetype');
|
phutil_require_module('phabricator', 'applications/herald/config/valuetype');
|
||||||
phutil_require_module('phabricator', 'applications/herald/controller/base');
|
phutil_require_module('phabricator', 'applications/herald/controller/base');
|
||||||
phutil_require_module('phabricator', 'applications/herald/storage/action');
|
phutil_require_module('phabricator', 'applications/herald/storage/action');
|
||||||
|
|
|
@ -51,11 +51,27 @@ class HeraldEngine {
|
||||||
|
|
||||||
$effects = array();
|
$effects = array();
|
||||||
foreach ($rules as $id => $rule) {
|
foreach ($rules as $id => $rule) {
|
||||||
|
|
||||||
|
|
||||||
$this->stack = array();
|
$this->stack = array();
|
||||||
try {
|
try {
|
||||||
|
if (($rule->getRepetitionPolicy() ==
|
||||||
|
HeraldRepetitionPolicyConfig::FIRST) &&
|
||||||
|
$rule->getRuleApplied($object->getPHID())) {
|
||||||
|
// This rule is only supposed to be applied a single time, and it's
|
||||||
|
// aleady been applied, so this is an automatic failure.
|
||||||
|
$xscript = id(new HeraldRuleTranscript())
|
||||||
|
->setRuleID($id)
|
||||||
|
->setResult(false)
|
||||||
|
->setRuleName($rule->getName())
|
||||||
|
->setRuleOwner($rule->getAuthorPHID())
|
||||||
|
->setReason(
|
||||||
|
"This rule is only supposed to be repeated a single time, ".
|
||||||
|
"and it has already been applied."
|
||||||
|
);
|
||||||
|
$this->transcript->addRuleTranscript($xscript);
|
||||||
|
$rule_matches = false;
|
||||||
|
} else {
|
||||||
$rule_matches = $this->doesRuleMatch($rule, $object);
|
$rule_matches = $this->doesRuleMatch($rule, $object);
|
||||||
|
}
|
||||||
} catch (HeraldRecursiveConditionsException $ex) {
|
} catch (HeraldRecursiveConditionsException $ex) {
|
||||||
$names = array();
|
$names = array();
|
||||||
foreach ($this->stack as $rule_id => $ignored) {
|
foreach ($this->stack as $rule_id => $ignored) {
|
||||||
|
@ -101,12 +117,10 @@ class HeraldEngine {
|
||||||
}
|
}
|
||||||
|
|
||||||
public function applyEffects(array $effects, HeraldObjectAdapter $object) {
|
public function applyEffects(array $effects, HeraldObjectAdapter $object) {
|
||||||
if ($object instanceof DryRunHeraldable) {
|
$this->transcript->setDryRun($object instanceof HeraldDryRunAdapter);
|
||||||
$this->transcript->setDryRun(true);
|
|
||||||
} else {
|
$xscripts = $object->applyHeraldEffects($effects);
|
||||||
$this->transcript->setDryRun(false);
|
foreach ($xscripts as $apply_xscript) {
|
||||||
}
|
|
||||||
foreach ($object->applyHeraldEffects($effects) as $apply_xscript) {
|
|
||||||
if (!($apply_xscript instanceof HeraldApplyTranscript)) {
|
if (!($apply_xscript instanceof HeraldApplyTranscript)) {
|
||||||
throw new Exception(
|
throw new Exception(
|
||||||
"Heraldable must return HeraldApplyTranscripts from ".
|
"Heraldable must return HeraldApplyTranscripts from ".
|
||||||
|
@ -114,6 +128,17 @@ class HeraldEngine {
|
||||||
}
|
}
|
||||||
$this->transcript->addApplyTranscript($apply_xscript);
|
$this->transcript->addApplyTranscript($apply_xscript);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!$this->transcript->getDryRun()) {
|
||||||
|
// Mark all the rules that have had their effects applied as having been
|
||||||
|
// executed for the current object.
|
||||||
|
$ruleIDs = mpull($xscripts, 'getRuleID');
|
||||||
|
foreach ($ruleIDs as $ruleID) {
|
||||||
|
id(new HeraldRule())
|
||||||
|
->setID($ruleID)
|
||||||
|
->saveRuleApplied($object->getPHID());
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getTranscript() {
|
public function getTranscript() {
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
|
|
||||||
phutil_require_module('phabricator', 'applications/herald/config/condition');
|
phutil_require_module('phabricator', 'applications/herald/config/condition');
|
||||||
phutil_require_module('phabricator', 'applications/herald/config/field');
|
phutil_require_module('phabricator', 'applications/herald/config/field');
|
||||||
|
phutil_require_module('phabricator', 'applications/herald/config/repetitionpolicy');
|
||||||
phutil_require_module('phabricator', 'applications/herald/engine/effect');
|
phutil_require_module('phabricator', 'applications/herald/engine/effect');
|
||||||
phutil_require_module('phabricator', 'applications/herald/engine/engine/exception');
|
phutil_require_module('phabricator', 'applications/herald/engine/engine/exception');
|
||||||
phutil_require_module('phabricator', 'applications/herald/storage/rule');
|
phutil_require_module('phabricator', 'applications/herald/storage/rule');
|
||||||
|
|
|
@ -18,13 +18,18 @@
|
||||||
|
|
||||||
class HeraldRule extends HeraldDAO {
|
class HeraldRule extends HeraldDAO {
|
||||||
|
|
||||||
|
const TABLE_RULE_APPLIED = 'herald_ruleapplied';
|
||||||
|
|
||||||
protected $name;
|
protected $name;
|
||||||
protected $authorPHID;
|
protected $authorPHID;
|
||||||
|
|
||||||
protected $contentType;
|
protected $contentType;
|
||||||
protected $mustMatchAll;
|
protected $mustMatchAll;
|
||||||
|
protected $repetitionPolicy;
|
||||||
|
|
||||||
protected $configVersion = 7;
|
protected $configVersion = 8;
|
||||||
|
|
||||||
|
private $ruleApplied = array(); // phids for which this rule has been applied
|
||||||
|
|
||||||
public static function loadAllByContentTypeWithFullData($content_type) {
|
public static function loadAllByContentTypeWithFullData($content_type) {
|
||||||
$rules = id(new HeraldRule())->loadAllWhere(
|
$rules = id(new HeraldRule())->loadAllWhere(
|
||||||
|
@ -45,10 +50,19 @@ class HeraldRule extends HeraldDAO {
|
||||||
'ruleID in (%Ld)',
|
'ruleID in (%Ld)',
|
||||||
$rule_ids);
|
$rule_ids);
|
||||||
|
|
||||||
|
$applied = queryfx_all(
|
||||||
|
id(new HeraldRule())->establishConnection('r'),
|
||||||
|
'SELECT * FROM %T WHERE ruleID in (%Ld)',
|
||||||
|
self::TABLE_RULE_APPLIED, $rule_ids
|
||||||
|
);
|
||||||
|
|
||||||
$conditions = mgroup($conditions, 'getRuleID');
|
$conditions = mgroup($conditions, 'getRuleID');
|
||||||
$actions = mgroup($actions, 'getRuleID');
|
$actions = mgroup($actions, 'getRuleID');
|
||||||
|
$applied = igroup($applied, 'ruleID');
|
||||||
|
|
||||||
|
|
||||||
foreach ($rules as $rule) {
|
foreach ($rules as $rule) {
|
||||||
|
$rule->attachAllRuleApplied(idx($applied, $rule->getID(), array()));
|
||||||
$rule->attachConditions(idx($conditions, $rule->getID(), array()));
|
$rule->attachConditions(idx($conditions, $rule->getID(), array()));
|
||||||
$rule->attachActions(idx($actions, $rule->getID(), array()));
|
$rule->attachActions(idx($actions, $rule->getID(), array()));
|
||||||
}
|
}
|
||||||
|
@ -56,6 +70,35 @@ class HeraldRule extends HeraldDAO {
|
||||||
return $rules;
|
return $rules;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getRuleApplied($phid) {
|
||||||
|
// defaults to false because (ruleID, phid) pairs not in the db imply
|
||||||
|
// a rule that's not been applied before
|
||||||
|
return idx($this->ruleApplied, $phid, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setRuleApplied($phid) {
|
||||||
|
$this->ruleApplied[$phid] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function attachAllRuleApplied(array $applied) {
|
||||||
|
// turn array of array(ruleID, phid) into array of ruleID => true
|
||||||
|
$this->ruleApplied = array_fill_keys(ipull($applied, 'phid'), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function saveRuleApplied($phid) {
|
||||||
|
if (!$this->getID()) {
|
||||||
|
throw new Exception("Save rule before saving children.");
|
||||||
|
}
|
||||||
|
|
||||||
|
queryfx(
|
||||||
|
$this->establishConnection('w'),
|
||||||
|
'INSERT IGNORE INTO %T (phid, ruleID) VALUES (%s, %d)',
|
||||||
|
self::TABLE_RULE_APPLIED, $phid, $this->getID()
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->setRuleApplied($phid);
|
||||||
|
}
|
||||||
|
|
||||||
public function loadConditions() {
|
public function loadConditions() {
|
||||||
if (!$this->getID()) {
|
if (!$this->getID()) {
|
||||||
return array();
|
return array();
|
||||||
|
|
Loading…
Reference in a new issue