| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267 | <?php/** * @file * Support for taxonomy term destinations. */// TODO:// Make sure this works with updates, explicit destination keys// taxonomy_term_save() is doing a cache_clear_all and an automatic insertion for parent./** * Destination class implementing migration into terms. */class MigrateDestinationTerm extends MigrateDestinationEntity {  static public function getKeySchema() {    return array(      'tid' => array(        'type' => 'int',        'unsigned' => TRUE,        'description' => 'ID of destination term',      ),    );  }  /**   * Return an options array for term destinations.   *   * @param string $language   *  Default language for terms created via this destination class.   * @param string $text_format   *  Default text format for terms created via this destination class.   */  static public function options($language, $text_format) {    return compact('language', 'text_format');  }  /**   * Basic initialization   *   * @param array $options   *  Options applied to terms.   */  public function __construct($bundle, array $options = array()) {    parent::__construct('taxonomy_term', $bundle, $options);  }  /**   * Returns a list of fields available to be mapped for this vocabulary (bundle)   *   * @param Migration $migration   *  Optionally, the migration containing this destination.   * @return array   *  Keys: machine names of the fields (to be passed to addFieldMapping)   *  Values: Human-friendly descriptions of the fields.   */  public function fields($migration = NULL) {    $fields = array();    // First the core (taxonomy_term_data table) properties    $fields['tid'] = t('Term: <a href="@doc">Existing term ID</a>',                array('@doc' => 'http://drupal.org/node/1349702#tid'));    $fields['name'] = t('Term: <a href="@doc">Name</a>',                    array('@doc' => 'http://drupal.org/node/1349702#name'));    $fields['description'] = t('Term: <a href="@doc">Description</a>',                    array('@doc' => 'http://drupal.org/node/1349702#description'));    $fields['parent'] = t('Term: <a href="@doc">Parent (by Drupal term ID)</a>',                    array('@doc' => 'http://drupal.org/node/1349702#parent'));    // TODO: Remove parent_name, implement via arguments    $fields['parent_name'] = t('Term: <a href="@doc">Parent (by name)</a>',                    array('@doc' => 'http://drupal.org/node/1349702#parent_name'));    $fields['format'] = t('Term: <a href="@doc">Format</a>',                    array('@doc' => 'http://drupal.org/node/1349702#format'));    $fields['weight'] = t('Term: <a href="@doc">Weight</a>',                    array('@doc' => 'http://drupal.org/node/1349702#weight'));    // Then add in anything provided by handlers    $fields += migrate_handler_invoke_all('entity', 'fields', $this->entityType, $this->bundle, $migration);    $fields += migrate_handler_invoke_all('taxonomy_term', 'fields', $this->entityType, $this->bundle, $migration);    return $fields;  }  /**   * Delete a migrated term   *   * @param $ids   *  Array of fields representing the key (in this case, just tid).   */  public function rollback(array $key) {    $tid = reset($key);    /*     * This load() happens soon delete() anyway. We load here in order to     * avoid notices when term has already been deleted. That is easily possible     * considering how deleting a term parent also deletes children in same call.     */    migrate_instrument_start('taxonomy_term_load');    if (taxonomy_term_load($tid)) {      migrate_instrument_stop('taxonomy_term_load');      migrate_instrument_start('taxonomy_term_delete');      $this->prepareRollback($tid);      $result = (bool) taxonomy_term_delete($tid);      $this->completeRollback($tid);      migrate_instrument_stop('taxonomy_term_delete');    }    else {      migrate_instrument_stop('taxonomy_term_load');      // If it didn't exist, consider this a success      $result = TRUE;    }    return $result;  }  /**   * Import a single term.   *   * @param $term   *  Term object to build. Prefilled with any fields mapped in the Migration.   * @param $row   *  Raw source data object - passed through to prepare/complete handlers.   * @return array   *  Array of key fields (tid only in this case) of the term that was saved if   *  successful. FALSE on failure.   */  public function import(stdClass $term, stdClass $row) {    $migration = Migration::currentMigration();    // Updating previously-migrated content?    if (isset($row->migrate_map_destid1)) {      $term->tid = $row->migrate_map_destid1;      if (isset($term->tid)) {        if ($term->tid != $row->migrate_map_destid1) {          throw new MigrateException(t("Incoming tid !tid and map destination nid !destid1 don't match",            array('!tid' => $term->tid, '!destid1' => $row->migrate_map_destid1)));        }      }      else {        $term->tid = $row->migrate_map_destid1;      }    }    if ($migration->getSystemOfRecord() == Migration::DESTINATION) {      if (!isset($term->tid)) {        throw new MigrateException(t('System-of-record is DESTINATION, but no destination tid provided'));      }      $rawterm = $term;      $this->prepare($term, $row);      $old_term = taxonomy_term_load($term->tid);      if (empty($old_term)) {        throw new MigrateException(t('System-of-record is DESTINATION, but term !tid does not exist',          array('!tid' => $term->tid)));      }      foreach ($rawterm as $field => $value) {        $old_term->$field = $term->$field;      }      $term = $old_term;    }    else {      // Default to bundle if no vocabulary machine name provided      if (!isset($term->vocabulary_machine_name)) {        $term->vocabulary_machine_name = $this->bundle;      }      // vid is required      if (empty($term->vid)) {        static $vocab_map = array();        if (!isset($vocab_map[$term->vocabulary_machine_name])) {          // The keys of the returned array are vids          $vocabs = taxonomy_vocabulary_load_multiple(array(),            array('machine_name' => $term->vocabulary_machine_name));          $vids = array_keys($vocabs);          if (isset($vids[0])) {            $vocab_map[$term->vocabulary_machine_name] = $vids[0];          }          else {            $migration->saveMessage(t('No vocabulary found with machine_name !name',              array('!name' => $term->vocabulary_machine_name)));            return FALSE;          }        }        $term->vid = $vocab_map[$term->vocabulary_machine_name];      }      // Look up parent name if provided      if (isset($term->parent_name) && trim($term->parent_name)) {        // Look for the name in the same vocabulary.        // Note that hierarchies may have multiples of the same name...        $terms = taxonomy_term_load_multiple(array(),          array('name' => trim($term->parent_name), 'vid' => $term->vid));        $tids = array_keys($terms);        $term->parent = array($tids[0]);        unset($term->parent_name);      }      if (empty($term->parent)) {        $term->parent = array(0);      }      if (is_array($term->parent) && isset($term->parent['arguments'])) {        // Unset arguments here to avoid duplicate entries in the        // term_hierarchy table.        unset($term->parent['arguments']);      }      if (!isset($term->format)) {        $term->format = $this->textFormat;      }      $this->prepare($term, $row);      // See if the term, with the same parentage, already exists - if so,      // load it      $candidates = taxonomy_term_load_multiple(array(),        array('name' => trim($term->name), 'vid' => $term->vid));      foreach ($candidates as $candidate) {        $parents = taxonomy_get_parents($candidate->tid);        // We need to set up $parents as a simple array of tids        if (empty($parents)) {          $parents = array(0);        }        else {          // Parents array is tid => term object, make into list of tids          $new_parents = array();          foreach ($parents as $parent) {            $new_parents[] = $parent->tid;          }          $parents = $new_parents;        }        if ($term->parent == $parents) {          // We've found a matching term, we'll use that          $term = $candidate;          break;        }      }    }    // Trying to update an existing term    if ($migration->getSystemOfRecord() == Migration::DESTINATION) {      $existing_term = taxonomy_term_load($term->tid);      if ($existing_term) {        // Incoming data overrides existing data, so only copy non-existent fields        foreach ($existing_term as $field => $value) {          if (!isset($term->$field)) {            $term->$field = $existing_term->$field;          }        }      }    }    if (isset($term->tid)) {      $updating = TRUE;    }    else {      $updating = FALSE;    }    migrate_instrument_start('taxonomy_term_save');    $status = taxonomy_term_save($term);    migrate_instrument_stop('taxonomy_term_save');    $this->complete($term, $row);    if (isset($term->tid)) {      if ($updating) {        $this->numUpdated++;      }      else {        $this->numCreated++;      }      $return = array($term->tid);    }    else {      $return = FALSE;    }    return $return;  }}
 |