cd.go.contrib.elasticagents.dockerswarm.elasticagent.DockerService.java Source code

Java tutorial

Introduction

Here is the source code for cd.go.contrib.elasticagents.dockerswarm.elasticagent.DockerService.java

Source

/*
 * Copyright 2016 ThoughtWorks, Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package cd.go.contrib.elasticagents.dockerswarm.elasticagent;

import cd.go.contrib.elasticagents.dockerswarm.elasticagent.requests.CreateAgentRequest;
import cd.go.contrib.elasticagents.dockerswarm.elasticagent.utils.Size;
import com.google.gson.Gson;
import com.spotify.docker.client.DockerClient;
import com.spotify.docker.client.exceptions.DockerException;
import com.spotify.docker.client.exceptions.ServiceNotFoundException;
import com.spotify.docker.client.messages.ServiceCreateOptions;
import com.spotify.docker.client.messages.ServiceCreateResponse;
import com.spotify.docker.client.messages.swarm.*;
import org.apache.commons.lang.StringUtils;
import org.joda.time.DateTime;

import java.io.IOException;
import java.util.*;

import static cd.go.contrib.elasticagents.dockerswarm.elasticagent.Constants.*;
import static cd.go.contrib.elasticagents.dockerswarm.elasticagent.DockerPlugin.LOG;
import static cd.go.contrib.elasticagents.dockerswarm.elasticagent.utils.Util.splitIntoLinesAndTrimSpaces;
import static org.apache.commons.lang.StringUtils.isBlank;

public class DockerService {
    private static final Gson GSON = new Gson();
    private final DateTime createdAt;
    private final Map<String, String> properties;
    private final String environment;
    private String name;

    public DockerService(String name, Date createdAt, Map<String, String> properties, String environment) {
        this.name = name;
        this.createdAt = new DateTime(createdAt);
        this.properties = properties;
        this.environment = environment;
    }

    public String name() {
        return name;
    }

    public DateTime createdAt() {
        return createdAt;
    }

    public String environment() {
        return environment;
    }

    public Map<String, String> properties() {
        return properties;
    }

    public void terminate(DockerClient docker) throws DockerException, InterruptedException {
        try {
            LOG.debug("Terminating service " + this.name());
            docker.removeService(name);
        } catch (ServiceNotFoundException ignore) {
            LOG.warn("Cannot terminate a service that does not exist " + name);
        }
    }

    public static DockerService fromService(Service service) {
        Map<String, String> labels = service.spec().labels();
        return new DockerService(service.spec().name(), service.createdAt(),
                GSON.fromJson(labels.get(CONFIGURATION_LABEL_KEY), HashMap.class),
                labels.get(ENVIRONMENT_LABEL_KEY));
    }

    public static DockerService create(CreateAgentRequest request, PluginSettings settings, DockerClient docker)
            throws InterruptedException, DockerException, IOException {
        String serviceName = UUID.randomUUID().toString();

        HashMap<String, String> labels = labelsFrom(request);
        String imageName = image(request.properties());
        String[] env = environmentFrom(request, settings, serviceName);

        ContainerSpec.Builder containerSpecBuilder = ContainerSpec.builder();
        if (StringUtils.isNotBlank(request.properties().get("Command"))) {
            containerSpecBuilder.withCommands(
                    splitIntoLinesAndTrimSpaces(request.properties().get("Command")).toArray(new String[] {}));
        }

        ContainerSpec containerSpec = containerSpecBuilder.withImage(imageName).withEnv(env).build();

        TaskSpec taskSpec = TaskSpec.builder().withContainerSpec(containerSpec).withResources(requirements(request))
                .build();

        ServiceSpec serviceSpec = ServiceSpec.builder().withName(serviceName).withLabels(labels)
                .withTaskTemplate(taskSpec).build();
        ServiceCreateResponse service = docker.createService(serviceSpec, new ServiceCreateOptions());

        String id = service.id();

        Service serviceInfo = docker.inspectService(id);

        LOG.debug("Created service " + serviceInfo.spec().name());
        return new DockerService(serviceName, serviceInfo.createdAt(), request.properties(), request.environment());
    }

    private static ResourceRequirements requirements(CreateAgentRequest request) {
        ResourceRequirements resourceRequirements = null;
        Resources.Builder limits = null;
        Resources.Builder reservations = null;

        if (request.properties().containsKey("MaxMemory")) {
            long maxMemory = Size.parse(request.properties().get("MaxMemory")).toBytes();
            limits = Resources.builder().withMemoryBytes(maxMemory);
        }

        if (request.properties().containsKey("ReservedMemory")) {
            long reservedMemory = Size.parse(request.properties().get("ReservedMemory")).toBytes();
            reservations = Resources.builder().withMemoryBytes(reservedMemory);

        }

        if (limits != null || reservations != null) {
            ResourceRequirements.Builder resourceRequirementsBuilder = ResourceRequirements.builder();

            if (limits != null) {
                resourceRequirementsBuilder.withLimits(limits.build());
            }

            if (reservations != null) {
                resourceRequirementsBuilder.withReservations(reservations.build());
            }

            resourceRequirements = resourceRequirementsBuilder.build();
        }
        return resourceRequirements;
    }

    private static String[] environmentFrom(CreateAgentRequest request, PluginSettings settings,
            String containerName) {
        Set<String> env = new HashSet<>();

        env.addAll(settings.getEnvironmentVariables());
        if (StringUtils.isNotBlank(request.properties().get("Environment"))) {
            env.addAll(splitIntoLinesAndTrimSpaces(request.properties().get("Environment")));
        }

        env.addAll(Arrays.asList("GO_EA_MODE=" + mode(), "GO_EA_SERVER_URL=" + settings.getGoServerUrl(),
                "GO_EA_GUID=" + "docker-swarm." + containerName));

        env.addAll(request.autoregisterPropertiesAsEnvironmentVars(containerName));

        return env.toArray(new String[env.size()]);
    }

    private static HashMap<String, String> labelsFrom(CreateAgentRequest request) {
        HashMap<String, String> labels = new HashMap<>();

        labels.put(CREATED_BY_LABEL_KEY, Constants.PLUGIN_ID);
        if (StringUtils.isNotBlank(request.environment())) {
            labels.put(ENVIRONMENT_LABEL_KEY, request.environment());
        }
        labels.put(CONFIGURATION_LABEL_KEY, GSON.toJson(request.properties()));
        return labels;
    }

    @Override
    public boolean equals(Object o) {
        if (this == o)
            return true;
        if (o == null || getClass() != o.getClass())
            return false;

        DockerService that = (DockerService) o;

        return name != null ? name.equals(that.name) : that.name == null;
    }

    @Override
    public int hashCode() {
        return name != null ? name.hashCode() : 0;
    }

    private static String mode() {
        if ("false".equals(System.getProperty("rails.use.compressed.js"))) {
            return "dev";
        }

        if ("true".equalsIgnoreCase(System.getProperty("rails.use.compressed.js"))) {
            return "prod";
        }

        return "";
    }

    private static String image(Map<String, String> properties) {
        String image = properties.get("Image");

        if (isBlank(image)) {
            throw new IllegalArgumentException("Must provide `Image` attribute.");
        }

        if (!image.contains(":")) {
            return image + ":latest";
        }
        return image;
    }

}