Skip to content

Commit

Permalink
Merge pull request #164 from opentable/testcontainers
Browse files Browse the repository at this point in the history
[OTPL-6466] Using PostgreSQLContainer as a back end with same API
  • Loading branch information
mikebell90 authored Jan 31, 2022
2 parents 9c63a3b + d95b128 commit d228aaf
Show file tree
Hide file tree
Showing 25 changed files with 929 additions and 1,213 deletions.
45 changes: 45 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,48 @@

1.0.0
-----

All the RC release features and fixes plus:

* Add network alias option
* Expose full bind mount function
* fix equals/hashcode in builder.

1.0.0RC3
-----
* Stop excluding junit4 from testcontainers - check the README for the sordid details.
* Pass host and port to the ConnectionInfo bean. We strongly recommend you prefer getDatasource or getUrl, these will be more portable in usage. We ran into
a few use cases where this was handy, however.
* LegacySingleInstancePostgresExtension to do the old Junit5 lifecycle behavior.
* 60 seconds default startup wait.
* Expose bind mounts (optional, none by default) in builder, currently hard coded as Read only.
* Expose network (optional, none by default) in builder.

1.0.0RC2
-------
* Restore Java 8 compatibility
* Update to testcontainers 1.16.3

1.0.0RC1
-----
* A completely rewritten version of `otj-pg-embedded`. Uses "testcontainers" for docker, while preserving API compatibility.

Advantages

* multi arch (m1 etc) support
* Works the same way on every OS - Mac, Windows, Linux. Please note the maintainers only test on Mac Linux
* You need a tarball for every linux distribution as PG 10+ no longer ship a "universal binary" for linux.
* Easy to switch docker image tag to upgrade versions.
* More maintainable and secure (you can pull docker images you trust, instead of trusting our tarballs)

Admittedly, a few disadvantages

* Slower than running a tarball
* A few compatibility drops and options have probably disappeared. Feel free to submit PRs
* Docker in Docker can be dodgy to get running.

=== legacy tarball versions ===

0.13.4
------
* POM 287, Flyway 7
Expand Down
163 changes: 136 additions & 27 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,18 +1,50 @@
OpenTable Embedded PostgreSQL Component
=======================================

Allows embedding PostgreSQL into Java application code with
no external dependencies. Excellent for allowing you to unit
test with a "real" Postgres without requiring end users to install
and set up a database cluster.
Note: This library requires Java 8+.

Allows embedding PostgreSQL into Java application code, using Docker containers.
Excellent for allowing you to unit
test with a "real" Postgres without requiring end users to install and set up a database cluster.

