Fix commerce_invoice update errors on Drupal 11

Created on 16 March 2025, 20 days ago

Added checks in update hooks 8200–8203 to verify that required fields and tables exist before applying schema changes. This prevents fatal errors during database updates (drush updb) on Drupal 11 when fields or tables are missing or already updated.

<?php

/**
 * @file
 * Contains install and update functions for Commerce Invoice.
 */

use Drupal\commerce_invoice\Entity\Invoice;
use Drupal\Core\Field\BaseFieldDefinition;
use Drupal\Core\StreamWrapper\PrivateStream;

/**
 * Implements hook_requirements().
 */
function commerce_invoice_requirements($phase) {
  $requirements = [];

  // Ensure the private file system path is configured.
  if (in_array($phase, ['install', 'runtime']) && !PrivateStream::basePath()) {
    $requirements['commerce_invoice_private_path'] = [
      'title' => t('Private file system path'),
      'description' => t('Commerce Invoice requires the private file system path to be configured.'),
      'severity' => REQUIREMENT_ERROR,
    ];
  }

  return $requirements;
}

/**
 * Mark the invoice owner field as non translatable.
 */
function commerce_invoice_update_8200() {
  $definition_update_manager = \Drupal::entityDefinitionUpdateManager();
  $field_definition = $definition_update_manager->getFieldStorageDefinition('uid', 'commerce_invoice');
  if ($field_definition) {
    $field_definition->setTranslatable(FALSE);
    $definition_update_manager->updateFieldStorageDefinition($field_definition);
  }
  else {
    \Drupal::logger('commerce_invoice')->notice('Skipping update 8200: invoice owner field definition not found.');
  }
}

/**
 * Update the 'uid' field.
 */
function commerce_invoice_update_8201() {
  $definition_update_manager = \Drupal::entityDefinitionUpdateManager();
  $base_field_override_storage = \Drupal::entityTypeManager()->getStorage('base_field_override');
  $storage_definition = $definition_update_manager->getFieldStorageDefinition('uid', 'commerce_invoice');
  if (!$storage_definition) {
    \Drupal::logger('commerce_invoice')->notice('Skipping update 8201: "uid" field storage definition not found.');
    return;
  }
  $default_value_callback = Invoice::class . '::getCurrentUserId';

  $base_field_overrides = $base_field_override_storage->loadByProperties([
    'entity_type' => 'commerce_invoice',
    'field_name' => 'uid',
  ]);
  /** @var \Drupal\Core\Field\FieldDefinitionInterface $base_field_override */
  foreach ($base_field_overrides as $base_field_override) {
    if ($base_field_override->getDefaultValueCallback() !== $storage_definition->getDefaultValueCallback()) {
      continue;
    }
    // Update the "default_value_callback" for base field overrides, as long
    // as they're using the default one.
    $base_field_override->setDefaultValueCallback($default_value_callback);
    $base_field_override->save();
  }

  $storage_definition->setDefaultValueCallback($default_value_callback);
  $definition_update_manager->updateFieldStorageDefinition($storage_definition);
}

/**
 * Add a file field to invoices that holds a reference to the PDF file.
 */
function commerce_invoice_update_8202() {
  $database = \Drupal::database();
  // commerce_invoice_field_data is the table for the commerce_invoice entity.
  if (!$database->schema()->tableExists('commerce_invoice_field_data')) {
    \Drupal::logger('commerce_invoice')->notice('Skipping update 8202: commerce_invoice_field_data table does not exist.');
    return;
  }

  $definition_manager = \Drupal::entityDefinitionUpdateManager();
  $existing_definition = $definition_manager->getFieldStorageDefinition('invoice_file', 'commerce_invoice');
  if ($existing_definition) {
    \Drupal::logger('commerce_invoice')->notice('Skipping update 8202: invoice_file field already exists.');
    return;
  }
  $storage_definition = BaseFieldDefinition::create('entity_reference')
    ->setLabel(t('Invoice PDF'))
    ->setSetting('target_type', 'file')
    ->setDescription(t('The invoice PDF file.'))
    ->setDisplayConfigurable('view', TRUE);

  $definition_manager->installFieldStorageDefinition('invoice_file', 'commerce_invoice', 'commerce_invoice', $storage_definition);
}

/**
 * Convert the invoice_file from entity_reference to file field.
 */
function commerce_invoice_update_8203() {
  $entity_type_manager = \Drupal::entityTypeManager();
  $bundle_of = 'commerce_invoice';

  $storage = $entity_type_manager->getStorage($bundle_of);
  $bundle_definition = $entity_type_manager->getDefinition($bundle_of);
  // Sometimes the primary key isn't 'id'. e.g. 'eid' or 'item_id'.
  $id_key = $bundle_definition->getKey('id');
  // If there is no data table defined then use the base table.
  $table_name = $storage->getDataTable() ?: $storage->getBaseTable();
  
  $database = \Drupal::database();
  if (!$database->schema()->tableExists($table_name)) {
    \Drupal::logger('commerce_invoice')->notice("Skipping update 8203: Table $table_name does not exist.");
    return;
  }

  $definition_manager = \Drupal::entityDefinitionUpdateManager();

  // Store the existing values.
  $file_values = $database->select($table_name)
    ->fields($table_name, [$id_key, 'invoice_file'])
    ->execute()
    ->fetchAllKeyed();

  // Clear out the values.
  $database->update($table_name)
    ->fields(['invoice_file' => NULL])
    ->execute();

  // Uninstall the field.
  $field_storage_definition = $definition_manager->getFieldStorageDefinition('invoice_file', $bundle_of);
  if (!$field_storage_definition) {
    \Drupal::logger('commerce_invoice')->notice('Skipping update 8203: invoice_file field storage definition not found.');
    return;
  }
  $definition_manager->uninstallFieldStorageDefinition($field_storage_definition);

  // Create a new field definition.
  $new_invoice_file = BaseFieldDefinition::create('file')
    ->setLabel(t('Invoice PDF'))
    ->setDescription(t('The invoice PDF file.'))
    ->setDisplayConfigurable('view', TRUE);

  // Install the new definition.
  $definition_manager->installFieldStorageDefinition('invoice_file', $bundle_of, $bundle_of, $new_invoice_file);

  // Restore the values.
  foreach ($file_values as $id => $value) {
    $database->update($table_name)
      ->fields(['invoice_file__target_id' => $value])
      ->condition($id_key, $id)
      ->execute();
  }
}
🐛 Bug report
Status

Active

Version

2.0

Component

Code

Created by

🇺🇸United States w01f

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Comments & Activities

  • Issue created by @w01f
  • 🇮🇱Israel jsacksick

    But why would the updates run if the fields are already there? I'm confused?

  • 🇺🇸United States w01f

    I just ran into this, errors when attempting to run drush updb, when attempting to install commerce_invoice and bee_hotel together on a new D11 instance. These changes allowed me to run updb as normal, though I am still having issues getting everything to work as expected.

Production build 0.71.5 2024