Managing databases
Syncing development databases between team members
Introduction
If a project has more than one developer working on it, keeping up with changes to development databases can be challenging. Syncing those changes across the whole development team can be tedious and time consuming. When various team members work on a database schema or with data from development databases that isn't reflective of the most recent state, it can cause frustration and lead to bugs that might end up in production.
In particular, these problems can arise in the following scenarios:
- Developers are working on different iterations of a database schema
- Data in a development database is not reflective of the current state of the schema
In this guide, we'll look at some ways that teams can keep their database schemas and development data up-to-date and in-sync across teams.
Setting up development databases during onboarding
Before changes to database schemas and development data can be shared across team members, those team members need to be bootstrapped with some initial database state. For most teams, this is part of the onboarding process when they begin working on the project. The level of sophistication for this process largely depends on the size and maturity of the organization and its needs.
The starting point for most developers who are onboarding is to clone and install the project locally. Next, they usually need to connect to a development database. The two most common ways that new developers get set up with a development database are to:
- Run a script to seed a local development database
- Connect to a shared remote development database
Both of these approaches come with their own set of benefits and drawbacks.
Seeding a local database
Seeding a local database is beneficial because it alleviates the need to have a remote development database running. Developers can have more control over the data that goes into their development database as they get started. They can also reset the database more easily than if a shared remote database is used.
If this approach is taken, it requires extra effort to keep seeding data and logic up-to-date with the current schema. If changes occur frequently, this can pose a challenge and may cause problems getting developers onboarded smoothly.
If you are using Prisma Client, you can use the integrated seeding functionality to easily populate a database.
Connecting to a shared remote database
Having developers connect to a shared remote development database can alleviate the burden of populating the new developer's database locally. This can be helpful because it makes the initial onboarding easier, allowing the developer to be productive faster. Using a shared remote database might also be beneficial because changes to the database can be migrated on a regular basis by a single developer and those changes can then be consumed by all team members.
The downside to using a shared remote development database is that it requires additional resources to be running, possibly incurring additional cost. Additionally, if all developers are working against the same database, it can be easy to overwrite data. Another potential issue is if schema changes are adopted by team members asynchronously, out-of-band changes might negatively affect the state of the data, making it difficult to work with.
Ongoing database updates
Whether locally or remotely, once developers are set up with a database to work against, they will need to continuously check for updates to the database schema and possibly the data in the database itself. This can be difficult if team sizes are large and changes are made frequently.
There's no one-size-fits-all approach for keeping team members up-to-date with schema changes. The right approach for your team is context-dependent.
The following list contains some approaches that can be combined to build your overall strategy for keeping data and schema changes in-sync across your teams.
Using migrations
When working with relational databases in a team setting, it's vital to use migrations as a starting point for your database schema change strategy.
A database migration is a set of artifacts that, through automation, can change a database's structure. At the simplest, a migration is a SQL file that includes the statements necessary to apply a schema change. For example, the addition of a field to a database model might produce the following SQL file:
ALTER TABLE User ADD bio varchar(255);
While this command could be run on its own by a single developer within the context of their local development environment, having it exist as a database migration file allows all other developers on the team to apply the exact same changes through automation.
Specific database migration strategies differ between languages and frameworks. However, in general, they follow a similar pattern:
- Alter the code describing a database model
- Generate a set of migration files
- Run the migrations to execute the statements and have the changes take effect
Prisma Migrate makes it easy to describe your database models and produce migrations from them. It offers a very simple way to keep data in-sync between team members.
Notifying team members of changes
Using database migrations as a change strategy will help to keep database structure changes in-sync between team members. However, notifying developers of those changes can be challenging.
This becomes a problem when two or more team members are working on the same part of the database schema or part of the application that consumes a given part of the schema. Such a scenario is likely to produce merge conflicts and cause frustration.
While this situation is best addressed by improving team communication, technology can help to some extent.
Git hooks can be used to notify developers about changes to a schema at various times. For example, a hook can be used to check for schema changes that have already been merged and applied upstream before allowing a developer to push their changes. This can help to alleviate merge conflicts and allow the team to be more in-sync.
This behavior can be scripted and applied to your team's specific scenario using git-migration-hook as a starting point.
Automating data syncing
When a database schema changes, it's likely that the whole team will need to re-seed the database such that the data within abides by the new schema. It can be tedious to maintain a seed script, distribute it, and have everyone on the team apply it in an ongoing fashion.
One option for automating this process is to use a service that provides bi-directional data syncing between databases. While such a service is more often meant to sync remote production databases for various purposes, it could also be used to sync development data across teams. Doing so allows for teams to use their own local or remote development databases that are continually updated. One example of such a service is SQL Data Sync for Azure.
Conclusion
Keeping development database changes in-sync between team members can be a challenging task. When teams move quickly and collaborate on the same schema, opportunities for collisions are plentiful.
By introducing workflows with Git hooks and other tools to provide automation, teams can more easily keep schema changes and data in-sync.