Use the compare_schema API to monitor database schema changes in CI/CD pipelines and agentic systems
Docs/Use cases/Dev/Test

Dev/Test Environments on Neon

Speed up time-to-launch by running your non-prod workflows on Neon

What sets Neon apart from other Postgres providers, beyond its true serverless nature, is its exceptional developer experience. Teams often experience a boost in developer velocity when switching to Neon. In Neon, it takes less time to create and synchronize environments, run tests, and tedious database tasks are suddenly automated—all of which accelerate software lifecycles.

However, not every team is ready for a full database migration. Moving a production database to a new vendor is a significant project that requires careful planning—but even if teams aren't ready to migrate their production databases, they can still get more efficiency by moving their non-production environments to Neon. As a cherry on top, this also significantly saves non-production costs (up to 80%).

In this guide, we’ll walk you through the steps needed to set up Neon for development and testing, including how to move data from your current Postgres provider to Neon, keep it in sync, and set up ephemeral environments as Neon branches for optimal cost-efficiency and productivity.

Benefits of running Dev/Test on Neon

  • New ephemeral environments are provisioned instantly, with an updated copy of the testing dataset. In Neon, each ephemeral dev/test environment is a database branch. Database branches in Neon include a full copy of the dataset, and they’re created instantly. Ephemeral environments are available for immediate use, and they can be programmatically created and deleted via CI/CD without delays. When there’s an update in the testing dataset, ephemeral environments can be reset in one click.

  • You only pay for storage once, across all your environments—and when environments are not being used, you aren’t billed for compute. Ephemeral environments in Neon are also extremely affordable. Neon branches autosuspend when they’re not being used, saving compute costs in idle hours. And since all branches within a Neon project share the same storage, storage is only billed once across all dev/test environments, avoiding the costs and work of duplicating storage for each setup.

Methodology

Most teams running dev/test workloads on Neon while keeping production on another Postgres platform implement a workflow similar to this:

  1. Set up a single Neon Project for dev/test ephemeral environments

    Teams start by creating a single Neon project to host multiple dev/test environments. In Neon, a project is the logical equivalent of an “instance.” Thanks to Neon branching, many non-production instances in RDS or Aurora can be replaced with a single Neon project, as we'll see next.

  2. Create a Neon Twin

    Next, teams create a Neon Twin —a copy of their production or staging dataset from the other platform (or a subset of this dataset) that remains automatically synchronized. There are various methods to keep the dataset in sync, which we’ll cover in this guide, but the process generally looks like this: 1. The testing dataset is loaded into the main branch within the Neon project. 2. Automation is set up so that data in the main branch is refreshed periodically, such as nightly.

    This main branch serves as the primary source for all dev/test environments, and it's the only location that needs to be updated with new data or schema changes, as we’ll see later.

  3. Set up ephemeral environments as child branches

    Once the Neon Twin is set up within the main branch, teams can instantly create ephemeral environments by deriving child branches from this main branch. These branches are fully isolated and provide teams with a complete copy of the testing dataset immediately.

    After the main branch is refreshed (e.g., nightly), all environments can be synced in one click. Neon includes a reset from parent feature, which instantly resets all child branches with data from the main branch. This allows teams to get the latest testing data (including schema changes) without needing to reload testing datasets in every single environment.

    Teams configure their workflows so that after development or testing is complete (e.g., a PR is closed), child branches are deleted automatically via the API. But since Neon's autosuspend automatically pauses these environments when unused, teams don’t have to worry too much about inactive branches.

Step 1: Set up a Neon project for Dev/Test

  1. Sign up to Neon if you haven’t already.
  2. Provide a descriptive name for your project, such as Dev/Test Environments
  3. Select your desired Postgres version, cloud service provider, and region.
  4. Click Create Project
  5. Once inside the Neon console:
    1. By default, Neon creates a branch named main
    2. To distinguish it as your primary development branch, rename it to main_dev or similar by navigating to the Branches screen, clicking on the options menu (three dots), and selecting Rename

Step 2: Create a Neon Twin

Using dump/restore with GitHub Actions

