CRM-6845 Allow API (and even BAO's) to refer to custom fields as "groupName:fieldName"

    Details

    • Type: New Feature
    • Status: Done/Fixed
    • Priority: Trivial
    • Resolution: Won't Fix
    • Affects Version/s: 3.2.3
    • Fix Version/s: 4.3.0
    • Component/s: Core CiviCRM
    • Labels:
      None

      Description

      This allow code that makes any CiviCRM calls to be ID agnostic. We should use two helper functions:

      1. given a parameter array with groupName:fieldName add the custom_ID value

      2. given a result array with custom_ID add the groupName:fieldName

      lobo

        Attachments

          Activity

          [CRM-6845] Allow API (and even BAO's) to refer to custom fields as "groupName:fieldName"
          Tim Otten added a comment -

          To add a little more about our current approach: we discussed adding support for name-based keys to the Civi API. For example, one might write:

          $params = array(
          'contact_type' => 'Individual',
          'first_name' => 'Bob',
          'last_name' => 'Roberts',
          'mygroup:myfield' => 'My Value', // used to require "custom_123"
          );
          $result = civicrm_contact_create($params);

          This goes a little deeper than our first conversation, e.g. the $result should also use named-based keys, and control parameters such as 'return.*' should also respect the name-based keys. We also need to maintain backward compatibility. To avoid confusion among API consumers, the decision to use name-based keys or id-based keys should be all-or-nothing. I think a more fully-fleshed snippet would be:

          $params = array(
          '#format' => 'group:field',
          'contact_type' => 'Individual',
          'mygroup:myfield' => 'My Value',
          'return.mygroup:myfield' => 1,
          'return.mygroup:myfield2' => 1,
          'return.display_name' => 1,
          );
          $result = civicrm_contact_get($params);
          // assert isset($params[*]['mygroup:myfield'])
          // assert isset($params[*]['mygroup:myfield2'])
          // assert isset($params[*]['display_name'])

          The attached file offers some more steps to help us get there. There are some revisions to CRM_Core_BAO_CustomField:

          1. There's a new function, getFullnameIdMappings(), which is an alternative to getCustomFieldID(). In contexts with a large number of custom-data fields or a large-number of operations, getFullnameIdMappings should offer a performance improvement. Note the two levels of caching (PHP "static" and CRM_Core_BAO_Cache).

          2. There's a commented-out revision to getKeyID. This should have the domino-effect that we discussed – e.g. _civicrm_custom_format_params calls getKeyID, so now several API calls can accept inbound $params data with name-based keys.

          In theory, we might get a similar domino affect to provide name-based keys for outbound $result data by modifying CRM_Core_BAO_CustomGroup::formatGroupTree and CRM_Contact_BAO_Contact. My concern (as someone who reads code) is that it would make the data-structures harder to predict/understand – e.g. sometimes the data uses "custom_123" format, and sometimes it uses "custom_123_4" format, and sometimes "mygroup:myfield", and sometimes a "custom" data-tree.

          As an alternative, we could do all of the translation in the API layer. This is like _civicrm_custom_format_params, but it also address control-parameters and output-data. The attached draft is an idea for how to implement that layer. Here are a few notable things about the approach:

          • It doesn't change any core functions, so all existing application logic, unit tests, etc. remain correct.
          • It can rewrite input data, control parameters, and output data.
          • The CRM_API_Formatter_FieldRewrite should be very amenable to unit testing – e.g. you can inject a mock-value for _nameIdMapping.
          • If we encounter some issue with the new naming convention, then we can put our fixes in the CRM_API_Formatter layer – without revising application logic or core functions.
          • It leaves me some options in plotting a migration path from my current mechanism to the official mechanism.

          Let me know what you think of the approach/patch. If the basic approach works, I can clean it up a bit (e.g. make a proper patch, add unit tests, and provide a backward-compatible version).

          Donald A. Lobo added a comment -

          These 448 issues have not been worked on for the past 18 months.

          Doing a bulk close of old issues to make the issue queue more manageable. We should do this on a periodic basis.

            People

            • Assignee:
              Donald A. Lobo
              Reporter:
              Donald A. Lobo

              Dates

              • Created:
                Updated:
                Resolved: