'Medtr4g030710' * ); * $feature = chado_generate_var('feature', $values); * @endcode * * The $values array passed to this function can be of the same format used * by the chado_select_record() function. * * If a field is a foreign key then its value is an object that contains * key/value pairs for that record. The following code provides examples * for retrieving values associated with the record, either as columns in the * original Chado table or as columns in linked records through foreign keys: * @code * // Get the feature name. * $name = $feature->name; * // Get the feature unique name. * $uniquename = $feature->uniquename; * // Get the feature type. Because the type name is obtained via * // a foreign key with the cvterm table, the objects are nested * // and we can follow the foreign key fields to retrieve those values * $type = $feature->type_id->name; * // Get the name of the vocabulary. * $cv = $feature->type_id->cv_id->name; * // Get the vocabulary id. * $cv_id = $feature->type_id->cv_id->cv_id; * @endcode * * * This will return an object if there is only one feature with the name * Medtr4g030710 or it will return an array of feature objects if more than one * feature has that name. * * Note to Module Designers: Fields can be excluded by default from these * objects by implementing one of the following hooks: * - hook_exclude_field_from_tablename_by_default (where tablename is the * name of the table): This hook allows you to add fields to be excluded * on a per table basis. Simply implement this hook to return an array of * fields to be excluded. The following example will ensure that * feature.residues is excluded from a feature object by default: * @code * mymodule_exclude_field_from_feature_by_default() { * return array('residues' => TRUE); * } * @endcode * - hook_exclude_type_by_default: * This hook allows you to exclude fields using conditional. This * function should return an array of postgresql types mapped to criteria. * If the field types of any table match the criteria then the field * is excluded. Tokens available in criteria are >field_value< * and >field_name<. The following example will exclude all text * fields with a length > 50. Thus if $feature.residues is longer than * 50 it will be excluded, otherwise it will be added. * @code * mymodule_exclude_type_by_default() { * return array('text' => 'length(>field_value< ) > 50'); * } * @endcode * * * @param $table * The name of the base table to generate a variable for * @param $values * A select values array that selects the records you want from the base table * (this has the same form as chado_select_record) * @param $base_options * An array containing options for the base table. For example, an * option of 'order_by' may be used to sort results in the base table * if more than one are returned. The options must be compatible with * the options accepted by the chado_select_record() function. * Additionally, These options are available for this function: * -return_array: * can be provided to force the function to always return an array. Default * behavior is to return a single record if only one record exists or to * return an array if multiple records exist. * - include_fk: * an array of FK relationships to follow. By default, the * chado_select_record function will follow all FK relationships but this * may generate more queries then is desired slowing down this function * call when there are lots of FK relationships to follow. Provide an * array specifying the fields to include. For example, if expanding a * property table (e.g. featureprop) and you want the CV and accession * but do not want the DB the following array would work: * * $table_options = array( * 'include_fk' => array( * 'type_id' => array( * 'cv_id' => 1, * 'dbxref_id' => 1, * ) * ) * ); * * The above array will expand the 'type_id' of the property table but only * further expand the cv_id and the dbxref_id and will go no further. * - pager: * Use this option if it is desired to return only a subset of results * so that they may be shown within a Drupal-style pager. This should be * an array with two keys: 'limit' and 'element'. The value of 'limit' * should specify the number of records to return and 'element' is a * unique integer to differentiate between pagers when more than one * appear on a page. The 'element' should start with zero and increment by * one for each pager. * * @return * Either an object (if only one record was selected from the base table) * or an array of objects (if more than one record was selected from the * base table). If the option 'return_array' is provided the function * always returns an array. * * @ingroup tripal_chado_variables_api */ function chado_generate_var($table, $values, $base_options = []) { $all = new stdClass(); $return_array = 0; if (array_key_exists('return_array', $base_options)) { $return_array = 1; } $include_fk = FALSE; if (array_key_exists('include_fk', $base_options)) { $include_fk = $base_options['include_fk']; } $pager = []; if (array_key_exists('pager', $base_options)) { $pager = $base_options['pager']; } // get description for the current table------------------------------------- $table_desc = chado_get_schema($table); if (!$table_desc or count($table_desc) == 0) { tripal_report_error('tripal_chado', TRIPAL_ERROR, "chado_generate_var: The table '%table' has not been defined " . "and cannot be expanded. If this is a custom table, please add it using the Tripal " . "custom table interface. Values: %values", ['%table' => $table, '%values' => print_r($values, TRUE)]); if ($return_array) { return []; } return FALSE; } $table_primary_key = $table_desc['primary key'][0]; $table_columns = array_keys($table_desc['fields']); // Expandable fields without value needed for criteria----------------------- // Add in the default expandable arrays // These are used for later expanding fields, tables, foreign keys and nodes $all->expandable_fields = []; $all->expandable_foreign_keys = []; if (array_key_exists('referring_tables', $table_desc) and $table_desc['referring_tables']) { $all->expandable_tables = $table_desc['referring_tables']; } else { $all->expandable_tables = []; } $all->expandable_nodes = []; // Get fields to be removed by name................................. // This gets all implementations of hook_exclude_field_from__by_default() // where
is the current table a variable is being created for. // This allows modules to specify that some fields should be excluded by default // For example, tripal core provides a tripal_chado_exclude_field_from_feature_by_default() // which says that we usually don't want to include the residues field by // default since it can be very large and cause performance issues. // If a field is excluded by default it can always be expanded at a later // point by calling chado_expand_var($chado_var, 'field', // ); // First get an array of all the fields to be removed for the current table // module_invoke_all() is drupal's way of invoking all implementations of the // specified hook and merging all of the results. // $fields_to_remove should be an array with the keys matching field names // and the values being strings to be executed using php_eval() to determine // whether to exclude the field (evaluates to TRUE) or not (evaluates to FALSE) $fields_to_remove = module_invoke_all('exclude_field_from_' . $table . '_by_default'); // Now, for each field to be removed foreach ($fields_to_remove as $field_name => $criteria) { //Replace with the current field name $field_name_safe = preg_replace("/\'\"\\\/", '\\1', $field_name); $criteria = preg_replace('/ /', $field_name_safe, $criteria); // If field_value needed we can't deal with this field yet. if (preg_match('/ /', $criteria)) { break; } // If criteria then remove from query // @coder-ignore: only module designers can populate $criteria -not a // security risk. $success = php_eval(''); if ($success) { unset($table_columns[array_search($field_name, $table_columns)]); unset($fields_to_remove[$field_name]); $all->expandable_fields[] = $table . '.' . $field_name; } } // Get fields to be removed by type................................ // This gets all implementations of hook_exclude_type_by_default(). // This allows modules to specify that some types of fields should be excluded // by default For example, tripal core provides a // tripal_chado_exclude_type_by_default() which says that text fields are // often very large and if they are longer than 250 characters then // we want to exclude them by default // If a field is excluded by default it can always be expanded at a later // point by calling chado_expand_var($chado_var, 'field', //); // First get an array of all the types of fields to be removed for the current // table module_invoke_all() is drupal's way of invoking all implementations // of the specified hook and merging all of the results. // $types_to_remove should be an array with the keys matching field names // and the values being strings to be executed using php_eval() to determine // whether to exclude the field (evaluates to TRUE) or not (evaluates to FALSE) // (ie: array('text' => 'strlen(" ") > 100'); $types_to_remove = module_invoke_all('exclude_type_by_default'); // Get a list of all the types of fields // the key is the type of field and the value is an array of fields of this // type. $field_types = []; foreach ($table_desc['fields'] as $field_name => $field_array) { $field_types[$field_array['type']][] = $field_name; } // We want to use the types to remove in conjunction with our table field // descriptions to determine which fields might need to be removed. foreach ($types_to_remove as $field_type => $criteria) { // If there are fields of that type to remove. if (isset($field_types[$field_type])) { // Do any processing needed on the php criteria //replace with the current field name. $field_name_safe = preg_replace('/\'|"|\\\/', '\\1', $field_name); $criteria = preg_replace('/ /', $field_name_safe, $criteria); foreach ($field_types[$field_type] as $field_name) { // if field_value needed we can't deal with this field yet if (preg_match('//', $criteria)) { $fields_to_remove[$field_name] = $criteria; continue; } // If criteria then remove from query // (as long as is not needed for the criteria to be // evaluated) @coder-ignore: only module designers can populate //$criteria -not a security risk. $success = php_eval(''); if ($success) { unset($table_columns[array_search($field_name, $table_columns)]); $all->expandable_fields[] = $table . '.' . $field_name; } } // End of foreach field of that type. } } // End of foreach type to be removed. // Get the values for the record in the current table------------------------- $results = chado_select_record($table, $table_columns, $values, $base_options); if ($results) { // Iterate through each result. foreach ($results as $key => $object) { // Add empty expandable_x arrays. $object->expandable_fields = $all->expandable_fields; $object->expandable_foreign_keys = $all->expandable_foreign_keys; $object->expandable_tables = $all->expandable_tables; $object->expandable_nodes = $all->expandable_nodes; // add curent table $object->tablename = $table; // For Tripal v2 compatibility // check if the current table maps to a node type------------------------- // If this table is connected to a node there will be a chado_tablename // table in drupal. $base_tables = chado_get_base_tables(); if (module_exists('tripal_core') and db_table_exists('chado_' . $table)) { // That has a foreign key to this one ($table_desc['primary key'][0] // and to the node table (nid). $sql = " SELECT $table_primary_key, nid FROM {chado_$table} WHERE $table_primary_key = :$table_primary_key "; $mapping = db_query($sql, [":$table_primary_key" => $object->{$table_primary_key}])->fetchObject(); if ($mapping and $mapping->$table_primary_key) { $object->nid = $mapping->nid; $object->expandable_nodes[] = $table; } } // Check to see if the current record maps to an entity. Because // multiple bundles can map to the same table we have to check // all bundles for this table. $entity_id = chado_get_record_entity_by_table($table, $object->{$table_primary_key}); if ($entity_id) { $object->entity_id = $entity_id; } // Remove any fields where criteria needs to be evalulated---------------- // The fields to be removed can be populated by implementing either // hook_exclude_field_from_
_by_default() where
is the // current table OR hook_exclude_type_by_default() where there are fields // of the specified type in the current table It only reaches this point // if the criteria specified for whether or not to exclude the field // includes which means it has to be evaluated after // the query has been executed. foreach ($fields_to_remove as $field_name => $criteria) { // If the field is an object then we don't support exclusion of it // For example, if the field is a foreign key if (!isset($object->{$field_name})) { break; } // Replace with the actual value of the field from the // query. $field_name_safe = preg_replace('/\'|"|\\\/', '\\1', $object->{$field_name}); $criteria = preg_replace('//', $field_name_safe, $criteria); // evaluate the criteria, if TRUE is returned then exclude the field // excluded fields can be expanded later by calling // chado_expand_var($var, 'field', ); $success = php_eval(''); if ($success) { unset($object->{$field_name}); $object->expandable_fields[] = $table . '.' . $field_name; } } // Recursively follow foreign key relationships nesting objects as we go------------------------ if (array_key_exists('foreign keys', $table_desc) and $table_desc['foreign keys']) { foreach ($table_desc['foreign keys'] as $foreign_key_array) { $foreign_table = $foreign_key_array['table']; foreach ($foreign_key_array['columns'] as $foreign_key => $primary_key) { // Note: Foreign key is the field in the current table whereas // primary_key is the field in the table referenced by the foreign // key, don't do anything if the foreign key is empty if (empty($object->{$foreign_key})) { continue; } if (is_array($include_fk)) { // Don't recurse if the callee has supplied an $fk_include list // and this FK table is not in the list. if (is_array($include_fk) and !array_key_exists($foreign_key, $include_fk)) { $object->expandable_foreign_keys[] = $table . '.' . $foreign_key . ' => ' . $foreign_table; continue; } } // If we have the option but it is not an array then we don't // recurse any further. if ($include_fk === TRUE) { $object->expandable_foreign_keys[] = $table . '.' . $foreign_key . ' => ' . $foreign_table; continue; } // Get the record from the foreign table. $foreign_values = [$primary_key => $object->{$foreign_key}]; $options = []; if (is_array($include_fk)) { $options['include_fk'] = $include_fk[$foreign_key]; } $foreign_object = chado_generate_var($foreign_table, $foreign_values, $options); // Add the foreign record to the current object in a nested manner. $object->{$foreign_key} = $foreign_object; // Flatten expandable_x arrays so only in the bottom object. if (property_exists($object->{$foreign_key}, 'expandable_fields') and is_array($object->{$foreign_key}->expandable_fields)) { $object->expandable_fields = array_merge( $object->expandable_fields, $object->{$foreign_key}->expandable_fields ); unset($object->{$foreign_key}->expandable_fields); } if (property_exists($object->{$foreign_key}, 'expandable_foreign_keys') and is_array($object->{$foreign_key}->expandable_foreign_keys)) { $object->expandable_foreign_keys = array_merge( $object->expandable_foreign_keys, $object->{$foreign_key}->expandable_foreign_keys ); unset($object->{$foreign_key}->expandable_foreign_keys); } if (property_exists($object->{$foreign_key}, 'expandable_tables') and is_array($object->{$foreign_key}->expandable_tables)) { $object->expandable_tables = array_merge( $object->expandable_tables, $object->{$foreign_key}->expandable_tables ); unset($object->{$foreign_key}->expandable_tables); } if (property_exists($object->{$foreign_key}, 'expandable_nodes') and is_array($object->{$foreign_key}->expandable_nodes)) { $object->expandable_nodes = array_merge( $object->expandable_nodes, $object->{$foreign_key}->expandable_nodes ); unset($object->{$foreign_key}->expandable_nodes); } } } $results[$key] = $object; } } } // Convert the results into an array. $results_arr = []; foreach ($results as $record) { $results_arr[] = $record; } // Check only one result returned. if (!$return_array) { if (sizeof($results_arr) == 1) { // Add results to object. return $results_arr[0]; } elseif (!empty($results_arr)) { return $results_arr; } else { // No results returned. } } // The caller has requested results are always returned as // an array. else { if (!$results_arr) { return []; } else { return $results_arr; } } } /** * Retrieves fields, or tables that were excluded by default from a variable. * * The chado_generate_var() function automatically excludes some * fields and tables from the default form of a variable. Fields that are * extremely long, such as text fields are automatically excluded to prevent * long page loads. Linking tables that have a many-to-one relationship with * the record are also excluded. This function allows for custom expansion * of the record created by chado_generate_var() by specifyin the field and * tables that should be added. * * Example Usage: * * @code * // Get a chado object to be expanded * $values = array( * 'name' => 'Medtr4g030710' * ); * $features = chado_generate_var('feature', $values); * // Expand the feature.residues field * $feature = chado_expand_var($feature, 'field', 'feature.residues'); * // Expand the feature properties (featureprop table) * $feature = chado_expand_var($feature, 'table', 'featureprop'); * @endcode * * If a field is requested, it's value is added where it normally is expected * in the record. If a table is requested then a new key/value element is * added to the record. The key is the table's name and the value is an * array of records (of the same type created by chado_generate_var()). For * example, expanding a 'feature' record to include a 'pub' record via the * 'feature_pub' table. The following provides a simple example for how * the 'feature_pub' table is added. * * @code * array( * 'feature_id' => 1 * 'name' => 'blah', * 'uniquename' => 'blah', * .... * 'feature_pub => array( * [pub object], * [pub object], * [pub object], * [pub object], * ) * ) * @endcode * * where [pub object] is a record of a publication as created by * chado_generate_var(). * * If the requested table has multiple foreign keys, such as the 'featureloc' * or 'feature_genotype' tables, then an additional level is added to the * array where the foreign key column names are added. An example feature * record with an expanded featureloc table is shown below: * * @code * array( * 'feature_id' => 1 * 'name' => 'blah', * 'uniquename' => 'blah', * .... * 'featureloc => array( * 'srcfeature_id' => array( * [feature object], * ... * ) * 'feature_id' => array( * [feature object], * ... * ) * ) * ) * @endcode * * @param $object * This must be an object generated using chado_generate_var() * @param $type * Indicates what is being expanded. Must be one of 'field', 'foreign_key', * 'table', 'node'. While field and node are self-explanitory, it might help * to note that 'table' refers to tables that have a foreign key pointing to * the current table (ie: featureprop is a table that can be expanded for * features) and 'foreign_key' expands a foreign key in the current table * that might have been excluded (ie: feature.type_id for features). * @param $to_expand * The name of the field/foreign_key/table/node to be expanded * @param $table_options * - order_by: * An array containing options for the base table. For example, an * option of 'order_by' may be used to sort results in the base table * if more than one are returned. The options must be compatible with * the options accepted by the chado_select_record() function. * - return_array: * Additionally, The option 'return_array' can be provided to force * the function to expand tables as an array. Default behavior is to expand * a table as single record if only one record exists or to expand as an * array if multiple records exist. * - include_fk: * an array of FK relationships to follow. By default, the * chado_expand_var function will follow all FK relationships but this * may generate more queries then is desired slowing down this function call * when there are lots of FK relationships to follow. Provide an array * specifying the fields to include. For example, if expanding a property * table (e.g. featureprop) and you want the CV and accession but do not * want the DB the following array would work: * $table_options = array( * 'include_fk' => array( * 'type_id' => array( * 'cv_id' => 1, * 'dbxref_id' => 1, * ) * ) * ); * * The above array will expand the 'type_id' of the property table but only * further expand the cv_id and the dbxref_id and will go no further. * - pager: * Use this option if it is desired to return only a subset of results * so that they may be shown within a Drupal-style pager. This should be * an array with two keys: 'limit' and 'element'. The value of 'limit' * should specify the number of records to return and 'element' is a * unique integer to differentiate between pagers when more than one * appear on a page. The 'element' should start with zero and increment by * one for each pager. This only works when type is a 'table'. * - filter: * This options is only used where type=table and allows you to * expand only a subset of results based on the given criteria. Criteria * should provided as an array of [field name] => [value] similar to the * values array provided to chado_generate_var(). For example, when * expanding the featureprop table for a feature, you will already get only * properties for that feature, this option allows you to further get only * properties of a given type by passing in * array('type_id' => array('name' => [name of type])) * * @return * A chado object supplemented with the field/table/node requested to be * expanded. If the type is a table and it has already been expanded no * changes is made to the returned object * * * @ingroup tripal_chado_variables_api */ function chado_expand_var($object, $type, $to_expand, $table_options = []) { // Make sure we have a value. if (!$object) { $trace = (debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS)); tripal_report_error('tripal_chado', TRIPAL_ERROR, 'Missing $object argument to the chado_expand_var function from %caller (line: %line)', ['%caller' => $trace[1]['function'], '%line' => $trace[0]['line']]); return $object; } // Check to see if we are expanding an array of objects. if (is_array($object)) { foreach ($object as $index => $o) { $object[$index] = chado_expand_var($o, $type, $to_expand); } return $object; } // Get the base table name. $base_table = $object->tablename; switch ($type) { case "field": //------------------------------------------------------------ if (preg_match('/(\w+)\.(\w+)/', $to_expand, $matches)) { $tablename = $matches[1]; $fieldname = $matches[2]; $table_desc = chado_get_schema($tablename); // BASE CASE: the field is from the current table. if ($base_table == $tablename) { // Use the table description to fully describe the current object // in a $values array to be used to select the field from chado. $values = []; foreach ($table_desc['primary key'] as $key) { if (property_exists($object, $key)) { $values[$key] = $object->{$key}; } } // Retrieve the field from Chado. $results = chado_select_record($tablename, [$fieldname], $values); // Check that the field was retrieved correctly. if (isset($results[0])) { $object->{$fieldname} = $results[0]->{$fieldname}; $object->expanded = $to_expand; } // If it wasn't retrieved correctly, we need to warn the administrator. } // RECURSIVE CASE: the field is in a nested object. else { // We want to look at each field and if it's an object then we want to // attempt to expand the field in it via recursion. foreach ((array) $object as $field_name => $field_value) { if (is_object($field_value)) { $object->{$field_name} = chado_expand_var( $field_value, 'field', $to_expand ); } } // End of for each field in the current object. } } // Otherwise we weren't able to extract the parts of the field to expand // Thus we will warn the administrator. else { tripal_report_error('tripal_chado', TRIPAL_ERROR, 'chado_expand_var: Field (%field) not in the right format. " . "It should be .', ['%field' => $to_expand]); } break; case "foreign_key": //----------------------------------------------------- if (preg_match('/(\w+)\.(\w+) => (\w+)/', $to_expand, $matches)) { $table_name = $matches[1]; $field_name = $matches[2]; $foreign_table = $matches[3]; $table_desc = chado_get_schema($table_name); // BASE CASE: The foreign key is from the current table. if ($base_table == $table_name) { // Get the value of the foreign key from the object $field_value = $object->{$field_name}; // Get the name of the field in the foreign table using the table // description For example, with the // feature.type_id => cvterm.cvterm_id we need cvterm_id $foreign_field_name = FALSE; foreach ($table_desc['foreign keys'][$foreign_table]['columns'] as $left => $right) { if ($right == $field_name) { $foreign_field_name = $left; } } // Check that we were able to determine the field name in the foreign // table. if ($foreign_field_name) { // Generate a chado variable of the foreign key // For example, if the foreign key to expand is feature.type_id // then we want to generate a chado cvterm variable that matches the // feature.type_id. $foreign_var = chado_generate_var( $foreign_table, // thus in the example above, generate a cvterm var [$foreign_field_name => $field_value], // where the cvterm.cvterm_id = feature.type_id value $table_options //pass in the same options given to this function ); // Check that the foreign object was returned. if ($foreign_var) { // It was so now we can add this chado variable to our current // object in place of the key value. $object->{$field_name} = $foreign_var; $object->expanded = $to_expand; } // Otherwise we weren't able to expand the foreign key else { tripal_report_error('tripal_chado', TRIPAL_ERROR, 'chado_expand_var: unable to retrieve the object desribed by the foreign key while trying to expand %fk.', ['%fk' => $to_expand]); } } // Else we were unable to determine the field name in the foreign table. else { tripal_report_error('tripal_chado', TRIPAL_ERROR, 'chado_expand_var: unable to determine the field name in the table the foreign key points to while trying to expand %fk.', ['%fk' => $to_expand]); } } // RECURSIVE CASE: Check any nested objects. else { foreach ((array) $object as $field_name => $field_value) { if (is_object($field_value)) { $object->{$field_name} = chado_expand_var( $field_value, 'foreign_key', $to_expand ); } } //End of for each field in the current object. } } // Otherwise we weren't able to extract the parts of the foreign key to // expand thus we will warn the administrator. else { tripal_report_error('tripal_chado', TRIPAL_ERROR, 'chado_expand_var: foreign_key (%fk) not in the right format. " . "It should be .', ['%fk' => $to_expand]); } break; case "table": //------------------------------------------------------------ $foreign_table = $to_expand; // BASE CASE: don't expand the table it already is expanded if (property_exists($object, $foreign_table)) { return $object; } $foreign_table_desc = chado_get_schema($foreign_table); // If we don't get a foreign_table (which could happen of a custom // table is not correctly defined or the table name is mispelled then we // should return gracefully. if (!is_array($foreign_table_desc)) { return $object; } // BASE CASE: If it's connected to the base table via a FK constraint // then we have all the information needed to expand it now. if (array_key_exists($base_table, $foreign_table_desc['foreign keys'])) { foreach ($foreign_table_desc['foreign keys'][$base_table]['columns'] as $left => $right) { // if the FK value in the base table is not there then we can't expand // it, so just skip it. if (!$object->{$right}) { continue; } // If the user wants to limit the results they expand, make sure // those criteria are taken into account. if (isset($table_options['filter'])) { if (is_array($table_options['filter'])) { $filter_criteria = $table_options['filter']; $filter_criteria[$left] = $object->{$right}; } else { // If they supplied criteria but it's not in the correct format // then warn them but proceed as though criteria was not supplied. $filter_criteria = [$left => $object->{$right}]; tripal_report_error('tripal_chado', TRIPAL_WARNING, 'chado_expand_var: unable to apply supplied filter criteria since it should be an array. You supplied %criteria', ['%criteria' => print_r($table_options['filter'], TRUE)] ); } } else { $filter_criteria = [$left => $object->{$right}]; } // Generate a new object for this table using the FK values in the // base table. $new_options = $table_options; $foreign_object = chado_generate_var($foreign_table, $filter_criteria, $new_options); // If the generation of the object was successful, update the base // object to include it. if ($foreign_object) { // In the case where the foreign key relationship exists more // than once with the same table we want to alter the array // structure to include the field name. if (count($foreign_table_desc['foreign keys'][$base_table]['columns']) > 1) { if (!property_exists($object, $foreign_table)) { $object->{$foreign_table} = new stdClass(); } $object->{$foreign_table}->{$left} = $foreign_object; $object->expanded = $to_expand; } else { if (!property_exists($object, $foreign_table)) { $object->{$foreign_table} = new stdClass(); } $object->{$foreign_table} = $foreign_object; $object->expanded = $to_expand; } } // If the object returned is NULL then handle that. else { // In the case where the foreign key relationship exists more // than once with the same table we want to alter the array // structure to include the field name. if (count($foreign_table_desc['foreign keys'][$base_table]['columns']) > 1) { if (!property_exists($object, $foreign_table)) { $object->{$foreign_table} = new stdClass(); } $object->{$foreign_table}->{$left} = NULL; } else { $object->{$foreign_table} = NULL; } } } } // RECURSIVE CASE: if the table is not connected directly to the current // base table through a foreign key relationship, then maybe it has a // relationship to one of the nested objects. else { // We need to recurse -the table has a relationship to one of the nested // objects. We assume it's a nested object if the value of the field is // an object. $did_expansion = 0; foreach ((array) $object as $field_name => $field_value) { // CASE #1: This field is an already expanded foreign key and the // table to be expanded is in the table referenced by the foreign key. // First of all it can only be expanded if it's an object // And if it's a foreign key it should have a tablename property. if (is_object($field_value) AND property_exists($field_value, 'tablename')) { $object->{$field_name} = chado_expand_var($field_value, 'table', $foreign_table); } // CASE #2: This field is an already expanded object (ie: the field is // actually the expanded table name) and the table to be expanded is // related to it. // Check to see if the $field_name is a valid chado table, we don't // need to call chado_expand_var on fields that aren't tables. $check = chado_get_schema($field_name); if ($check) { $did_expansion = 1; $object->{$field_name} = chado_expand_var($field_value, 'table', $foreign_table); } } // If we did not expand this table we should return a message that the // foreign tabl could not be expanded. if (!$did_expansion) { tripal_report_error('tripal_chado', TRIPAL_ERROR, 'chado_expand_var: Could not expand %table. ' . 'The table is either not related to the base object through a foreign key relationships or ' . 'it is already expanded. First check the object to ensure it doesn’t already contain the ' . 'data needed and otherwise check the table definition using chado_get_schema() to ensure ' . 'a proper foreign key relationship is present.', ['%table' => $foreign_table]); } } break; case "node": //------------------------------------------------------------- // BASE CASE: if the node to be expanded is for our base table, then just // expand it. if ($object->tablename == $to_expand) { // Load the node based on the current objects nid (node primary key). $node = NULL; if (property_exists($object, 'nid')) { $node = node_load($object->nid); } // Try to get the nid based on the tablename. else { // Invoke all hook_node_info to avoid hard-coding the chado_$table // assumption.. foreach (module_invoke_all('node_info') as $node_info) { if (array_key_exists('chado_node_api', $node_info)) { if ($node_info['chado_node_api']['base_table'] == $object->tablename) { $key_name = $node_info['chado_node_api']['base_table'] . '_id'; $nid = chado_get_nid_from_id( $node_info['chado_node_api']['base_table'], $object->{$key_name}, $node_info['base']); if ($nid > 0) { $object->nid = $nid; $node = node_load($nid); break; } } } } } // If we have successfully loaded the node... if ($node) { // Move expandable arrays from the object into the node. $object->expanded = $to_expand; $node->expandable_fields = $object->expandable_fields; unset($object->expandable_fields); $node->expandable_tables = $object->expandable_tables; unset($object->expandable_tables); $node->expandable_nodes = $object->expandable_nodes; unset($object->expandable_nodes); // The node becomes the base object with the obejct added to it. // For example, we may start with a feature object with a name, // uniquename , type, etc. After expanding we will return the node and // at $node->feature you will find the original object. $node->{$base_table} = $object; $object = $node; } // Else we were unable to load the node. else { // Warn the administrator if (isset($object->nid)) { tripal_report_error('tripal_chado', TRIPAL_ERROR, 'chado_expand_var: No node matches the nid (%nid) supplied.', ['%nid' => $object->nid]); } else { tripal_report_error('tripal_chado', TRIPAL_NOTICE, 'chado_expand_var: There is no node for the current object:
%object
', ['%object' => print_r($object, TRUE)]); } } //End of if node. } // RECURSIVE CASE: check to see if the node to be expanded associates with // a chado table within one of the nested objects. else { // We need to recurse -the node to expand is one of the nested objects // We assume it's a nested object if the field value is an object. foreach ((array) $object as $field_name => $field_value) { if (is_object($field_value)) { $object->{$field_name} = chado_expand_var( $field_value, 'node', $to_expand ); } } // End of for each field in the current object. } break; // The $type to be expanded is not yet supported. default: tripal_report_error('tripal_chado', TRIPAL_ERROR, 'chado_expand_var: Unrecognized type (%type). Should be one of "field", "table", "node".', ['%type' => $type]); return FALSE; } // Move expandable arrays downwards ------------------------------- // If the type was either table or foreign key then a new chado variable was // generated this variable will have it's own expandable array's which need to // be moved down and merged with the base objects expandable arrays. // Thus, check all nested objects for expandable arrays // and if they have them, move them downwards. foreach ((array) $object as $field_name => $field_value) { if (is_object($field_value)) { // The current nested object has expandable arrays. if (isset($field_value->expandable_fields)) { // Move expandable fields downwards. if (isset($field_value->expandable_fields) and is_array($field_value->expandable_fields)) { // If the current object has it's own expandable fields then merge them. if (isset($object->expandable_fields)) { $object->expandable_fields = array_merge( $object->expandable_fields, $object->{$field_name}->expandable_fields ); unset($object->{$field_name}->expandable_fields); } // Otherwise, just move the expandable fields downwards. else { $object->expandable_fields = $object->{$field_name}->expandable_fields; unset($object->{$field_name}->expandable_fields); } } // Move expandable foreign keys downwards. if (isset($field_value->expandable_foreign_keys) and is_array($field_value->expandable_foreign_keys)) { // If the current object has it's own expandable foreign keys then // merge them. if (isset($object->expandable_foreign_keys)) { $object->expandable_foreign_keys = array_merge( $object->expandable_foreign_keys, $object->{$field_name}->expandable_foreign_keys ); unset($object->{$field_name}->expandable_foreign_keys); } // Otherwise, just move the expandable foreign keys downwards. else { $object->expandable_foreign_keys = $object->{$field_name}->expandable_foreign_keys; unset($object->{$field_name}->expandable_foreign_keys); } } // Move expandable tables downwards. if (isset($field_value->expandable_tables) and is_array($field_value->expandable_tables)) { // If the current object has it's own expandable tables then merge them. if (isset($object->expandable_tables)) { $object->expandable_tables = array_merge( $object->expandable_tables, $object->{$field_name}->expandable_tables ); unset($object->{$field_name}->expandable_tables); } // Otherwise, just move the expandable tables downwards. else { $object->expandable_tables = $object->{$field_name}->expandable_tables; unset($object->{$field_name}->expandable_tables); } } // Move expandable nodes downwards. if (isset($field_value->expandable_nodes) and is_array($field_value->expandable_nodes)) { // If the current object has it's own expandable tables then merge them. if (isset($object->expandable_nodes)) { $object->expandable_nodes = array_merge( $object->expandable_nodes, $object->{$field_name}->expandable_nodes ); unset($object->{$field_name}->expandable_nodes); } // Otherwise, just move the expandable tables downwards. else { $object->expandable_nodes = $object->{$field_name}->expandable_nodes; unset($object->{$field_name}->expandable_nodes); } } } } } // Move extended array downwards ---------------------------------- // This tells us what we have expanded (ie: that we succeeded) // and is needed to remove the entry from the expandable array. // If there is no expanded field in the current object then check any of the // nested objects and move it down. if (!property_exists($object, 'expanded')) { // It's a nested object if the value is an object. foreach ((array) $object as $field_name => $field_value) { if (is_object($field_value)) { // Check if the current nested object has an expanded array. if (isset($field_value->expanded)) { // If so, then move it downwards. $object->expanded = $field_value->expanded; unset($field_value->expanded); } } } } // Check again if there is an expanded field in the current object // We check again because it might have been moved downwards above. if (property_exists($object, 'expanded')) { // If so, then remove the expanded identifier from the correct expandable // array.. $expandable_name = 'expandable_' . $type . 's'; if (property_exists($object, $expandable_name) and $object->{$expandable_name}) { $key_to_remove = array_search($object->expanded, $object->{$expandable_name}); unset($object->{$expandable_name}[$key_to_remove]); unset($object->expanded); } } // Finally, Return the object! return $object; }