tripal_entities.admin.inc 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544
  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_data_types_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/data_types');
  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_data', '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. $cvterm = chado_generate_var('cvterm', array('cvterm_id' => $entity->cvterm_id));
  72. $author = user_load($entity->uid);
  73. $rows[] = array(
  74. l($entity->title, 'data/' . $entity->id),
  75. $entity->type . ' (' . $cvterm->cv_id->name . ')',
  76. $entity->bundle . ' (' . $cvterm->name . ')',
  77. l($author->name, 'user/' . $entity->uid),
  78. $entity->status == 1 ? 'published' : 'unpublished',
  79. format_date($entity->changed, 'short'),
  80. l('edit', 'data/' . $entity->id . '/edit') . '&nbsp;&nbsp;' .
  81. l('delete', 'data/' . $entity->id . '/delete')
  82. );
  83. }
  84. $table_vars = array(
  85. 'header' => $headers,
  86. 'rows' => $rows,
  87. 'attributes' => array(),
  88. 'sticky' => TRUE,
  89. 'caption' => '',
  90. 'colgroups' => array(),
  91. 'empty' => '',
  92. );
  93. $results_table = theme('table', $table_vars);
  94. $form['results'] = array(
  95. '#markup' => $results_table
  96. );
  97. return $form;
  98. }
  99. /**
  100. *
  101. * @param unknown $form
  102. * @param unknown $form_state
  103. * @return multitype:
  104. */
  105. function tripal_entities_admin_data_types_form($form, &$form_state) {
  106. $form = array();
  107. // Set the defaults.
  108. $cv_id = NULL;
  109. $term_name = NULL;
  110. // Set defaults using the form state.
  111. if (array_key_exists('values', $form_state)) {
  112. $cv_id = array_key_exists('cv_id', $form_state['values']) ? $form_state['values']['cv_id'] : NULL;
  113. $term_name = array_key_exists('term_name', $form_state['values']) ? $form_state['values']['term_name'] : NULL;
  114. }
  115. // Let the user select the vocabulary and tripal_data but only if they haven't
  116. // already selected a tripal_data.
  117. $sql = "
  118. SELECT CV.cv_id, CV.name
  119. FROM {cv} CV
  120. ORDER BY CV.name
  121. ";
  122. $vocabs = chado_query($sql);
  123. $cvs = array();
  124. while ($vocab = $vocabs->fetchObject()) {
  125. $cvs[$vocab->cv_id] = $vocab->name;
  126. }
  127. $form['cv_id'] = array(
  128. '#type' => 'select',
  129. '#title' => t('Vocabulary'),
  130. '#options' => $cvs,
  131. '#required' => FALSE,
  132. '#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.'),
  133. '#default_value' => $cv_id,
  134. '#ajax' => array(
  135. 'callback' => "tripal_entities_admin_data_types_form_ajax_callback",
  136. 'wrapper' => 'tripal_entities_admin_data_types_form',
  137. 'effect' => 'fade',
  138. 'method' => 'replace'
  139. )
  140. );
  141. $form['refresh_data_types'] = array(
  142. '#type' => 'submit',
  143. '#value' => t('Refresh Data Types'),
  144. '#submit' => array('tripal_entity_admin_data_types_form_submit_refresh_data_types'),
  145. );
  146. $form['publish_new_data'] = array(
  147. '#type' => 'submit',
  148. '#value' => t('Publish New Data'),
  149. '#submit' => array('tripal_entity_admin_data_types_form_submit_publish_new_data'),
  150. );
  151. $form['#prefix'] = '<div id="tripal_entities_admin_data_type_form">';
  152. $form['#suffix'] = '</div>';
  153. return $form;
  154. }
  155. /**
  156. * Submit a job to populate the entity tables
  157. * This operation makes available data types in the database publishable
  158. */
  159. function tripal_entity_admin_data_types_form_submit_refresh_data_types () {
  160. global $user;
  161. tripal_add_job('Create publishable data types', 'tripal_entity', 'tripal_entities_populate_entity_tables', array(), $user->uid);
  162. }
  163. /**
  164. *
  165. * @param unknown $form
  166. * @param unknown $form_state
  167. * @return multitype:
  168. */
  169. function tripal_entities_admin_publish_form($form, &$form_state) {
  170. $form = array();
  171. // Set the defaults.
  172. $cv_id = NULL;
  173. $term_name = NULL;
  174. // Set defaults using the form state.
  175. if (array_key_exists('values', $form_state)) {
  176. $cv_id = array_key_exists('cv_id', $form_state['values']) ? $form_state['values']['cv_id'] : NULL;
  177. $term_name = array_key_exists('term_name', $form_state['values']) ? $form_state['values']['term_name'] : NULL;
  178. }
  179. // Let the user select the vocabulary and tripal_data but only if they haven't
  180. // already selected a tripal_data.
  181. $sql = "
  182. SELECT CV.cv_id, CV.name
  183. FROM {tripal_vocabulary} TET
  184. INNER JOIN {cv} CV on CV.cv_id = TET.cv_id
  185. ORDER BY CV.name
  186. ";
  187. $vocabs = chado_query($sql);
  188. $cvs = array();
  189. while ($vocab = $vocabs->fetchObject()) {
  190. $cvs[$vocab->cv_id] = $vocab->name;
  191. }
  192. $form['cv_id'] = array(
  193. '#type' => 'select',
  194. '#title' => t('Vocabulary'),
  195. '#options' => $cvs,
  196. '#required' => TRUE,
  197. '#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.'),
  198. '#default_value' => $cv_id,
  199. '#ajax' => array(
  200. 'callback' => "tripal_entities_admin_publish_form_ajax_callback",
  201. 'wrapper' => 'tripal_entities_admin_publish_form',
  202. 'effect' => 'fade',
  203. 'method' => 'replace'
  204. )
  205. );
  206. // If we have a CV ID then we want to provide an autocomplete field
  207. if ($cv_id) {
  208. $form['cvterm_select']['term_name'] = array(
  209. '#title' => t('Data Type'),
  210. '#type' => 'textfield',
  211. '#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."),
  212. '#required' => TRUE,
  213. '#default_value' => $term_name,
  214. '#autocomplete_path' => "admin/tripal/chado/tripal_cv/cvterm/auto_name/$cv_id",
  215. );
  216. $form['cvterm_select']['select_button'] = array(
  217. '#type' => 'submit',
  218. '#value' => t('Publish Data'),
  219. '#name' => 'publish',
  220. );
  221. }
  222. $form['#prefix'] = '<div id="tripal_entities_admin_publish_form">';
  223. $form['#suffix'] = '</div>';
  224. return $form;
  225. }
  226. /**
  227. * An Ajax callback for the tripal_entities_admin_publish_form..
  228. */
  229. function tripal_entities_admin_publish_form_ajax_callback($form, $form_state) {
  230. // return the form so Drupal can update the content on the page
  231. return $form;
  232. }
  233. /**
  234. * Implements hook_validate() for the tripal_entities_admin_publish_form.
  235. *
  236. */
  237. function tripal_entities_admin_publish_form_validate($form, &$form_state) {
  238. $cv_id = $form_state['values']['cv_id'];
  239. $term_name = $form_state['values']['term_name'];
  240. // Make sure the term_name is a real term in the vocabulary.
  241. $type = tripal_get_cvterm(array(
  242. 'name' => $term_name,
  243. 'cv_id' => $cv_id
  244. ));
  245. if (!$type) {
  246. form_set_error('term_name', t("The data type is not a valid name for the selected vocabulary."));
  247. }
  248. // Make sure the term is used in the site:
  249. $values = array(
  250. 'cvterm_id' => $type->cvterm_id,
  251. );
  252. $bundles = chado_select_record('tripal_term', array('term_id'), $values);
  253. if (count($bundles) == 0) {
  254. 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)));
  255. }
  256. // Make sure the term is not already published.
  257. $values = array(
  258. 'cvterm_id' => $type->cvterm_id,
  259. 'publish' => 1,
  260. );
  261. $bundles = chado_select_record('tripal_term', array('term_id'), $values);
  262. if (count($bundles) > 0) {
  263. form_set_error('term_name', t("This data type is already set as publishable."));
  264. }
  265. }
  266. /**
  267. * Implements hook_submit() for the tripal_entities_admin_publish_form.
  268. *
  269. */
  270. function tripal_entities_admin_publish_form_submit($form, &$form_state) {
  271. $cv_id = $form_state['values']['cv_id'];
  272. $term_name = $form_state['values']['term_name'];
  273. // Get the data type using the $term_name and $cv_id.
  274. $type = chado_generate_var('cvterm', array('cv_id' => $cv_id, 'name' => $term_name));
  275. // Start the transaction.
  276. $transaction = db_transaction();
  277. try {
  278. // We don't need to check if the vocabulary is used because the
  279. // form only shows those vocabs that are used.
  280. // Mark this entity as published.
  281. $match = array('cv_id' => $cv_id);
  282. $values = array('publish' => 1);
  283. $success = chado_update_record('tripal_vocabulary', $match, $values);
  284. if (!$success) {
  285. throw new Exception('Cannot set the vocabulary as publishable');
  286. }
  287. // We have already checked in the validate if the term already exists
  288. // as a bundle. So, if we're here we can enable it.
  289. $match = array('cvterm_id' => $type->cvterm_id);
  290. $values = array('publish' => 1);
  291. $success = chado_update_record('tripal_term', $match, $values);
  292. if (!$success) {
  293. throw new Exception('Cannot set the data type as publishable');
  294. }
  295. // Get the bundle
  296. $records = chado_select_record('tripal_term', array('*'), $match);
  297. $bundle = $records[0];
  298. $bundle_name = $type->dbxref_id->db_id->name . '_' . $type->dbxref_id->accession;
  299. $entity_type_name = $type->dbxref_id->db_id->name;
  300. // Get the list of tables where this cvterm is used.
  301. $values = array('term_id' => $bundle->term_id);
  302. $tables = chado_select_record('tripal_term_usage', array('*'), $values);
  303. // Iterate through the tables.
  304. foreach ($tables as $table) {
  305. $tablename = $table->data_table;
  306. // We only want to look at base tables.
  307. if ($tablename == 'cvterm_dbxref' || $tablename == 'cvterm_relationship' ||
  308. $tablename == 'cvtermpath' || $tablename == 'cvtermprop' || $tablename == 'chadoprop' ||
  309. $tablename == 'cvtermsynonym' || preg_match('/_relationship$/', $tablename) ||
  310. preg_match('/_cvterm$/', $tablename)) {
  311. continue;
  312. }
  313. // Add in the bundle's fields.
  314. tripal_entities_add_bundle_fields($tablename, $entity_type_name, $bundle_name);
  315. // The TripalDataType Entity manages the bundles we have available.
  316. // Therefore, we need to add a new entity for each bundle "type".
  317. $vals = array(
  318. 'label' => $bundle_name . ' (' . $type->name . ')',
  319. 'type' => $entity_type_name,
  320. 'bundle' => $bundle_name,
  321. 'data' => serialize(array()),
  322. 'module' => 'tripal_entities'
  323. );
  324. $tripal_data_type = new TripalDataType($vals, $entity_type_name . '_type');
  325. $tripal_data_type->save();
  326. }
  327. drupal_set_message(t('Data type, %type, is now set as publishable.', array('%type' => $term_name)));
  328. }
  329. catch (Exception $e) {
  330. $transaction->rollback();
  331. drupal_set_message('Failure publishing this data type: ' . $e->getMessage(), 'error');
  332. watchdog_exception('trp_entities', $e);
  333. }
  334. }
  335. /**
  336. *
  337. * @param $table
  338. * @param $entity_type
  339. * @param $bundle_name
  340. */
  341. function tripal_entities_add_bundle_fields($tablename, $entity_type_name, $bundle_name) {
  342. // Iterate through the columns of the table and see if fields have been
  343. // created for each one. If not, then create them.
  344. $schema = chado_get_schema($tablename);
  345. $columns = $schema['fields'];
  346. foreach ($columns as $column_name => $details) {
  347. $field_name = $tablename . '__' . $column_name;
  348. $field = field_info_field($field_name);
  349. // Skip the primary key field.
  350. if ($column_name == $schema['primary key'][0]) {
  351. continue;
  352. }
  353. // Determine if the field is required.
  354. $is_required = 0;
  355. if (array_key_exists('not null', $details)) {
  356. $is_required = $details['not null'] ? 1 : 0;
  357. }
  358. // Determine what type of field this should be.
  359. // Drupal data types are: https://www.drupal.org/node/159605.
  360. // Field types are here: https://www.drupal.org/node/1879542
  361. $field_type = '';
  362. $widget_type = '';
  363. $settings = array();
  364. $label = '';
  365. switch($details['type']) {
  366. case 'char':
  367. // unsupported by postgres
  368. break;
  369. case 'varchar':
  370. $field_type = 'text';
  371. $widget_type = 'text_textfield';
  372. $settings['max_length'] = $details['length'];
  373. break;
  374. case 'text':
  375. $field_type = 'text';
  376. $widget_type = 'text_textarea';
  377. $settings['max_length'] = '';
  378. break;
  379. case 'blob':
  380. // not sure how to support a blob field.
  381. continue;
  382. break;
  383. case 'int':
  384. $field_type = 'number_integer';
  385. $widget_type = 'number';
  386. break;
  387. case 'float':
  388. $field_type = 'number_float';
  389. $widget_type = 'number';
  390. $settings['precision'] = 10;
  391. $settings['scale'] = 2;
  392. $settings['decimal_separator'] = '.';
  393. break;
  394. case 'numeric':
  395. $field_type = 'number_decimal';
  396. $widget_type = 'number';
  397. break;
  398. case 'serial':
  399. // Serial fields are most likely not needed as a field.
  400. break;
  401. case 'boolean':
  402. // TODO: what is the proper field for booleans?????
  403. break;
  404. case 'datetime':
  405. // TODO: What is the proper datetime fields ??????
  406. break;
  407. default:
  408. drupal_set_message(t("Unhandled field type: %type", array('%type' => $details['type'])), 'warning');
  409. $field_type = 'text';
  410. $widget_type = 'text_textarea';
  411. if (array_key_exists('length', $details)) {
  412. $settings['max_length'] = $details['length'];
  413. }
  414. }
  415. // If we don't have a field type then we don't need to create a field.
  416. if (!$field_type) {
  417. // If we don't have a field type but it is required and doesn't have
  418. // a default value then we are in trouble.
  419. if ($is_required and !array_key_exists('default', $details)) {
  420. throw new Exception(t('The %table.%field type, %type, is not yet supported for Entity fields, but it is required,',
  421. array('%table' => $tablename, '%field' => $column_name, '%type' => $details['type'])));
  422. }
  423. continue;
  424. }
  425. // If this field is a foreign key field then we will have a special custom
  426. // field provided by Tripal.
  427. $is_fk = FALSE;
  428. if (array_key_exists('foreign keys', $schema)) {
  429. foreach ($schema['foreign keys'] as $remote_table => $fk_details) {
  430. if (array_key_exists($column_name, $fk_details['columns'])) {
  431. $is_fk = TRUE;
  432. }
  433. }
  434. }
  435. // If this column is a FK relationship then use a custom Tripal
  436. // defined field type for it.
  437. if ($is_fk) {
  438. // TODO: We need a better way to get the fields for FK relationships.
  439. // It's not a good idea to enumerate them all here. We need some sort
  440. // of hook or something that will let us lookup the correct field.
  441. switch ($column_name) {
  442. case 'organism_id':
  443. $field_type = 'organism_id';
  444. $label = 'Organism';
  445. $widget_type = 'tripal_entities_organism_select_widget';
  446. break;
  447. }
  448. }
  449. // If the field doesn't exist then create it.
  450. if (!$field) {
  451. $field = array(
  452. 'field_name' => $field_name,
  453. 'type' => $field_type,
  454. 'cardinality' => 1,
  455. 'locked' => TRUE,
  456. 'storage' => array(
  457. 'type' => 'field_chado_storage'
  458. ),
  459. );
  460. field_create_field($field);
  461. }
  462. // Attach the field to the bundle.
  463. $field_instance = array(
  464. 'field_name' => $field_name,
  465. 'label' => $label ? $label : ucwords(preg_replace('/_/', ' ', $column_name)),
  466. 'widget' => array(
  467. 'type' => $widget_type
  468. ),
  469. 'entity_type' => $entity_type_name,
  470. 'required' => $is_required,
  471. 'settings' => $settings,
  472. 'bundle' => $bundle_name,
  473. );
  474. field_create_instance($field_instance);
  475. }
  476. }
  477. /**
  478. *
  479. * @param unknown $form
  480. * @param unknown $form_state
  481. * @return multitype:
  482. */
  483. function tripal_entities_admin_access_form($form, &$form_state) {
  484. $form = array();
  485. return $form;
  486. }