| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322 | <?php/** * Retrieves an entity that matches the given table and record id. * * @param $table *   The name of the Chado table. * @param $record_id *   The record's primary key in the table specified by $table. * * @return *   A chado_entity object. */function tripal_load_chado_entity($table, $record_id) {   $entity_id = db_select('chado_entity', 'ce')     ->fields('ce', array('entity_id'))     ->condition('ce.record_id', $record_id)     ->condition('ce.data_table', $table)     ->execute()     ->fetchField();   if ($entity_id) {     $entity = entity_load('TripalEntity', array($entity_id));     return reset($entity);   }   return NULL;}/** * Retrieves the entity ID using a record ID. * * @param unknown $data_table * @param unknown $record_id */function tripal_get_chado_entity_id($data_table, $record_id) {   return db_select('chado_entity', 'ce')     ->fields('ce', array('entity_id'))     ->condition('data_table', $data_table)     ->condition('record_id', $record_id)     ->execute()     ->fetchField();}/** * 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_id *   (Optional) The numeric job ID as provided by the Tripal jobs system. There *   is no need to specify this argument if this function is being called *   outside of the jobs systems. * * @return boolean *   TRUE if all of the records of the given bundle type were published, and *   FALSE if a failure occured. */function tripal_chado_publish_records($values, $job_id = NULL) {  // 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",      array('@error' => 'The bundle name must be provided'));    return FALSE;  }  $bundle_name = $values['bundle_name'];  $sync_node = array_key_exists('sync_node', $values) ? $values['sync_node'] : '';  // Load the bundle entity so we can get information about which Chado  // table/field this entity belongs to.  $bundle = tripal_load_bundle_entity(array('name' => $bundle_name));  if (!$bundle) {    tripal_report_error('tripal_chado', TRIPAL_ERROR,        "Unknown bundle. Could not publish record: @error",        array('@error' => 'The bundle name must be provided'));    return FALSE;  }  $table = $bundle->data_table;  $type_column = $bundle->type_column;  $type_linker_table = $bundle->type_linker_table;  $cvterm_id  = $bundle->type_id;  // 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 = array();  $select = "SELECT $pkey_field as record_id ";  $from = "    FROM {" . $table . "} T      LEFT JOIN public.chado_entity CE on CE.record_id = T.$pkey_field         AND CE.data_table = :table  ";  // 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 public.chado_$table CT ON CT.$pkey_field = T.$pkey_field";  }  $where = " WHERE CE.record_id IS NULL ";  $args[':table'] = $table;  // Handle bundles that use a linker property table for identifying the type  // of record to publish.  if ($type_linker_table and $type_column) {    $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";    $args[':cvterm_id'] = $cvterm_id;  }  // If the type column is in the base table then add in the SQL for that.  if (!type_linker_table and $type_column) {    $where .= "AND T.$type_column = :cvterm_id";    $args[':cvterm_id'] = $cvterm_id;  }  // If no type column is specified then we have a problem.  if ($type_linker_table and !$type_column) {    tripal_report_error('tripal_chado', TRIPAL_ERROR,        "Could not publish record: @error",        array('@error' => 'The bundle does not properly map to Chado.'));    return FALSE;  }  // First get the count  $sql = "SELECT count(*) as num_records " . $from . $where;  $result = chado_query($sql, $args);  $count = $result->fetchField();  // calculate the interval for updates  $interval = intval($count / 1000);  if ($interval < 1) {    $interval = 1;  }  // Perform the query.  $sql = $select . $from . $where;  $records = chado_query($sql, $args);  $transaction  = db_transaction();  print "\nNOTE: publishing records is performed using a database transaction. \n" .      "If the load fails or is terminated prematurely then the entire set of \n" .      "is rolled back with no changes to the database\n\n";  printf("%d of %d records. (%0.2f%%) Memory: %s bytes\r", $i, $count, 0, number_format(memory_get_usage()));  try {    $i = 0;    while($record = $records->fetchObject()) {      // update the job status every interval      //if ($jobid and $i % $interval == 0) {        $complete = ($i / $count) * 33.33333333;        //tripal_set_job_progress($jobid, intval($complete + 33.33333333));        printf("%d of %d records. (%0.2f%%) Memory: %s bytes\r", $i, $count, $complete * 3, number_format(memory_get_usage()));      //}      // First save the tripal_entity record.      $record_id = $record->record_id;      $ec = entity_get_controller('TripalEntity');      $entity = $ec->create(array(        'bundle' => $bundle_name,        'term_id' => $bundle->term_id,        'chado_record' => chado_generate_var($table, array($pkey_field => $record_id)),        'chado_record_id' => $record_id,      ));      $entity = $entity->save();      if (!$entity) {        throw new Exception('Could not create entity.');      }      // Next save the chado_entity record.      $entity_record = array(        'entity_id' => $entity->id,        'record_id' => $record_id,        'data_table' => $table,      );      // 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;      }      $success = drupal_write_record('chado_entity', $entity_record);      $i++;    }  }  catch (Exception $e) {    $transaction->rollback();    $error = $e->getMessage();    tripal_report_error('tripal_chado', TRIPAL_ERROR, "Could not publish record: @error", array('@error' => $error));    drupal_set_message('Failed publishing record. See recent logs for more details.', 'error');    return FALSE;  }  drupal_set_message("Succesfully published $i " . $bundle->label . " record(s).");  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. */function tripal_get_chado_tokens($base_table) {  $tokens = array();  $table_descrip = chado_get_schema($base_table);  foreach ($table_descrip['fields'] as $field_name => $field_details) {    $token = '[' . $base_table . '.' . $field_name . ']';    $location = implode(' > ',array($base_table, $field_name));    $tokens[$token] = array(      '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_location_prefix = implode(' > ',array($base_table, $left_field));        $sub_tokens = tripal_get_chado_tokens($table);        if (is_array($sub_tokens)) {          $tokens = array_merge($tokens, $sub_tokens);        }      }    }  }  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. */function tripal_replace_chado_tokens($string, $record) {  // Get the list of tokens  $tokens = tripal_get_chado_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 the $node object or a table          // that has been expanded.          if (is_object($var)) {            // check to see if the index is a member of the object. If so,            // then reset the $var to this value.            if (property_exists($var, $index)) {              $value = $var->$index;            }          }          // if the $var is an array then there are multiple instances of the same          // table in a FK relationship (e.g. relationship tables)          elseif (is_array($var)) {            $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\'.',              array('%token' => $token, '%index' => $index, '%var' => print_r($var,TRUE))            );          }        }        $string = str_replace($token, $value, $string);      }    }  }  return $string;}
 |