The release of 1.0 brings major changes to the innards of this library.
Previous pre 1.x versions used an embedded tarball. This was extemely fast (a major plus(, but we switched to a docker based version
for these reasons:

Advantages
---

* multi architecture support. This has become a huge issue for us with the introduction of the Mac M1 (and Windows ARM, Linux ARM)/
* The same container works the same way on every OS - Mac, Windows, Linux.
* You need a tarball for every linux distribution as PG 10+ no longer ship a "universal binary" for linux. This means a lot of support and maintenance work.
* Easy to switch docker image tag to upgrade versions - no need for a whole new pg-embedded version.
* More maintainable and secure (you can pull docker images you trust, instead of trusting our tarballs running in your security context)
* Trivial to do a build oneself based on the official Postgres image adding extensions, setup scripts etc - see https://github.com/docker-library/docs/blob/master/postgres/README.md for details.

Admittedly, a few disadvantages
---

* Slower than running a tarball (2-5x slower).
* A few API compatibility changes and options have probably disappeared. Feel free to submit PRs.
* Docker in Docker can be dodgy to get running. (See below for one thing we discovered))

## Before filing tickets.

1. Before filing tickets, please test your docker environment etc. If using podman or lima instead of "true docker", state so, and realize that the
docker socket api provided by these apps is not 100% compatible, as we've found to our sadness. We'll be revisiting
testing these in the future. We've managed to get PodMan working, albeit not 100% reliably.
2. **No further PRs or tickets will be accepted for the pre 1.0.0 release, unless community support arises for the `legacy` branch.**
3. We primarily use Macs and Ubuntu Linux at OpenTable. We'll be happy to try to help out otherwise, but other platforms, such
as Windows depend primarily on community support. We simply don't have the time or hardware. Happy to merge PRs though

See "Alternatives Considered" as well if this library doesn't appear to fit your needs.

[![Build Status](https://travis-ci.org/opentable/otj-pg-embedded.svg)](https://travis-ci.org/opentable/otj-pg-embedded)

## Basic Usage

In your JUnit test just add (for JUnit 5 example see **Using JUnit5** below):

```java
```
@Rule
public SingleInstancePostgresRule pg = EmbeddedPostgresRules.singleInstance();
```
Expand All @@ -23,19 +55,32 @@ Additionally you may use the [`EmbeddedPostgres`](src/main/java/com/opentable/db

Default username/password is: postgres/postgres and the default database is 'postgres'

## Sample of Embedded Postgres direct Usage

```
public void testDatabaseName() throws IOException,SQLException{
EmbeddedPostgres db=EmbeddedPostgres.builder().start();
Datasource dataSource = db.getPostgresDatabase();
.... use the datasource then ...
db.close();
}
```

The builder includes options to set the image, the tag, the database name, and various configuration options.

## Migrators (Flyway or Liquibase)

You can easily integrate Flyway or Liquibase database schema migration:
##### Flyway
```java
```
@Rule
public PreparedDbRule db =
EmbeddedPostgresRules.preparedDatabase(
FlywayPreparer.forClasspathLocation("db/my-db-schema"));
```

##### Liquibase
```java
```
@Rule
public PreparedDbRule db =
EmbeddedPostgresRules.preparedDatabase(
Expand All @@ -48,31 +93,38 @@ independent databases gives you.

## Postgres version

The JAR file contains bundled version of Postgres. You can pass different Postgres version by implementing [`PgBinaryResolver`](src/main/java/com/opentable/db/postgres/embedded/PgBinaryResolver.java).
The default is to use the docker hub registry and pull a tag, hardcoded in `EmbeddedPostgres`. Currently this is "13-latest",
as this fits the needs of OpenTable, however you can change this easily. This is super useful, both to use a newer version
of Postgres, or to build your own DockerFile with additional extensions.

Example:
```java
class ClasspathBinaryResolver implements PgBinaryResolver {
public InputStream getPgBinary(String system, String machineHardware) throws IOException {
ClassPathResource resource = new ClassPathResource(format("postgresql-%s-%s.txz", system, machineHardware));
return resource.getInputStream();
}
}
You may change this either by environmental variables or by explicit builder usage

### Environmental Variables

EmbeddedPostgreSQL
.builder()
.setPgBinaryResolver(new ClasspathBinaryResolver())
.start();
1. If `PG_FULL_IMAGE` is set, then this will be used and is assumed to include the full docker image name. So for example this might be set to `docker.otenv.com/postgres:mytag`
2. Otherwise, if `TESTCONTAINERS_HUB_IMAGE_NAME_PREFIX` is set, this is prefixed to "postgres" (adding a slash if it doesn't exist). So for example this might be set to "docker.otenv.com/"
3. Otherwise, the default is used as defined above.

### Explicit builder

It is possible to change postgres image and tag in the builder:

```
EmbeddedPostgres.builder()
.setTag("10")
.start();
```

## Windows
or use custom image:

If you experience difficulty running `otj-pg-embedded` tests on Windows, make sure
you've installed the appropriate MFC redistributables.
```
EmbeddedPostgres.builder()
.setImage(DockerImageName.parse("docker.otenv.com/super-postgres"))
.start();
```

* [Microsoft Site](https://support.microsoft.com/en-us/help/2977003/the-latest-supported-visual-c-downloads])
* [Github issue discussing this](https://github.com/opentable/otj-pg-embedded/issues/65)
There are also options to set the initDB configuration parameters, or other functional params, the bind mounts, and
the network.

## Using JUnit5

Expand Down Expand Up @@ -130,5 +182,62 @@ class DaoTestUsingJunit5 {
}
```

## Yes, Junit4 is a compile time dependency

This is because TestContainers has a long outstanding bug to remove this -https://github.com/testcontainers/testcontainers-java/issues/970
If you exclude Junit4, you get nasty NoClassDefFound errors.

If you only use Junit5 in your classpath, and bringing in Junit4 bothers you (it does us, sigh), then
you can do the following:

* add maven exclusions to the testcontainers modules you declare dependencies on to strip out junit:junit. This by itself
would still lead to NoClassDefFound errors.
* add a dependency on io.quarkus:quarkus-junit4-mock , which imports empty interfaces of the required classes. This is
a hack and a cheat, but what can you do?

We initially excluded junit4 ourselves, which led to confusing breakages for junit5 users...

## Some new options and some lost from Pre 1.0

* You can't wire to a local postgres, since that concept doesn't make sense here. So that's gone.
* You can add bind mounts and a Network (between two containers), since those are docker concepts, and can
be very useful.
* By the way, TestContainers does support ~/.docker/config.json for setting authenticated access to Docker, but we've not tested it.

## Docker in Docker, authentication notes

We've been able to get this working in our CICD pipeline with the following

`TESTCONTAINERS_HOST_OVERRIDE=localhost`
`TESTCONTAINERS_HUB_IMAGE_NAME_PREFIX=dockerhub.otenv.com/`

The first parameter corrects for testcontainers getting confused whether to address the hosting container or the "container inside the container".
The second parameter (which outside OpenTable would point to your private Docker Registry) avoids much of the Docker Rate Limiting issues.


## Alternatives considered

We updated this library primarily for convenience of current users to allow them to make a reasonably smooth transition to a Docker based
test approach.

* Why not just use Testcontainers directly?

You can, and it should work well for you. The builders, the api compatibility, the wrapping around Flyway - that's the added value.
But certainly there's no real reason you can't use TestContainers directly - they have their own Junit4 and Junit5 Rules/Extensions.

* Why not _use a maven plugin approach like fabric8-docker-maven?

Honestly I suspect this is a better approach in that it doesn't try to maintain it's own version of the Docker API, and
runs outside the tests, reducing issues like forking and threading conflicts. However it would have been too major an overhaul
for our users.

* "I really prefer the old embedded postgres approach. It's faster."
* We recommend those who prefer the embedded tarball use https://github.com/zonkyio/embedded-postgres which was forked a couple
years ago from the embedded branch and is kept reasonably up to date.
* Another alternative is flapdoodle's embedded postgres.

Both libraries suffer from many of the cons that bedeviled upkeep of this library for years, but they are certainly viable options
for many.

----
Copyright (C) 2017 OpenTable, Inc
Copyright (C) 2017-2022 OpenTable, Inc
86 changes: 28 additions & 58 deletions pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -16,10 +16,11 @@
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

<modelVersion>4.0.0</modelVersion>

<parent>
<groupId>com.opentable</groupId>
<artifactId>otj-parent-spring</artifactId>
<version>287</version>
<version>308</version>
</parent>

<scm>
Expand All @@ -31,11 +32,13 @@

<groupId>com.opentable.components</groupId>
<artifactId>otj-pg-embedded</artifactId>
<version>0.13.5-SNAPSHOT</version>
<version>1.0.0.RC4-SNAPSHOT</version>
<description>Embedded PostgreSQL driver</description>

<properties>
<project.build.targetJdk>1.8</project.build.targetJdk>
<!-- Remove once parent POM updated -->
<dep.testcontainers.version>1.16.3</dep.testcontainers.version>
<!-- end remove -->
<basepom.test.timeout>1800</basepom.test.timeout>
<basepom.oss.skip-scala-doc>true</basepom.oss.skip-scala-doc>
<basepom.check.skip-javadoc>false</basepom.check.skip-javadoc>
Expand All @@ -51,27 +54,6 @@
</license>
</licenses>

<build>
<plugins>
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>exec-maven-plugin</artifactId>
<version>1.3.2</version>
<executions>
<execution>
<phase>generate-resources</phase>
<goals>
<goal>exec</goal>
</goals>
</execution>
</executions>
<configuration>
<executable>./repack-postgres.sh</executable>
</configuration>
</plugin>
</plugins>
</build>

<dependencies>
<dependency>
<groupId>org.slf4j</groupId>
Expand All @@ -83,27 +65,6 @@
<artifactId>commons-lang3</artifactId>
</dependency>

<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-compress</artifactId>
</dependency>

<dependency>
<groupId>org.tukaani</groupId>
<artifactId>xz</artifactId>
<version>1.5</version>
</dependency>

<dependency>
<groupId>commons-io</groupId>
<artifactId>commons-io</artifactId>
</dependency>

<dependency>
<groupId>commons-codec</groupId>
<artifactId>commons-codec</artifactId>
</dependency>

<dependency>
<groupId>org.flywaydb</groupId>
<artifactId>flyway-core</artifactId>
Expand All @@ -120,9 +81,14 @@
<groupId>org.postgresql</groupId>
<artifactId>postgresql</artifactId>
</dependency>

<dependency>
<groupId>com.github.spotbugs</groupId>
<artifactId>spotbugs-annotations</artifactId>
<groupId>org.testcontainers</groupId>
<artifactId>postgresql</artifactId>
</dependency>
<dependency>
<groupId>org.testcontainers</groupId>
<artifactId>testcontainers</artifactId>
</dependency>

<dependency>
Expand All @@ -144,17 +110,21 @@
<artifactId>slf4j-simple</artifactId>
<scope>test</scope>
</dependency>
</dependencies>

<dependency>
<groupId>org.objenesis</groupId>
<artifactId>objenesis</artifactId>
<scope>test</scope>
</dependency>
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.testcontainers</groupId>
<artifactId>postgresql</artifactId>
<version>${dep.testcontainers.version}</version>
</dependency>
<dependency>
<groupId>org.testcontainers</groupId>
<artifactId>testcontainers</artifactId>
<version>${dep.testcontainers.version}</version>
</dependency>
</dependencies>
</dependencyManagement>

<dependency>
<groupId>org.mockito</groupId>
<artifactId>mockito-core</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
</project>
Loading

0 comments on commit d228aaf

Please sign in to comment.