tripal_core.chado_variables.api.inc 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670
  1. <?php
  2. /**
  3. * @file
  4. * This API generates objects containing the full details of a record(s) in chado.
  5. */
  6. /**
  7. * Implements hook_exclude_type_by_default()
  8. *
  9. * This hooks allows fields of a specified type that match a specified criteria to be excluded by
  10. * default from any table when chado_generate_var() is called. Keep in mind that if
  11. * fields are excluded by default they can always be expanded at a later date using
  12. * chado_expand_var().
  13. *
  14. * Criteria are php strings that evaluate to either TRUE or FALSE. These strings are evaluated using
  15. * drupal_eval() which suppresses syntax errors and throws watchdog entries of type php. There are
  16. * also watchdog entries of type tripal_core stating the exact criteria evaluated. Criteria can
  17. * contain the following tokens:
  18. * - &gt;field_name&lt;
  19. * Replaced by the name of the field to be excluded
  20. * - &gt;field_value&lt;
  21. * Replaced by the value of the field in the current record
  22. * Also keep in mind that if your criteria doesn't contain the &gt;field_value&lt; token then it will be
  23. * evaluated before the query is executed and if the field is excluded it won't be included in the
  24. * query.
  25. *
  26. * @return
  27. * An array of type => criteria where the type is excluded if the criteria evaluates to TRUE
  28. *
  29. * @ingroup tripal_chado_query_api
  30. */
  31. function tripal_core_exclude_type_by_default() {
  32. return array('text' => 'strlen("&gt;field_value&lt; ") > 100');
  33. }
  34. /**
  35. * Implements hook_exclude_field_from_<tablename>_by_default()
  36. *
  37. * This hooks allows fields from a specified table that match a specified criteria to be excluded by
  38. * default from any table when chado_generate_var() is called. Keep in mind that if
  39. * fields are excluded by default they can always be expanded at a later date using
  40. * chado_expand_var().
  41. *
  42. * Criteria are php strings that evaluate to either TRUE or FALSE. These strings are evaluated using
  43. * drupal_eval() which suppresses syntax errors and throws watchdog entries of type php. There are
  44. * also watchdog entries of type tripal_core stating the exact criteria evaluated. Criteria can
  45. * contain the following tokens:
  46. * - &gt;field_name&lt;
  47. * Replaced by the name of the field to be excluded
  48. * - &gt;field_value&lt;
  49. * Replaced by the value of the field in the current record
  50. * Also keep in mind that if your criteria doesn't contain the &gt;field_value&lt; token then it will be
  51. * evaluated before the query is executed and if the field is excluded it won't be included in the
  52. * query.
  53. *
  54. * @return
  55. * An array of type => criteria where the type is excluded if the criteria evaluates to TRUE
  56. *
  57. * @ingroup tripal_chado_query_api
  58. */
  59. function tripal_core_exclude_field_from_feature_by_default() {
  60. return array();
  61. }
  62. /**
  63. * Generates an object containing the full details of a record(s) in chado.
  64. *
  65. * This differs from the objects returned by chado_select_record in so far as all foreign key
  66. * relationships have been followed meaning you have more complete details. Thus this function
  67. * should be used whenever you need a full variable and chado_select_record should be used if
  68. * you only case about a few columns.
  69. *
  70. * @param $table
  71. * The name of the base table to generate a variable for
  72. * @param $values
  73. * A select values array that selects the records you want from the base table
  74. * (this has the same form as chado_select_record)
  75. * @param $base_options
  76. * An array containing options for the base table. For example, an
  77. * option of 'order_by' may be used to sort results in the base table
  78. * if more than one are returned. The options must be compatible with
  79. * the options accepted by the chado_select_record() function.
  80. * Additionally, These options are available for this function:
  81. * -return_array:
  82. * can be provided to force the function to always return an array. Default
  83. * behavior is to return a single record if only one record exists or to return
  84. * an array if multiple records exist.
  85. * - include_fk:
  86. * an array of FK relationships to follow. By default, the
  87. * chado_select_record function will follow all FK relationships but this
  88. * may generate more queries then is desired slowing down this function call when
  89. * there are lots of FK relationships to follow. Provide an array specifying the
  90. * fields to include. For example, if expanding a property table (e.g. featureprop)
  91. * and you want the CV and accession but do not want the DB the following
  92. * array would work:
  93. *
  94. * $table_options = array(
  95. * 'include_fk' => array(
  96. * 'type_id' => array(
  97. * 'cv_id' => 1,
  98. * 'dbxref_id' => 1,
  99. * )
  100. * )
  101. * );
  102. *
  103. * The above array will expand the 'type_id' of the property table but only
  104. * further expand the cv_id and the dbxref_id and will go no further.
  105. * - pager:
  106. * Use this option if it is desired to return only a subset of results
  107. * so that they may be shown within a Drupal-style pager. This should be
  108. * an array with two keys: 'limit' and 'element'. The value of 'limit'
  109. * should specify the number of records to return and 'element' is a
  110. * unique integer to differentiate between pagers when more than one
  111. * appear on a page. The 'element' should start with zero and increment by
  112. * one for each pager. This only works when type is a 'table'.
  113. * @return
  114. * Either an object (if only one record was selected from the base table)
  115. * or an array of objects (if more than one record was selected from the base table).
  116. * If the option 'return_array' is provided the function always returns an array.
  117. *
  118. * Example Usage:
  119. * @code
  120. $values = array(
  121. 'name' => 'Medtr4g030710'
  122. );
  123. $features = chado_generate_var('feature', $values);
  124. * @endcode
  125. * This will return an object if there is only one feature with the name Medtr4g030710 or it will
  126. * return an array of feature objects if more than one feature has that name.
  127. *
  128. * Note to Module Designers: Fields can be excluded by default from these objects by implementing
  129. * one of the following hooks:
  130. * - hook_exclude_field_from_tablename_by_default (where tablename is the name of the table):
  131. * This hook allows you to add fields to be excluded on a per table basis. Simply implement
  132. * this hook to return an array of fields to be excluded. For example:
  133. * @code
  134. mymodule_exclude_field_from_feature_by_default() {
  135. return array('residues' => TRUE);
  136. }
  137. * @endcode
  138. * will ensure that feature.residues is ecluded from a feature object by default.
  139. * - hook_exclude_type_by_default:
  140. * This hook allows you to exclude fields from all tables that are of a given postgresql field
  141. * type. Simply implement this hook to return an array of postgresql types mapped to criteria.
  142. * Then all fields of that type where the criteria supplied returns TRUE will be excluded from
  143. * any table. Tokens available in criteria are &gt;field_value&lt; and &gt;field_name&lt; . For example:
  144. * @code
  145. mymodule_exclude_type_by_default() {
  146. return array('text' => 'length(&gt;field_value&lt; ) > 50');
  147. }
  148. * @endcode
  149. * 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.
  150. *
  151. * @ingroup tripal_chado_query_api
  152. */
  153. function chado_generate_var($table, $values, $base_options = array()) {
  154. $all = new stdClass();
  155. $return_array = 0;
  156. if (array_key_exists('return_array', $base_options)) {
  157. $return_array = 1;
  158. }
  159. $include_fk = 0;
  160. if (array_key_exists('include_fk', $base_options)) {
  161. $include_fk = $base_options['include_fk'];
  162. }
  163. $pager = array();
  164. if (array_key_exists('pager', $base_options)) {
  165. $pager = $base_options['pager'];
  166. }
  167. // get description for the current table----------------------------------------------------------
  168. $table_desc = chado_get_schema($table);
  169. if (!$table_desc or count($table_desc) == 0) {
  170. tripal_report_error('tripal_core', TRIPAL_ERROR,
  171. "chado_generate_var: The table '%table' has not been defined. " .
  172. "and cannot be expanded. If this is a custom table, please add it using the Tripal " .
  173. "custom table interface.", array('%table' => $table));
  174. if ($return_array) {
  175. return array();
  176. }
  177. return FALSE;
  178. }
  179. $table_primary_key = $table_desc['primary key'][0];
  180. $table_columns = array_keys($table_desc['fields']);
  181. // Expandable fields without value needed for criteria--------------------------------------------
  182. $all->expandable_fields = array();
  183. if (array_key_exists('referring_tables', $table_desc) and $table_desc['referring_tables']) {
  184. $all->expandable_tables = $table_desc['referring_tables'];
  185. }
  186. else {
  187. $all->expandable_tables = array();
  188. }
  189. $all->expandable_nodes = array();
  190. /*
  191. // Get fields to be removed by name.................................
  192. $fields_to_remove = module_invoke_all('exclude_field_from_' . $table . '_by_default');
  193. foreach ($fields_to_remove as $field_name => $criteria) {
  194. //replace &gt;field_name&lt; with the current field name &
  195. $criteria = preg_replace('/&gt;field_name&lt; /', addslashes($field_name), $criteria);
  196. // if field_value needed we can't deal with this field yet
  197. if (preg_match('/&gt;field_value&lt; /', $criteria)) {
  198. break;
  199. }
  200. //if criteria then remove from query
  201. // @coder-ignore: only module designers can populate $criteria -not security risk
  202. $success = php_eval('<?php return ' . $criteria . '; ?>');
  203. if ($success) {
  204. unset($table_columns[array_search($field_name, $table_columns)]);
  205. unset($fields_to_remove[$field_name]);
  206. $all->expandable_fields[] = $table . '.' . $field_name;
  207. }
  208. }
  209. //Get fields to be removed by type................................
  210. $types_to_remove = module_invoke_all('exclude_type_by_default');
  211. $field_types = array();
  212. foreach ($table_desc['fields'] as $field_name => $field_array) {
  213. $field_types[$field_array['type']][] = $field_name;
  214. }
  215. foreach ($types_to_remove as $field_type => $criteria) {
  216. // if there are fields of that type to remove
  217. if (is_array($field_types[$field_type])) {
  218. //replace &gt;field_name&lt; with the current field name &
  219. $criteria = preg_replace('/&gt;field_name&lt; /', addslashes($field_name), $criteria);
  220. foreach ($field_types[$field_type] as $field_name) {
  221. // if field_value needed we can't deal with this field yet
  222. if (preg_match('/&gt;field_value&lt; /', $criteria)) {
  223. $fields_to_remove[$field_name] = $criteria;
  224. continue;
  225. }
  226. // if field_value needed we can't deal with this field yet
  227. if (preg_match('/&gt;field_value&lt; /', $criteria)) {
  228. break;
  229. }
  230. //if criteria then remove from query
  231. // @coder-ignore: only module designers can populate $criteria -not security risk
  232. $success = php_eval('<?php return ' . $criteria . '; ?>');
  233. if ($success) {
  234. unset($table_columns[array_search($field_name, $table_columns)]);
  235. $all->expandable_fields[] = $table . '.' . $field_name;
  236. }
  237. } //end of foreach field of that type
  238. }
  239. } //end of foreach type to be removed
  240. */
  241. // get the values for the record in the current table---------------------------------------------
  242. $results = chado_select_record($table, $table_columns, $values, $base_options);
  243. if ($results) {
  244. foreach ($results as $key => $object) {
  245. // Add empty expandable_x arrays
  246. $object->expandable_fields = $all->expandable_fields;
  247. $object->expandable_tables = $all->expandable_tables;
  248. $object->expandable_nodes = $all->expandable_nodes;
  249. // add curent table
  250. $object->tablename = $table;
  251. // check if the current table maps to a node type-----------------------------------------------
  252. // if this table is connected to a node there will be a chado_tablename table in drupal
  253. if (db_table_exists('chado_' . $table)) {
  254. // that has a foreign key to this one ($table_desc['primary key'][0]
  255. // and to the node table (nid)
  256. $sql = "
  257. SELECT $table_primary_key, nid
  258. FROM {chado_$table}
  259. WHERE $table_primary_key = :$table_primary_key
  260. ";
  261. $mapping = db_query($sql, array(":$table_primary_key" => $object->{$table_primary_key}))->fetchObject();
  262. if ($mapping and $mapping->$table_primary_key) {
  263. $object->nid = $mapping->nid;
  264. $object->expandable_nodes[] = $table;
  265. }
  266. }
  267. // remove any fields where criteria needs to be evalulated---------------------------------------
  268. /* foreach ($fields_to_remove as $field_name => $criteria) {
  269. if (!isset($object->{$field_name})) {
  270. break;
  271. }
  272. $criteria = preg_replace('/&gt;field_value&lt; /', addslashes($object->{$field_name}), $criteria);
  273. $success = php_eval('<?php return ' . $criteria . '; ?>');
  274. if ($success) {
  275. unset($object->{$field_name});
  276. $object->expandable_fields[] = $table . '.' . $field_name;
  277. }
  278. }
  279. */
  280. // recursively follow foreign key relationships nesting objects as we go------------------------
  281. if ($table_desc['foreign keys']) {
  282. foreach ($table_desc['foreign keys'] as $foreign_key_array) {
  283. $foreign_table = $foreign_key_array['table'];
  284. foreach ($foreign_key_array['columns'] as $foreign_key => $primary_key) {
  285. // Note: Foreign key is the field in the current table whereas primary_key is the field in
  286. // the table referenced by the foreign key
  287. //Dont do anything if the foreign key is empty
  288. if (empty($object->{$foreign_key})) {
  289. continue;
  290. }
  291. if ($include_fk) {
  292. // don't recurse if the callee has supplied an $fk_include list and this
  293. // FK table is not in the list.
  294. if (is_array($include_fk) and !array_key_exists($foreign_key, $include_fk)) {
  295. continue;
  296. }
  297. // if we have the option but it is not an array then we don't recurse any furutehr
  298. if (!is_array($include_fk)) {
  299. continue;
  300. }
  301. }
  302. // get the record from the foreign table
  303. $foreign_values = array($primary_key => $object->{$foreign_key});
  304. $options = array();
  305. if (is_array($include_fk)) {
  306. $options['include_fk'] = $include_fk[$foreign_key];
  307. }
  308. $foreign_object = chado_generate_var($foreign_table, $foreign_values, $options);
  309. // add the foreign record to the current object in a nested manner
  310. $object->{$foreign_key} = $foreign_object;
  311. // Flatten expandable_x arrays so only in the bottom object
  312. if (property_exists($object->{$foreign_key}, 'expandable_fields') and
  313. is_array($object->{$foreign_key}->expandable_fields)) {
  314. $object->expandable_fields = array_merge(
  315. $object->expandable_fields,
  316. $object->{$foreign_key}->expandable_fields
  317. );
  318. unset($object->{$foreign_key}->expandable_fields);
  319. }
  320. if (property_exists($object->{$foreign_key}, 'expandable_tables') and
  321. is_array($object->{$foreign_key}->expandable_tables)) {
  322. $object->expandable_tables = array_merge(
  323. $object->expandable_tables,
  324. $object->{$foreign_key}->expandable_tables
  325. );
  326. unset($object->{$foreign_key}->expandable_tables);
  327. }
  328. if (property_exists($object->{$foreign_key}, 'expandable_nodes') and
  329. is_array($object->{$foreign_key}->expandable_nodes)) {
  330. $object->expandable_nodes = array_merge(
  331. $object->expandable_nodes,
  332. $object->{$foreign_key}->expandable_nodes
  333. );
  334. unset($object->{$foreign_key}->expandable_nodes);
  335. }
  336. }
  337. }
  338. $results[$key] = $object;
  339. }
  340. }
  341. }
  342. // convert the results into an array
  343. $results_arr = array();
  344. foreach ($results as $record) {
  345. $results_arr[] = $record;
  346. }
  347. // check only one result returned
  348. if (!$return_array) {
  349. if (sizeof($results_arr) == 1) {
  350. // add results to object
  351. return $results_arr[0];
  352. }
  353. elseif (!empty($results_arr)) {
  354. return $results_arr;
  355. }
  356. else {
  357. // no results returned
  358. }
  359. }
  360. // the caller has requested results are always returned as
  361. // an array
  362. else {
  363. if (!$results_arr) {
  364. return array();
  365. }
  366. else {
  367. return $results_arr;
  368. }
  369. }
  370. }
  371. /**
  372. * Retrieves fields/tables/nodes that were excluded by default from a variable and adds them
  373. *
  374. * This function exists to allow chado_generate_var() to excldue some
  375. * fields/tables/nodes from the default form of a variable without making it extremely difficult for
  376. * the tripal admin to get at these variables if he/she wants them.
  377. *
  378. * @param $object
  379. * This must be an object generated using chado_generate_var()
  380. * @param $type
  381. * Must be one of 'field', 'table', 'node'. Indicates what is being expanded.
  382. * @param $to_expand
  383. * The name of the field/table/node to be expanded
  384. * @param $table_options
  385. * - order_by:
  386. * An array containing options for the base table. For example, an
  387. * option of 'order_by' may be used to sort results in the base table
  388. * if more than one are returned. The options must be compatible with
  389. * the options accepted by the chado_select_record() function.
  390. * - return_array:
  391. * Additionally, The option 'return_array' can be provided to force
  392. * the function to expand tables as an array. Default behavior is to expand
  393. * a table as single record if only one record exists or to expand as an array if
  394. * multiple records exist.
  395. * - include_fk:
  396. * an array of FK relationships to follow. By default, the
  397. * chado_expand_var function will follow all FK relationships but this
  398. * may generate more queries then is desired slowing down this function call when
  399. * there are lots of FK relationships to follow. Provide an array specifying the
  400. * fields to include. For example, if expanding a property table (e.g. featureprop)
  401. * and you want the CV and accession but do not want the DB the following
  402. * array would work:
  403. * $table_options = array(
  404. * 'include_fk' => array(
  405. * 'type_id' => array(
  406. * 'cv_id' => 1,
  407. * 'dbxref_id' => 1,
  408. * )
  409. * )
  410. * );
  411. *
  412. * The above array will expand the 'type_id' of the property table but only
  413. * further expand the cv_id and the dbxref_id and will go no further.
  414. * - pager:
  415. * Use this option if it is desired to return only a subset of results
  416. * so that they may be shown within a Drupal-style pager. This should be
  417. * an array with two keys: 'limit' and 'element'. The value of 'limit'
  418. * should specify the number of records to return and 'element' is a
  419. * unique integer to differentiate between pagers when more than one
  420. * appear on a page. The 'element' should start with zero and increment by
  421. * one for each pager. This only works when type is a 'table'.
  422. * @return
  423. * A chado object supplemented with the field/table/node requested to be expanded.
  424. * If the type is a table and it has already been expanded no changes is made to the
  425. * returned object
  426. *
  427. * Example Usage:
  428. * @code
  429. // Get a chado object to be expanded
  430. $values = array(
  431. 'name' => 'Medtr4g030710'
  432. );
  433. $features = chado_generate_var('feature', $values);
  434. // Expand the organism node
  435. $feature = chado_expand_var($feature, 'node', 'organism');
  436. // Expand the feature.residues field
  437. $feature = chado_expand_var($feature, 'field', 'feature.residues');
  438. // Expand the feature properties (featureprop table)
  439. $feature = chado_expand_var($feature, 'table', 'featureprop');
  440. * @endcode
  441. *
  442. * @ingroup tripal_chado_query_api
  443. */
  444. function chado_expand_var($object, $type, $to_expand, $table_options = array()) {
  445. // make sure we have a value
  446. if (!$object) {
  447. tripal_report_error('tripal_core', TRIPAL_ERROR,
  448. 'Cannot pass non array as argument, $object, to chado_expand_var function.', array());
  449. return $object;
  450. }
  451. // check to see if we are expanding an array of objects
  452. if (is_array($object)) {
  453. foreach ($object as $index => $o) {
  454. $object[$index] = chado_expand_var($o, $type, $to_expand);
  455. }
  456. return $object;
  457. }
  458. // get the base table name
  459. $base_table = $object->tablename;
  460. switch ($type) {
  461. case "field": //--------------------------------------------------------------------------------
  462. if (preg_match('/(\w+)\.(\w+)/', $to_expand, $matches)) {
  463. $tablename = $matches[1];
  464. $fieldname = $matches[2];
  465. $table_desc = chado_get_schema($tablename);
  466. $values = array();
  467. foreach ($table_desc['primary key'] as $key) {
  468. if(property_exists($object, $key)) {
  469. $values[$key] = $object->{$key};
  470. }
  471. }
  472. if ($base_table == $tablename) {
  473. //get the field
  474. $results = chado_select_record($tablename, array($fieldname), $values);
  475. $object->{$fieldname} = $results[0]->{$fieldname};
  476. $object->expanded = $to_expand;
  477. }
  478. else {
  479. //We need to recurse -the field is in a nested object
  480. foreach ((array) $object as $field_name => $field_value) {
  481. if (is_object($field_value)) {
  482. $object->{$field_name} = chado_expand_var(
  483. $field_value,
  484. 'field',
  485. $to_expand
  486. );
  487. }
  488. } //end of for each field in the current object
  489. }
  490. }
  491. else {
  492. tripal_report_error('tripal_core', TRIPAL_ERROR,
  493. 'chado_expand_var: Field (%field) not in the right format. " .
  494. "It should be <tablename>.<fieldname>');
  495. }
  496. break;
  497. case "table": //--------------------------------------------------------------------------------
  498. $foreign_table = $to_expand;
  499. // don't expand the table it already is expanded
  500. if (array_key_exists($foreign_table, $object)) {
  501. return $object;
  502. }
  503. $foreign_table_desc = chado_get_schema($foreign_table);
  504. // If it's connected to the base table via a FK constraint
  505. if (array_key_exists($base_table, $foreign_table_desc['foreign keys'])) {
  506. foreach ($foreign_table_desc['foreign keys'][$base_table]['columns'] as $left => $right) {
  507. // if the FK value in the base table is not there then we can't expand it, so just skip it.
  508. if (!$object->{$right}) {
  509. continue;
  510. }
  511. // generate a new object for this table using the FK values in the base table.
  512. $new_options = $table_options;
  513. $foreign_object = chado_generate_var($foreign_table, array($left => $object->{$right}), $new_options);
  514. // if the generation of the object was successful, update the base object to include it.
  515. if ($foreign_object) {
  516. // in the case where the foreign key relationship exists more
  517. // than once with the same table we want to alter the array structure to
  518. // include the field name.
  519. if (count($foreign_table_desc['foreign keys'][$base_table]['columns']) > 1) {
  520. if (!property_exists($object, $foreign_table)) {
  521. $object->{$foreign_table} = new stdClass();
  522. }
  523. $object->{$foreign_table}->{$left} = $foreign_object;
  524. $object->expanded = $to_expand;
  525. }
  526. else {
  527. if (!property_exists($object, $foreign_table)) {
  528. $object->{$foreign_table} = new stdClass();
  529. }
  530. $object->{$foreign_table} = $foreign_object;
  531. $object->expanded = $to_expand;
  532. }
  533. }
  534. // if the object returned is NULL then handle that
  535. else {
  536. // in the case where the foreign key relationship exists more
  537. // than once with the same table we want to alter the array structure to
  538. // include the field name.
  539. if (count($foreign_table_desc['foreign keys'][$base_table]['columns']) > 1) {
  540. if (!property_exists($object, $foreign_table)) {
  541. $object->{$foreign_table} = new stdClass();
  542. }
  543. $object->{$foreign_table}->{$left} = NULL;
  544. }
  545. else {
  546. $object->{$foreign_table} = NULL;
  547. }
  548. }
  549. }
  550. }
  551. // if the foreign table is not connected to the base table through a FK constraint
  552. else {
  553. // We need to recurse -the table has a relationship to one of the nested objects
  554. $did_expansion = 0;
  555. foreach ((array) $object as $field_name => $field_value) {
  556. // if we have a nested object ->expand the table in it
  557. // check to see if the $field_name is a valid chado table, we don't need
  558. // to call chado_expand_var on fields that aren't tables
  559. $check = chado_get_schema($field_name);
  560. if ($check) {
  561. $did_expansion = 1;
  562. $object->{$field_name} = chado_expand_var($field_value, 'table', $foreign_table);
  563. }
  564. }
  565. // if we did not expand this table we should return a message that the foreign table
  566. // could not be expanded
  567. if (!$did_expansion) {
  568. tripal_report_error('tripal_core', TRIPAL_ERROR, 'chado_expand_var: Could not expand table, %table. It is ' .
  569. 'not in a foreign key relationship with the base object nor with any other expanded table. ' .
  570. 'Check the table definition to ensure that a proper foreign key relationship is present.',
  571. array('%table' => $foreign_table));
  572. }
  573. }
  574. break;
  575. case "node": //---------------------------------------------------------------------------------
  576. //if the node to be expanded is for our base table, then just expand it
  577. if ($object->tablename == $to_expand) {
  578. $node = node_load($object->nid);
  579. if ($node) {
  580. $object->expanded = $to_expand;
  581. $node->expandable_fields = $object->expandable_fields;
  582. unset($object->expandable_fields);
  583. $node->expandable_tables = $object->expandable_tables;
  584. unset($object->expandable_tables);
  585. $node->expandable_nodes = $object->expandable_nodes;
  586. unset($object->expandable_nodes);
  587. $node->{$base_table} = $object;
  588. $object = $node;
  589. }
  590. else {
  591. tripal_report_error('tripal_core', TRIPAL_ERROR, 'chado_expand_var: No node matches the nid (%nid) supplied.',
  592. array('%nid' => $object->nid));
  593. } //end of if node
  594. }
  595. else {
  596. //We need to recurse -the node to expand is one of the nested objects
  597. foreach ((array) $object as $field_name => $field_value) {
  598. if (is_object($field_value)) {
  599. $object->{$field_name} = chado_expand_var(
  600. $field_value,
  601. 'node',
  602. $to_expand
  603. );
  604. }
  605. } //end of for each field in the current object
  606. }
  607. break;
  608. default:
  609. tripal_report_error('tripal_core', TRIPAL_ERROR, 'chado_expand_var: Unrecognized type (%type). Should be one of "field", "table", "node".',
  610. array('%type' => $type));
  611. return FALSE;
  612. }
  613. // move extended array downwards
  614. if (!property_exists($object, 'expanded')) {
  615. //if there's no extended field then go hunting for it
  616. foreach ( (array)$object as $field_name => $field_value) {
  617. if (is_object($field_value)) {
  618. if (isset($field_value->expanded)) {
  619. $object->expanded = $field_value->expanded;
  620. unset($field_value->expanded);
  621. }
  622. }
  623. }
  624. }
  625. //try again becasue now we might have moved it down
  626. if (property_exists($object, 'expanded')) {
  627. $expandable_name = 'expandable_' . $type . 's';
  628. if (property_exists($object, $expandable_name) and $object->{$expandable_name}) {
  629. $key_to_remove = array_search($object->expanded, $object->{$expandable_name});
  630. unset($object->{$expandable_name}[$key_to_remove]);
  631. unset($object->expanded);
  632. }
  633. else {
  634. // if there is an expandable array then we've reached the base object
  635. // if we get here and don't have anything expanded then something went wrong
  636. // tripal_report_error('tripal_core', TRIPAL_ERROR,
  637. // 'chado_expand_var: Unable to expand the %type %to_expand',
  638. // array('%type'=>$type, '%to_expand'=>$to_expand),
  639. // );
  640. } //end of it we've reached the base object
  641. }
  642. return $object;
  643. }