A Neon Twin is a synchronized copy of your production or staging database within your main_dev branch, which will serve as the source for your development and testing environments. One of the methods to build a Neon Twin is automating nightly data synchronization using pg_dump, pg_restore, and GitHub Actions. Detailed instructions can be found in this blog post series.

  • Set up the initial data import. Use pg_dump to create a dump of your production or staging database.

    pg_dump -Fc -v -d postgresql://[user]:[password]@[source_host]/[database] -f source_dump.bak
  • Import data into Neon. Use pg_restore to load the dump into your Neon main_dev branch:

    pg_restore -v -d postgresql://[user]:[password]@[neon_host]/[database] source_dump.bak
  • Automate nightly synchronization with GitHub Actions.

    • Prerequisites:

      • Ensure you have a GitHub repository to host the workflow.
      • Store your database credentials securely using GitHub Secrets.
    • Steps:

      1. In your GitHub repository, navigate to .github/workflows/ and create a file named neon_twin_sync.yml.

      2. Insert the following content into neon_twin_sync.yml:

        name: Neon Twin Sync
        
        on:
          schedule:
            - cron: '0 0 * * *' # Runs daily at midnight
        
        jobs:
          sync:
            runs-on: ubuntu-latest
            steps:
              - name: Checkout repository
                uses: actions/checkout@v3
        
              - name: Set up PostgreSQL
                uses: postgres-actions/setup-postgresql@v2
                with:
                  postgresql-version: '17'
        
              - name: Dump source database
                run: |
                  pg_dump -Fc -v -d ${{ secrets.SOURCE_DB_URL }} -f source_dump.bak
        
              - name: Restore to Neon
                run: |
                  pg_restore -v -d ${{ secrets.NEON_DB_URL }} source_dump.bak
      3. In your GitHub repository, navigate to Settings > Secrets and variables > Actions.

      4. Add the following secrets:

        • SOURCE_DB_URL: Connection string for your source database.
        • NEON_DB_URL: Connection string for your Neon main_dev branch.
      5. Commit the neon_twin_sync.yml file to your repository. GitHub Actions will execute this workflow nightly, synchronizing your Neon Twin with the source database.

Using AWS DMS

If your source database lives in Amazon RDS or Aurora, you can also use AWS Database Migration Service (DMS) to build a Neon Twin. AWS DMS is a service that allows you to migrate data between platforms.

For instructions on setting up AWS DMS with Neon, refer to this guide.

Using logical replication

If you’d like to ensure a more continuous and real-time synchronization between your source dataset and Neon, you could implement logical replication in your Neon Twin. This method will replicate all data changes as they occur; however, you must remember that logical replication won’t reflect changes in schema between your source database and the Neon Twin. You’ll still have to implement a method that makes sure schema stays in sync.

For more instructions on logical replication, refer to this guide.

Step 3: Set up ephemeral environments as child branches

With your Neon Twin setup, you’re ready to set up isolated development and testing environments within your Neon project by creating database branches off the primary branch, main_dev. This process is often automated via the Neon API.

In the Neon Console, navigate to the Branches tab within your project to manage and view existing branches.

  • Creating child branches manually
    • To create a new environment manually, select "New Branch"
    • You can name each branch to reflect its purpose, such as feature_xyz for branches related to specific feature development or test_xyz for testing environments
    • Select "main_dev" as the parent branch so that the new branch inherits the latest data from the primary dataset.
  • Automating branch creation via Neon API and GitHub Actions
    • Most likely, you’ll want to use the Neon API to automate branch creation as part of your CI/CD pipeline.
    • For example: Create a new branch in Neon when a new PR is opened in GitHub. This can be done using GitHub Actions to trigger the API call when a pull request is opened. Naviagate to our documentation on the GitHub Integration.
    • Similarly, you automate the deletion of branches when a pull request is closed or merged, which removes unused environments and keeps your project clean.
  • Synchronize branches with parent updates
    • Neon's Reset from Parent feature allows for easy syncing of each child branch with the latest data from main.
    • To do this manually, select the child branch in the Branches section and choose "Reset from Parent" to bring it up-to-date with the latest state of main_dev.
    • Most likely, you might want to schedule this as a nightly task in your CI/CD pipeline to reset all child branches, ensuring all environments are consistently aligned with the main dataset after updates, such as schema changes.

Let's Connect

We’re happy to give you a hand with any technical questions about how to set this up. We can also discuss pricing options, annual contracts, and migration assistance.

Contact us

Last updated on

Was this page helpful?