Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Document Database Setup and Migrations #2994

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
92 changes: 92 additions & 0 deletions doc/Database_Setup_and_Migrations.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,92 @@
# How the database setup and migration works

The Fundraising Application has two different database environments:

- The **local development environment**. We're running a MariaDB database in a Docker container for local user and acceptance testing. We are using an in-memory SQLite database for running unit tests. This makes running the tests fast and allows for a clear separation between interactive testing and automated testing.
- The **server environment** has a dedicated database server with two databases (test and production) that our test and production web servers connect to. The Fundraising App and the Fundraising Operation Center both share the same production or testing database.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
- The **server environment** has a dedicated database server with two databases (test and production) that our test and production web servers connect to. The Fundraising App and the Fundraising Operation Center both share the same production or testing database.
- The **server environment** is running on a dedicated database server with two databases (test and production) that our test and production web servers connect to. The Fundraising App and the Fundraising Operation Center both share the same production or testing database.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, maybe "consists of" or just "uses"? Pinging @Abban for a comment

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"Consists of" is good


The two environments have different strategies for initial setup and migration of existing database structure and data:

- The local environment is a "throwaway" environment - When one developer on the team changes the database structure, they notify the rest of the team that they should throw away their docker container and docker volumes and restart the development environment. The mechanism we use is putting **SQL files** with structure and data into the directory `.docker/database/`. The `docker-compose.yml` file defines a database volume that we can throw away. When the docker container starts and has no volume from previous runs, MariaDB will execute the files (in alphabetical order).
gbirke marked this conversation as resolved.
Show resolved Hide resolved
- The server environment needs to preserve the existing data (all donations since 2011, etc.), so we can't throw the database away when we want to make changes to the structure. Therefore we use **Database Migration Scripts** that modify the database structure (and copy or modify the existing data if it needs adaptation to the new structure).


## How to regenerate the local environment

Run the command `docker compose down -v`. This will stop all containers and will also clear out their data volumes.

Afterwards, run `docker compose up -d` to start the environment again.

### Troubleshooting

If you're getting errors from the application after the containers are up, check if the database tables exist by running the following command:

```shell
docker compose exec database mysql -u fundraising -p"INSECURE PASSWORD" fundraising -e "SHOW TABLES"
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
docker compose exec database mysql -u fundraising -p"INSECURE PASSWORD" fundraising -e "SHOW TABLES"
docker compose exec database mysql -u fundraising -p "INSECURE PASSWORD" fundraising -e "SHOW TABLES"

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nope, this is one of the pecularities of the MySQL client: if you specify -p something it will interpret that as "ask for a password and use parameter 'something'". If its -psomething it will interpret that as "Use password 'something'". And sometimes will give a warning about that practice because that command can then end up in the bash history, revealing the password.

```

If there are no tables, run `docker compose logs database` to see if there are any errors.


## How to integrate a database change in the local environment

To integrate a database change from a bounded context you need to do the following steps:

1. Create an integration branch in in the application repository, pointing the version of the bounded context in `composer.json` to the `-dev` branch of the bounded context.
2. Run CI and fix failing tests.
3. Run the database migration on the *old* version of the database schema
4. User-test the changed database (to check if the migrated database is correct)
5. Re-generate SQL for the local environment.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

does this point refer to ### Regenerating the local SQL ? Maybe we could link there or add the command here as well

4. User-test the changed database (again, this time for the auto-generated local environment)
6. Create a new minor/major release of the bounded context and rebase/change your integration branch, pointing the version of the bounded context in `composer.json` to the new release.

**Important:** You have to do the steps 3 and 4 exactly in this order, because the migration won't work on the new schema!

### Testing the migration

When checking out a branch (or creating a new integration branch for updating a version from a bounded context), your local environment Docker container should still have the old version of the database. If you want to be extra sure to "reset" your local container, do the following steps:

1. Check out the `main` branch
2. Follow the steps from the "How to regenerate the local environment" section of this document.
3. Check out your branch again.

To check if the applications "sees" the new migration, run the following command:

```shell
docker compose exec app php bin/doctrine migrations:list
```

You should see a list of migrations, none them applied. In the local environment this is fine, because we throw away the Doctrine migration metadata table whenever we regenerate the local environment. If the migration does not show up, check the migrations directory in `vendor/wmde/your-bounded-context` and the configuration file `app/config/migrations.php`.

To test / run the new migration, run the following command:

```shell
docker-compose exec app vendor/bin/doctrine-migrations migrations:migrations:execute 'Your\\Migration\\Name'
gbirke marked this conversation as resolved.
Show resolved Hide resolved
```

Replace the placeholder `Your\\Migration\\Name` with the namespace name of the actual migration, e.g. `'WMDE\\Fundraising\\MembershipContext\\DataAccess\\Migrations\\Version20240528122513'`.

**Important:** The `migrations:list` command lists the migrations with single backslashes! If you copy-paste the migration name form the list or the code, remember to use double backslashes! Otherwise your command line shell will interpret the combination of backslash and the following character as a Characters and the resulting name passed to PHP will not contain any backslashes which will make the command fail.

After running the migration you can user-test the application (to see if the changed database works with the code) and inspect the changed database in the IDE or CLI.

If the migration fails for syntactic reasons (e.g. wrong column or table names) or semantic reasons (code does not work with changed schema) you need to change the migration in the branch of the bounded context and run `make update-php` in the integration branch of the application.

### Regenerating the local SQL

Run the following command:

```shell
make generate-database-schema
```

This will regenerate the file `.docker/database/01.Database_Schema.sql`. Check the file to make sure it only contains SQL commands and no PHP error messages/warnings.

Follow the steps from the "How to regenerate the local environment" section of this document. Check that the database tables are still there in the container.

User-test the application in your local environment. If the application does not work correctly, you'll have to edit the code in the branch of the bounded context.

## How to migrate the database on the server

See our [infrastructure documentation how to deploy a new version of the Fundraising application that needs migrations](https://github.com/wmde/fundraising-infrastructure/blob/main/docs/deployment/Fundraising_Application.md#deploying-a-release-that-needs-database-changes)