Groovy Integration Testing
with Spock and Docker
data:image/s3,"s3://crabby-images/09fed/09fed0e5660e345e8252945293d5bdae92afba74" alt=""
Kevin Wittek
Software Developer @ GDATA Advanced Analytics
Consultant @ Styracosoft GbR
https://groovy-coder.com
@Kiview
https://github.com/kiview
data:image/s3,"s3://crabby-images/a048f/a048f69674cddb61f881a7f949476f34ff416519" alt=""
data:image/s3,"s3://crabby-images/afb3e/afb3e0c7ddb18050a1ffec05052efae7ed6a5dce" alt=""
data:image/s3,"s3://crabby-images/43abd/43abded27334f60263fdf3b3d63c7ecd80b4965a" alt=""
data:image/s3,"s3://crabby-images/7e97c/7e97ca40f0b448e94811c88897e50f4a9ab21823" alt=""
Key learnings
- Easy setup of complete test environments
- Not a lot effort to integration test using real external applications
- Development history of Docker Spock-Extension
$ git clone https://github.com/foo/bar.git
$ cd bar
$ ./gradlew build
data:image/s3,"s3://crabby-images/1b5c4/1b5c42090668010a370a016e1f59cf76ef27ccee" alt=""
And what about Microservices?
data:image/s3,"s3://crabby-images/9422e/9422e89b711fef8aa385fb70ce3b0aab6ef80690" alt=""
My definiton
For example:
- Launch application server
- Framework/Library interactions
- Interact with external ports (e.g. network, file system, database)
"Tests which interact with external systems/dependencies"
Which tools to use?
(in the JVM world)
data:image/s3,"s3://crabby-images/c3a2a/c3a2aef99c92f58b1b989c9ed5ace9dd472aa1e4" alt=""
data:image/s3,"s3://crabby-images/99153/991536c6c32f69aead8cf0fb009c7b9247fcbc16" alt=""
data:image/s3,"s3://crabby-images/598a5/598a5feda08a76cf03641e8a56b7b7b511cbf751" alt=""
data:image/s3,"s3://crabby-images/0028d/0028d23f2fd76f0d229963f3ae844dfc347b9388" alt=""
data:image/s3,"s3://crabby-images/2746c/2746cb431e98b4d5058eec067b5c00ee1e3d85d7" alt=""
class HelloSpockSpec extends spock.lang.Specification {
def "length of Spock's and his friends' names"() {
expect:
name.size() == length
where:
name | length
"Spock" | 5
"Kirk" | 4
"Scotty" | 6
}
}
data:image/s3,"s3://crabby-images/13f64/13f643462e83f3a4e00d9a70c6d08f3df0e71efe" alt=""
data:image/s3,"s3://crabby-images/a3bd0/a3bd0c4cb89b798ea0c4989edd8cf8771bfc6257" alt=""
data:image/s3,"s3://crabby-images/0bb9c/0bb9c416e1bb4492e38cc476cdf009fbdefd6de4" alt=""
data:image/s3,"s3://crabby-images/5150b/5150b4b2027bf23c90517eef6b9cbbd18fd62a0c" alt=""
data:image/s3,"s3://crabby-images/e5699/e56993134b57aa846871d8f62d1478356eab44fa" alt=""
Why combine?
- Easy setup of dev environment
- Uniform build and test environments
- Also self contained and portable!
- No installation and setup of external software
- Except Docker ;)
$ git clone https://github.com/foo/bar.git
$ cd bar
$ ./gradlew build
How to fuse both worlds?
Spock + Docker = Spocker?
@Docker(image = "emilevauge/whoami", ports = @PortBinding("8080:80"))
class DockerExtensionIT extends Specification {
def "should start accessible docker container"() {
given: "a http client"
def client = HttpClientBuilder.create().build()
when: "accessing web server"
def response = client.execute(new HttpGet("http://localhost:8080"))
then: "docker container is running and returns http status code 200"
response.statusLine.statusCode == 200
}
}
First commit
class DockerMethodInterceptor extends AbstractMethodInterceptor {
private final DockerClientFacade dockerClient
DockerMethodInterceptor(Docker docker) {
this(new DockerClientFacade(docker))
}
DockerMethodInterceptor(DockerClientFacade dockerClient) {
this.dockerClient = dockerClient
}
@Override
void interceptSpecExecution(IMethodInvocation invocation) throws Throwable {
dockerClient.startContainer()
invocation.proceed()
dockerClient.stopContainer()
}
}
First commit
class DockerClientFacade {
private final Docker imageConfig
private final DockerClient dockerClient
private static final Logger LOGGER = LoggerFactory.getLogger(DockerClientFacade)
private String id
DockerClientFacade(Docker containerConfig) {
this.imageConfig = containerConfig
dockerClient = initializeClient()
}
private static DockerClient initializeClient() {
def clientConfig = DefaultDockerClientConfig.createDefaultConfigBuilder()
return DockerClientBuilder.getInstance(clientConfig).build()
}
void startContainer() {
def portBindings = parsePortBindings()
ensureImageExists()
id = createContainer(portBindings)
dockerClient.startContainerCmd(id).exec()
}
private String createContainer(List<PortBinding> portBindings) {
dockerClient.createContainerCmd(imageConfig.image())
.withPortBindings(portBindings)
.exec().id
}
private void ensureImageExists() {
try {
dockerClient.inspectImageCmd(imageConfig.image()).exec()
} catch (NotFoundException e) {
dockerClient.pullImageCmd(imageConfig.image()).exec(new PullImageResultCallback()).awaitSuccess()
}
}
void stopContainer() {
dockerClient.stopContainerCmd(id).exec()
try {
dockerClient.removeContainerCmd(id).exec()
} catch(InternalServerErrorException e) {
LOGGER.error("error while removing the container", e)
}
}
List<PortBinding> parsePortBindings() {
def ports = imageConfig.ports().first()
return [parsePortBinding(ports)]
}
private PortBinding parsePortBinding(com.groovycoder.spockdockerextension.PortBinding ports) {
def parsedBinding = PortBinding.parse(ports.value())
return parsedBinding
}
}
First commit
class DockerClientFacade {
DockerClient dockerClient
String image
Map clientSpecificContainerConfig
def containerStatus
DockerClientFacade(Docker containerConfig) {
image = containerConfig.image()
dockerClient = new DockerClientImpl()
clientSpecificContainerConfig = new DockerContainerConfigBuilder(containerConfig).build()
}
void startContainer() {
containerStatus = dockerClient.run(image, clientSpecificContainerConfig, "latest")
}
void stopContainer() {
def id = containerStatus.container.content.Id
dockerClient.stop(id)
dockerClient.rm(id)
}
}
Switch to gesellix/docker-client
data:image/s3,"s3://crabby-images/1cfa5/1cfa575f5e839cfa4312ef441fcedd72f825d51d" alt=""
data:image/s3,"s3://crabby-images/2746c/2746cb431e98b4d5058eec067b5c00ee1e3d85d7" alt=""
// Set up a redis container
@ClassRule
public static GenericContainer redis =
new GenericContainer("redis:3.0.2")
.withExposedPorts(6379);
- Generic Containers
- Specialized Containers (Databases, Selenium)
- Dockerfile
- Dockerfile DSL
- Docker-Compose
- WaitStrategy
- Docker environment discovery (e.g. docker-machine, DOCKER_HOST, ...)
Using testcontainers as docker-client
class DockerClientFacade {
FixedHostPortGenericContainer dockerClient
final String name
final String[] ports
final String image
final Env[] env
WaitStrategy waitStrategy
DockerClientFacade(Docker containerConfig) {
name = containerConfig.name()
ports = containerConfig.ports()
env = containerConfig.env()
this.image = concatImageWithDefaultTagIfNeeded(containerConfig.image())
waitStrategy = ((Closure) containerConfig.waitStrategy().newInstance(this, this))()
}
private static String concatImageWithDefaultTagIfNeeded(String image) {
(image.contains(":")) ? image : image + ":latest"
}
void run() {
try {
start()
} catch (ContainerLaunchException e) {
throw new DockerRunException("Error running the container", e)
}
}
void rm() {
dockerClient.stop()
}
void start() {
dockerClient = new FixedHostPortGenericContainer(image)
ports.each { String portMapping ->
def split = portMapping.split(":")
dockerClient.withFixedExposedPort(split[0].toInteger(), split[1].toInteger())
}
env.each { Env e ->
dockerClient.withEnv(e.key(), e.value())
}
dockerClient.waitingFor(waitStrategy)
dockerClient.start()
}
void stop() {
dockerClient.stop()
}
String getIp() {
//noinspection GrDeprecatedAPIUsage
dockerClient.containerInfo.networkSettings.ipAddress
}
}
Annotations a good idea?
@Docker(image = "postgres", ports = ["5432:5432"], env = [
@Env(key = "POSTGRES_USER", value = "foo"),
@Env(key = "POSTGRES_PASSWORD", value = "secret")
],
waitStrategy = { new JdbcWaitStrategy(username: "foo",
password: "secret",
jdbcUrl: "jdbc:postgresql:foo",
driverClassName: "org.postgresql.Driver",
testQueryString: "SELECT 1") })
def "waits until postgres accepts jdbc connections"() {
// tests
}
Fully embrace testcontainers
@Testcontainers
class TestContainersClassIT extends Specification {
WaitStrategy postgresWaitStrategy = new JdbcWaitStrategy(
username: "foo",
password: "secret",
jdbcUrl: "jdbc:postgresql:foo",
driverClassName: "org.postgresql.Driver",
testQueryString: "SELECT 1")
@Shared
GenericContainer genericContainer = new GenericContainer("postgres:latest")
.withExposedPorts(5432)
.withEnv([
POSTGRES_USER: "foo"
POSTGRES_PASSWORD: "secret"
])
Demo!
Docker-Compose
data:image/s3,"s3://crabby-images/6e977/6e977fe4f724c44a765840326505f0b5f97dd3c4" alt=""
data:image/s3,"s3://crabby-images/cb918/cb9188afa1689201355a11490e4ee1aae124d061" alt=""
version: "3"
services:
vote:
image: vote-frontend
command: python app.py
volumes:
- ./vote:/app
ports:
- "5000:80"
redis:
image: redis:alpine
ports: ["6379"]
worker:
image: worker
db:
image: postgres:9.4
result:
image: result-frontend
command: nodemon --debug server.js
volumes:
- ./result:/app
ports:
- "5001:80"
- "5858:5858"
data:image/s3,"s3://crabby-images/598a5/598a5feda08a76cf03641e8a56b7b7b511cbf751" alt=""
class GebHomepageSpec extends GebSpec {
def "can access The Book of Geb via homepage"() {
when:
to GebHomePage
and:
highlights.jQueryLikeApi.click()
then:
sectionTitles == ["Navigating Content", "Form Control Shortcuts"]
highlights.jQueryLikeApi.selected
}
}
Moar Demo!
CI inside containers?
data:image/s3,"s3://crabby-images/a9ea8/a9ea8a84b2bbbe771eed838ec7452e0c85e2f8bf" alt=""
Docker in Docker - DinD
data:image/s3,"s3://crabby-images/4aa81/4aa816cad6030846e9e1a88c573eb31e60f92421" alt=""
data:image/s3,"s3://crabby-images/4a411/4a41166b7e4153a70e169156bc0d968da909a8ac" alt=""
data:image/s3,"s3://crabby-images/b68e5/b68e53df97466fcaeeb6472f163f0e981e72b19d" alt=""
data:image/s3,"s3://crabby-images/4501a/4501a949926937c58d685820f2aba39bd6a47c33" alt=""
Docker Wormhole
Docker Wormhole
data:image/s3,"s3://crabby-images/f58ad/f58ad692051c3caf8589c811bf747033f287f993" alt=""
data:image/s3,"s3://crabby-images/c9f51/c9f5172c6eec91e975e0dae8e61a79874a1dd416" alt=""
Future
- Better support for Docker networking
- Improve Windows support
- Build and test inside containers with full IDE support
Resources
data:image/s3,"s3://crabby-images/c208e/c208e3629f0cd665c5c257394cf9248795a8fb7a" alt=""
data:image/s3,"s3://crabby-images/c0863/c0863b3b7cfdc0af6aa3da62b35e1c84aa5dc997" alt=""
Integration testing with Spock and Docker (Gr8conf)
By Kevin Wittek
Integration testing with Spock and Docker (Gr8conf)
- 3,645