tripal_core.chado_nodes.api.inc 48 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315
  1. <?php
  2. /**
  3. * @file
  4. * API to handle much of the common functionality implemented when creating a drupal node type.
  5. */
  6. /**
  7. * @defgroup tripal_chado_node_api Chado Node API
  8. * @ingroup tripal_chado_api
  9. * @{
  10. * Many Tripal modules implement Drupal node types as a means of displaying chado
  11. * records individually through Drupal as a single web page. In order to do this, many of
  12. * the same drupal hooks are implemented and the code between modules is actually quite
  13. * similar. This API aims to abstract much of the common functionality in order to make
  14. * it easier for new Tripal modules to implement drupal node types and to centralize the
  15. * maintenance effort as much as possible.
  16. *
  17. * A generic sync form has been created. See chado_node_sync_form() for
  18. * instructions on how to implement this form in your module.
  19. *
  20. * Many of the base chado tables also have associated prop, _dbxref and _relationship
  21. * tables. Generic mini-forms have been created to help you handle these forms. To
  22. * implement this functionality you call the mini-form from your module node form and
  23. * then call the associated update functions from both your hook_insert and hook_update.
  24. * The functions of interest are as follows:
  25. * - chado_add_node_form_properties() and chado_update_node_form_properties()
  26. * to provide an interface for adding/removing properties
  27. * - chado_add_node_form_dbxrefs() and chado_update_node_form_dbxrefs()
  28. * to provide an interface for adding/removing additional database references
  29. * - chado_add_node_form_relationships() and chado_update_node_form_relationships()
  30. * to provide an interface for adding/removing relationships between chado records
  31. * from your base table
  32. * @}
  33. */
  34. /**
  35. * Get chado id for a node. E.g, if you want to get 'analysis_id' from the
  36. * 'analysis' table for a synced 'chado_analysis' node, (the same for
  37. * organisms and features):
  38. * $analysis_id = chado_get_id_from_nid ('analysis', $node->nid)
  39. * $organism_id = chado_get_id_from_nid ('organism', $node->nid)
  40. * $feature_id = chado_get_id_from_nid ('feature', $node->nid)
  41. *
  42. * @param $table
  43. * The chado table the chado record is from
  44. * @param $nid
  45. * The value of the primary key of node
  46. * @param $linking_table
  47. * The Drupal table linking the chado record to it's node.
  48. * This field is optional and defaults to chado_$table
  49. *
  50. * @return
  51. * The chado id of the associated chado record
  52. *
  53. * @ingroup tripal_chado_node_api
  54. */
  55. function chado_get_id_from_nid($table, $nid, $linking_table = NULL) {
  56. if (empty($linking_table)) {
  57. $linking_table = 'chado_' . $table;
  58. }
  59. $sql = "SELECT " . $table . "_id as id FROM {$linking_table} WHERE nid = :nid";
  60. return db_query($sql, array(':nid' => $nid))->fetchField();
  61. }
  62. /**
  63. * Get node id for a chado feature/organism/analysis. E.g, if you want to
  64. * get the node id for an analysis, use:
  65. * $nid = chado_get_nid_from_id ('analysis', $analysis_id)
  66. * Likewise,
  67. * $nid = chado_get_nid_from_id ('organism', $organism_id)
  68. * $nid = chado_get_nid_from_id ('feature', $feature_id)
  69. *
  70. * @param $table
  71. * The chado table the id is from
  72. * @param $id
  73. * The value of the primary key from the $table chado table (ie: feature_id)
  74. * @param $linking_table
  75. * The Drupal table linking the chado record to it's node.
  76. * This field is optional and defaults to chado_$table
  77. *
  78. * @return
  79. * The nid of the associated node
  80. *
  81. * @ingroup tripal_chado_node_api
  82. */
  83. function chado_get_nid_from_id($table, $id, $linking_table = NULL) {
  84. if (empty($linking_table)) {
  85. $linking_table = 'chado_' . $table;
  86. }
  87. $sql = "SELECT nid FROM {" . $linking_table . "} WHERE " . $table . "_id = :" . $table . "_id";
  88. return db_query($sql, array(":" . $table . "_id" => $id))->fetchField();
  89. }
  90. /**
  91. * Determine the chado base table for a given content type
  92. *
  93. * @param $content_type
  94. * The machine name of the content type (node type) you want to
  95. * determine the base chado table of
  96. * @param $module
  97. * (Optional) The machine-name of the module implementing the
  98. * content type
  99. *
  100. * @return
  101. * The name of the chado base table for the specified content type
  102. *
  103. * @ingroup tripal_chado_node_api
  104. */
  105. function chado_node_get_base_table($content_type, $module = FALSE) {
  106. if ($module) {
  107. $node_info = call_user_func($details['module'] . '_node_info');
  108. }
  109. else {
  110. $node_types = module_invoke_all('node_info');
  111. if (isset($node_types[$content_type])) {
  112. $node_info = $node_types[$content_type];
  113. }
  114. else {
  115. return FALSE;
  116. }
  117. }
  118. if (isset($node_info['chado_node_api']['base_table'])) {
  119. return $node_info['chado_node_api']['base_table'];
  120. }
  121. else {
  122. return FALSE;
  123. }
  124. }
  125. /**
  126. * @section
  127. * Common Functionality for Properties, Dbxrefs and relationships chado node API
  128. */
  129. /**
  130. * Validate the Triggering element from a node form.
  131. *
  132. * We are going to inspect the post to determine what PHP knows is the triggering
  133. * element and if it doesn't agree with Drupal then we are actually going to
  134. * change it in Drupal.
  135. *
  136. * This fixes an obscure bug triggered when a property is added and then
  137. * a relationship removed, Drupal thinks the first property remove button was
  138. * clicked and instead removes a property (not a relationship) and renders the new
  139. * property table in the relationship table page space.
  140. *
  141. * NOTE: Many Drupal issues state that this problem is solved if the #name
  142. * of the button is unique (which it is in our case) but we are still experiencing
  143. * incorrectly determined triggering elements so we need to handle it ourselves.
  144. */
  145. function chado_validate_node_form_triggering_element($form, &$form_state) {
  146. // We are going to inspect the post to determine what PHP knows is the triggering
  147. // element and if it doesn't agree with Drupal then we are actually going to
  148. // change it in Drupal.
  149. if ($_POST['_triggering_element_name'] != $form_state['triggering_element']['#name']) {
  150. $form_state['triggering_element']['#name'] = $_POST['_triggering_element_name'];
  151. }
  152. }
  153. /**
  154. * Validate Adding Subtables entries from the node forms.
  155. * Supported subtables: Properties, Relationships, Additional DBxrefs.
  156. *
  157. * @param array $form
  158. * @param array $form_state
  159. */
  160. function chado_add_node_form_subtables_add_button_validate($form, &$form_state) {
  161. // Based on triggering element call the correct validation function
  162. // ASUMPTION #1: each of the buttons must have property, dbxref or relationship
  163. // as the first part of the #name to uniquely identify the subsection.
  164. if (preg_match('/^([a-z]+).*/', $form_state['triggering_element']['#name'], $matches)) {
  165. $subsection = $matches[1];
  166. switch($subsection) {
  167. case 'properties':
  168. chado_add_node_form_properties_add_button_validate($form, $form_state);
  169. break;
  170. case 'dbxrefs':
  171. chado_add_node_form_dbxrefs_add_button_validate($form, $form_state);
  172. break;
  173. case 'relationships':
  174. chado_add_node_form_relationships_add_button_validate($form, $form_state);
  175. break;
  176. }
  177. }
  178. }
  179. /**
  180. * Add subtable entries to the node forms.
  181. * Supported subtables: Properties, Relationships, Additional DBxrefs.
  182. *
  183. * @param array $form
  184. * @param array $form_state
  185. */
  186. function chado_add_node_form_subtables_add_button_submit($form, &$form_state) {
  187. // Based on triggering element call the correct submit function
  188. // ASUMPTION #1: each of the buttons must have properties, dbxrefs or relationships
  189. // as the first part of the #name to uniquely identify the subsection.
  190. if (preg_match('/^([a-z]+).*/', $form_state['triggering_element']['#name'], $matches)) {
  191. $subsection = $matches[1];
  192. switch($subsection) {
  193. case 'properties':
  194. chado_add_node_form_properties_add_button_submit($form, $form_state);
  195. break;
  196. case 'dbxrefs':
  197. chado_add_node_form_dbxrefs_add_button_submit($form, $form_state);
  198. break;
  199. case 'relationships':
  200. chado_add_node_form_relationships_add_button_submit($form, $form_state);
  201. break;
  202. }
  203. }
  204. // This is needed to ensure the form builder function is called for the node
  205. // form in order for any of these changes to be seen.
  206. $form_state['rebuild'] = TRUE;
  207. }
  208. /**
  209. * Validate Removing Subtables entries from the node forms.
  210. * Supported subtables: Properties, Relationships, Additional DBxrefs.
  211. *
  212. * Since Removing isn't associated with any user input the only thing we
  213. * need to validate is that Drupal has determined the triggering element correctly.
  214. * That said, we will call each subtables associated validate function just incase
  215. * there is some case-specific validation we do not know of or have not anticipated.
  216. *
  217. * @param array $form
  218. * @param array $form_state
  219. */
  220. function chado_add_node_form_subtables_remove_button_validate($form, &$form_state) {
  221. // We need to validate the trigerring element since Drupal has known
  222. // issues determining this correctly when there are multiple buttons
  223. // with the same label.
  224. chado_validate_node_form_triggering_element($form, $form_state);
  225. // Based on triggering element call the correct validation function
  226. // ASUMPTION #1: each of the buttons must have property, dbxref or relationship
  227. // as the first part of the #name to uniquely identify the subsection.
  228. if (preg_match('/^([a-z]+).*/', $form_state['triggering_element']['#name'], $matches)) {
  229. $subsection = $matches[1];
  230. switch($subsection) {
  231. case 'properties':
  232. chado_add_node_form_properties_remove_button_validate($form, $form_state);
  233. break;
  234. case 'dbxrefs':
  235. chado_add_node_form_dbxrefs_remove_button_validate($form, $form_state);
  236. break;
  237. case 'relationships':
  238. chado_add_node_form_relationships_remove_button_validate($form, $form_state);
  239. break;
  240. }
  241. }
  242. }
  243. /**
  244. * Remove subtable entries to the node forms.
  245. * Supported subtables: Properties, Relationships, Additional DBxrefs.
  246. *
  247. * @param array $form
  248. * @param array $form_state
  249. */
  250. function chado_add_node_form_subtables_remove_button_submit($form, &$form_state) {
  251. // Based on triggering element call the correct submit function
  252. // ASUMPTION #1: each of the buttons must have properties, dbxrefs or relationships
  253. // as the first part of the #name to uniquely identify the subsection.
  254. if (preg_match('/^([a-z]+).*/', $form_state['triggering_element']['#name'], $matches)) {
  255. $subsection = $matches[1];
  256. switch($subsection) {
  257. case 'properties':
  258. chado_add_node_form_properties_remove_button_submit($form, $form_state);
  259. break;
  260. case 'dbxrefs':
  261. chado_add_node_form_dbxrefs_remove_button_submit($form, $form_state);
  262. break;
  263. case 'relationships':
  264. chado_add_node_form_relationships_remove_button_submit($form, $form_state);
  265. break;
  266. }
  267. }
  268. // This is needed to ensure the form builder function is called for the node
  269. // form in order for any of these changes to be seen.
  270. $form_state['rebuild'] = TRUE;
  271. }
  272. /**
  273. * Ajax function which returns the section of the form to be re-rendered
  274. * for either the properties, dbxref or relationship sub-sections.
  275. *
  276. * @ingroup tripal_core
  277. */
  278. function chado_add_node_form_subtable_ajax_update($form, &$form_state) {
  279. // We need to validate the trigerring element since Drupal has known
  280. // issues determining this correctly when there are multiple buttons
  281. // with the same label.
  282. chado_validate_node_form_triggering_element($form, $form_state);
  283. // Based on triggering element render the correct part of the form.
  284. // ASUMPTION: each of the buttons must have property, dbxref or relationship
  285. // as the first part of the #name to uniquely identify the subsection.
  286. if (preg_match('/^([a-z]+).*/', $form_state['triggering_element']['#name'], $matches)) {
  287. $subsection = $matches[1];
  288. switch($subsection) {
  289. case 'properties':
  290. return $form['properties']['property_table'];
  291. break;
  292. case 'dbxrefs':
  293. return $form['addtl_dbxrefs']['dbxref_table'];
  294. break;
  295. case 'relationships':
  296. return $form['relationships']['relationship_table'];
  297. break;
  298. }
  299. }
  300. }
  301. /**
  302. * @section
  303. * Sync Form
  304. */
  305. /**
  306. * Generic Sync Form to aid in sync'ing (create drupal nodes linking to chado content)
  307. * any chado node type.
  308. *
  309. * To use this you need to add a call to it from your hook_menu() and
  310. * add some additional information to your hook_node_info(). The Following code gives an
  311. * example of how this might be done:
  312. * @code
  313. function modulename_menu() {
  314. // the machine name of your module
  315. $module_name = 'tripal_example';
  316. // the base specified in hook_node_info
  317. $node_type = 'chado_example';
  318. // This menu item will be a tab on the admin/tripal/chado/tripal_example page
  319. // that is not selected by default
  320. $items['admin/tripal/chado/tripal_example/sync'] = array(
  321. 'title' => ' Sync',
  322. 'description' => 'Sync examples from Chado with Drupal',
  323. 'page callback' => 'drupal_get_form',
  324. 'page arguments' => array('chado_node_sync_form', $module_name, $node_type),
  325. 'access arguments' => array('administer tripal examples'),
  326. 'type' => MENU_LOCAL_TASK,
  327. 'weight' => 0
  328. );
  329. return $items;
  330. }
  331. function modulename_node_info() {
  332. return array(
  333. 'chado_example' => array(
  334. 'name' => t('example'),
  335. 'base' => 'chado_example',
  336. 'description' => t('A Chado example is a collection of material that can be sampled and have experiments performed on it.'),
  337. 'has_title' => TRUE,
  338. 'locked' => TRUE,
  339. // this is what differs from the regular Drupal-documented hook_node_info()
  340. 'chado_node_api' => array(
  341. 'base_table' => 'example', // The name of the chado base table
  342. 'hook_prefix' => 'chado_example', // Usually the name of the node type
  343. 'linking_table' => 'chado_example', // Specifies the linking table used
  344. // to map records to Drupal nodes.
  345. // if 'linking_table' is not specified
  346. // it defaults to the node_type name.
  347. 'record_type_title' => array(
  348. 'singular' => t('Example'), // Singular human-readable title
  349. 'plural' => t('Examples') // Plural human-readable title
  350. ),
  351. 'sync_filters' => array( // filters for syncing
  352. 'type_id' => TRUE, // TRUE if there is an example.type_id field
  353. 'organism_id' => TRUE, // TRUE if there is an example.organism_id field
  354. 'checkboxes' => array('name') // If the 'checkboxes' key is present then the
  355. // value must be an array of column names in
  356. // base table. The values from these columns will
  357. // be retreived, contentated with a space delimeter
  358. // and provided in a list of checkboxes
  359. // for the user to choose which to sync.
  360. ),
  361. )
  362. ),
  363. );
  364. }
  365. * @endcode
  366. *
  367. * For more information on how you can override some of this behaviour while still
  368. * benifiting from as much of the common architecture as possible see the following
  369. * functions: hook_chado_node_sync_create_new_node(), hook_chado_node_sync_form(),
  370. * hook_chado_node_sync_select_query().
  371. *
  372. * @ingroup tripal_chado_node_api
  373. */
  374. function chado_node_sync_form($form, &$form_state) {
  375. $form = array();
  376. if (isset($form_state['build_info']['args'][0])) {
  377. $module = $form_state['build_info']['args'][0];
  378. $node_type = $form_state['build_info']['args'][1];
  379. $node_info = call_user_func($module . '_node_info');
  380. // If a linking table is set in the node_info array then use that,
  381. // otherwise ues the node_type as the linking table.
  382. if (array_key_exists('linking_table', $node_info[$node_type]['chado_node_api'])) {
  383. $linking_table = $node_info[$node_type]['chado_node_api']['linking_table'];
  384. }
  385. else {
  386. $linking_table = 'chado_' . $node_info[$node_type]['chado_node_api']['base_table'];
  387. }
  388. $args = $node_info[$node_type]['chado_node_api'];
  389. $form_state['chado_node_api'] = $args;
  390. }
  391. $form['linking_table'] = array(
  392. '#type' => 'hidden',
  393. '#value' => $linking_table
  394. );
  395. $form['node_type'] = array(
  396. '#type' => 'hidden',
  397. '#value' => $node_type
  398. );
  399. // define the fieldsets
  400. $form['sync'] = array(
  401. '#type' => 'fieldset',
  402. '#title' => 'Sync ' . $args['record_type_title']['plural'],
  403. '#descrpition' => '',
  404. );
  405. $form['sync']['description'] = array(
  406. '#type' => 'item',
  407. '#value' => t("%title_plural of the types listed ".
  408. "below in the %title_singular Types box will be synced (leave blank to sync all types). You may limit the ".
  409. "%title_plural to be synced by a specific organism. Depending on the ".
  410. "number of %title_plural in the chado database this may take a long ".
  411. "time to complete. ",
  412. array(
  413. '%title_singular' => $args['record_type_title']['singular'],
  414. '%title_plural' => $args['record_type_title']['plural']
  415. )),
  416. );
  417. if ($args['sync_filters']['type_id']) {
  418. $form['sync']['type_ids'] = array(
  419. '#title' => t('%title_singular Types',
  420. array(
  421. '%title_singular' => $args['record_type_title']['singular'],
  422. '%title_plural' => $args['record_type_title']['plural']
  423. )),
  424. '#type' => 'textarea',
  425. '#description' => t("Enter the names of the %title_singular types to sync. " .
  426. "Leave blank to sync all %title_plural. Separate each type with a comma ".
  427. "or new line. Pages for these %title_singular ".
  428. "types will be created automatically for %title_plural that exist in the ".
  429. "chado database. The names must match ".
  430. "exactly (spelling and case) with terms in the ontologies",
  431. array(
  432. '%title_singular' => strtolower($args['record_type_title']['singular']),
  433. '%title_plural' => strtolower($args['record_type_title']['plural'])
  434. )),
  435. '#default_value' => (isset($form_state['values']['type_id'])) ? $form_state['values']['type_id'] : '',
  436. );
  437. }
  438. // get the list of organisms
  439. if ($args['sync_filters']['organism_id']) {
  440. $sql = "SELECT * FROM {organism} ORDER BY genus, species";
  441. $results = chado_query($sql);
  442. $organisms[] = '';
  443. foreach ($results as $organism) {
  444. $organisms[$organism->organism_id] = "$organism->genus $organism->species ($organism->common_name)";
  445. }
  446. $form['sync']['organism_id'] = array(
  447. '#title' => t('Organism'),
  448. '#type' => t('select'),
  449. '#description' => t("Choose the organism for which %title_plural types set above will be synced.",
  450. array(
  451. '%title_singular' => $args['record_type_title']['singular'],
  452. '%title_plural' => $args['record_type_title']['plural']
  453. )),
  454. '#options' => $organisms,
  455. '#default_value' => (isset($form_state['values']['organism_id'])) ? $form_state['values']['organism_id'] : 0,
  456. );
  457. }
  458. // get the list of organisms
  459. if (array_key_exists('checkboxes', $args['sync_filters'])) {
  460. // get the base schema
  461. $base_table = $args['base_table'];
  462. $table_info = chado_get_schema($base_table);
  463. // if the base table does not have a primary key or has more than one then
  464. // we can't proceed, otherwise, generate the checkboxes
  465. if (array_key_exists('primary key', $table_info) and count($table_info['primary key']) == 1) {
  466. $pkey = $table_info['primary key'][0];
  467. $columns = $args['sync_filters']['checkboxes'];
  468. $select_cols = '';
  469. foreach ($columns as $column) {
  470. $select_cols .= $base_table . '.' . $column . "|| ' ' ||";
  471. }
  472. // Remove trailing || ' ' ||
  473. $select_cols = substr($select_cols, 0, -9);
  474. $base_table_id = $base_table . '_id';
  475. $select = array($base_table . '.' . $pkey, $select_cols . ' as value');
  476. $joins = array();
  477. $where_clauses = array();
  478. $where_args = array();
  479. // Allow module to update the query.
  480. $hook_query_alter = $node_type . '_chado_node_sync_select_query';
  481. if (function_exists($hook_query_alter)) {
  482. $update = call_user_func($hook_query_alter, array(
  483. 'select' => $select,
  484. 'joins' => $joins,
  485. 'where_clauses' => $where_clauses,
  486. 'where_args' => $where_args,
  487. ));
  488. // Now add in any new changes
  489. if ($update and is_array($update)) {
  490. $select = $update['select'];
  491. $joins = $update['joins'];
  492. $where_clauses = $update['where_clauses'];
  493. $where_args = $update['where_args'];
  494. }
  495. }
  496. // Build Query, we do a left join on the chado_xxxx table in the Drupal schema
  497. // so that if no criteria are specified we only get those items that have not
  498. // yet been synced.
  499. $query = "SELECT " . implode(', ', $select) . ' ' .
  500. 'FROM {' . $base_table . '} ' . $base_table . ' ' . implode(' ', $joins) . ' '.
  501. " LEFT JOIN public.$linking_table CT ON CT.$base_table_id = $base_table.$base_table_id " .
  502. "WHERE CT.$base_table_id IS NULL";
  503. // extend the where clause if needed
  504. $where = '';
  505. $sql_args = array();
  506. foreach ($where_clauses as $category => $items) {
  507. $where .= ' AND (';
  508. foreach ($items as $item) {
  509. $where .= $item . ' OR ';
  510. }
  511. $where = substr($where, 0, -4); // remove the trailing 'OR'
  512. $where .= ') ';
  513. $sql_args = array_merge($sql_args, $where_args[$category]);
  514. }
  515. if ($where) {
  516. $query .= $where;
  517. }
  518. $query .= " ORDER BY $base_table." . implode(", $base_table.", $columns);
  519. $results = chado_query($query, $sql_args);
  520. $values = array();
  521. foreach ($results as $result) {
  522. $values[$result->$pkey] = $result->value;
  523. }
  524. if (count($values) > 0) {
  525. $form['sync']['ids'] = array(
  526. '#title' => 'Avaliable ' . $args['record_type_title']['plural'],
  527. '#type' => 'checkboxes',
  528. '#options' => $values,
  529. '#default_value' => (isset($form_state['values']['ids'])) ? $form_state['values']['ids'] : array(),
  530. '#suffix' => '</div><br>',
  531. '#prefix' => t("The following %title_plural have not been synced. Check those to be synced or leave all unchecked to sync them all.",
  532. array(
  533. '%title_singular' => strtolower($args['record_type_title']['singular']),
  534. '%title_plural' => strtolower($args['record_type_title']['plural'])
  535. )) . '<div style="height: 200px; overflow: scroll">',
  536. );
  537. }
  538. else {
  539. $form['sync']['no_ids'] = array(
  540. '#markup' => "<p>There are no " . strtolower($args['record_type_title']['plural']) . " to sync.</p>",
  541. );
  542. }
  543. }
  544. }
  545. // if we provide a list of checkboxes we shouldn't need a max_sync
  546. else {
  547. $form['sync']['max_sync'] = array(
  548. '#type' => 'textfield',
  549. '#title' => t('Maximum number of records to Sync'),
  550. '#description' => t('Leave this field empty to sync all records, regardless of number'),
  551. '#default_value' => (isset($form_state['values']['max_sync'])) ? $form_state['values']['max_sync'] : '',
  552. );
  553. }
  554. $form['sync']['button'] = array(
  555. '#type' => 'submit',
  556. '#value' => t('Sync ' . $args['record_type_title']['plural']),
  557. '#weight' => 3,
  558. );
  559. $form['cleanup'] = array(
  560. '#type' => 'fieldset',
  561. '#title' => t('Clean Up')
  562. );
  563. $form['cleanup']['description'] = array(
  564. '#markup' => t("<p>With Drupal and chado residing in different databases " .
  565. "it is possible that nodes in Drupal and " . strtolower($args['record_type_title']['plural']) . " in Chado become " .
  566. "\"orphaned\". This can occur if a node in Drupal is " .
  567. "deleted but the corresponding chado records is not and/or vice " .
  568. "versa. Click the button below to resolve these discrepancies.</p>"),
  569. '#weight' => -10,
  570. );
  571. $form['cleanup']['cleanup_batch_size'] = array(
  572. '#type' => 'textfield',
  573. '#title' => t('Batch Size'),
  574. '#description' => t('The number of records to analyze together in a batch. If you are having memory issues you might want to decrease this number.'),
  575. '#default_value' => variable_get('chado_node_api_cleanup_batch_size', 25000),
  576. );
  577. $form['cleanup']['button'] = array(
  578. '#type' => 'submit',
  579. '#value' => 'Clean up orphaned ' . strtolower($args['record_type_title']['plural']),
  580. '#weight' => 2,
  581. );
  582. // Allow each module to alter this form as needed
  583. $hook_form_alter = $args['hook_prefix'] . '_chado_node_sync_form';
  584. if (function_exists($hook_form_alter)) {
  585. $form = call_user_func($hook_form_alter, $form, $form_state);
  586. }
  587. return $form;
  588. }
  589. /**
  590. * Generic Sync Form Validate
  591. *
  592. * @ingroup tripal_core
  593. */
  594. function chado_node_sync_form_validate($form, &$form_state) {
  595. if (empty($form_state['values']['cleanup_batch_size'])) {
  596. $form_state['values']['cleanup_batch_size'] = 25000;
  597. drupal_set_message('You entered a Batch Size of 0 for Cleaning-up orphaned nodes. Since this is not valid, we reset it to the default of 25,000.', 'warning');
  598. }
  599. elseif (!is_numeric($form_state['values']['cleanup_batch_size'])) {
  600. form_set_error('cleanup_batch_size', 'The batch size must be a postitive whole number.');
  601. }
  602. else {
  603. // Round the value just to make sure.
  604. $form_state['values']['cleanup_batch_size'] = abs(round($form_state['values']['cleanup_batch_size']));
  605. }
  606. }
  607. /**
  608. * Generic Sync Form Submit
  609. *
  610. * @ingroup tripal_core
  611. */
  612. function chado_node_sync_form_submit($form, $form_state) {
  613. global $user;
  614. if (preg_match('/^Sync/', $form_state['values']['op'])) {
  615. // get arguments
  616. $args = $form_state['chado_node_api'];
  617. $module = $form_state['chado_node_api']['hook_prefix'];
  618. $base_table = $form_state['chado_node_api']['base_table'];
  619. $linking_table = $form_state['values']['linking_table'];
  620. $node_type = $form_state['values']['node_type'];
  621. // Allow each module to hijack the submit if needed
  622. $hook_form_hijack_submit = $args['hook_prefix'] . '_chado_node_sync_form_submit';
  623. if (function_exists($hook_form_hijack_submit)) {
  624. return call_user_func($hook_form_hijack_submit, $form, $form_state);
  625. }
  626. // Get the types separated into a consistent string
  627. $types = array();
  628. if (isset($form_state['values']['type_ids'])) {
  629. // seperate by new line or comma.
  630. $temp_types = preg_split("/[,\n\r]+/", $form_state['values']['type_ids']);
  631. // remove any extra spacing around the types
  632. for($i = 0; $i < count($temp_types); $i++) {
  633. // skip empty types
  634. if (trim($temp_types[$i]) == '') {
  635. continue;
  636. }
  637. $types[$i] = trim($temp_types[$i]);
  638. }
  639. }
  640. // Get the ids to be synced
  641. $ids = array();
  642. if (array_key_exists('ids', $form_state['values'])){
  643. foreach ($form_state['values']['ids'] as $id => $selected) {
  644. if ($selected) {
  645. $ids[] = $id;
  646. }
  647. }
  648. }
  649. // get the organism to be synced
  650. $organism_id = FALSE;
  651. if (array_key_exists('organism_id', $form_state['values'])) {
  652. $organism_id = $form_state['values']['organism_id'];
  653. }
  654. // Job Arguments
  655. $job_args = array(
  656. 'base_table' => $base_table,
  657. 'max_sync' => (!empty($form_state['values']['max_sync'])) ? $form_state['values']['max_sync'] : FALSE,
  658. 'organism_id' => $organism_id,
  659. 'types' => $types,
  660. 'ids' => $ids,
  661. 'linking_table' => $linking_table,
  662. 'node_type' => $node_type
  663. );
  664. $title = "Sync " . $args['record_type_title']['plural'];
  665. tripal_add_job($title, $module, 'chado_node_sync_records', $job_args, $user->uid);
  666. }
  667. if (preg_match('/^Clean up orphaned/', $form_state['values']['op'])) {
  668. $module = $form_state['chado_node_api']['hook_prefix'];
  669. $base_table = $form_state['chado_node_api']['base_table'];
  670. $linking_table = $form_state['values']['linking_table'];
  671. $node_type = $form_state['values']['node_type'];
  672. $job_args = array($base_table, $form_state['values']['cleanup_batch_size'], $linking_table, $node_type);
  673. variable_set('chado_node_api_cleanup_batch_size', $form_state['values']['cleanup_batch_size']);
  674. tripal_add_job($form_state['values']['op'], $module, 'chado_cleanup_orphaned_nodes', $job_args, $user->uid);
  675. }
  676. }
  677. /**
  678. * Generic function for syncing records in Chado with Drupal nodes.
  679. *
  680. * @param $base_table
  681. * The name of the Chado table containing the record that should be synced
  682. * @param $max_sync
  683. * Optional: A numeric value to indicate the maximum number of records to sync.
  684. * @param $organism_id
  685. * Optional: Limit the list of records to be synced to only those that
  686. * are associated with this organism_id. If the record is not assocaited
  687. * with an organism then this field is not needed.
  688. * @param $types
  689. * Optional: Limit the list of records to be synced to only those that
  690. * match the types listed in this array.
  691. * @param $ids
  692. * Optional: Limit the list of records to bye synced to only those whose
  693. * primary key value matches the ID provided in this array.
  694. * @param $linking_table
  695. * Optional: Tripal maintains "linking" tables in the Drupal schema
  696. * to link Drupal nodes with Chado records. By default these tables
  697. * are named as 'chado_' . $base_table. But if for some reason the
  698. * linking table is not named in this way then it can be provided by this
  699. * argument.
  700. * @param $job_id
  701. * Optional. Used by the Trpial Jobs system when running this function
  702. * as a job. It is not needed othewise.
  703. *
  704. * @ingroup tripal_chado_node_api
  705. */
  706. function chado_node_sync_records($base_table, $max_sync = FALSE,
  707. $organism_id = FALSE, $types = array(), $ids = array(),
  708. $linking_table = FALSE, $node_type = FALSE, $job_id = NULL) {
  709. global $user;
  710. $base_table_id = $base_table . '_id';
  711. if (!$linking_table) {
  712. $linking_table = 'chado_' . $base_table;
  713. }
  714. if (!$node_type) {
  715. $node_type = 'chado_' . $base_table;
  716. }
  717. print "\nSync'ing $base_table records. ";
  718. // START BUILDING QUERY TO GET ALL RECORD FROM BASE TABLE THAT MATCH
  719. $select = array("$base_table.*");
  720. $joins = array();
  721. $where_clauses = array();
  722. $where_args = array();
  723. // If types are supplied then handle them
  724. $restrictions = '';
  725. if (count($types) > 0) {
  726. $restrictions .= " Type(s): " . implode(', ',$types) . "\n";
  727. $select[] = 'cvterm.name as cvtname';
  728. $joins[] = "LEFT JOIN {cvterm} cvterm ON $base_table.type_id = cvterm.cvterm_id";
  729. foreach ($types as $type) {
  730. $sanitized_type = str_replace(' ','_',$type);
  731. $where_clauses['type'][] = "cvterm.name = :type_name_$sanitized_type";
  732. $where_args['type'][":type_name_$sanitized_type"] = $type;
  733. }
  734. }
  735. // if IDs have been supplied
  736. if ($ids) {
  737. $restrictions .= " Specific Records: " . count($ids) . " recored(s) specified.\n";
  738. foreach ($ids as $id) {
  739. $where_clauses['id'][] = "$base_table.$base_table_id = :id_$id";
  740. $where_args['id'][":id_$id"] = $id;
  741. }
  742. }
  743. // If Organism is supplied
  744. if ($organism_id) {
  745. $organism = chado_select_record('organism', array('*'), array('organism_id' => $organism_id));
  746. $restrictions .= " Organism: " . $organism[0]->genus . " " . $organism[0]->species . "\n";
  747. $select[] = 'organism.*';
  748. $joins[] = "LEFT JOIN {organism} organism ON organism.organism_id = $base_table.organism_id";
  749. $where_clauses['organism'][] = 'organism.organism_id = :organism_id';
  750. $where_args['organism'][':organism_id'] = $organism_id;
  751. }
  752. // Allow module to add to query
  753. $hook_query_alter = $node_type . '_chado_node_sync_select_query';
  754. if (function_exists($hook_query_alter)) {
  755. $update = call_user_func($hook_query_alter, array(
  756. 'select' => $select,
  757. 'joins' => $joins,
  758. 'where_clauses' => $where_clauses,
  759. 'where_args' => $where_args,
  760. ));
  761. // Now add in any new changes
  762. if ($update and is_array($update)) {
  763. $select = $update['select'];
  764. $joins = $update['joins'];
  765. $where_clauses = $update['where_clauses'];
  766. $where_args = $update['where_args'];
  767. }
  768. }
  769. // Build Query, we do a left join on the chado_xxxx table in the Drupal schema
  770. // so that if no criteria are specified we only get those items that have not
  771. // yet been synced.
  772. $query = "
  773. SELECT " . implode(', ', $select) . ' ' .
  774. 'FROM {' . $base_table . '} ' . $base_table . ' ' . implode(' ', $joins) . ' '.
  775. " LEFT JOIN public.$linking_table CT ON CT.$base_table_id = $base_table.$base_table_id " .
  776. "WHERE CT.$base_table_id IS NULL ";
  777. // extend the where clause if needed
  778. $where = '';
  779. $sql_args = array();
  780. foreach ($where_clauses as $category => $items) {
  781. $where .= ' AND (';
  782. foreach ($items as $item) {
  783. $where .= $item . ' OR ';
  784. }
  785. $where = substr($where, 0, -4); // remove the trailing 'OR'
  786. $where .= ') ';
  787. $sql_args = array_merge($sql_args, $where_args[$category]);
  788. }
  789. if ($where) {
  790. $query .= $where;
  791. }
  792. $query .- " ORDER BY " . $base_table_id;
  793. // If Maximum number to Sync is supplied
  794. if ($max_sync) {
  795. $query .= " LIMIT $max_sync";
  796. $restrictions .= " Limited to $max_sync records.\n";
  797. }
  798. if ($restrictions) {
  799. print "Records matching these criteria will be synced: \n$restrictions";
  800. }
  801. else {
  802. print "\n";
  803. }
  804. // execute the query
  805. $results = chado_query($query, $sql_args);
  806. // Iterate through records that need to be synced
  807. $count = $results->rowCount();
  808. $interval = intval($count * 0.01);
  809. if ($interval < 1) {
  810. $interval = 1;
  811. }
  812. print "\n$count $base_table records found.\n";
  813. $i = 0;
  814. $transaction = db_transaction();
  815. print "\nNOTE: Syncing is performed using a database transaction. \n" .
  816. "If the sync fails or is terminated prematurely then the entire set of \n" .
  817. "synced items is rolled back and will not be found in the database\n\n";
  818. try {
  819. $percent = 0;
  820. foreach ($results as $record) {
  821. // Update the job status every 1% features.
  822. if ($job_id and $i % $interval == 0) {
  823. $percent = sprintf("%.2f", (($i + 1) / $count) * 100);
  824. print "Syncing $base_table " . ($i + 1) . " of $count (" . $percent . "%). Memory: " . number_format(memory_get_usage()) . " bytes.\r";
  825. tripal_set_job_progress($job_id, intval(($i/$count)*100));
  826. }
  827. // Check if the record is already in the chado linking table
  828. // (ie: check to see if it is already linked to a node).
  829. $result = db_select($linking_table, 'lnk')
  830. ->fields('lnk',array('nid'))
  831. ->condition($base_table_id, $record->{$base_table_id}, '=')
  832. ->execute()
  833. ->fetchObject();
  834. if (empty($result)) {
  835. // Create generic new node.
  836. $new_node = new stdClass();
  837. $new_node->type = $node_type;
  838. $new_node->uid = $user->uid;
  839. $new_node->{$base_table_id} = $record->{$base_table_id};
  840. $new_node->$base_table = $record;
  841. $new_node->language = LANGUAGE_NONE;
  842. // TODO: should we get rid of this hook and use hook_node_presave() instead?
  843. // allow base module to set additional fields as needed
  844. $hook_create_new_node = $node_type . '_chado_node_sync_create_new_node';
  845. if (function_exists($hook_create_new_node)) {
  846. $new_node = call_user_func($hook_create_new_node, $new_node, $record);
  847. }
  848. // Validate and Save New Node
  849. $form = array();
  850. $form_state = array();
  851. node_validate($new_node, $form, $form_state);
  852. if (!form_get_errors()) {
  853. $node = node_submit($new_node);
  854. // If there are memory leaks on the node_save it is probably
  855. // caused by the hook_node_insert() function.
  856. node_save($node);
  857. }
  858. else {
  859. throw new Exception(t("Failed to insert $base_table: %title", array('%title' => $new_node->title)));
  860. }
  861. }
  862. $i++;
  863. }
  864. print "\n\nComplete!\n";
  865. }
  866. catch (Exception $e) {
  867. $transaction->rollback();
  868. print "\n"; // make sure we start errors on new line
  869. watchdog_exception('trp-fsync', $e);
  870. print "FAILED: Rolling back database changes...\n";
  871. }
  872. }
  873. /**
  874. * This function is a wrapper for the chado_cleanup_orphaned_nodes function.
  875. * It breaks up the work of chado_cleanup_orphaned_nodes into smaller pieces
  876. * that are more managable for servers that may have low php memory settings.
  877. *
  878. * @param $table
  879. * The name of the table that corresonds to the node type we want to clean up.
  880. * @param $nentries
  881. * The number of entries to parse at one time (ie: the batch size).
  882. * @param $job_id
  883. * This should be the job id from the Tripal jobs system. This function
  884. * will update the job status using the provided job ID.
  885. *
  886. * @ingroup tripal_chado_node_api
  887. */
  888. function chado_cleanup_orphaned_nodes($table, $nentries = 25000, $linking_table, $node_type, $job_id = NULL) {
  889. $count = 0;
  890. // Find the total number of entries in the table.
  891. $dsql = "SELECT COUNT(*) FROM {node} WHERE type = :node_type";
  892. $clsql= "SELECT COUNT(*) FROM {$linking_table}";
  893. // Find the number nodes of type chado_$table and find the number of entries
  894. // in chado_$table; keep the larger of the two numbers.
  895. $ndat = db_query($dsql, array(':node_type' => $node_type));
  896. $temp = $ndat->fetchObject();
  897. $count = $temp->count;
  898. $cdat = db_query($clsql);
  899. $temp = $cdat->fetchObject();
  900. if (count < $temp->count) {
  901. $count = $temp->count;
  902. }
  903. $transaction = db_transaction();
  904. print "\nNOTE: Syncing is performed using a database transaction. \n" .
  905. "If the sync fails or is terminated prematurely then the entire set of \n" .
  906. "synced items is rolled back and will not be found in the database\n\n";
  907. try {
  908. $m = ceil($count / $nentries);
  909. for ($i = 0; $i < $m; $i++) {
  910. $offset = ($nentries * $i);
  911. chado_cleanup_orphaned_nodes_part($table, $job_id, $nentries, $offset, $linking_table, $node_type);
  912. }
  913. }
  914. catch (Exception $e) {
  915. $transaction->rollback();
  916. print "\n"; // make sure we start errors on new line
  917. watchdog_exception('trp-fsync', $e);
  918. print "FAILED: Rolling back database changes...\n";
  919. }
  920. return '';
  921. }
  922. /**
  923. * This function will delete Drupal nodes for any sync'ed table (e.g.
  924. * feature, organism, analysis, stock, library) if the chado record has been
  925. * deleted or the entry in the chado_[table] table has been removed.
  926. *
  927. * @param $table
  928. * The name of the table that corresonds to the node type we want to clean up.
  929. * @param $job_id
  930. * This should be the job id from the Tripal jobs system. This function
  931. * will update the job status using the provided job ID.
  932. *
  933. * @ingroup tripal_chado_node_api
  934. */
  935. function chado_cleanup_orphaned_nodes_part($table, $job_id = NULL, $nentries,
  936. $offset, $linking_table, $node_type) {
  937. $count = 0;
  938. // Retrieve all of the entries in the linker table for a given node type
  939. // and place into an array.
  940. print "Verifying $linking_table records...\n";
  941. $cnodes = array();
  942. $clsql= "
  943. SELECT *
  944. FROM {" . $linking_table . "} LT
  945. INNER JOIN {node} N ON N.nid = LT.nid
  946. WHERE N.type = :node_type
  947. ORDER BY LT.nid LIMIT $nentries OFFSET $offset";
  948. $res = db_query($clsql, array(':node_type' => $node_type));
  949. foreach ($res as $node) {
  950. $cnodes[$count] = $node;
  951. $count++;
  952. }
  953. // Iterate through all of the $linking_table entries and remove those
  954. // that don't have a node or don't have a $table record.
  955. $deleted = 0;
  956. if ($count > 0) {
  957. $i = 0;
  958. $interval = intval($count * 0.01);
  959. if ($interval < 1) {
  960. $interval = 1;
  961. }
  962. foreach ($cnodes as $nid) {
  963. // Update the job status every 1% analyses
  964. if ($job_id and $i % $interval == 0) {
  965. $percent = sprintf("%.2f", ($i / $count) * 100);
  966. tripal_set_job_progress($job_id, intval($percent));
  967. print "Percent complete: $percent%. Memory: " . number_format(memory_get_usage()) . " bytes.\r";
  968. }
  969. // See if the node exits, if not remove the entry from linking table table.
  970. $nsql = "SELECT * FROM {node} WHERE nid = :nid";
  971. $results = db_query($nsql, array(':nid' => $nid->nid));
  972. $node = $results->fetchObject();
  973. if (!$node) {
  974. $deleted++;
  975. db_query("DELETE FROM {" . $linking_table . "} WHERE nid = :nid", array(':nid' => $nid->nid));
  976. //print "$linking_table missing node.... DELETING: $nid->nid\n";
  977. }
  978. // Does record in chado exists, if not remove entry from $linking_table.
  979. $table_id = $table . "_id";
  980. $lsql = "SELECT * FROM {" . $table . "} where " . $table . "_id = :" . $table . "_id";
  981. $results = chado_query($lsql, array(":" . $table . "_id" => $nid->$table_id));
  982. $record = $results->fetchObject();
  983. if (!$record) {
  984. $deleted++;
  985. $sql = "DELETE FROM {" . $linking_table . "} WHERE " . $table . "_id = :" . $table . "_id";
  986. db_query($sql, array(":" . $table . "_id" => $nid->$table_id));
  987. //print "$linking_table missing $table.... DELETING entry.\n";
  988. }
  989. $i++;
  990. }
  991. $percent = sprintf("%.2f", ($i / $count) * 100);
  992. tripal_set_job_progress($job_id, intval($percent));
  993. print "Percent complete: $percent%. Memory: " . number_format(memory_get_usage()) . " bytes.\r";
  994. }
  995. print "\nDeleted $deleted record(s) from $linking_table missing either a node or chado entry.\n";
  996. // Build the SQL statements needed to check if nodes point to valid record.
  997. print "Verifying nodes...\n";
  998. $dsql = "
  999. SELECT *
  1000. FROM {node}
  1001. WHERE type = :node_type
  1002. ORDER BY nid
  1003. LIMIT $nentries OFFSET $offset
  1004. ";
  1005. $dsql_args = array(':node_type' => $node_type);
  1006. $nodes = array();
  1007. $res = db_query($dsql, $dsql_args);
  1008. $count = 0;
  1009. foreach ($res as $node) {
  1010. $nodes[$count] = $node;
  1011. $count++;
  1012. }
  1013. // Iterate through all of the nodes and delete those that don't
  1014. // have a corresponding entry in the linking table.
  1015. $deleted = 0;
  1016. if ($count > 0) {
  1017. $i = 0;
  1018. $interval = intval($count * 0.01);
  1019. if ($interval < 1) {
  1020. $interval = 1;
  1021. }
  1022. foreach ($nodes as $node) {
  1023. // update the job status every 1%
  1024. if ($job_id and $i % $interval == 0) {
  1025. $percent = sprintf("%.2f", ($i / $count) * 100);
  1026. tripal_set_job_progress($job_id, intval($percent));
  1027. print "Percent complete: $percent%. Memory: " . number_format(memory_get_usage()) . " bytes.\r";
  1028. }
  1029. // check to see if the node has a corresponding entry
  1030. // in the $linking_table table. If not then delete the node.
  1031. $csql = "SELECT * FROM {" . $linking_table . "} WHERE nid = :nid ";
  1032. $results = db_query($csql, array(':nid' => $node->nid));
  1033. $link = $results->fetchObject();
  1034. if (!$link) {
  1035. // Checking node_access creates a memory leak. Commenting out for now
  1036. // assuming that this code can only be run by a site administrator
  1037. // anyway.
  1038. // if (node_access('delete', $node)) {
  1039. $deleted++;
  1040. node_delete($node->nid);
  1041. // }
  1042. // else {
  1043. // print "\nNode missing in $linking_table table.... but cannot delete due to improper permissions (node $node->nid)\n";
  1044. // }
  1045. }
  1046. $i++;
  1047. }
  1048. $percent = sprintf("%.2f", ($i / $count) * 100);
  1049. tripal_set_job_progress($job_id, intval($percent));
  1050. print "Percent complete: $percent%. Memory: " . number_format(memory_get_usage()) . " bytes.\r";
  1051. print "\nDeleted $deleted node(s) that did not have corresponding $linking_table entries.\n";
  1052. }
  1053. return '';
  1054. }
  1055. /**
  1056. * Create New Node
  1057. *
  1058. * Note: For your own module, replace hook in the function name with the
  1059. * machine-name of your chado node type (ie: chado_feature).
  1060. *
  1061. * @param $new_node:
  1062. * a basic new node object
  1063. * @param $record:
  1064. * the record object from chado specifying the biological data for this node
  1065. *
  1066. * @return
  1067. * A node object containing all the fields necessary to create a new node
  1068. * during sync
  1069. *
  1070. * @ingroup tripal_chado_node_api
  1071. */
  1072. function hook_chado_node_sync_create_new_node($new_node, $record) {
  1073. // Add relevant chado details to the new node object. This really only
  1074. // needs to be the fields from the node used during node creation
  1075. // including values used to generate the title, etc. All additional chado
  1076. // data will be added via nodetype_load when the node is later used
  1077. $new_node->uniquename = $record->uniquename;
  1078. return $new_node;
  1079. }
  1080. /**
  1081. * Alter the Chado node sync form.
  1082. *
  1083. * This might be necessary if you need additional filtering options for
  1084. * choosing which chado records to sync or even if you just want to further
  1085. * customize the help text provided by the form.
  1086. *
  1087. * Note: For your own module, replace hook in the function name with the
  1088. * machine-name of your chado node type (ie: chado_feature).
  1089. *
  1090. * @ingroup tripal_chado_node_api
  1091. */
  1092. function hook_chado_node_sync_form($form, &$form_state) {
  1093. // Change or add to the form array as needed.
  1094. // Any changes should be made in accordance with the Drupal Form API.
  1095. return $form;
  1096. }
  1097. /**
  1098. * Bypass chado node api sync form submit.
  1099. *
  1100. * Allows you to use this function as your own submit.
  1101. *
  1102. * This might be necessary if you want to add additional arguements to the
  1103. * tripal job or to call your own sync'ing function if the generic
  1104. * chado_node_sync_records() is not sufficient.
  1105. *
  1106. * Note: For your own module, replace hook in the function name with the
  1107. * machine-name of your chado node type (ie: chado_feature).
  1108. *
  1109. * @ingroup tripal_chado_node_api
  1110. */
  1111. function hook_chado_node_sync_form_submit ($form, $form_state) {
  1112. global $user;
  1113. $job_args = array(
  1114. // The base chado table (ie: feature).
  1115. $base_table,
  1116. // The maximum number of records to sync or FALSE for sync all that match.
  1117. $max_sync,
  1118. // The organism_id to restrict records to or FALSE if not to restrict by organism_id.
  1119. $organism_id,
  1120. // A string with the cvterm.name of the types to restrict to separated by |||
  1121. $types
  1122. );
  1123. // You should register a tripal job
  1124. tripal_add_job(
  1125. // The title of the job -be descriptive.
  1126. $title,
  1127. // The name of your module.
  1128. $module,
  1129. // The chado node api sync function.
  1130. 'chado_node_sync_records',
  1131. // An array with the arguments to pass to the above function.
  1132. $job_args,
  1133. // The user who submitted the job.
  1134. $user->uid
  1135. );
  1136. }
  1137. /**
  1138. * Alter the query that retrieves records to be sync'd (optional)
  1139. *
  1140. * This might be necessary if you need fields from other chado tables to
  1141. * create your node or if your chado node type only supports a subset of a
  1142. * given table (ie: a germplasm node type might only support node creation for
  1143. * cerain types of stock records in which case you would need to filter the
  1144. * results to only those types).
  1145. *
  1146. * Note: For your own module, replace hook in the function name with the
  1147. * machine-name of your chado node type (ie: chado_feature).
  1148. *
  1149. * @param $query
  1150. * An array containing the following:
  1151. * 'select': An array of select clauses
  1152. * 'joins: An array of joins (ie: a single join could be
  1153. * 'LEFT JOIN {chadotable} alias ON base.id=alias.id')
  1154. * 'where_clauses: An array of where clauses which will all be AND'ed
  1155. * together. Use :placeholders for values.
  1156. * 'where_args: An associative array of arguments to be subbed in to the
  1157. * where clause where the
  1158. *
  1159. * @ingroup tripal_chado_node_api
  1160. */
  1161. function hook_chado_node_sync_select_query($query) {
  1162. // You can add fields to be selected. Be sure to prefix each field with the
  1163. // tale name.
  1164. $query['select'][] = 'example.myfavfield';
  1165. // Provide any join you may need to the joins array. Be sure to wrap the
  1166. // table name in curly brackets.
  1167. $query['joins'][] = 'LEFT JOIN {exampleprop} PROP ON PROP.example_id=EXAMPLE.example_id';
  1168. // The category should be a unique id for a group of items that will be
  1169. // concatenated together via an SQL 'OR'. By default the $where_clases
  1170. // variable will come with categories of 'id', 'organism' and 'type'.
  1171. // you can add your own unique category or alter the contents of the existing
  1172. // categories. Be sure to make sure the category doesn't already exist
  1173. // in the $query['where_clauses']
  1174. $category = 'my_category';
  1175. // Provide any aditionall where clauses and their necessary arguments.
  1176. // Be sure to prefix the field with the table name. Be sure that the
  1177. // placeholder is unique across all categories (perhaps add a unique
  1178. // prefix/suffix).
  1179. $query['where_clauses'][$category][] = 'example.myfavfield = :favvalue';
  1180. $query['where_args'][$category][':favvalue'] = 'awesome-ness';
  1181. // Must return the updated query
  1182. return $query;
  1183. }