All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.bmuschko.gradle.docker.tasks.AbstractDockerRemoteApiTask.groovy Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2014 the original author or authors.
 *
 * 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 com.bmuschko.gradle.docker.tasks

import com.bmuschko.gradle.docker.DockerExtension
import com.bmuschko.gradle.docker.DockerRemoteApiPlugin
import com.bmuschko.gradle.docker.internal.RegistryAuthLocator
import com.github.dockerjava.api.DockerClient
import com.github.dockerjava.core.DefaultDockerClientConfig
import com.github.dockerjava.core.DockerClientBuilder
import com.github.dockerjava.core.DockerClientConfig
import com.github.dockerjava.httpclient5.ApacheDockerHttpClient
import groovy.transform.CompileStatic
import groovy.transform.Memoized
import org.gradle.api.Action
import org.gradle.api.DefaultTask
import org.gradle.api.file.DirectoryProperty
import org.gradle.api.provider.Property
import org.gradle.api.tasks.Input
import org.gradle.api.tasks.InputDirectory
import org.gradle.api.tasks.Internal
import org.gradle.api.tasks.Optional
import org.gradle.api.tasks.PathSensitive
import org.gradle.api.tasks.PathSensitivity
import org.gradle.api.tasks.TaskAction

@CompileStatic
abstract class AbstractDockerRemoteApiTask extends DefaultTask {

    /**
     * Docker remote API server URL. Defaults to "http://localhost:2375".
     */
    @Input
    @Optional
    final Property url = project.objects.property(String)

    /**
     * Path to the Docker certificate and key.
     */
    @InputDirectory
    @PathSensitive(PathSensitivity.RELATIVE)
    @Optional
    final DirectoryProperty certPath = project.objects.directoryProperty()

    /**
     * The Docker remote API version.
     */
    @Input
    @Optional
    final Property apiVersion = project.objects.property(String)

    private Action errorHandler
    private Action nextHandler
    private Runnable completeHandler

    @TaskAction
    void start() {
        boolean commandFailed = false
        try {
            runRemoteCommand()
        } catch (Exception possibleException) {
            commandFailed = true
            if (errorHandler) {
                errorHandler.execute(possibleException)
            } else {
                throw possibleException
            }
        }
        if(!commandFailed && completeHandler) {
            completeHandler.run()
        }
    }

    /**
     * Reacts to a potential error occurring during the operation.
     *
     * @param action The action handling the error
     * @since 4.0.0
     */
    void onError(Action action) {
        errorHandler = action
    }

    /**
     * Reacts to data returned by an operation.
     *
     * @param action The action handling the data
     * @since 4.0.0
     */
    void onNext(Action action) {
        nextHandler = action
    }

    @Internal
    protected Action getNextHandler() {
        nextHandler
    }

    /**
     * Reacts to the completion of the operation.
     *
     * @param callback The callback to be executed
     * @since 4.0.0
     */
    void onComplete(Runnable callback) {
        completeHandler = callback
    }

    /**
     * Gets the Docker client uses to communicate with Docker via its remote API.
     * Initialized instance upon first request.
     * Returns the same instance for any successive method call.
     * 

* Before accessing the Docker client, all data used for configuring its runtime behavior needs to be evaluated. * The data includes: *

    *
  1. The property values of this class
  2. *
  3. The plugin's extension property values
  4. *
*

* It is safe to access the Docker client under the following conditions: *

    *
  1. In the task action
  2. *
  3. In the task's constructor if used in {@code Action} or {@code Closure} of {@code outputs.upToDateWhen}
  4. *
* * @return The Docker client */ @Internal @Memoized DockerClient getDockerClient() { DockerClientConfiguration dockerClientConfiguration = createDockerClientConfig() DockerExtension dockerExtension = (DockerExtension) project.extensions.getByName(DockerRemoteApiPlugin.EXTENSION_NAME) String dockerUrl = getDockerHostUrl(dockerClientConfiguration, dockerExtension) File dockerCertPath = dockerClientConfiguration.certPath?.asFile ?: dockerExtension.certPath.getOrNull()?.asFile String apiVersion = dockerClientConfiguration.apiVersion ?: dockerExtension.apiVersion.getOrNull() // Create configuration DefaultDockerClientConfig.Builder dockerClientConfigBuilder = DefaultDockerClientConfig.createDefaultConfigBuilder() dockerClientConfigBuilder.withDockerHost(dockerUrl) if (dockerCertPath) { dockerClientConfigBuilder.withDockerTlsVerify(true) dockerClientConfigBuilder.withDockerCertPath(dockerCertPath.canonicalPath) } else { dockerClientConfigBuilder.withDockerTlsVerify(false) } if (apiVersion) { dockerClientConfigBuilder.withApiVersion(apiVersion) } DefaultDockerClientConfig dockerClientConfig = dockerClientConfigBuilder.build() DockerClient dockerClient = createDefaultDockerClient(dockerClientConfig) // register buildFinished-hook to close docker client. project.gradle.buildFinished { dockerClient.close() } dockerClient } private DockerClient createDefaultDockerClient(DockerClientConfig config) { ApacheDockerHttpClient dockerClient = new ApacheDockerHttpClient.Builder() .dockerHost(config.getDockerHost()) .sslConfig(config.getSSLConfig()) .build() DockerClientBuilder.getInstance(config) .withDockerHttpClient(dockerClient) .build() } /** * Returns the instance of {@link RegistryAuthLocator}. *

* Unless other credentials information provided, the instance returns authConfig object provided by the Docker client. * * @return The registry authentication locator */ @Internal @Memoized protected RegistryAuthLocator getRegistryAuthLocator() { new RegistryAuthLocator() } private DockerClientConfiguration createDockerClientConfig() { DockerClientConfiguration dockerClientConfig = new DockerClientConfiguration() dockerClientConfig.url = url.getOrNull() dockerClientConfig.certPath = certPath.getOrNull() dockerClientConfig.apiVersion = apiVersion.getOrNull() dockerClientConfig } /** * Checks if Docker host URL starts with http(s) and if so, converts it to tcp * which is accepted by docker-java library. * * @param dockerClientConfiguration docker client configuration * @return Docker host URL as string */ private String getDockerHostUrl(DockerClientConfiguration dockerClientConfiguration, DockerExtension dockerExtension) { String url = (dockerClientConfiguration.url ?: dockerExtension.url.getOrNull()).toLowerCase() url.startsWith('http') ? 'tcp' + url.substring(url.indexOf(':')) : url } abstract void runRemoteCommand() }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy