Skip to content

Hooks

Most DDEV commands provide hooks to run tasks before or after the main command executes. To automate setup tasks specific to your project, define them in the project’s config.yaml file.

To define command tasks in your configuration, specify the desired command hook as a subfield to hooks, then a list of tasks to run:

hooks:
  post-start:
    - exec: "simple command expression"
    - exec: "ls >/dev/null && touch /var/www/html/somefile.txt"
    - exec-host: "simple command expression"
  post-import-db:
    - exec: "drush uli"

Supported Command Hooks

  • pre-start: Hooks into ddev start. Execute tasks before the project environment starts.

    Tip

    Only exec-host tasks can run during pre-start because the containers are not yet running. See Supported Tasks below.

  • post-start: Execute tasks after the project environment has started.

  • pre-import-db and post-import-db: Execute tasks before or after database import.
  • pre-import-files and post-import-files: Execute tasks before or after files are imported.
  • pre-composer and post-composer: Execute tasks before or after the composer command.
  • pre-share and post-share: Execute tasks before or after the share command.
  • pre-stop, pre-config, post-config, pre-exec, post-exec, pre-pull, post-pull, pre-push, post-push, pre-snapshot, post-snapshot, pre-delete-snapshot, post-delete-snapshot, pre-restore-snapshot, post-restore-snapshot: Execute as the name suggests.
  • post-stop: Hooks into ddev stop. Execute tasks after the project environment stopped.

    Tip

    Only exec-host tasks can run during post-stop. See Supported Tasks below.

Supported Tasks

DDEV currently supports these tasks:

  • exec to execute a command in any service/container.
  • exec-host to execute a command on the host.
  • composer to execute a Composer command in the web container.

exec: Execute a shell command in a container (defaults to web container)

Value: string providing the command to run. Commands requiring user interaction are not supported.

Optional keys:

  • service: Specify which container to run the command in (defaults to web)
  • user: Specify which user to run the command as (username or UID, defaults to container’s default user)
  • exec_raw: Array of command arguments for direct execution without shell interpretation (alternative to string command)

Example: Use Drush to rebuild all caches and get a user login link after database import.

hooks:
  post-import-db:
    - exec: drush cache:rebuild
    - exec: drush user:login

Example: Use wp-cli to replace the production URL with development URL in a WordPress project’s database.

hooks:
  post-import-db:
    - exec: wp search-replace https://www.myproductionsite.com http://mydevsite.ddev.site

Example: Use Drush to sanitize a database by removing or obfuscating user data.

hooks:
  post-import-db:
    - exec: drush sql:sanitize

Example: Add an extra database before import-db, executing in db container.

hooks:
  pre-import-db:
    - exec: mysql -uroot -proot -e "CREATE DATABASE IF NOT EXISTS some_new_database;"
      service: db

Example: Execute a command as root user in the db container.

hooks:
  post-start:
    - exec: ls -la /root
      service: db
      user: root

Example: Add the common ll alias into the web container’s .bashrc file.

hooks:
  post-start:
    - exec: sudo echo alias ll=\"ls -lhA\" >> ~/.bashrc

Tip

This could be done more efficiently via .ddev/web-build/Dockerfile as explained in Customizing Images.

Advanced usages may require running commands directly with explicit arguments. This approach is useful when Bash interpretation is not required (no environment variables, no redirection, etc.).

hooks:
  post-start:
    - exec:
      exec_raw: [ls, -lR, /var/www/html]

exec-host: Execute a shell command on the host system

Value: string providing the command to run. Commands requiring user interaction are not supported.

hooks:
  pre-start:
    - exec-host: "command to run"

composer: Execute a Composer command in the web container

Value: string providing the Composer command to run.

Optional keys:

  • exec_raw: Array of Composer command arguments (alternative to string command)

Example:

hooks:
  post-start:
    - composer: config discard-changes true

Example with exec_raw:

hooks:
  post-start:
    - composer:
      exec_raw: [install, --no-dev]

WordPress Example

hooks:
  post-start:
    # Install WordPress after start
    - exec: "wp config create --dbname=db --dbuser=db --dbpass=db --dbhost=db"
    - exec: "wp core install --url=http://mysite.ddev.site --title=MySite --admin_user=admin --admin_email=admin@mail.test"
  post-import-db:
    # Update the URL of your project throughout your database after import
    - exec: "wp search-replace https://www.myproductionsite.com http://mydevsite.ddev.site"

Drupal 7 Example

hooks:
  post-start:
    # Install Drupal after start if not installed already
    - exec: "(drush status bootstrap | grep -q Successful) || drush site-install -y --db-url=db:db@db/db"
    # Generate a one-time login link for the admin account
    - exec: "drush uli"
  post-import-db:
    # Set the project name
    - exec: "drush vset site_name MyDevSite"
    # Enable the environment indicator module
    - exec: "drush en -y environment_indicator"
    # Clear the cache
    - exec: "drush cc all"

Drupal 10 Example

hooks:
  post-start:
    # Install Composer dependencies from the web container
    - composer: install
    # Generate a one-time login link for the admin account
    - exec: "drush user:login"
  post-import-db:
    # Sanitize the database
    - exec: "drush sql:sanitize"
    # Apply any database updates
    - exec: "drush updatedb"
    # Rebuild all caches
    - exec: "drush cache:rebuild"

TYPO3 Example

hooks:
  post-start:
    - composer: install