Details
-
Type:
Bug
-
Status: Done/Fixed
-
Priority:
Trivial
-
Resolution: Fixed/Completed
-
Affects Version/s: 4.5.5
-
Fix Version/s: 4.6
-
Component/s: Core CiviCRM
-
Labels:None
-
Documentation Required?:None
Description
This seems to be a regression caused by CRM-15061.
CRM-15061 creates a new data structure, as saved in the civicrm_saved_search form_values field, for custom group/multi-select fields:
custom_NN = array('value1', 'value4', 'value5');
custon_NN_operator = 'or'
In 4.4, the data structure is:
custom_NN = array(
'value1' => 1,
'value2' =>
'value3' =>
'value4' => 1,
'value5' => 1
'CiviCRM_OP_OR' => 1
)
The 4.5 code properly saves new searches using the new data structure, but smart groups that have saved searches made during 4.4 still have the old structure which is incompatible with the new structure.
One option is to change the code to auto-detect which data structure is coming in. But that seems like it will simply preserve cruft we could do with out.
Here's some code that updates the saved search table converting to the new structure. I think this should be carefully reviewed since it could result in data loss:
$sql = "SELECT id, form_values FROM civicrm_saved_search";
$dao = CRM_Core_DAO::executeQuery($sql);
while($dao->fetch()) {
echo "Considering " . $dao->id . "\n";
$new = $data = unserialize($dao->form_values);
$update = FALSE;
while(list($field, $data_value) = each($data)) {
if(preg_match('/^custom_/', $field) && is_array($data_value)) {
if(array_key_exists('CiviCRM_OP_OR', $data_value)) {
// This indicates old-style data format. We need to fix it.
$update = TRUE;
$new_value = array();
$op = 'and';
if($data_value['CiviCRM_OP_OR'] == 1)
while(list($k, $v) = each($data_value)) {
if($v == 1)
}
// Overwrite old value.
$new[$field] = $new_value;
// Add new key for the operator.
$new["$
_operator"] = $op;
}
}
}
if($update)
}
I'm still banging my head against this one.
In an effort to fix our smart groups - I'm temporarily patching our sites with this code that adjusts for this problem at the code level rather than the data level. It's far safer to implement, but I don't think it's the right long term solution:
diff --git a/CRM/Core/BAO/CustomQuery.php b/CRM/Core/BAO/CustomQuery.php
{ + $sqlOP = ' OR '; + }index 8ca1620..920182c 100644
— a/CRM/Core/BAO/CustomQuery.php
+++ b/CRM/Core/BAO/CustomQuery.php
@@ -364,6 +364,26 @@ SELECT label, value
$qillValue = '';
$sqlOP = $wildcard ? ' OR ' : ' AND ';
$sqlValue = array();
+ // Pre 4.5 custom fields with multiple options are in the format:
+ // array(value1 => '', value2 => 1, value3 => '', value4 => 1, CiviCRM_OP_OR => 1 )
+ // indicating that value2 and value4 are chosen.
+ // 4.5 changes the structure to just include the values chosen, e.g.
+ // array(value2, value4).
+ // The presence of the CiviCRM_OP_OR operator should be a clue as to which format
+ // we are gettting.
+ if(array_key_exists('CiviCRM_OP_OR', $value)) {
+ // Overwrite the sqlOP variable
+ if($value['CiviCRM_OP_OR'] == 1)
+ else
{ + $sqlOP = ' AND '; + }+ // Now unset it
+ unset($value['civiCRM_OP_OR']);
+ // And re-create the $value array
+ $value = array_keys($value, 1);
+ }
foreach ($value as $num => &$v) {
$sep = count($value) > (1 + $num) ? ', ' : (' ' . ($wildcard ? ts('OR') : ts('AND')) . ' ');
$qillValue .= ($num ? $sep : '') . $options[$v];