Bootiful Containers

with Spring-Boot and Docker

Agenda

  • Run Spring-Boot Apps inside a Docker container
    • Make Spring-Boot Apps configurable using Environment Variables
    • Build your Spring-Boot Docker image using Gradle
    • Deploy the image via docker-compose
  • Unit- and Integration-Test your Spring-Boot App with Spock and Docker
  • Deploy and configure Spring-Boot microservices using docker-compose

Run Spring-Boot Apps inside a Docker container

1

Spring-Boot

  • Opinionated view on Spring
  • Lots of autoconfigure magic
  • Useful features and defaults for microservices and 12-factor apps
    • Supports externalized configuration using environment variables
    • Easy to use health and metrics endpoints
    • Use of embedded Tomcat allows for-self contained application bundle
  • Works well inside a container
@Component
public class MyBean {

    @Value("${foo.bar}")
    private String name;

    // ...

}
$ export FOO_BAR=mobydock

Externalized Configuration

@ConfigurationProperties("foo")
@Component
public class MyBeanProperties {

    private String name;

    public String getName() { ... }

    public void setName(String name) { ... }

}

Externalized Configuration

Type-safe Configuration Properties

Hands-On!

Refactoring "Hello World"

Use Type-safe Configuration Properties

  • Create a GreeterConfiguration component using the @ConfigurationProperties annotation

  • Inject your bean into the greeter
  • Run the hello-world application to verify
$ GREETER_NAME=gradle-runner ./gradlew bootrun
$ git clone https://github.com/devops-gathering/bootiful-hello-world.git
  • Groovy DSL for build management
  • Rich plugin ecosystem
  • Allows to use Maven dependencies
  • A bit like Maven if Maven was cool ;)

buildscript {
  ext {
    springBootVersion = '1.5.1.RELEASE'
  }
  repositories {
    jcenter()
  }
  dependencies {
    classpath("org.springframework.boot:spring-boot-gradle-plugin:${springBootVersion}")
    classpath 'com.bmuschko:gradle-docker-plugin:3.0.5'
  }
}

apply plugin: 'java'
apply plugin: 'org.springframework.boot'

sourceCompatibility = 1.8

repositories {
    jcenter()
}


dependencies {
    compile('org.springframework.boot:spring-boot-starter')
    testCompile('org.springframework.boot:spring-boot-starter-test')
}
apply plugin: 'com.bmuschko.docker-remote-api'

import com.bmuschko.gradle.docker.tasks.image.Dockerfile
import com.bmuschko.gradle.docker.tasks.image.DockerBuildImage

def dockerBuildDir = 'build/docker/'
def applicationJar = "${archivesBaseName}.jar"

task copyJar(type: Copy) {
    dependsOn bootRepackage
    file(dockerBuildDir).mkdirs()
    from "$libsDir/$applicationJar"
    into dockerBuildDir
}

task createDockerfile(type: Dockerfile) {
    dependsOn copyJar
    destFile = project.file(dockerBuildDir + "Dockerfile")

    from 'openjdk:8-jre'
    copyFile(applicationJar, "/")
    entryPoint "sh", "-c", 
      "java \$JAVA_OPTS -Djava.security.egd=file:/dev/./urandom -jar /$applicationJar"
}

task buildImage(type: DockerBuildImage) {
    dependsOn createDockerfile
    inputDir = createDockerfile.destFile.parentFile
    tag = 'devops/hello-world'
}

Hands-on!

"Hello World" from a container with externalized configuration

$ docker run \
--rm \
--name "hello-world-test" \
-e "GREETER_NAME=docker-command-line" \
devops/hello-world
$ ./gradlew buildImage

Build your image

(or via IDE...)

Run a container from your image

  • Try using different names!
  • What happens if you omit the environment variable?
version: "3"

services:
  greeter:
    image: dog.bootcamp/greeter
    environment: 
      GREETER_NAME: docker-compose

Deployment with Docker-Compose

Like before:

  • image is the tag from the build.gradle

  • Configure application properties via environment variable

Hands-On!

