From 4c23d5cfae549b9caf277c41a94e11b5c2e55f91 Mon Sep 17 00:00:00 2001 From: epriestley Date: Mon, 6 Jul 2015 13:15:58 -0700 Subject: [PATCH] Make Herald custom field integration modular Summary: Ref T8726. The existing implementation is largely made redundant by the newer modular implementation. Test Plan: Created a custom field rule, set custom field to various values until it matched, saw effect and proper transcripts. Reviewers: btrahan Reviewed By: btrahan Subscribers: eadler, joshuaspence, epriestley Maniphest Tasks: T8726 Differential Revision: https://secure.phabricator.com/D13497 --- src/__phutil_library_map__.php | 2 + .../herald/adapter/HeraldAdapter.php | 222 +----------------- src/applications/herald/field/HeraldField.php | 12 +- .../PhabricatorCustomFieldHeraldField.php | 68 ++++++ 4 files changed, 83 insertions(+), 221 deletions(-) create mode 100644 src/infrastructure/customfield/herald/PhabricatorCustomFieldHeraldField.php diff --git a/src/__phutil_library_map__.php b/src/__phutil_library_map__.php index 012d958cd9..b46ffd5d07 100644 --- a/src/__phutil_library_map__.php +++ b/src/__phutil_library_map__.php @@ -1709,6 +1709,7 @@ phutil_register_library_map(array( 'PhabricatorCustomFieldAttachment' => 'infrastructure/customfield/field/PhabricatorCustomFieldAttachment.php', 'PhabricatorCustomFieldConfigOptionType' => 'infrastructure/customfield/config/PhabricatorCustomFieldConfigOptionType.php', 'PhabricatorCustomFieldDataNotAvailableException' => 'infrastructure/customfield/exception/PhabricatorCustomFieldDataNotAvailableException.php', + 'PhabricatorCustomFieldHeraldField' => 'infrastructure/customfield/herald/PhabricatorCustomFieldHeraldField.php', 'PhabricatorCustomFieldImplementationIncompleteException' => 'infrastructure/customfield/exception/PhabricatorCustomFieldImplementationIncompleteException.php', 'PhabricatorCustomFieldIndexStorage' => 'infrastructure/customfield/storage/PhabricatorCustomFieldIndexStorage.php', 'PhabricatorCustomFieldInterface' => 'infrastructure/customfield/interface/PhabricatorCustomFieldInterface.php', @@ -5339,6 +5340,7 @@ phutil_register_library_map(array( 'PhabricatorCustomFieldAttachment' => 'Phobject', 'PhabricatorCustomFieldConfigOptionType' => 'PhabricatorConfigOptionType', 'PhabricatorCustomFieldDataNotAvailableException' => 'Exception', + 'PhabricatorCustomFieldHeraldField' => 'HeraldField', 'PhabricatorCustomFieldImplementationIncompleteException' => 'Exception', 'PhabricatorCustomFieldIndexStorage' => 'PhabricatorLiskDAO', 'PhabricatorCustomFieldList' => 'Phobject', diff --git a/src/applications/herald/adapter/HeraldAdapter.php b/src/applications/herald/adapter/HeraldAdapter.php index 62f9087c1d..ac72bb97b6 100644 --- a/src/applications/herald/adapter/HeraldAdapter.php +++ b/src/applications/herald/adapter/HeraldAdapter.php @@ -1,8 +1,5 @@ isHeraldCustomKey($field_name)) { - return $this->getCustomFieldValue($field_name); - } - throw new Exception(pht("Unknown field '%s'!", $field_name)); } } @@ -365,17 +357,7 @@ abstract class HeraldAdapter extends Phobject { } public function getFields() { - $fields = array_keys($this->getFieldImplementationMap()); - - $custom_fields = $this->getCustomFields(); - if ($custom_fields) { - foreach ($custom_fields->getFields() as $custom_field) { - $key = $custom_field->getFieldKey(); - $fields[] = $this->getHeraldKeyFromCustomKey($key); - } - } - - return $fields; + return array_keys($this->getFieldImplementationMap()); } public function getFieldNameMap() { @@ -417,7 +399,7 @@ abstract class HeraldAdapter extends Phobject { self::FIELD_TASK_STATUS => pht('Task status'), self::FIELD_PUSHER_IS_COMMITTER => pht('Pusher same as committer'), self::FIELD_PATH => pht('Path'), - ) + $this->getCustomFieldNameMap(); + ); } @@ -552,9 +534,6 @@ abstract class HeraldAdapter extends Phobject { self::CONDITION_IS_FALSE, ); default: - if ($this->isHeraldCustomKey($field)) { - return $this->getCustomFieldConditions($field); - } throw new Exception( pht( "This adapter does not define conditions for field '%s'!", @@ -933,15 +912,6 @@ abstract class HeraldAdapter extends Phobject { return $impl->getHeraldFieldValueType($condition); } - if ($this->isHeraldCustomKey($field)) { - $value_type = $this->getCustomFieldValueTypeForFieldAndCondition( - $field, - $condition); - if ($value_type !== null) { - return $value_type; - } - } - switch ($condition) { case self::CONDITION_CONTAINS: case self::CONDITION_NOT_CONTAINS: @@ -1205,9 +1175,7 @@ abstract class HeraldAdapter extends Phobject { $field_type = $condition->getFieldName(); - $default = $this->isHeraldCustomKey($field_type) - ? pht('(Unknown Custom Field "%s")', $field_type) - : pht('(Unknown Field "%s")', $field_type); + $default = pht('(Unknown Field "%s")', $field_type); $field_name = idx($this->getFieldNameMap(), $field_type, $default); @@ -1226,9 +1194,7 @@ abstract class HeraldAdapter extends Phobject { $action_type = $action->getAction(); - $default = $this->isHeraldCustomKey($action_type) - ? pht('(Unknown Custom Action "%s") equals', $action_type) - : pht('(Unknown Action "%s") equals', $action_type); + $default = pht('(Unknown Action "%s") equals', $action_type); $action_name = idx( $this->getActionNameMap($rule_global), @@ -1362,186 +1328,6 @@ abstract class HeraldAdapter extends Phobject { return $phids; } -/* -( Custom Field Integration )------------------------------------------- */ - - - /** - * Returns the prefix used to namespace Herald fields which are based on - * custom fields. - * - * @return string Key prefix. - * @task customfield - */ - private function getCustomKeyPrefix() { - return 'herald.custom/'; - } - - - /** - * Determine if a field key is based on a custom field or a regular internal - * field. - * - * @param string Field key. - * @return bool True if the field key is based on a custom field. - * @task customfield - */ - private function isHeraldCustomKey($key) { - $prefix = $this->getCustomKeyPrefix(); - return (strncmp($key, $prefix, strlen($prefix)) == 0); - } - - - /** - * Convert a custom field key into a Herald field key. - * - * @param string Custom field key. - * @return string Herald field key. - * @task customfield - */ - private function getHeraldKeyFromCustomKey($key) { - return $this->getCustomKeyPrefix().$key; - } - - - /** - * Get custom fields for this adapter, if appliable. This will either return - * a field list or `null` if the adapted object does not implement custom - * fields or the adapter does not support them. - * - * @return PhabricatorCustomFieldList|null List of fields, or `null`. - * @task customfield - */ - private function getCustomFields() { - if ($this->customFields === false) { - $this->customFields = null; - - - $template_object = $this->newObject(); - if ($template_object instanceof PhabricatorCustomFieldInterface) { - $object = $this->getObject(); - if (!$object) { - $object = $template_object; - } - - $fields = PhabricatorCustomField::getObjectFields( - $object, - PhabricatorCustomField::ROLE_HERALD); - $fields->setViewer(PhabricatorUser::getOmnipotentUser()); - $fields->readFieldsFromStorage($object); - - $this->customFields = $fields; - } - } - - return $this->customFields; - } - - - /** - * Get a custom field by Herald field key, or `null` if it does not exist - * or custom fields are not supported. - * - * @param string Herald field key. - * @return PhabricatorCustomField|null Matching field, if it exists. - * @task customfield - */ - private function getCustomField($herald_field_key) { - $fields = $this->getCustomFields(); - if (!$fields) { - return null; - } - - foreach ($fields->getFields() as $custom_field) { - $key = $custom_field->getFieldKey(); - if ($this->getHeraldKeyFromCustomKey($key) == $herald_field_key) { - return $custom_field; - } - } - - return null; - } - - - /** - * Get the field map for custom fields. - * - * @return map Map of Herald field keys to field names. - * @task customfield - */ - private function getCustomFieldNameMap() { - $fields = $this->getCustomFields(); - if (!$fields) { - return array(); - } - - $map = array(); - foreach ($fields->getFields() as $field) { - $key = $field->getFieldKey(); - $name = $field->getHeraldFieldName(); - $map[$this->getHeraldKeyFromCustomKey($key)] = $name; - } - - return $map; - } - - - /** - * Get the value for a custom field. - * - * @param string Herald field key. - * @return wild Custom field value. - * @task customfield - */ - private function getCustomFieldValue($field_key) { - $field = $this->getCustomField($field_key); - if (!$field) { - return null; - } - - return $field->getHeraldFieldValue(); - } - - - /** - * Get the Herald conditions for a custom field. - * - * @param string Herald field key. - * @return list List of Herald conditions. - * @task customfield - */ - private function getCustomFieldConditions($field_key) { - $field = $this->getCustomField($field_key); - if (!$field) { - return array( - self::CONDITION_NEVER, - ); - } - - return $field->getHeraldFieldConditions(); - } - - - /** - * Get the Herald value type for a custom field and condition. - * - * @param string Herald field key. - * @param const Herald condition constant. - * @return const|null Herald value type constant, or null to use the default. - * @task customfield - */ - private function getCustomFieldValueTypeForFieldAndCondition( - $field_key, - $condition) { - - $field = $this->getCustomField($field_key); - if (!$field) { - return self::VALUE_NONE; - } - - return $field->getHeraldFieldValueType($condition); - } - - /* -( Applying Effects )--------------------------------------------------- */ diff --git a/src/applications/herald/field/HeraldField.php b/src/applications/herald/field/HeraldField.php index a66ce5f62f..422bd3ea30 100644 --- a/src/applications/herald/field/HeraldField.php +++ b/src/applications/herald/field/HeraldField.php @@ -71,19 +71,25 @@ abstract class HeraldField extends Phobject { 'FIELDCONST')); } - if (!is_string($const) || (strlen($const) > 32)) { + $limit = self::getFieldConstantByteLimit(); + if (!is_string($const) || (strlen($const) > $limit)) { throw new Exception( pht( '"%s" class "%s" has an invalid "%s" property. Field constants '. - 'must be strings and no more than 32 bytes in length.', + 'must be strings and no more than %s bytes in length.', __CLASS__, get_class($this), - 'FIELDCONST')); + 'FIELDCONST', + new PhutilNumber($limit))); } return $const; } + final public static function getFieldConstantByteLimit() { + return 64; + } + final public static function getAllFields() { return id(new PhutilClassMapQuery()) ->setAncestorClass(__CLASS__) diff --git a/src/infrastructure/customfield/herald/PhabricatorCustomFieldHeraldField.php b/src/infrastructure/customfield/herald/PhabricatorCustomFieldHeraldField.php new file mode 100644 index 0000000000..772228c5cd --- /dev/null +++ b/src/infrastructure/customfield/herald/PhabricatorCustomFieldHeraldField.php @@ -0,0 +1,68 @@ +customField = $custom_field; + return $this; + } + + public function getCustomField() { + return $this->customField; + } + + public function supportsObject($object) { + return ($object instanceof PhabricatorCustomFieldInterface); + } + + public function getFieldsForObject($object) { + $field_list = PhabricatorCustomField::getObjectFields( + $object, + PhabricatorCustomField::ROLE_HERALD); + $field_list->setViewer(PhabricatorUser::getOmnipotentUser()); + $field_list->readFieldsFromStorage($object); + + $prefix = 'herald.custom/'; + $limit = self::getFieldConstantByteLimit(); + + $map = array(); + foreach ($field_list->getFields() as $field) { + $key = $field->getFieldKey(); + + // NOTE: This use of digestToLength() isn't preferred (you should + // normally digest a key unconditionally, so that it isn't possible to + // arrange a collision) but preserves backward compatibility. + + $full_key = $prefix.$key; + if (strlen($full_key) > $limit) { + $full_key = PhabricatorHash::digestToLength($full_key, $limit); + } + + $map[$full_key] = id(new PhabricatorCustomFieldHeraldField()) + ->setCustomField($field); + } + + return $map; + } + + public function getHeraldFieldName() { + return $this->getCustomField()->getHeraldFieldName(); + } + + public function getHeraldFieldValue($object) { + return $this->getCustomField()->getHeraldFieldValue(); + } + + public function getHeraldFieldConditions() { + return $this->getCustomField()->getHeraldFieldConditions(); + } + + public function getHeraldFieldValueType($condition) { + return $this->getCustomField()->getHeraldFieldValueType($condition); + } + +}