123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282 |
- <?php
- /**
- * Destination class for the webform_submissions table.
- *
- * Working component types:
- * - email
- * - date ('Y-m-d')
- * - file (use the file id)
- * - markup
- * - pagebreak (content is ignored)
- * - select (looks up the key by default, pass 'source_type' => 'value' as an
- * argument to try to match the value)
- * - textfield
- * - textarea
- * - time ('H:i:s')
- * Untested/needs work:
- * - grid
- * - hidden
- */
- class MigrateDestinationWebformSubmission extends MigrateDestination {
- static public function getKeySchema() {
- return array(
- 'sid' => array(
- 'type' => 'int',
- 'not null' => TRUE,
- 'unsigned' => TRUE,
- ),
- );
- }
- /**
- * The webform of the destination.
- *
- * @var string
- */
- protected $node;
- public function getWebform() {
- return $this->node;
- }
- /**
- * An array mapping our custom names to component ids.
- *
- * @var array
- */
- protected $component_cids;
- /**
- * Constructs a destination for a given webform node.
- *
- * @param object $node
- * A node object that's type has been enabled for webform use.
- */
- public function __construct($node) {
- parent::__construct();
- if (empty($node)) {
- throw new Exception(t("You must provide a webform node"));
- }
- // Make sure it's a webform node.
- $types = webform_variable_get('webform_node_types');
- if (!in_array($node->type, $types)) {
- throw new Exception(t("The node must be configured to accept webform submissions but %type was not", array('%type' => $node->type)));
- }
- $this->node = $node;
- // Webform expects the component values to be keyed by cid, so we need a
- // hash to map prefixed field names to cid.
- $this->component_cids = array();
- foreach ($this->node->webform['components'] as $component) {
- $this->component_cids['data_' . $component['form_key']] = $component['cid'];
- }
- // We use the functions in this file in import() but load it here so we
- // only do it once.
- module_load_include('inc', 'webform', 'includes/webform.submissions');
- }
- public function __toString() {
- return t('Submission for the <a href="!link">%title</a> Webform', array(
- '!link' => url('node/' . $this->node->nid),
- '%title' => $this->node->title,
- ));
- }
- /**
- * Returns a list of fields available to be mapped.
- *
- * @return array
- * Keys: machine names of the fields (to be passed to addFieldMapping)
- * Values: Human-friendly descriptions of the fields.
- */
- public function fields() {
- // Fields defined by the schema. nid is omitted since it should come from
- // $this->node.
- $fields = array(
- 'sid' => t('The unique identifier for this submission.'),
- 'uid' => t('The id of the user that completed this submission.'),
- 'is_draft' => t('Is this a draft of the submission?'),
- 'submitted' => t('Timestamp of when the form was submitted.'),
- 'remote_addr' => t('The IP address of the user that submitted the form.'),
- );
- // Create a field for each component on the webform.
- foreach ($this->node->webform['components'] as $component) {
- // TODO: Seems like we should skip over page break components.
- $fields['data_' . $component['form_key']] = t('@type: @name', array('@type' => $component['type'], '@name' => $component['name']));
- }
- // Then add in anything provided by handlers.
- $fields += migrate_handler_invoke_all('WebformSubmission', 'fields', $this->node);
- return $fields;
- }
- /**
- * Give handlers a shot at modifying the object before saving it.
- *
- * @param $entity
- * Webform submission object to build. Prefilled with any fields mapped in
- * the Migration.
- * @param $source_row
- * Raw source data object - passed through to prepare handlers.
- */
- public function prepare($entity, stdClass $source_row) {
- $migration = Migration::currentMigration();
- $entity->migrate = array(
- 'machineName' => $migration->getMachineName(),
- );
- // Call any general object handlers.
- migrate_handler_invoke_all('WebformSubmission', 'prepare', $entity, $source_row, $this->node);
- // Then call any prepare handler for this specific Migration.
- if (method_exists($migration, 'prepare')) {
- $migration->prepare($entity, $source_row, $this->node);
- }
- }
- /**
- * Give handlers a shot at modifying the object (or taking additional action)
- * after saving it.
- *
- * @param $entity
- * Webform submission object to build. This is the complete object after
- * saving.
- * @param $source_row
- * Raw source data object - passed through to complete handlers.
- */
- public function complete($entity, stdClass $source_row) {
- $migration = Migration::currentMigration();
- // Call any general object handlers.
- migrate_handler_invoke_all('WebformSubmission', 'complete', $entity, $source_row, $this->node);
- // Then call any complete handler for this specific Migration.
- if (method_exists($migration, 'complete')) {
- $migration->complete($entity, $source_row, $this->node);
- }
- }
- /**
- * Import a record.
- *
- * @param $entity
- * Webform submission object to build. This is the complete object after
- * saving.
- * @param $source_row
- * Raw source data object - passed through to complete handlers.
- */
- public function import(stdClass $entity, stdClass $row) {
- // Updating previously-migrated content?
- $migration = Migration::currentMigration();
- if (isset($row->migrate_map_destid1)) {
- if (isset($entity->sid) && $entity->sid != $row->migrate_map_destid1) {
- throw new MigrateException(t("Incoming sid !sid and map destination sid !destid1 don't match",
- array('!sid' => $entity->sid, '!destid1' => $row->migrate_map_destid1)));
- }
- else {
- $entity->sid = $row->migrate_map_destid1;
- }
- }
- $entity->nid = $this->node->nid;
- // Move the data from our custom keys back to webform's component ids.
- $data = array();
- foreach ($this->component_cids as $field_name => $cid) {
- if (isset($entity->$field_name)) {
- // Move the arguments out and kill any extraneous wrapper arrays.
- $value = $entity->$field_name;
- $arguments = array();
- if (is_array($value) && isset($value['arguments'])) {
- $arguments = (array) $value['arguments'];
- unset($value['arguments']);
- $value = count($value) ? reset($value) : $value;
- }
- // Avoid a warning if they passed in an empty array.
- $arguments += array('source_type' => 'key');
- // By default passed to select components are assumed to be the
- // key. If the key should be looked up use the add a
- // array('source_type' => 'value') argument to the field mapping.
- $component = $this->node->webform['components'][$cid];
- if ($component['type'] == 'select' && $arguments['source_type'] == 'value') {
- $options = _webform_select_options($component);
- $id = array_search($value, $options);
- $data[$cid] = ($id === FALSE) ? NULL : $id;
- }
- else {
- $data[$cid] = $value;
- }
- unset($entity->$field_name);
- }
- }
- $entity->data = webform_submission_data($this->node, $data);
- // Invoke migration prepare handlers
- $this->prepare($entity, $row);
- migrate_instrument_start('webform_submission_update/insert');
- // Determine if it's an insert or update.
- if (empty($entity->sid)) {
- $updating = FALSE;
- $sid = webform_submission_insert($this->node, $entity);
- }
- else {
- // If the sid was specified but doesn't exist we'll need to stick an
- // empty record in so webform's update has something to stick to.
- $status = db_merge('webform_submissions')
- ->key(array(
- 'sid' => $entity->sid,
- ))
- ->insertFields(array(
- 'sid' => $entity->sid,
- 'nid' => $entity->nid,
- 'submitted' => $entity->submitted,
- 'remote_addr' => $entity->remote_addr,
- 'is_draft' => $entity->is_draft,
- 'bundle' => $entity->bundle,
- ))
- ->execute();
- // If db_merge() makes no changes $status is NULL so make a less
- // elegant comparison.
- $updating = MergeQuery::STATUS_INSERT !== $status;
- $sid = webform_submission_update($this->node, $entity);
- }
- migrate_instrument_stop('webform_submission_update/insert');
- if (isset($sid)) {
- $entity->sid = $sid;
- if ($updating) {
- $this->numUpdated++;
- }
- else {
- $this->numCreated++;
- }
- $return = array($sid);
- }
- else {
- $return = FALSE;
- }
- // Invoke migration complete handlers
- $this->complete($entity, $row);
- return $return;
- }
- /**
- * Delete a batch of submissions at once.
- *
- * @param $sids
- * Array of submission IDs to be deleted.
- */
- public function bulkRollback(array $sids) {
- migrate_instrument_start(__METHOD__);
- foreach (webform_get_submissions(array('sid' => $sids)) as $submission) {
- webform_submission_delete($this->node, $submission);
- }
- migrate_instrument_stop(__METHOD__);
- }
- }
|