tripal_entities.admin.inc 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587
  1. <?php
  2. /**
  3. * Launchpad for feature administration.
  4. *
  5. * @ingroup tripal_feature
  6. */
  7. function tripal_entities_admin_view() {
  8. $form = drupal_get_form('tripal_entities_admin_bundles_form');
  9. $output = drupal_render($form) . "<br>[ Image Place Holder for Data Type Summary ]<br>";
  10. // set the breadcrumb
  11. $breadcrumb = array();
  12. $breadcrumb[] = l('Home', '<front>');
  13. $breadcrumb[] = l('Administration', 'admin');
  14. $breadcrumb[] = l('Tripal', 'admin/tripal');
  15. $breadcrumb[] = l('Biological Data', 'admin/tripal/bundles');
  16. drupal_set_breadcrumb($breadcrumb);
  17. /* // Add the view
  18. $view = views_embed_view('tripal_feature_admin_features','default');
  19. if (isset($view)) {
  20. $output .= $view;
  21. }
  22. else {
  23. $output .= '<p>The Feature module uses primarily views to provide an '
  24. . 'administrative interface. Currently one or more views needed for this '
  25. . 'administrative interface are disabled. <strong>Click each of the following links to '
  26. . 'enable the pertinent views</strong>:</p>';
  27. $output .= '<ul>';
  28. $output .= '<li>'.l('Features View', 'admin/tripal/chado/tripal_feature/views/features/enable').'</li>';
  29. $output .= '</ul>';
  30. }
  31. // Add a summary chart.
  32. //-----------------------------------
  33. $output .= theme('tripal_feature_bar_chart_type_organism_summary');
  34. drupal_add_js('
  35. Drupal.behaviors.tripalFeature_moveAdminSummaryChart = {
  36. attach: function (context, settings) {
  37. jQuery("#tripal-feature-admin-summary").insertBefore( jQuery(".view-filters") );
  38. }};
  39. ', 'inline'); */
  40. return $output;
  41. }
  42. /**
  43. *
  44. */
  45. function tripal_entities_content_view() {
  46. $form = drupal_get_form('tripal_entities_content_overview_form');
  47. $output = drupal_render($form);
  48. // set the breadcrumb
  49. $breadcrumb = array();
  50. $breadcrumb[] = l('Home', '<front>');
  51. $breadcrumb[] = l('Administration', 'admin');
  52. drupal_set_breadcrumb($breadcrumb);
  53. return $output;
  54. }
  55. /**
  56. *
  57. * @param unknown $form
  58. * @param unknown $form_state
  59. * @return multitype:
  60. */
  61. function tripal_entities_content_overview_form($form, &$form_state) {
  62. $form = array();
  63. $entities = db_select('tripal_entity', 'td')
  64. ->fields('td')
  65. ->orderBy('created', 'DESC')//ORDER BY created
  66. ->range(0,25)
  67. ->execute();
  68. $headers = array('Title', 'Vocabulary', 'Term', 'Author', 'Status', 'Updated', 'Operations');
  69. $rows = array();
  70. while ($entity = $entities->fetchObject()) {
  71. $author = user_load($entity->uid);
  72. $rows[] = array(
  73. l($entity->title, 'data/' . $entity->id),
  74. $entity->type,
  75. $entity->bundle,
  76. l($author->name, 'user/' . $entity->uid),
  77. $entity->status == 1 ? 'published' : 'unpublished',
  78. format_date($entity->changed, 'short'),
  79. l('edit', 'data/' . $entity->id . '/edit') . '&nbsp;&nbsp;' .
  80. l('delete', 'data/' . $entity->id . '/delete')
  81. );
  82. }
  83. $table_vars = array(
  84. 'header' => $headers,
  85. 'rows' => $rows,
  86. 'attributes' => array(),
  87. 'sticky' => TRUE,
  88. 'caption' => '',
  89. 'colgroups' => array(),
  90. 'empty' => '',
  91. );
  92. $results_table = theme('table', $table_vars);
  93. $form['results'] = array(
  94. '#markup' => $results_table
  95. );
  96. return $form;
  97. }
  98. /**
  99. *
  100. * @param unknown $form
  101. * @param unknown $form_state
  102. * @return multitype:
  103. */
  104. function tripal_entities_admin_bundles_form($form, &$form_state) {
  105. $form = array();
  106. // Set the defaults.
  107. $cv_id = NULL;
  108. $term_name = NULL;
  109. // Set defaults using the form state.
  110. if (array_key_exists('values', $form_state)) {
  111. $cv_id = array_key_exists('cv_id', $form_state['values']) ? $form_state['values']['cv_id'] : NULL;
  112. $term_name = array_key_exists('term_name', $form_state['values']) ? $form_state['values']['term_name'] : NULL;
  113. }
  114. // Let the user select the vocabulary and tripal_entity but only if they haven't
  115. // already selected a tripal_entity.
  116. $sql = "
  117. SELECT CV.cv_id, CV.name
  118. FROM {cv} CV
  119. ORDER BY CV.name
  120. ";
  121. $vocabs = chado_query($sql);
  122. $cvs = array();
  123. while ($vocab = $vocabs->fetchObject()) {
  124. $cvs[$vocab->cv_id] = $vocab->name;
  125. }
  126. $form['cv_id'] = array(
  127. '#type' => 'select',
  128. '#title' => t('Vocabulary'),
  129. '#options' => $cvs,
  130. '#required' => FALSE,
  131. '#description' => t('Select a vocabulary to view potential data types in the chart below. Limit the chart to only published data types by selecting the checkbox.'),
  132. '#default_value' => $cv_id,
  133. '#ajax' => array(
  134. 'callback' => "tripal_entities_admin_bundles_form_ajax_callback",
  135. 'wrapper' => 'tripal_entities_admin_bundles_form',
  136. 'effect' => 'fade',
  137. 'method' => 'replace'
  138. )
  139. );
  140. $form['refresh_bundles'] = array(
  141. '#type' => 'submit',
  142. '#value' => t('Refresh Data Types'),
  143. '#name' => 'refresh_bundles',
  144. );
  145. $form['publish_new_data'] = array(
  146. '#type' => 'submit',
  147. '#value' => t('Publish New Data'),
  148. '#name' => 'publish_new_data',
  149. );
  150. $form['#prefix'] = '<div id="tripal_entities_admin_bundle_form">';
  151. $form['#suffix'] = '</div>';
  152. return $form;
  153. }
  154. /**
  155. * Submit a job to populate the entity tables
  156. * This operation makes available data types in the database publishable
  157. */
  158. function tripal_entities_admin_bundles_form_submit($form, $form_state) {
  159. global $user;
  160. if ($form_state['clicked_button']['#name'] == 'refresh_bundles') {
  161. tripal_add_job('Create publishable data types', 'tripal_entity', 'tripal_entities_populate_entity_tables', array(), $user->uid);
  162. }
  163. if ($form_state['clicked_button']['#name'] == 'publish_new_data') {
  164. }
  165. }
  166. /**
  167. *
  168. * @param unknown $form
  169. * @param unknown $form_state
  170. * @return multitype:
  171. */
  172. function tripal_entities_admin_publish_form($form, &$form_state) {
  173. $form = array();
  174. // Set the defaults.
  175. $cv_id = NULL;
  176. $term_name = NULL;
  177. // Set defaults using the form state.
  178. if (array_key_exists('values', $form_state)) {
  179. $cv_id = array_key_exists('cv_id', $form_state['values']) ? $form_state['values']['cv_id'] : NULL;
  180. $term_name = array_key_exists('term_name', $form_state['values']) ? $form_state['values']['term_name'] : NULL;
  181. }
  182. // Let the user select the vocabulary and tripal_entity but only if they haven't
  183. // already selected a tripal_entity.
  184. $sql = "
  185. SELECT CV.cv_id, CV.name
  186. FROM {tripal_vocabulary} TET
  187. INNER JOIN {cv} CV on CV.cv_id = TET.cv_id
  188. ORDER BY CV.name
  189. ";
  190. $vocabs = chado_query($sql);
  191. $cvs = array();
  192. while ($vocab = $vocabs->fetchObject()) {
  193. $cvs[$vocab->cv_id] = $vocab->name;
  194. }
  195. $form['cv_id'] = array(
  196. '#type' => 'select',
  197. '#title' => t('Vocabulary'),
  198. '#options' => $cvs,
  199. '#required' => TRUE,
  200. '#description' => t('Select a vocabulary that contains the term you would like to set as publishable. Only vocabularies that are linked to data are shown.'),
  201. '#default_value' => $cv_id,
  202. '#ajax' => array(
  203. 'callback' => "tripal_entities_admin_publish_form_ajax_callback",
  204. 'wrapper' => 'tripal_entities_admin_publish_form',
  205. 'effect' => 'fade',
  206. 'method' => 'replace'
  207. )
  208. );
  209. // If we have a CV ID then we want to provide an autocomplete field
  210. if ($cv_id) {
  211. $form['cvterm_select']['term_name'] = array(
  212. '#title' => t('Data Type'),
  213. '#type' => 'textfield',
  214. '#description' => t("Please enter the name of the data type to set as publishable. The data type must be a valid term in the selected vocabulary. This field will autopopulate as you type to help find available data types."),
  215. '#required' => TRUE,
  216. '#default_value' => $term_name,
  217. '#autocomplete_path' => "admin/tripal/chado/tripal_cv/cvterm/auto_name/$cv_id",
  218. );
  219. $form['cvterm_select']['select_button'] = array(
  220. '#type' => 'submit',
  221. '#value' => t('Publish Data'),
  222. '#name' => 'publish',
  223. );
  224. }
  225. $form['#prefix'] = '<div id="tripal_entities_admin_publish_form">';
  226. $form['#suffix'] = '</div>';
  227. return $form;
  228. }
  229. /**
  230. * An Ajax callback for the tripal_entities_admin_publish_form..
  231. */
  232. function tripal_entities_admin_publish_form_ajax_callback($form, $form_state) {
  233. // return the form so Drupal can update the content on the page
  234. return $form;
  235. }
  236. /**
  237. * Implements hook_validate() for the tripal_entities_admin_publish_form.
  238. *
  239. */
  240. function tripal_entities_admin_publish_form_validate($form, &$form_state) {
  241. $cv_id = $form_state['values']['cv_id'];
  242. $term_name = $form_state['values']['term_name'];
  243. // Make sure the term_name is a real term in the vocabulary.
  244. $type = tripal_get_cvterm(array(
  245. 'name' => $term_name,
  246. 'cv_id' => $cv_id
  247. ));
  248. if (!$type) {
  249. form_set_error('term_name', t("The data type is not a valid name for the selected vocabulary."));
  250. }
  251. // Make sure the term is used in the site:
  252. $values = array(
  253. 'cvterm_id' => $type->cvterm_id,
  254. );
  255. $bundles = chado_select_record('tripal_term', array('term_id'), $values);
  256. if (count($bundles) == 0) {
  257. form_set_error('term_name', t("The data type, %type, is not associated with data on this site and thus cannot be set as publishable.", array('%type' => $term_name)));
  258. }
  259. // Make sure the term is not already published.
  260. $values = array(
  261. 'cvterm_id' => $type->cvterm_id,
  262. 'publish' => 1,
  263. );
  264. $bundles = chado_select_record('tripal_term', array('term_id'), $values);
  265. if (count($bundles) > 0) {
  266. form_set_error('term_name', t("This data type is already set as publishable."));
  267. }
  268. }
  269. /**
  270. * Implements hook_submit() for the tripal_entities_admin_publish_form.
  271. *
  272. */
  273. function tripal_entities_admin_publish_form_submit($form, &$form_state) {
  274. $cv_id = $form_state['values']['cv_id'];
  275. $term_name = $form_state['values']['term_name'];
  276. // Get the data type using the $term_name and $cv_id.
  277. $cvterm = chado_generate_var('cvterm', array('cv_id' => $cv_id, 'name' => $term_name));
  278. // Start the transaction.
  279. $transaction = db_transaction();
  280. try {
  281. // We don't need to check if the vocabulary is used because the
  282. // form only shows those vocabs that are used.
  283. // Mark this entity as published.
  284. $match = array('cv_id' => $cv_id);
  285. $values = array('publish' => 1);
  286. $success = chado_update_record('tripal_vocabulary', $match, $values);
  287. if (!$success) {
  288. throw new Exception('Cannot set the vocabulary as publishable');
  289. }
  290. // We have already checked in the validate if the term already exists
  291. // as a bundle. So, if we're here we can enable it.
  292. $match = array('cvterm_id' => $cvterm->cvterm_id);
  293. $values = array('publish' => 1);
  294. $success = chado_update_record('tripal_term', $match, $values);
  295. if (!$success) {
  296. throw new Exception('Cannot set the data type as publishable');
  297. }
  298. // Create the bundle name and entity type name.
  299. $bundle_name = $cvterm->dbxref_id->db_id->name . '_' . $cvterm->dbxref_id->accession;
  300. $entity_type_name = $cvterm->dbxref_id->db_id->name;
  301. // Clear the entity cache so that Drupal will read our
  302. // hook_entity_info() implementation which now will have the entities
  303. // described because we set the publish column to 1 in the tripal_term
  304. // table.
  305. global $language;
  306. $langcode = $language->language;
  307. cache_clear_all("entity_info:$langcode", 'cache');
  308. // The TripalBundle Entity manages the bundles we have available.
  309. // Therefore, we need to add a new entity for each bundle "type".
  310. $vals = array(
  311. 'label' => $bundle_name . ' (' . $cvterm->name . ')',
  312. 'type' => $entity_type_name,
  313. 'bundle' => $bundle_name,
  314. 'data' => serialize(array()),
  315. 'module' => 'tripal_entities'
  316. );
  317. $tripal_bundle = new TripalBundle($vals, $entity_type_name . '_bundle');
  318. $tripal_bundle->save();
  319. // Allow modules to now add fields to the bundle
  320. module_invoke_all('add_bundle_fields', $entity_type_name, $bundle_name, $cvterm);
  321. drupal_set_message(t('Data type, %type, is now set as publishable.', array('%type' => $term_name)));
  322. }
  323. catch (Exception $e) {
  324. $transaction->rollback();
  325. drupal_set_message('Failure publishing this data type: ' . $e->getMessage(), 'error');
  326. watchdog_exception('trp_entities', $e);
  327. }
  328. }
  329. /**
  330. *
  331. * @param $table
  332. * @param $entity_type
  333. * @param $bundle_name
  334. */
  335. function tripal_entities_add_bundle_fields($entity_type_name, $bundle_name, $cvterm) {
  336. // Get the list of tables where this cvterm is used.
  337. $match = array('cvterm_id' => $cvterm->cvterm_id);
  338. $term = chado_select_record('tripal_term', array('*'), $match);
  339. $values = array('term_id' => $term[0]->term_id);
  340. $tables = chado_select_record('tripal_term_usage', array('*'), $values);
  341. // Iterate through the tables.
  342. foreach ($tables as $table) {
  343. $tablename = $table->data_table;
  344. $type_table = $table->type_table;
  345. $type_field = $table->field;
  346. // We only want to look at base tables.
  347. if ($tablename == 'cvterm_dbxref' || $tablename == 'cvterm_relationship' ||
  348. $tablename == 'cvtermpath' || $tablename == 'cvtermprop' || $tablename == 'chadoprop' ||
  349. $tablename == 'cvtermsynonym' || preg_match('/_relationship$/', $tablename) ||
  350. preg_match('/_cvterm$/', $tablename)) {
  351. continue;
  352. }
  353. // Iterate through the columns of the table and see if fields have been
  354. // created for each one. If not, then create them.
  355. $schema = chado_get_schema($tablename);
  356. $columns = $schema['fields'];
  357. foreach ($columns as $column_name => $details) {
  358. $field_name = $tablename . '__' . $column_name;
  359. $field = field_info_field($field_name);
  360. // Skip the primary key field.
  361. if ($column_name == $schema['primary key'][0]) {
  362. continue;
  363. }
  364. // Skip the type field.
  365. if ($tablename == $type_table and $column_name == $type_field) {
  366. continue;
  367. }
  368. // Determine if the field is required.
  369. $is_required = 0;
  370. if (array_key_exists('not null', $details) and $details['not null'] === TRUE) {
  371. $is_required = array_key_exists('default', $details) ? 0 : 1;
  372. }
  373. // Determine what type of field this should be.
  374. // Drupal data types are: https://www.drupal.org/node/159605.
  375. // Field types are here: https://www.drupal.org/node/1879542
  376. $field_type = '';
  377. $widget_type = '';
  378. $settings = array();
  379. $label = '';
  380. $desc = '';
  381. switch($details['type']) {
  382. case 'char':
  383. $field_type = 'text';
  384. $widget_type = 'text_textfield';
  385. $settings['max_length'] = $details['length'];
  386. break;
  387. case 'varchar':
  388. $field_type = 'text';
  389. $widget_type = 'text_textfield';
  390. $settings['max_length'] = $details['length'];
  391. break;
  392. case 'text':
  393. $field_type = 'text';
  394. $widget_type = 'text_textarea';
  395. $settings['max_length'] = '';
  396. break;
  397. case 'blob':
  398. // not sure how to support a blob field.
  399. continue;
  400. break;
  401. case 'int':
  402. $field_type = 'number_integer';
  403. $widget_type = 'number';
  404. break;
  405. case 'float':
  406. $field_type = 'number_float';
  407. $widget_type = 'number';
  408. $settings['precision'] = 10;
  409. $settings['scale'] = 2;
  410. $settings['decimal_separator'] = '.';
  411. break;
  412. case 'numeric':
  413. $field_type = 'number_decimal';
  414. $widget_type = 'number';
  415. break;
  416. case 'serial':
  417. // Serial fields are most likely not needed as a field.
  418. break;
  419. case 'boolean':
  420. $field_type = 'list_boolean';
  421. $widget_type = 'options_onoff';
  422. $settings['allowed_values'] = array(0 => "No", 1 => "Yes");
  423. break;
  424. case 'datetime':
  425. // Use the Drupal Date and Date API to create the field/widget
  426. $field_type = 'datetime';
  427. $widget_type = 'date_select';
  428. break;
  429. default:
  430. drupal_set_message(t("Unhandled field type: %type", array('%type' => $details['type'])), 'warning');
  431. $field_type = 'text';
  432. $widget_type = 'text_textarea';
  433. if (array_key_exists('length', $details)) {
  434. $settings['max_length'] = $details['length'];
  435. }
  436. }
  437. // If we don't have a field type then we don't need to create a field.
  438. if (!$field_type) {
  439. // If we don't have a field type but it is required and doesn't have
  440. // a default value then we are in trouble.
  441. if ($is_required and !array_key_exists('default', $details)) {
  442. throw new Exception(t('The %table.%field type, %type, is not yet supported for Entity fields, but it is required,',
  443. array('%table' => $tablename, '%field' => $column_name, '%type' => $details['type'])));
  444. }
  445. continue;
  446. }
  447. // If this field is a foreign key field then we will have a special custom
  448. // field provided by Tripal.
  449. $is_fk = FALSE;
  450. if (array_key_exists('foreign keys', $schema)) {
  451. foreach ($schema['foreign keys'] as $remote_table => $fk_details) {
  452. if (array_key_exists($column_name, $fk_details['columns'])) {
  453. $is_fk = TRUE;
  454. }
  455. }
  456. }
  457. // If this column is a FK relationship then use a custom Tripal
  458. // defined field type for it.
  459. if ($is_fk) {
  460. // TODO: We need a better way to get the fields for FK relationships.
  461. // It's not a good idea to enumerate them all here. We need some sort
  462. // of hook or something that will let us lookup the correct field.
  463. switch ($column_name) {
  464. case 'organism_id':
  465. $field_type = 'organism_id';
  466. $label = 'Organism';
  467. $desc = 'Select an organism.';
  468. $widget_type = 'tripal_entities_organism_select_widget';
  469. break;
  470. case 'dbxref_id':
  471. $field_type = 'dbxref_id';
  472. $label = 'Primary Cross Reference';
  473. $desc = 'This record can be cross-referenced with a record in another online database. This field is intended for the most prominent reference. At a minimum, the database and accession must be provided.';
  474. $widget_type = 'tripal_entities_primary_dbxref_widget';
  475. break;
  476. }
  477. }
  478. // If this column is the md5checksum
  479. if ($column_name == 'md5checksum') {
  480. $field_type = 'md5checksum';
  481. $label = 'MD5 Checksum';
  482. $desc = 'Generating MD5 checksum for the sequence.';
  483. $widget_type = 'tripal_entities_md5checksum_checkbox_widget';
  484. }
  485. // If the field doesn't exist then create it.
  486. if (!$field) {
  487. $field = array(
  488. 'field_name' => $field_name,
  489. 'type' => $field_type,
  490. 'cardinality' => 1,
  491. 'locked' => FALSE,
  492. 'storage' => array(
  493. 'type' => 'field_chado_storage'
  494. ),
  495. 'settings' => $settings,
  496. );
  497. field_create_field($field);
  498. }
  499. // Attach the field to the bundle.
  500. $field_instance = array(
  501. 'field_name' => $field_name,
  502. 'label' => $label ? $label : ucwords(preg_replace('/_/', ' ', $column_name)),
  503. 'description' => $desc,
  504. 'widget' => array(
  505. 'type' => $widget_type,
  506. 'settings' => array('display_label' => 1)
  507. ),
  508. 'entity_type' => $entity_type_name,
  509. 'required' => $is_required,
  510. 'settings' => $settings,
  511. 'bundle' => $bundle_name,
  512. );
  513. field_create_instance($field_instance);
  514. }
  515. }
  516. }
  517. /**
  518. *
  519. * @param unknown $form
  520. * @param unknown $form_state
  521. * @return multitype:
  522. */
  523. function tripal_entities_admin_access_form($form, &$form_state) {
  524. $form = array();
  525. return $form;
  526. }