123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442 |
- <?php
- /**
- * @file
- *
- * This file contains miscellaneous API functions specific to working with
- * records in Chado that do not have a home in any other sub category of
- * API functions.
- */
- /**
- * @defgroup tripal_chado_api Chado
- *
- * @ingroup tripal_api
- * The Tripal Chado API is a set of functions for interacting with data
- * inside of a Chado relational database. Entities (or pages) in Drupal
- * that are provided by Tripal can supply data from any supported database
- * back-end, and Chado is the default. This API contains a variety of sub
- * categories (or groups) where functions are organized. Any extension module
- * that desires to work with data in Chado will find these functions useful.
- */
- /**
- * Publishes content in Chado as a new TripalEntity entity.
- *
- * @param $values
- * A key/value associative array that supports the following keys:
- * - bundle_name: The name of the the TripalBundle (e.g. bio_data-12345).
- * @param $job
- * The jobs management object for the job if this function is run as a job.
- * This argument is added by Tripal during a job run and is not needed if
- * this function is run directly.
- *
- * @return boolean
- * TRUE if all of the records of the given bundle type were published, and
- * FALSE if a failure occured.
- *
- * @ingroup tripal_chado_api
- */
- function chado_publish_records($values, $job = NULL) {
- // Used for adding runtime to the progress report.
- $started_at = microtime(TRUE);
- // We want the job object in order to report progress with the job.
- if (is_numeric($job)) {
- $job_id = $job;
- $job = new TripalJob();
- $job->load($job_id);
- }
- $report_progress = TRUE;
- if (!is_object($job)) {
- $report_progress = FALSE;
- }
- // These are options for the tripal_report_error function. We do not
- // want to log messages to the watchdog but we do for the job and to
- // the terminal
- $message_type = 'publish_records';
- $message_opts = [
- 'watchdog' => FALSE,
- 'job' => $job,
- 'print' => TRUE,
- ];
- // Start an array for caching objects to save performance.
- $cache = [];
- // Make sure we have the required options: bundle_name.
- if (!array_key_exists('bundle_name', $values) or !$values['bundle_name']) {
- tripal_report_error('tripal_chado', TRIPAL_ERROR,
- "Could not publish record: @error",
- ['@error' => 'The bundle name must be provided'], $message_opts);
- return FALSE;
- }
- // Get the incoming arguments from the $values array.
- $bundle_name = $values['bundle_name'];
- $filters = array_key_exists('filters', $values) ? $values['filters'] : [];
- $sync_node = array_key_exists('sync_node', $values) ? $values['sync_node'] : '';
- // We want to break the number of records to publish into chunks in order to ensure
- // transactions do not run for too long (performance issue). The number of records
- // to be processed per chunk is set here:
- $chunk_size = 500;
- // Load the bundle entity so we can get information about which Chado
- // table/field this entity belongs to.
- $bundle = tripal_load_bundle_entity(['name' => $bundle_name]);
- $cache['bundle'] = $bundle;
- if (!$bundle) {
- tripal_report_error($message_type, TRIPAL_ERROR,
- "Unknown bundle. Could not publish record: @error",
- ['@error' => 'The bundle name must be provided'], $message_opts);
- return FALSE;
- }
- $chado_entity_table = chado_get_bundle_entity_table($bundle);
- // Get the mapping of the bio data type to the Chado table.
- $chado_bundle = db_select('chado_bundle', 'cb')
- ->fields('cb')
- ->condition('bundle_id', $bundle->id)
- ->execute()
- ->fetchObject();
- if (!$chado_bundle) {
- tripal_report_error('tripal_chado', TRIPAL_ERROR,
- "Cannot find mapping of bundle to Chado tables. Could not publish record.");
- return FALSE;
- }
- // Load the term for use in setting the alias for each entity created.
- $term = entity_load('TripalTerm', ['id' => $bundle->term_id]);
- $cache['term'] = $term;
- $table = $chado_bundle->data_table;
- $type_column = $chado_bundle->type_column;
- $type_linker_table = $chado_bundle->type_linker_table;
- $cvterm_id = $chado_bundle->type_id;
- $type_value = $chado_bundle->type_value;
- // Get the table information for the Chado table.
- $table_schema = chado_get_schema($table);
- $pkey_field = $table_schema['primary key'][0];
- // Construct the SQL for identifying which records should be published.
- $args = [];
- $select = "SELECT T.$pkey_field as record_id ";
- $from = "
- FROM {" . $table . "} T
- LEFT JOIN [" . $chado_entity_table . "] CE on CE.record_id = T.$pkey_field
- ";
- // For migration of Tripal v2 nodes to entities we want to include the
- // coresponding chado linker table.
- if ($sync_node && db_table_exists('chado_' . $table)) {
- $select = "SELECT T.$pkey_field as record_id, CT.nid ";
- $from .= " INNER JOIN [chado_" . $table . "] CT ON CT.$pkey_field = T.$pkey_field";
- }
- $where = " WHERE CE.record_id IS NULL ";
- // Handle records that are mapped to property tables.
- if ($type_linker_table and $type_column and $type_value) {
- $propschema = chado_get_schema($type_linker_table);
- $fkeys = $propschema['foreign keys'][$table]['columns'];
- foreach ($fkeys as $leftkey => $rightkey) {
- if ($rightkey == $pkey_field) {
- $from .= " INNER JOIN {" . $type_linker_table . "} LT ON T.$pkey_field = LT.$leftkey ";
- }
- }
- $where .= "AND LT.$type_column = :cvterm_id and LT.value = :prop_value";
- $args[':cvterm_id'] = $cvterm_id;
- $args[':prop_value'] = $type_value;
- }
- // Handle records that are mapped to cvterm linking tables.
- if ($type_linker_table and $type_column and !$type_value) {
- $cvtschema = chado_get_schema($type_linker_table);
- $fkeys = $cvtschema['foreign keys'][$table]['columns'];
- foreach ($fkeys as $leftkey => $rightkey) {
- if ($rightkey == $pkey_field) {
- $from .= " INNER JOIN {" . $type_linker_table . "} LT ON T.$pkey_field = LT.$leftkey ";
- }
- }
- $where .= "AND LT.$type_column = :cvterm_id";
- $args[':cvterm_id'] = $cvterm_id;
- }
- // Handle records that are mapped via a type_id column in the base table.
- if (!$type_linker_table and $type_column) {
- $where .= "AND T.$type_column = :cvterm_id";
- $args[':cvterm_id'] = $cvterm_id;
- }
- // Handle the case where records are in the cvterm table and mapped via a single
- // vocab. Here we use the type_value for the cv_id.
- if ($table == 'cvterm' and $type_value) {
- $where .= "AND T.cv_id = :cv_id";
- $args[':cv_id'] = $type_value;
- }
- // Handle the case where records are in the cvterm table but we want to
- // use all of the child terms.
- if ($table == 'cvterm' and !$type_value) {
- $where .= "AND T.cvterm_id IN (
- SELECT CVTP.subject_id
- FROM {cvtermpath} CVTP
- WHERE CVTP.object_id = :cvterm_id)
- ";
- $args[':cvterm_id'] = $cvterm_id;
- }
- // Now add in any additional filters
- $fields = field_info_field_map();
- foreach ($fields as $field_name => $details) {
- if (array_key_exists('TripalEntity', $details['bundles']) and
- in_array($bundle_name, $details['bundles']['TripalEntity']) and
- in_array($field_name, array_keys($filters))) {
- $instance = field_info_instance('TripalEntity', $field_name, $bundle_name);
- $chado_table = $instance['settings']['chado_table'];
- $chado_column = $instance['settings']['chado_column'];
- if ($chado_table == $table) {
- $where .= " AND T.$chado_column = :$field_name";
- $args[":$field_name"] = $filters[$field_name];
- }
- }
- }
- // First get the count
- // @performance optimize, estimate or remove this. It's only used for reporting progress on the command-line.
- $sql = "SELECT count(*) as num_records " . $from . $where;
- $result = chado_query($sql, $args);
- $count = $result->fetchField();
- tripal_report_error($message_type, TRIPAL_INFO,
- "There are !count records to publish.",
- ['!count' => $count], $message_opts);
- if ($report_progress) {
- $job->setTotalItems($count);
- $job->setItemsHandled(0);
- $job->setInterval(1);
- }
- // Perform the query in chunks.
- $sql = $select . $from . $where . ' LIMIT ' . $chunk_size;
- $more_records_to_publish = TRUE;
- $num_actually_published = 0; // the full number of records published.
- $i = 0; //reset on each chunk.
- while ($more_records_to_publish) {
- $records = chado_query($sql, $args);
- // There is no need to cache transactions since Drupal handles nested
- // transactions "by performing no transactional operations (as far as the
- // database sees) within the inner nesting layers". Effectively, Drupal
- // ensures nested trasactions work the same as passing a transaction
- // through to the deepest level and not starting a new transaction if we
- // are already in one.
- $transaction = db_transaction();
- try {
- // Keep track of how many we've published so far
- // before clearing our chunk,
- $num_actually_published = $num_actually_published + $i;
- $i = 0;
- while ($record = $records->fetchObject()) {
- // First save the tripal_entity record.
- // @performace This is likely a bottleneck. Too bad we can't create
- // multiple entities at once... sort of like the copy method.
- $record_id = $record->record_id;
- $ec = entity_get_controller('TripalEntity');
- $entity = $ec->create([
- 'bundle' => $bundle_name,
- 'term_id' => $bundle->term_id,
- // Add in the Chado details for when the hook_entity_create()
- // is called and our tripal_chado_entity_create() implementation
- // can deal with it.
- 'chado_record' => chado_generate_var($table, [$pkey_field => $record_id], ['include_fk' => 0]),
- 'chado_record_id' => $record_id,
- 'publish' => TRUE,
- 'bundle_object' => $bundle,
- ]);
- $entity = $entity->save($cache);
- if (!$entity) {
- throw new Exception('Could not create entity.');
- }
- // Next save the chado entity record.
- $entity_record = [
- 'entity_id' => $entity->id,
- 'record_id' => $record_id,
- ];
- // For the Tv2 to Tv3 migration we want to add the nid to the
- // entity so we can associate the node with the entity.
- if (property_exists($record, 'nid')) {
- $entity_record['nid'] = $record->nid;
- }
- $result = db_insert($chado_entity_table)
- ->fields($entity_record)
- ->execute();
- if (!$result) {
- throw new Exception('Could not create mapping of entity to Chado record.');
- }
- $i++;
- if ($report_progress) {
- $job->setItemsHandled($i);
- }
- }
- } catch (Exception $e) {
- $transaction->rollback();
- $error = $e->getMessage();
- tripal_report_error($message_type, TRIPAL_ERROR, "Could not publish record: @error", ['@error' => $error]);
- drupal_set_message('Failed publishing record. See recent logs for more details.', 'error');
- return FALSE;
- }
- // If we get through the loop and haven't completed 100 records, then
- // we're done!
- if ($i < $chunk_size) {
- $more_records_to_publish = FALSE;
- }
- // Commit our current chunk.
- unset($transaction);
- }
- // Add the last number published to our total.
- $num_actually_published = $num_actually_published + $i;
- tripal_report_error($message_type, TRIPAL_INFO,
- "Successfully published !count !type record(s).",
- ['!count' => $num_actually_published, '!type' => $bundle->label], $message_opts);
- return TRUE;
- }
- /**
- * Returns an array of tokens based on Tripal Entity Fields.
- *
- * @param $base_table
- * The name of a base table in Chado.
- *
- * @return
- * An array of tokens where the key is the machine_name of the token.
- *
- * @ingroup tripal_chado_api
- */
- function chado_get_tokens($base_table) {
- $tokens = [];
- $table_descrip = chado_get_schema($base_table);
- foreach ($table_descrip['fields'] as $field_name => $field_details) {
- $token = '[' . $base_table . '.' . $field_name . ']';
- $location = implode(' > ', [$base_table, $field_name]);
- $tokens[$token] = [
- 'name' => ucwords(str_replace('_', ' ', $base_table)) . ': ' . ucwords(str_replace('_', ' ', $field_name)),
- 'table' => $base_table,
- 'field' => $field_name,
- 'token' => $token,
- 'description' => array_key_exists('description', $field_details) ? $field_details['description'] : '',
- 'location' => $location,
- ];
- if (!array_key_exists('description', $field_details) or preg_match('/TODO/', $field_details['description'])) {
- $tokens[$token]['description'] = 'The ' . $field_name . ' field of the ' . $base_table . ' table.';
- }
- }
- // RECURSION:
- // Follow the foreign key relationships recursively
- if (array_key_exists('foreign keys', $table_descrip)) {
- foreach ($table_descrip['foreign keys'] as $table => $details) {
- foreach ($details['columns'] as $left_field => $right_field) {
- $sub_token_prefix = $base_table . '.' . $left_field;
- $sub_tokens = chado_get_tokens($table);
- foreach ($sub_tokens as $key => $sub_token) {
- $key = preg_replace('/[\[\]]/','', $key);
- $new_key = '[' . $sub_token_prefix .'>' . $key . ']';
- $new_location = implode(' > ', [$base_table, $left_field, $sub_token['field']]);
- $tokens[$new_key] = $sub_token;
- $tokens[$new_key]['token'] = $new_key;
- $tokens[$new_key]['location'] = $new_location;
- }
- }
- }
- }
- return $tokens;
- }
- /**
- * Replace all Chado Tokens in a given string.
- *
- * NOTE: If there is no value for a token then the token is removed.
- *
- * @param string $string
- * The string containing tokens.
- * @param $record
- * A Chado record as generated by chado_generate_var()
- *
- * @return
- * The string will all tokens replaced with values.
- *
- * @ingroup tripal_chado_api
- */
- function chado_replace_tokens($string, $record) {
- // Get the list of tokens.
- $tokens = chado_get_tokens($record->tablename);
- // Determine which tokens were used in the format string.
- if (preg_match_all('/\[[^]]+\]/', $string, $used_tokens)) {
- // Get the value for each token used.
- foreach ($used_tokens[0] as $token) {
- $token_info = $tokens[$token];
- if (!empty($token_info)) {
- $table = $token_info['table'];
- $var = $record;
- $value = '';
- // Iterate through each portion of the location string. An example
- // string might be: stock > type_id > name.
- $location = explode('>', $token_info['location']);
- foreach ($location as $index) {
- $index = trim($index);
- // if $var is an object then it is and expanded table.
- // So, we'll reset $var to be this value and allow the
- // loop to continue to the next level.
- if (is_object($var) and property_exists($var, $index)) {
- if (is_object($var->$index)) {
- $var = $var->$index;
- }
- else {
- $value = $var->$index;
- }
- }
- else {
- tripal_report_error('tripal_chado', TRIPAL_WARNING,
- 'Tokens: Unable to determine the value of %token. Things went awry when trying ' .
- 'to access \'%index\' for the following: \'%var\'.',
- [
- '%token' => $token,
- '%index' => $index,
- '%var' => print_r($var, TRUE),
- ]
- );
- }
- }
- $string = str_replace($token, $value, $string);
- }
- }
- }
- return $string;
- }
|