Deploy "Hello World" via the docker-compose file

$ docker-compose up
  • Add a second "greeter"-service greeting another name
  • Run a few times
    • Make an observation about the greeting order!

Summary

  • We can externalize the configuration of spring-boot applications
  • We can configure those properties via environment variables
  • We can build a container from our app via gradle using a docker-plugin
  • We can pass the property values both from docker-cli and docker-compose
  • We can not predict any timing behaviour of docker-compose services on startup

Unit- and Integration-Test your Spring-Boot App

with Spock and Docker

2

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
  }
}  
@ContextConfiguration
@SpringBootTest
class ApplicationTests extends Specification {

    def "should boot up"() {
        expect:
        true
    }

}

Hands-On!

Writing Spock tests

  • Extend the BookRepositoryTests by a test that ensures that a book without IBAN can't be stored
  • Extend the BookRepositoryTests by a test that ensures that a book with IBAN can be found after it was stored
$ git clone https://github.com/devops-gathering/bootiful-book-store.git
$ gradlew check

Run the test suite with:

For better visualized results, run the gradle task from your IDE:

Testcontainers

  • Use Docker containers in your JUnit tests 
  • Spock extension also available ;)
@Testcontainers
class JpaTests extends Specification {

    @Shared
    PostgreSQLContainer postgresContainer = new PostgreSQLContainer()
            .withDatabaseName("bootiful")
            .withUsername("bootiful")
            .withPassword("secret")

    ...
}
  • PostgreSQLContainer will wait until the database is fully operational

Hands-On!

Using Testcontainers

  • Change the BookRepositoryTests to use a MySQL 5.5 container instead of a postgres
    • Keep track: How many project files do you have to change?
    • Also change to the latest Jitpack version of the library

Your client tells you they're - by some obscure policy - not allowed to run a postgres database. They are instead confined to using MySQL 5.5. No way you're going to install that on your local machine for testing.

Summary

  • We can use the Testcontainers library to use containers for integration tests
    • This brings our test and production environment closer together
  • We can easily simulate other deployment scenarios
  • We can isolate the test environment from the host system
    • This allows for more stable integration tests between developer machines and your continuous integration server
    • Any developer can run the same integration tests - no weird dependency management, no "works on my machine", no excuses

Deploy and configure Spring-Boot microservices

using docker-compose

3

Dockerize

  • Our bookstore application depends on creating the schema at startup...

  • In our integration test, using the PostgreSQLContainer ensures the database is ready.

  • But how do we ensure the database is ready in our deployment scenario?

'Correct' answer:

"We don't"

  • Rather build applications that can easily recover from network failure
  • Rather fail to boot dependent services in a swarm environment
    • service will be restarted until it works
  • BUT there are legacy applications...
  • ... and more often than not, framework mechanisms don't play nice with that idea

Introducing Dockerize

  • Delay your application start until a port is reachable
version: "2"
services:
  db:
    image: postgres
  myservice:
    image: devops/myservice
    depends_on:
      - "db"
    entrypoint:
      - dockerize
      - -wait
      - tcp://db:5432
      - -timeout 
      - 120s
      - java
      - -jar
      - myservice.jar

Installing Dockerize

  • In your createDockerfile-task:

task createDockerfile(type: Dockerfile) {
    [...]
	// Install dockerize
	environmentVariable('DOCKERIZE_VERSION', 'v0.3.0')
	runCommand 'apt-get update && apt-get install -y wget'
	runCommand 'wget \
 https://github.com/jwilder/dockerize/releases/download/$DOCKERIZE_VERSION/dockerize-linux-amd64-$DOCKERIZE_VERSION.tar.gz \
    		&& tar -C /usr/local/bin -xzvf dockerize-linux-amd64-$DOCKERIZE_VERSION.tar.gz \
    		&& rm dockerize-linux-amd64-$DOCKERIZE_VERSION.tar.gz'
    [...]
}

https://github.com/jwilder/dockerize#installation

Bootiful Containers

By Kevin Wittek

Bootiful Containers

  • 2,229