diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 000000000..5bbffc51f --- /dev/null +++ b/.editorconfig @@ -0,0 +1,23 @@ +root = true + +[*] +charset = utf-8 +end_of_line = lf +max_line_length = 140 +indent_style = space +indent_size = 4 +tab_width = 4 +insert_final_newline = true + +[*.java] +ij_java_class_count_to_use_import_on_demand = 9999 +ij_java_names_count_to_use_import_on_demand = 9999 + +[{*.pom,*.xml}] +indent_style = tab +ij_xml_attribute_wrap = off + + +[*.{yaml,yml}] +indent_size = 2 +ij_yaml_keep_indents_on_empty_lines = false diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS new file mode 100644 index 000000000..e5a6b0376 --- /dev/null +++ b/.github/CODEOWNERS @@ -0,0 +1,2 @@ +* @docker-java/team + diff --git a/.github/dependabot.yml b/.github/dependabot.yml new file mode 100644 index 000000000..9bcef2d88 --- /dev/null +++ b/.github/dependabot.yml @@ -0,0 +1,16 @@ +version: 2 +updates: +- package-ecosystem: maven + directory: "/" + schedule: + interval: weekly + day: monday + open-pull-requests-limit: 99 + rebase-strategy: disabled + ignore: + - dependency-name: "org.glassfish.jersey.connectors:jersey-apache-connector" + update-types: [ "version-update:semver-major" ] + - dependency-name: "org.glassfish.jersey.core:jersey-client" + update-types: [ "version-update:semver-major" ] + - dependency-name: "org.glassfish.jersey.inject:jersey-hk2" + update-types: [ "version-update:semver-major" ] diff --git a/.github/release-drafter.yml b/.github/release-drafter.yml new file mode 100644 index 000000000..f570cce43 --- /dev/null +++ b/.github/release-drafter.yml @@ -0,0 +1,19 @@ +tag-template: $NEXT_PATCH_VERSION +name-template: '$NEXT_PATCH_VERSION' +categories: + - title: '🚀 Features' + labels: + - 'type/feature' + - title: '📈 Enhancements' + labels: + - 'type/enhancement' + - title: '🐛 Bug Fixes' + labels: + - 'type/bug' + - title: '🧰 Maintenance' + label: 'type/housekeeping' +change-template: '- $TITLE @$AUTHOR (#$NUMBER)' +template: | + ## Changes + + $CHANGES diff --git a/.github/stale.yml b/.github/stale.yml new file mode 100644 index 000000000..23aefd1f3 --- /dev/null +++ b/.github/stale.yml @@ -0,0 +1,16 @@ + +daysUntilStale: 90 + +daysUntilClose: 30 + +exemptLabels: + - resolution/acknowledged + +staleLabel: resolution/stale + +markComment: > + This issue has been automatically marked as stale because it has not had + recent activity. It will be closed if no further activity occurs. Thank you + for your contributions. + +closeComment: false diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 000000000..d7b105d1d --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,55 @@ +name: CI + +on: + pull_request: {} + push: { branches: [ main ] } + workflow_dispatch: + +jobs: + build: + runs-on: ubuntu-24.04 + strategy: + fail-fast: false + matrix: + include: + - { name: "default", javaVersion: 8 } + - { name: "default", javaVersion: 17 } + - { name: "default", javaVersion: 21 } + steps: + - uses: actions/checkout@v4 + - name: Set up JDK + uses: actions/setup-java@v4 + with: + java-version: ${{matrix.javaVersion}} + distribution: temurin + cache: maven + - name: Configure Docker + id: setup_docker + uses: docker/setup-docker-action@v4 + with: + channel: stable + - name: Build with Maven + env: + DOCKER_HOST: ${{steps.setup_docker.outputs.sock}} + run: ./mvnw --no-transfer-progress verify + + tcp: + runs-on: ubuntu-24.04 + steps: + - uses: actions/checkout@v4 + - name: Set up JDK + uses: actions/setup-java@v4 + with: + java-version: 8 + distribution: temurin + cache: maven + - name: Configure Docker + id: setup_docker + uses: docker/setup-docker-action@v4 + with: + channel: stable + tcp-port: 2375 + - name: Build with Maven + env: + DOCKER_HOST: ${{steps.setup_docker.outputs.tcp}} + run: ./mvnw --no-transfer-progress verify diff --git a/.github/workflows/release-drafter.yml b/.github/workflows/release-drafter.yml new file mode 100644 index 000000000..99cd01cfc --- /dev/null +++ b/.github/workflows/release-drafter.yml @@ -0,0 +1,16 @@ +name: Release Drafter + +on: + push: + # branches to consider in the event; optional, defaults to all + branches: + - main + +jobs: + update_release_draft: + runs-on: ubuntu-latest + steps: + # Drafts your next Release notes as Pull Requests are merged into "master" + - uses: release-drafter/release-drafter@v5 + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml new file mode 100644 index 000000000..16777daa6 --- /dev/null +++ b/.github/workflows/release.yml @@ -0,0 +1,32 @@ +name: Release + +on: + release: + types: + - prereleased + - released + +jobs: + build: + runs-on: ubuntu-24.04 + steps: + - uses: actions/checkout@v4 + - name: Set up JDK 8 + uses: actions/setup-java@v4 + with: + java-version: 8 + distribution: temurin + server-id: central + server-username: MAVEN_USERNAME + server-password: MAVEN_CENTRAL_TOKEN + gpg-private-key: ${{ secrets.MAVEN_GPG_PRIVATE_KEY }} + gpg-passphrase: MAVEN_GPG_PASSPHRASE + - name: Set version + run: ./mvnw versions:set -DnewVersion="${{github.event.release.tag_name}}" + # TODO check main's CI status + - name: Deploy with Maven + env: + MAVEN_USERNAME: ${{ secrets.SONATYPE_CENTRAL_USERNAME }} + MAVEN_CENTRAL_TOKEN: ${{ secrets.SONATYPE_CENTRAL_PASSWORD }} + MAVEN_GPG_PASSPHRASE: ${{ secrets.MAVEN_GPG_PASSPHRASE }} + run: ./mvnw -Prelease deploy -DskipTests diff --git a/.gitignore b/.gitignore index 8f1fdc779..006641e8c 100644 --- a/.gitignore +++ b/.gitignore @@ -6,13 +6,14 @@ .project .settings .classpath +.factorypath # Ignore all build/dist directories target +dependency-reduced-pom.xml # Ignore InteliJ Idea project files -.idea -.idea/* +.idea/ *.iml *.iws *.ipr @@ -21,4 +22,5 @@ target *.log #Ignore Test Output -test-output \ No newline at end of file +test-output +/.checkstyle diff --git a/.mvn/wrapper/MavenWrapperDownloader.java b/.mvn/wrapper/MavenWrapperDownloader.java new file mode 100644 index 000000000..b901097f2 --- /dev/null +++ b/.mvn/wrapper/MavenWrapperDownloader.java @@ -0,0 +1,117 @@ +/* + * Copyright 2007-present 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. + */ +import java.net.*; +import java.io.*; +import java.nio.channels.*; +import java.util.Properties; + +public class MavenWrapperDownloader { + + private static final String WRAPPER_VERSION = "0.5.6"; + /** + * Default URL to download the maven-wrapper.jar from, if no 'downloadUrl' is provided. + */ + private static final String DEFAULT_DOWNLOAD_URL = "https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/" + + WRAPPER_VERSION + "/maven-wrapper-" + WRAPPER_VERSION + ".jar"; + + /** + * Path to the maven-wrapper.properties file, which might contain a downloadUrl property to + * use instead of the default one. + */ + private static final String MAVEN_WRAPPER_PROPERTIES_PATH = + ".mvn/wrapper/maven-wrapper.properties"; + + /** + * Path where the maven-wrapper.jar will be saved to. + */ + private static final String MAVEN_WRAPPER_JAR_PATH = + ".mvn/wrapper/maven-wrapper.jar"; + + /** + * Name of the property which should be used to override the default download url for the wrapper. + */ + private static final String PROPERTY_NAME_WRAPPER_URL = "wrapperUrl"; + + public static void main(String args[]) { + System.out.println("- Downloader started"); + File baseDirectory = new File(args[0]); + System.out.println("- Using base directory: " + baseDirectory.getAbsolutePath()); + + // If the maven-wrapper.properties exists, read it and check if it contains a custom + // wrapperUrl parameter. + File mavenWrapperPropertyFile = new File(baseDirectory, MAVEN_WRAPPER_PROPERTIES_PATH); + String url = DEFAULT_DOWNLOAD_URL; + if(mavenWrapperPropertyFile.exists()) { + FileInputStream mavenWrapperPropertyFileInputStream = null; + try { + mavenWrapperPropertyFileInputStream = new FileInputStream(mavenWrapperPropertyFile); + Properties mavenWrapperProperties = new Properties(); + mavenWrapperProperties.load(mavenWrapperPropertyFileInputStream); + url = mavenWrapperProperties.getProperty(PROPERTY_NAME_WRAPPER_URL, url); + } catch (IOException e) { + System.out.println("- ERROR loading '" + MAVEN_WRAPPER_PROPERTIES_PATH + "'"); + } finally { + try { + if(mavenWrapperPropertyFileInputStream != null) { + mavenWrapperPropertyFileInputStream.close(); + } + } catch (IOException e) { + // Ignore ... + } + } + } + System.out.println("- Downloading from: " + url); + + File outputFile = new File(baseDirectory.getAbsolutePath(), MAVEN_WRAPPER_JAR_PATH); + if(!outputFile.getParentFile().exists()) { + if(!outputFile.getParentFile().mkdirs()) { + System.out.println( + "- ERROR creating output directory '" + outputFile.getParentFile().getAbsolutePath() + "'"); + } + } + System.out.println("- Downloading to: " + outputFile.getAbsolutePath()); + try { + downloadFileFromURL(url, outputFile); + System.out.println("Done"); + System.exit(0); + } catch (Throwable e) { + System.out.println("- Error downloading"); + e.printStackTrace(); + System.exit(1); + } + } + + private static void downloadFileFromURL(String urlString, File destination) throws Exception { + if (System.getenv("MVNW_USERNAME") != null && System.getenv("MVNW_PASSWORD") != null) { + String username = System.getenv("MVNW_USERNAME"); + char[] password = System.getenv("MVNW_PASSWORD").toCharArray(); + Authenticator.setDefault(new Authenticator() { + @Override + protected PasswordAuthentication getPasswordAuthentication() { + return new PasswordAuthentication(username, password); + } + }); + } + URL website = new URL(urlString); + ReadableByteChannel rbc; + rbc = Channels.newChannel(website.openStream()); + FileOutputStream fos = new FileOutputStream(destination); + fos.getChannel().transferFrom(rbc, 0, Long.MAX_VALUE); + fos.close(); + rbc.close(); + } + +} diff --git a/.mvn/wrapper/maven-wrapper.jar b/.mvn/wrapper/maven-wrapper.jar new file mode 100644 index 000000000..2cc7d4a55 Binary files /dev/null and b/.mvn/wrapper/maven-wrapper.jar differ diff --git a/.mvn/wrapper/maven-wrapper.properties b/.mvn/wrapper/maven-wrapper.properties new file mode 100644 index 000000000..642d572ce --- /dev/null +++ b/.mvn/wrapper/maven-wrapper.properties @@ -0,0 +1,2 @@ +distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.6.3/apache-maven-3.6.3-bin.zip +wrapperUrl=https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index a9e93162a..000000000 --- a/.travis.yml +++ /dev/null @@ -1,52 +0,0 @@ -sudo: required -dist: trusty -language: java - -services: - - docker - -jdk: - - oraclejdk7 - -install: true - -env: - global: - - DOCKER_TLS_VERIFY="" - # The next declaration is the encrypted COVERITY_SCAN_TOKEN, created - # via the "travis encrypt" command using the project repo's public key - - secure: "GonzmzvnXsTNQV+6sKtBSSPiwbpMZjxumNt5LFp1g77/afLxw9kl2EQOXbUe308vFOwRVqeY7drBvNJa8aJkTUClfMaGRjfZ9DUwm6doMKMUYrdEkYoQTcH7yDX5K5w9MT6m+Izj+BK2gB7nK3yFlYG6COeXCdFbQ4/cf3/xfRc=" - - COVERITY_SCAN_PROJECT_NAME="docker-java/docker-java" - - COVERITY_SCAN_BRANCH_PATTERN="master" - - COVERITY_SCAN_NOTIFICATION_EMAIL="kanstantsin.sha@gmail.com" - - matrix: - - repo="main" DOCKER_HOST="tcp://127.0.0.1:2375" DOCKER_VERSION="1.12.0-0~trusty" - - repo="main" DOCKER_HOST="unix:///var/run/docker.sock" DOCKER_VERSION="1.12.0-0~trusty" - - repo="main" DOCKER_HOST="tcp://127.0.0.1:2375" DOCKER_VERSION="1.11.2-0~trusty" DEPLOY=true COVERITY=true CODECOV=true - - repo="main" DOCKER_HOST="unix:///var/run/docker.sock" DOCKER_VERSION="1.11.2-0~trusty" - - repo="main" DOCKER_HOST="tcp://127.0.0.1:2375" DOCKER_VERSION="1.10.3-0~trusty" - - repo="main" DOCKER_HOST="unix:///var/run/docker.sock" DOCKER_VERSION="1.10.3-0~trusty" -# - repo="testing" DOCKER_HOST="tcp://127.0.0.1:2375" -# - repo="testing" DOCKER_HOST="unix:///var/run/docker.sock" -# - repo="experimental" DOCKER_HOST="tcp://127.0.0.1:2375" -# - repo="experimental" DOCKER_HOST="unix:///var/run/docker.sock" - -cache: - directories: - - $HOME/.travis_cache - - /tmp/coverity-cache - - $HOME/.m2 # install will pollute it - -before_install: - - pip install --user codecov - - ./.travis/travis-before-install.sh - -script: - - ./.travis/travis-script.sh - -after_success: - - ./.travis/travis-after-success.sh - -after_script: - - sudo cat /var/log/upstart/docker.log diff --git a/.travis/travis-after-success.sh b/.travis/travis-after-success.sh deleted file mode 100755 index 695358122..000000000 --- a/.travis/travis-after-success.sh +++ /dev/null @@ -1,21 +0,0 @@ -#!/usr/bin/env bash - -if [[ $CODECOV == "true" ]]; then - codecov -fi - -if [[ $TRAVIS_BRANCH == "master" ]] && [[ $TRAVIS_PULL_REQUEST == "false" ]] && [[ $DEPLOY == "true" ]]; -then - cat <> ~/settings.xml - - - - ossrh - \${env.OSSRH_USER} - \${env.OSSRH_PASS} - - - -EOF - mvn deploy -DskipITs --settings ~/settings.xml - fi diff --git a/.travis/travis-before-install.sh b/.travis/travis-before-install.sh deleted file mode 100755 index 13034fc74..000000000 --- a/.travis/travis-before-install.sh +++ /dev/null @@ -1,48 +0,0 @@ -#!/usr/bin/env bash - - -sudo apt-get install -y -q ca-certificates - -echo -n | openssl s_client -connect scan.coverity.com:443 | sed -ne '/-BEGIN CERTIFICATE-/,/-END CERTIFICATE-/p' | sudo tee -a /etc/ssl/certs/ca-certificates.crt - - -if [ "$FAST_BUILD" == true ]; then - echo "Fast build, skipping docker installations." - exit 0 -fi - -set -exu - -docker info -docker version - -sudo -E apt-get update -sudo -E apt-get install -q -y wget -sudo -E apt-get -q -y --purge remove docker-engine -sudo -E apt-cache policy docker-engine - -./get-docker-com.sh -#mkdir "${HOME}/.cache" || : -#pushd "${HOME}/.cache" -# wget -N "https://apt.dockerproject.org/repo/pool/main/d/docker-engine/docker-engine_${DOCKER_VERSION}_amd64.deb" -# sudo apt-get -f install -# sudo dpkg -i "$(ls *${DOCKER_VERSION}*)" -#popd -#rm -f "src/test/resources/logback.xml" -mv "src/test/resources/travis-logback.xml" "src/test/resources/logback.xml" - -echo 'DOCKER_OPTS="-H=unix:///var/run/docker.sock -H=tcp://127.0.0.1:2375"' | sudo tee -a /etc/default/docker -sudo -E restart docker -sleep 10 -docker version -docker info - -set +u - -cat < "${HOME}/.docker-java.properties" -registry.username=${registry_username} -registry.password=${registry_password} -registry.email=${registry_email} -registry.url=https://index.docker.io/v1/ - -EOF diff --git a/.travis/travis-script.sh b/.travis/travis-script.sh deleted file mode 100755 index aeae555ef..000000000 --- a/.travis/travis-script.sh +++ /dev/null @@ -1,24 +0,0 @@ -#!/usr/bin/env bash - -set -ex - -IS_COVERITY_SCAN_BRANCH=`ruby -e "puts '${TRAVIS_BRANCH}' =~ /\\A$COVERITY_SCAN_BRANCH_PATTERN\\z/ ? 1 : 0"` - - -if [ "${FAST_BUILD}" == "true" ]; then - if [ "$TRAVIS_PULL_REQUEST" == "false" ] && [ "$COVERITY" == "true" ] && [ "$IS_COVERITY_SCAN_BRANCH" = "1" ]; then - export COVERITY_SCAN_BUILD_COMMAND="mvn package" - #curl -s "https://scan.coverity.com/scripts/travisci_build_coverity_scan.sh" | bash - ./.travis/travisci_build_coverity_scan.sh - else - mvn package - fi -else - if [ "$TRAVIS_PULL_REQUEST" == "false" ] && [ "$COVERITY" == "true" ] && [ "$IS_COVERITY_SCAN_BRANCH" = "1" ]; then - export COVERITY_SCAN_BUILD_COMMAND="mvn verify" - #curl -s "https://scan.coverity.com/scripts/travisci_build_coverity_scan.sh" | bash - ./.travis/travisci_build_coverity_scan.sh - else - mvn verify - fi -fi diff --git a/.travis/travisci_build_coverity_scan.sh b/.travis/travisci_build_coverity_scan.sh deleted file mode 100755 index 074d0a46f..000000000 --- a/.travis/travisci_build_coverity_scan.sh +++ /dev/null @@ -1,113 +0,0 @@ -#!/bin/bash - -set -e - -# Environment check -echo -e "\033[33;1mNote: COVERITY_SCAN_PROJECT_NAME and COVERITY_SCAN_TOKEN are available on Project Settings page on scan.coverity.com\033[0m" -[ -z "$COVERITY_SCAN_PROJECT_NAME" ] && echo "ERROR: COVERITY_SCAN_PROJECT_NAME must be set" && exit 1 -[ -z "$COVERITY_SCAN_NOTIFICATION_EMAIL" ] && echo "ERROR: COVERITY_SCAN_NOTIFICATION_EMAIL must be set" && exit 1 -[ -z "$COVERITY_SCAN_BRANCH_PATTERN" ] && echo "ERROR: COVERITY_SCAN_BRANCH_PATTERN must be set" && exit 1 -[ -z "$COVERITY_SCAN_BUILD_COMMAND" ] && echo "ERROR: COVERITY_SCAN_BUILD_COMMAND must be set" && exit 1 -[ -z "$COVERITY_SCAN_TOKEN" ] && echo "ERROR: COVERITY_SCAN_TOKEN must be set" && exit 1 - -PLATFORM=`uname` -TOOL_ARCHIVE=/tmp/coverity-cache/cov-analysis-${PLATFORM}.tgz -TOOL_URL=https://scan.coverity.com/download/${PLATFORM} -TOOL_BASE=/tmp/coverity-scan-analysis -UPLOAD_URL="https://scan.coverity.com/builds" -SCAN_URL="https://scan.coverity.com" - -# Do not run on pull requests -if [ "${TRAVIS_PULL_REQUEST}" = "true" ]; then - echo -e "\033[33;1mINFO: Skipping Coverity Analysis: branch is a pull request.\033[0m" - exit 0 -fi - -# Verify this branch should run -IS_COVERITY_SCAN_BRANCH=`ruby -e "puts '${TRAVIS_BRANCH}' =~ /\\A$COVERITY_SCAN_BRANCH_PATTERN\\z/ ? 1 : 0"` -if [ "$IS_COVERITY_SCAN_BRANCH" = "1" ]; then - echo -e "\033[33;1mCoverity Scan configured to run on branch ${TRAVIS_BRANCH}\033[0m" -else - echo -e "\033[33;1mCoverity Scan NOT configured to run on branch ${TRAVIS_BRANCH}\033[0m" - exit 1 -fi - -# Verify upload is permitted -AUTH_RES=`curl -s --form project="$COVERITY_SCAN_PROJECT_NAME" --form token="$COVERITY_SCAN_TOKEN" $SCAN_URL/api/upload_permitted` -if [ "$AUTH_RES" = "Access denied" ]; then - echo -e "\033[33;1mCoverity Scan API access denied. Check COVERITY_SCAN_PROJECT_NAME and COVERITY_SCAN_TOKEN.\033[0m" - exit 1 -else - AUTH=`echo $AUTH_RES | ruby -e "require 'rubygems'; require 'json'; puts JSON[STDIN.read]['upload_permitted']"` - if [ "$AUTH" = "true" ]; then - echo -e "\033[33;1mCoverity Scan analysis authorized per quota.\033[0m" - else - WHEN=`echo $AUTH_RES | ruby -e "require 'rubygems'; require 'json'; puts JSON[STDIN.read]['next_upload_permitted_at']"` - echo -e "\033[33;1mCoverity Scan analysis NOT authorized until $WHEN.\033[0m" - - exit 1 - fi -fi - -mkdir -p /tmp/coverity-cache || : - -if [ ! -d $TOOL_BASE ]; then - - # verify that binary is right - if file $TOOL_ARCHIVE | grep HTML ; then - echo "Removing $TOOL_ARCHIVE" - rm -f $TOOL_ARCHIVE - fi - - # Download Coverity Scan Analysis Tool - if [ ! -e $TOOL_ARCHIVE ]; then - echo -e "\033[33;1mDownloading Coverity Scan Analysis Tool...\033[0m" - wget -nv -N -O $TOOL_ARCHIVE $TOOL_URL --post-data "project=$COVERITY_SCAN_PROJECT_NAME&token=$COVERITY_SCAN_TOKEN" - fi - - # Extract Coverity Scan Analysis Tool - echo -e "\033[33;1mExtracting Coverity Scan Analysis Tool...\033[0m" - mkdir -p $TOOL_BASE - pushd $TOOL_BASE - du -sh $TOOL_ARCHIVE - file $TOOL_ARCHIVE - file $TOOL_ARCHIVE | grep HTML && cat $TOOL_ARCHIVE || : - ls -la $TOOL_ARCHIVE - tar -xf $TOOL_ARCHIVE #|& grep -v "Ignoring unknown extended header keyword" - popd -fi - -TOOL_DIR=`find $TOOL_BASE -type d -name 'cov-analysis*'` -export PATH=$TOOL_DIR/bin:$PATH - -# Build -echo -e "\033[33;1mRunning Coverity Scan Analysis Tool...\033[0m" -COV_BUILD_OPTIONS="" -#COV_BUILD_OPTIONS="--return-emit-failures 8 --parse-error-threshold 85" -RESULTS_DIR="cov-int" -eval "${COVERITY_SCAN_BUILD_COMMAND_PREPEND}" -COVERITY_UNSUPPORTED=1 cov-build --dir $RESULTS_DIR $COV_BUILD_OPTIONS $COVERITY_SCAN_BUILD_COMMAND -cov-import-scm --dir $RESULTS_DIR --scm git --log $RESULTS_DIR/scm_log.txt 2>&1 - -# Upload results -echo -e "\033[33;1mTarring Coverity Scan Analysis results...\033[0m" -RESULTS_ARCHIVE=analysis-results.tgz -tar czf $RESULTS_ARCHIVE $RESULTS_DIR -SHA=`git rev-parse --short HEAD` - -echo -e "\033[33;1mUploading Coverity Scan Analysis results...\033[0m" -response=$(curl \ - --silent --write-out "\n%{http_code}\n" \ - --form project=$COVERITY_SCAN_PROJECT_NAME \ - --form token=$COVERITY_SCAN_TOKEN \ - --form email=$COVERITY_SCAN_NOTIFICATION_EMAIL \ - --form file=@$RESULTS_ARCHIVE \ - --form version=$SHA \ - --form description="Travis CI build" \ - $UPLOAD_URL) -status_code=$(echo "$response" | sed -n '$p') -if [ "$status_code" != "201" ]; then - TEXT=$(echo "$response" | sed '$d') - echo -e "\033[33;1mCoverity Scan upload failed: $TEXT.\033[0m" - exit 1 -fi diff --git a/CHANGELOG.md b/CHANGELOG.md index 132f3c6b1..5d344d93b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,8 +1,157 @@ Change Log === -## 3.0.3-SNAPSHOT +## 3.2.0 +- **Changelog is not maintained in this file. Please follow git diff or github releases.** +- Library was split into multiple modules to get ability to choose transports. +Okhttp was added (say thanks to @bsideup). +- Various cleanup, tests de-duplication internally. Planned binary compatibility breakage was reverted by @testcontainers project, so migration should work smoothly. Please switch to non-deprecated methods. +- Appeared various new commands and Fields(command options for existing commands). + +## 3.1.2 +- update unix-socket to 2.2.0 +- Remove `JacksonJaxbJsonProvider` from `FiltersEncoder` +- BuildImageCmdImpl: Fix an exception message +- Add support for target parameter in BuildImgCmd +- Add prune operations +- Updating Jackson due to CVEs +- Make StatsConfig public +- Set 3 mb as limit for json responce. + +## 3.1.1 +- Patch save image with tag +- [api/healthcheck] startPeriod is now a long + +## 3.1.0 +- Release + +## 3.1.0-rc-8 +- Do awaitCompletion upon socket close exception +- Fix `X-Registry-Auth` base64 encoding + +## 3.1.0-rc-7 +- Fix NPE when docker config file doesn't exist + +## 3.1.0-rc-6 +- Add resize feature to container and exec +- Update part of apis to 1.37 +- Update dependencies +- Fix No serializer found for class com.githubdockerjava.api.model.ServiceGlobalModeOptions +- Add HostConfig.StorageOpt and ExecCreateCmd.Env +- Added GCPLOGS enum LoggingType +- Stop proxying HostConfig class (static helper provided) +- Add ExecCreateCmd.Env +- Add failcnt into memorystatsconfig +- Support some missing Engine APIs +- Added memory swappiness to create container command. +- Fix for ignore all files except specified + +## 3.1.0-rc-5 +- Add missing properties in InspectContainer response +- Add withFilter methods in ListNetworksCmd & ListVolumesCmd +- Add WorkingDir property in containers exec +- Add isolation property support in Info response +- Support platform option in image build/create/pull commands +- Add OSVersion and RootFS support in InspectImageResponse +- Fix double-marshalling of cachefrom +- make AuthConfig compatible with indentitytoken +- Allow netty to handle compressed response bodies + +## 3.1.0-rc-4 +- Update deps +- fix HTTP/1.1 compliance (missing Host header) +- support rollback_completed value in ServiceUpdateState +- Add Privileged property to ExecCreateCmd +- Encode spaces as %20 rather than + in URL params +- Add tmpfs mount support since v1.29 +- Add support for swarm service/task logs +- Add mapping annotations to custom constructor +- Support network mode as part of the docker build +- support "rollback" as value for UpdateFailureAction +- Add "Pid" field to InspectExecResponse +- Add DiskQuota to HostConfig and CreateContainerCmd +- Add AutoRemove to HostConfig +- follow symbolic links when walking dir +- Use path from the configured docker host instead of hardcoded "/var/run/docker.sock" in netty factory +- Configure JerseyDockerCmdExecFactory to avoid chunked encoding + +## 3.1.0-rc-3 +- export TmpFs configuration for HostConfig and DockerClient +- avoid double encoding for url query string + +## 3.1.0-rc-2 +- https://github.com/docker-java/docker-java/pulls?q=is%3Apr+milestone%3A3.1.0-rc-2+is%3Aclosed + +## 3.1.0-rc-1 + A lot of changes... +- Swarm Mode support. +- Classic swarm support. +- various netty improvements +- Implement AbstractDockerCmdExecFactory + + +## 3.0.14 +- Encode spaces as %20 rather than + in URL params + +## 3.0.13 +- Fix .dockerignore handling on Windows +- Include empty directories in build context + +## 3.0.12 +- Make NettyDockerCmdExecFactory has compatibility both Linux and OSX automatically +- Fix double encoding for netty. +- filter config.json before unmarshalling (creds/auth) + +## 3.0.11 +- Add labels and attachable properties to network. +- Set default socket timeout for RequestConfig. +- Netty skip instead of throw error on non-linux os. +- Clean tmp file after upload. +- Filters ignore application/x-tar. +- Allow user to call connectionManager's closeIdleConnections. + +## 3.0.10 +- Support for cache-from in build image command +- Allow multiple tags in build image command +- Custom `db` logging type +- Allow an explicit Dockerfile location string to be specified to theuild command +- Fix image build for docker 17 with 'tagged' word. + +## 3.0.9 +- NettyDockerCmdExecFactory ignores API version +- exclude commons-logging from httpclient since docker-java uses slf4j/logback +- Generate OSGi compliant manifest +- AuthResponse may contains token. + +## 3.0.8 + - Use TLSv1.2 by default + - Health api + - Labels + - Support for multiple certificates + +## 3.0.7 + * https://github.com/docker-java/docker-java/milestone/17?closed=1 + * HostConfig pidLimits + * Label image during build + * Expose 'User' property on ExecCreateCmd #707 #708 + +## 3.0.6 + * Fixed issue with jersey and unix domain sockets. + * [#703](https://github.com/docker-java/docker-java/pull/703) Allow to configure connection pool timeout. + * Make all models Serializable. + * [NETTY] Fix loadImage responce on 1.24 API. + * LogPath field for inspect container. + * [#700] (https://github.com/docker-java/docker-java/pull/700) Bugfix:donot throw RuntimeException when a error occured in awaitCompletion(long,TimeUnit) + +## 3.0.5 + * Events updated to 1.24 API model. + +## 3.0.4 + * Make cert util methods public. + +## 3.0.3 + * [JERSEY] Don't send body for start container request. ## 3.0.2 * Enhanced Dockerignore filtering. @@ -152,7 +301,7 @@ v2.0.0 Release notes * Some commands APIs has been changed to be callback-driven now to simplify the processing of the result streams for the client application. This affects namely the events, stats, log, attach, build, push and pull commands. Look at the Wiki how to [process events](https://github.com/docker-java/docker-java/wiki#handle-events) or how to [build an image](https://github.com/docker-java/docker-java/wiki#build-image-from-dockerfile) from dockerfile for example. -* The `DockerClientConfig` API has changed to free it from implementation specific configuration options like `readTimeout`, `maxTotalConnections`, `maxPerRouteConnections` and `enableLoggingFilter`. Most options can be configured via `DockerCmdExecFactoryImpl` [programmatically](https://github.com/docker-java/docker-java/wiki#intialize-docker-client-advanced) now. Logging is configurable via [logback](https://github.com/docker-java/docker-java/blob/master/src/test/resources/logback.xml) configuration file in the classpath. +* The `DockerClientConfig` API has changed to free it from implementation specific configuration options like `readTimeout`, `maxTotalConnections`, `maxPerRouteConnections` and `enableLoggingFilter`. Most options can be configured via `DockerCmdExecFactoryImpl` [programmatically](https://github.com/docker-java/docker-java/wiki#intialize-docker-client-advanced) now. Logging is configurable via [logback](https://github.com/docker-java/docker-java/blob/main/src/test/resources/logback.xml) configuration file in the classpath. All changes diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md new file mode 100644 index 000000000..5072b0864 --- /dev/null +++ b/CONTRIBUTING.md @@ -0,0 +1,58 @@ +# Build with Maven + +#### Prerequisites: + +* Java min 1.8 +* Maven 3 + +Build and run integration tests as follows: + + $ mvn clean install + +If you do not have access to a Docker server or just want to execute the build quickly, you can run the build without the integration tests: + + $ mvn clean install -DskipITs + +By default the docker engine is using local UNIX sockets for communication with the docker CLI so docker-java +client also uses UNIX domain sockets to connect to the docker daemon by default. To make the docker daemon listening on a TCP (http/https) port you have to configure it by setting the DOCKER_OPTS environment variable to something like the following: + + DOCKER_OPTS="-H tcp://127.0.0.1:2375 -H unix:///var/run/docker.sock" + +More details about setting up Docker Engine can be found in the official documentation: https://docs.docker.com/engine/admin/ + +To force docker-java to use TCP (http) configure the following (see [Configuration](https://github.com/docker-java/docker-java#configuration) for details): + + DOCKER_HOST=tcp://127.0.0.1:2375 + +For secure tls (https) communication: + + DOCKER_HOST=tcp://127.0.0.1:2376 + DOCKER_TLS_VERIFY=1 + DOCKER_CERT_PATH=/Users/marcus/.docker/machine/machines/docker-1.11.2 + + +# Code Design + * Model is based on Objects and not primitives that allows nullify requests and have null values for data + that wasn't provided by docker daemon. + * For null safeness findbugs annotations are used. + ** Every method that may return `null` (and we are unsure in any fields as docker daemon may change something) + should be annotated with `@CheckForNull` return qualifier from `javax.annotation` package. + ** Methods that can't return `null` must be annotated with `@Nonnull`. + ** The same for Arguments. + ** `@Nullable` must be used only for changing inherited (other typed) qualifier. + * Setters in builder style must be prefixed with `withXX`. + * All classes should provide `toString()` `equals()` and `hashCode()` defined methods. + * Javadocs + ** Provide full information on field: + *** For models define API version with `@since {@link RemoteApiVersion#VERSION_1_X}`. + ** getters/setters should refernce to field `@see #$field`. + * If it is `Serializable` it shall have a `serialVersionUID` field. Unless code has shipped to users, the initial value of the `serialVersionUID` field shall be `1L`. + +# Coding style + * Some initial styling already enforced with checkstyle. Please aim for consistency with the existing code. + +# Testing + * Unit tests for serder (serialization-deserialization). + * Integration tests for commands. + * If model object has builders, then fill it with data and compare by `equals()` with expected response + from docker daemon. If failed, then some fields mappings are wrong. \ No newline at end of file diff --git a/README.md b/README.md index da474ff85..b1fa9c89e 100644 --- a/README.md +++ b/README.md @@ -1,140 +1,9 @@ -[![Maven Central](https://img.shields.io/maven-central/v/com.github.docker-java/docker-java.svg)]() -[![Bintray](https://api.bintray.com/packages/kostyasha/maven/com.github.docker-java%3Adocker-java/images/download.svg)](https://bintray.com/kostyasha/maven/com.github.docker-java%3Adocker-java/_latestVersion) -[![Reference Status](https://www.versioneye.com/java/com.github.docker-java:docker-java/reference_badge.svg?style=flat)](https://www.versioneye.com/java/com.github.docker-java:docker-java/references) -[![Build Status](https://travis-ci.org/docker-java/docker-java.svg?branch=master)](https://travis-ci.org/docker-java/docker-java) -[![Coverity Scan Build Status](https://scan.coverity.com/projects/9177/badge.svg?flat=1)](https://scan.coverity.com/projects/9177) -[![codecov.io](http://codecov.io/github/docker-java/docker-java/coverage.svg?branch=master)](http://codecov.io/github/docker-java/docker-java?branch=master) -[![License](http://img.shields.io/:license-apache-blue.svg?style=flat)](https://github.com/docker-java/docker-java/blob/master/LICENSE) - - +[![Join the chat at https://gitter.im/docker-java/docker-java](https://badges.gitter.im/docker-java/docker-java.svg)](https://gitter.im/docker-java/docker-java?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge) +[![Maven Central](https://img.shields.io/maven-central/v/com.github.docker-java/docker-java.svg)](https://mvnrepository.com/artifact/com.github.docker-java/docker-java) +[![codecov.io](http://codecov.io/github/docker-java/docker-java/coverage.svg?branch=main)](http://codecov.io/github/docker-java/docker-java?branch=master) +[![License](http://img.shields.io/:license-apache-blue.svg?style=flat)](https://github.com/docker-java/docker-java/blob/main/LICENSE) # docker-java Java API client for [Docker](http://docs.docker.io/ "Docker") -The current implementation is based on Jersey 2.x and therefore classpath incompatible with older Jersey 1.x dependent libraries! - -Developer forum for [docker-java](https://groups.google.com/forum/?#!forum/docker-java-dev "docker-java") - -[Changelog](https://github.com/docker-java/docker-java/blob/master/CHANGELOG.md)
-[Wiki](https://github.com/docker-java/docker-java/wiki) - -## Build with Maven - -###### Prerequisites: - -* Java min 1.7 -* Maven 3 - -Build and run integration tests as follows: - - $ mvn clean install - -If you do not have access to a Docker server or just want to execute the build quickly, you can run the build without the integration tests: - - $ mvn clean install -DskipITs - -By default the docker engine is using local UNIX sockets for communication with the docker CLI so docker-java -client also uses UNIX domain sockets to connect to the docker daemon by default. To make the docker daemon listening on a TCP (http/https) port you have to configure it by setting the DOCKER_OPTS environment variable to something like the following: - - DOCKER_OPTS="-H tcp://127.0.0.1:2375 -H unix:///var/run/docker.sock" - -More details about setting up Docker server can be found in official documentation: http://docs.docker.io/en/latest/use/basics/ - -To force docker-java to use TCP (http) configure the following (see [Configuration](https://github.com/docker-java/docker-java#configuration) for details): - - DOCKER_HOST=tcp://127.0.0.1:2375 - -For secure tls (https) communication: - - DOCKER_HOST=tcp://127.0.0.1:2376 - DOCKER_TLS_VERIFY=1 - DOCKER_CERT_PATH=/Users/marcus/.docker/machine/machines/docker-1.11.2 - -### Latest release version -Supports a subset of the Docker Remote API [v1.23](https://github.com/docker/docker/blob/master/docs/reference/api/docker_remote_api_v1.23.md), Docker Server version 1.11.x - - - com.github.docker-java - docker-java - 3.0.1 - - -### Latest development version -Supports a subset of the Docker Remote API [v1.23](https://github.com/docker/docker/blob/master/docs/reference/api/docker_remote_api_v1.23.md), Docker Server version 1.11.x - -You can find the latest development version including javadoc and source files on [Sonatypes OSS repository](https://oss.sonatype.org/content/groups/public/com/github/docker-java/docker-java/). - - - com.github.docker-java - docker-java - 3.0.2-SNAPSHOT - - - -## Documentation - -For code examples, please look at the [Wiki](https://github.com/docker-java/docker-java/wiki) or [Test cases](https://github.com/docker-java/docker-java/tree/master/src/test/java/com/github/dockerjava/core/command "Test cases") - -## Configuration - -There are a couple of configuration items, all of which have sensible defaults: - -* `DOCKER_HOST` The Docker Host URL, e.g. `tcp://localhost:2376` or `unix:///var/run/docker.sock` -* `DOCKER_TLS_VERIFY` enable/disable TLS verification (switch between `http` and `https` protocol) -* `DOCKER_CERT_PATH` Path to the certificates needed for TLS verification -* `DOCKER_CONFIG` Path for additional docker configuration files (like `.dockercfg`) -* `api.version` The API version, e.g. `1.23`. -* `registry.url` Your registry's address. -* `registry.username` Your registry username (required to push containers). -* `registry.password` Your registry password. -* `registry.email` Your registry email. - -There are three ways to configure, in descending order of precedence: - -#### Programmatic: -In your application, e.g. - - DockerClientConfig config = DefaultDockerClientConfig.createDefaultConfigBuilder() - .withDockerHost("tcp://my-docker-host.tld:2376") - .withDockerTlsVerify(true) - .withDockerCertPath("/home/user/.docker/certs") - .withDockerConfig("/home/user/.docker") - .withApiVersion("1.23") - .withRegistryUrl("https://index.docker.io/v1/") - .withRegistryUsername("dockeruser") - .withRegistryPassword("ilovedocker") - .withRegistryEmail("dockeruser@github.com") - .build(); - DockerClient docker = DockerClientBuilder.getInstance(config).build(); - -#### Properties (docker-java.properties) - - DOCKER_HOST=tcp://localhost:2376 - DOCKER_TLS_VERIFY=1 - DOCKER_CERT_PATH=/home/user/.docker/certs - DOCKER_CONFIG=/home/user/.docker - api.version=1.23 - registry.url=https://index.docker.io/v1/ - registry.username=dockeruser - registry.password=ilovedocker - registry.email=dockeruser@github.com - -##### System Properties: - - java -DDOCKER_HOST=tcp://localhost:2375 -Dregistry.username=dockeruser pkg.Main - -##### System Environment - - export DOCKER_HOST=tcp://localhost:2376 - export DOCKER_TLS_VERIFY=1 - export DOCKER_CERT_PATH=/home/user/.docker/certs - export DOCKER_CONFIG=/home/user/.docker - -##### File System - -In `$HOME/.docker-java.properties` - -##### Class Path - -In the class path at `/docker-java.properties` - +# [Read the documentation here](docs/README.md) diff --git a/circle.sh b/circle.sh index c84ca3fe6..b5b7cdbb0 100755 --- a/circle.sh +++ b/circle.sh @@ -6,7 +6,7 @@ case "$1" in mkdir .docker cp $CIRCLE_PROJECT_REPONAME/etc/certs/* .docker - # configure docker deamon to use SSL and provide the path to the certificates + # configure docker daemon to use SSL and provide the path to the certificates docker_opts='DOCKER_OPTS="$DOCKER_OPTS -H tcp://127.0.0.1:2376 --tlsverify --tlscacert='$HOME'/.docker/ca.pem --tlscert='$HOME'/.docker/server-cert.pem --tlskey='$HOME'/.docker/server-key.pem"' sudo sh -c "echo '$docker_opts' >> /etc/default/docker" diff --git a/docker-java-api/pom.xml b/docker-java-api/pom.xml new file mode 100644 index 000000000..dda682ab1 --- /dev/null +++ b/docker-java-api/pom.xml @@ -0,0 +1,102 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java-api + jar + + docker-java-api + https://github.com/docker-java/docker-java + Java API Client for Docker + + + com.github.dockerjava.api + + + + + com.fasterxml.jackson.core + jackson-annotations + 2.20 + + + + org.slf4j + slf4j-api + ${slf4j-api.version} + + + + com.google.code.findbugs + annotations + 3.0.1u2 + provided + + + + org.projectlombok + lombok + 1.18.38 + provided + + + + + org.junit.jupiter + junit-jupiter + 5.13.4 + test + + + + com.tngtech.archunit + archunit-junit5 + 1.4.1 + test + + + + com.tngtech.archunit + archunit + 0.18.0 + test + + + + + + + org.apache.felix + maven-bundle-plugin + true + + + com.github.dockerjava.api.* + + + + + com.github.siom79.japicmp + japicmp-maven-plugin + + + + com.github.dockerjava.api.command.UpdateContainerCmd#getCpuPeriod() + com.github.dockerjava.api.command.UpdateContainerCmd#withCpuPeriod(java.lang.Integer) + com.github.dockerjava.api.command.UpdateContainerCmd#getCpuQuota() + com.github.dockerjava.api.command.UpdateContainerCmd#withCpuQuota(java.lang.Integer) + com.github.dockerjava.api.command.InspectContainerResponse#getSizeRootFs() + com.github.dockerjava.api.command.InspectContainerResponse#getSizeRw() + + + + + + + diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/DockerClient.java b/docker-java-api/src/main/java/com/github/dockerjava/api/DockerClient.java new file mode 100644 index 000000000..e5f57e1bb --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/DockerClient.java @@ -0,0 +1,500 @@ +package com.github.dockerjava.api; + +import com.github.dockerjava.api.command.AttachContainerCmd; +import com.github.dockerjava.api.command.AuthCmd; +import com.github.dockerjava.api.command.BuildImageCmd; +import com.github.dockerjava.api.command.CommitCmd; +import com.github.dockerjava.api.command.ConnectToNetworkCmd; +import com.github.dockerjava.api.command.ContainerDiffCmd; +import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; +import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; +import com.github.dockerjava.api.command.CopyFileFromContainerCmd; +import com.github.dockerjava.api.command.CreateConfigCmd; +import com.github.dockerjava.api.command.CreateContainerCmd; +import com.github.dockerjava.api.command.CreateImageCmd; +import com.github.dockerjava.api.command.CreateNetworkCmd; +import com.github.dockerjava.api.command.CreateSecretCmd; +import com.github.dockerjava.api.command.CreateServiceCmd; +import com.github.dockerjava.api.command.CreateVolumeCmd; +import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; +import com.github.dockerjava.api.command.EventsCmd; +import com.github.dockerjava.api.command.ExecCreateCmd; +import com.github.dockerjava.api.command.ExecStartCmd; +import com.github.dockerjava.api.command.InfoCmd; +import com.github.dockerjava.api.command.InitializeSwarmCmd; +import com.github.dockerjava.api.command.InspectConfigCmd; +import com.github.dockerjava.api.command.InspectContainerCmd; +import com.github.dockerjava.api.command.InspectExecCmd; +import com.github.dockerjava.api.command.InspectImageCmd; +import com.github.dockerjava.api.command.InspectNetworkCmd; +import com.github.dockerjava.api.command.InspectServiceCmd; +import com.github.dockerjava.api.command.InspectSwarmCmd; +import com.github.dockerjava.api.command.InspectVolumeCmd; +import com.github.dockerjava.api.command.JoinSwarmCmd; +import com.github.dockerjava.api.command.KillContainerCmd; +import com.github.dockerjava.api.command.LeaveSwarmCmd; +import com.github.dockerjava.api.command.ListConfigsCmd; +import com.github.dockerjava.api.command.ListContainersCmd; +import com.github.dockerjava.api.command.ListImagesCmd; +import com.github.dockerjava.api.command.ListNetworksCmd; +import com.github.dockerjava.api.command.ListSecretsCmd; +import com.github.dockerjava.api.command.ListServicesCmd; +import com.github.dockerjava.api.command.ListSwarmNodesCmd; +import com.github.dockerjava.api.command.ListTasksCmd; +import com.github.dockerjava.api.command.ListVolumesCmd; +import com.github.dockerjava.api.command.LoadImageAsyncCmd; +import com.github.dockerjava.api.command.LoadImageCmd; +import com.github.dockerjava.api.command.LogContainerCmd; +import com.github.dockerjava.api.command.LogSwarmObjectCmd; +import com.github.dockerjava.api.command.PauseContainerCmd; +import com.github.dockerjava.api.command.PingCmd; +import com.github.dockerjava.api.command.PruneCmd; +import com.github.dockerjava.api.command.PullImageCmd; +import com.github.dockerjava.api.command.PushImageCmd; +import com.github.dockerjava.api.command.RemoveConfigCmd; +import com.github.dockerjava.api.command.RemoveContainerCmd; +import com.github.dockerjava.api.command.RemoveImageCmd; +import com.github.dockerjava.api.command.RemoveNetworkCmd; +import com.github.dockerjava.api.command.RemoveSecretCmd; +import com.github.dockerjava.api.command.RemoveServiceCmd; +import com.github.dockerjava.api.command.RemoveSwarmNodeCmd; +import com.github.dockerjava.api.command.RemoveVolumeCmd; +import com.github.dockerjava.api.command.RenameContainerCmd; +import com.github.dockerjava.api.command.ResizeContainerCmd; +import com.github.dockerjava.api.command.ResizeExecCmd; +import com.github.dockerjava.api.command.RestartContainerCmd; +import com.github.dockerjava.api.command.SaveImageCmd; +import com.github.dockerjava.api.command.SaveImagesCmd; +import com.github.dockerjava.api.command.SearchImagesCmd; +import com.github.dockerjava.api.command.StartContainerCmd; +import com.github.dockerjava.api.command.StatsCmd; +import com.github.dockerjava.api.command.StopContainerCmd; +import com.github.dockerjava.api.command.TagImageCmd; +import com.github.dockerjava.api.command.TopContainerCmd; +import com.github.dockerjava.api.command.UnpauseContainerCmd; +import com.github.dockerjava.api.command.UpdateContainerCmd; +import com.github.dockerjava.api.command.UpdateServiceCmd; +import com.github.dockerjava.api.command.UpdateSwarmCmd; +import com.github.dockerjava.api.command.UpdateSwarmNodeCmd; +import com.github.dockerjava.api.command.VersionCmd; +import com.github.dockerjava.api.command.WaitContainerCmd; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.Identifier; +import com.github.dockerjava.api.model.PruneType; +import com.github.dockerjava.api.model.SecretSpec; +import com.github.dockerjava.api.model.ServiceSpec; +import com.github.dockerjava.api.model.SwarmSpec; + +import javax.annotation.Nonnull; +import java.io.Closeable; +import java.io.File; +import java.io.IOException; +import java.io.InputStream; + +// https://godoc.org/github.com/fsouza/go-dockerclient +public interface DockerClient extends Closeable { + + AuthConfig authConfig() throws DockerException; + + /** + * Authenticate with the server, useful for checking authentication. + */ + AuthCmd authCmd(); + + InfoCmd infoCmd(); + + PingCmd pingCmd(); + + VersionCmd versionCmd(); + + /** + * * IMAGE API * + */ + + PullImageCmd pullImageCmd(@Nonnull String repository); + + PushImageCmd pushImageCmd(@Nonnull String name); + + PushImageCmd pushImageCmd(@Nonnull Identifier identifier); + + CreateImageCmd createImageCmd(@Nonnull String repository, @Nonnull InputStream imageStream); + + /** + * Loads a tarball with a set of images and tags into a Docker repository. + * + * Corresponds to POST /images/load API endpoint. + * + * @param imageStream + * stream of the tarball file + * @return created command + * @since {@link RemoteApiVersion#VERSION_1_7} + */ + LoadImageCmd loadImageCmd(@Nonnull InputStream imageStream); + + LoadImageAsyncCmd loadImageAsyncCmd(@Nonnull InputStream imageStream); + + SearchImagesCmd searchImagesCmd(@Nonnull String term); + + RemoveImageCmd removeImageCmd(@Nonnull String imageId); + + ListImagesCmd listImagesCmd(); + + InspectImageCmd inspectImageCmd(@Nonnull String imageId); + + /** + * @param name + * The name, e.g. "alexec/busybox" or just "busybox" if you want to default. Not null. + */ + SaveImageCmd saveImageCmd(@Nonnull String name); + + /** + * Command to download multiple images at once. + * @return command (builder) + */ + SaveImagesCmd saveImagesCmd(); + + /** + * * CONTAINER API * + */ + + ListContainersCmd listContainersCmd(); + + CreateContainerCmd createContainerCmd(@Nonnull String image); + + /** + * Creates a new {@link StartContainerCmd} for the container with the given ID. The command can then be further customized by using + * builder methods on it like {@link StartContainerCmd#withDns(String...)}. + *

+ * If you customize the command, any existing configuration of the target container will get reset to its default before applying the + * new configuration. To preserve the existing configuration, use an unconfigured {@link StartContainerCmd}. + *

+ * This command corresponds to the /containers/{id}/start endpoint of the Docker Remote API. + */ + StartContainerCmd startContainerCmd(@Nonnull String containerId); + + ExecCreateCmd execCreateCmd(@Nonnull String containerId); + + ResizeExecCmd resizeExecCmd(@Nonnull String execId); + + InspectContainerCmd inspectContainerCmd(@Nonnull String containerId); + + RemoveContainerCmd removeContainerCmd(@Nonnull String containerId); + + WaitContainerCmd waitContainerCmd(@Nonnull String containerId); + + AttachContainerCmd attachContainerCmd(@Nonnull String containerId); + + ExecStartCmd execStartCmd(@Nonnull String execId); + + InspectExecCmd inspectExecCmd(@Nonnull String execId); + + LogContainerCmd logContainerCmd(@Nonnull String containerId); + + /** + * Copy resource from container to local machine. + * + * @param containerId + * id of the container + * @param resource + * path to container's resource + * @return created command + * @since {@link RemoteApiVersion#VERSION_1_20} + */ + CopyArchiveFromContainerCmd copyArchiveFromContainerCmd(@Nonnull String containerId, @Nonnull String resource); + + /** + * Copy resource from container to local machine. + * + * @param containerId + * id of the container + * @param resource + * path to container's resource + * @return created command + * @see #copyArchiveFromContainerCmd(String, String) + * @deprecated since docker API version 1.20, replaced by {@link #copyArchiveFromContainerCmd(String, String)} + * since 1.24 fails. + */ + @Deprecated + CopyFileFromContainerCmd copyFileFromContainerCmd(@Nonnull String containerId, @Nonnull String resource); + + /** + * Copy archive from local machine to remote container + * + * @param containerId + * id of the container + * @return created command + * @since {@link RemoteApiVersion#VERSION_1_20} + */ + CopyArchiveToContainerCmd copyArchiveToContainerCmd(@Nonnull String containerId); + + ContainerDiffCmd containerDiffCmd(@Nonnull String containerId); + + StopContainerCmd stopContainerCmd(@Nonnull String containerId); + + KillContainerCmd killContainerCmd(@Nonnull String containerId); + + /** + * Update container settings + * + * @param containerId id of the container + * @return command + * @since {@link RemoteApiVersion#VERSION_1_22} + */ + UpdateContainerCmd updateContainerCmd(@Nonnull String containerId); + + /** + * Rename container. + * + * @param containerId id of the container + * @return command + * @since {@link RemoteApiVersion#VERSION_1_17} + */ + RenameContainerCmd renameContainerCmd(@Nonnull String containerId); + + RestartContainerCmd restartContainerCmd(@Nonnull String containerId); + + ResizeContainerCmd resizeContainerCmd(@Nonnull String containerId); + + CommitCmd commitCmd(@Nonnull String containerId); + + BuildImageCmd buildImageCmd(); + + BuildImageCmd buildImageCmd(File dockerFileOrFolder); + + BuildImageCmd buildImageCmd(InputStream tarInputStream); + + TopContainerCmd topContainerCmd(String containerId); + + TagImageCmd tagImageCmd(String imageId, String imageNameWithRepository, String tag); + + PauseContainerCmd pauseContainerCmd(String containerId); + + UnpauseContainerCmd unpauseContainerCmd(String containerId); + + EventsCmd eventsCmd(); + + StatsCmd statsCmd(String containerId); + + CreateVolumeCmd createVolumeCmd(); + + InspectVolumeCmd inspectVolumeCmd(String name); + + RemoveVolumeCmd removeVolumeCmd(String name); + + ListVolumesCmd listVolumesCmd(); + + ListNetworksCmd listNetworksCmd(); + + InspectNetworkCmd inspectNetworkCmd(); + + CreateNetworkCmd createNetworkCmd(); + + RemoveNetworkCmd removeNetworkCmd(@Nonnull String networkId); + + ConnectToNetworkCmd connectToNetworkCmd(); + + DisconnectFromNetworkCmd disconnectFromNetworkCmd(); + + /** + * Enables swarm mode for the docker engine and creates a new swarm cluster + * + * @since 1.24 + * @param swarmSpec the specification for the swarm + * @return the command + */ + InitializeSwarmCmd initializeSwarmCmd(SwarmSpec swarmSpec); + + /** + * Gets information about the swarm the docker engine is currently in + * + * @since 1.24 + * @return the command + */ + InspectSwarmCmd inspectSwarmCmd(); + + /** + * Enables swarm mode for the docker engine and joins an existing swarm cluster + * + * @since 1.24 + * @return the command + */ + JoinSwarmCmd joinSwarmCmd(); + + /** + * Disables swarm node for the docker engine and leaves the swarm cluster + * + * @since 1.24 + * @return the command + */ + LeaveSwarmCmd leaveSwarmCmd(); + + /** + * Updates the swarm specification + * + * @since 1.24 + * @param swarmSpec the specification for the swarm + * @return the command + */ + UpdateSwarmCmd updateSwarmCmd(SwarmSpec swarmSpec); + + /** + * Updates the swarm node + * + * @return the command + * @since 1.24 + */ + UpdateSwarmNodeCmd updateSwarmNodeCmd(); + + /** + * Remove the swarm node + * + * @param swarmNodeId swarmNodeId + * @return the command + * @since 1.24 + */ + RemoveSwarmNodeCmd removeSwarmNodeCmd(String swarmNodeId); + + /** + * List nodes in swarm + * + * @return the command + * @since 1.24 + */ + ListSwarmNodesCmd listSwarmNodesCmd(); + + /** + * Command to list all services in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + * @return command + */ + ListServicesCmd listServicesCmd(); + + /** + * Command to create a service in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + * @param serviceSpec the service specification + * @return command + */ + CreateServiceCmd createServiceCmd(ServiceSpec serviceSpec); + + /** + * Command to inspect a service + * @param serviceId service id or service name + * @return command + */ + InspectServiceCmd inspectServiceCmd(String serviceId); + + /** + * Command to update a service specification + * @param serviceId service id + * @param serviceSpec the new service specification + * @return command + */ + UpdateServiceCmd updateServiceCmd(String serviceId, ServiceSpec serviceSpec); + + /** + * Command to remove a service + * @param serviceId service id or service name + * @return command + */ + RemoveServiceCmd removeServiceCmd(String serviceId); + + /** + * List tasks in the swarm cluster + * + * @return the command + * @since 1.24 + */ + ListTasksCmd listTasksCmd(); + + /** + * Command to get service log + * + * @return the command + * @since 1.29 + */ + LogSwarmObjectCmd logServiceCmd(String serviceId); + + /** + * Command to get task log + * + * @return the command + * @since 1.29 + */ + LogSwarmObjectCmd logTaskCmd(String taskId); + + /** + * Command to delete unused containers/images/networks/volumes + * + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + PruneCmd pruneCmd(PruneType pruneType); + + /** + * Command to list all secrets. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_25} + * @return command + */ + ListSecretsCmd listSecretsCmd(); + + /** + * Command to create a secret in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_25} + * @param secretSpec the secret specification + * @return command + */ + CreateSecretCmd createSecretCmd(SecretSpec secretSpec); + + /** + * Command to remove a secret + * + * @since {@link RemoteApiVersion#VERSION_1_25} + * @param secretId secret id or secret name + * @return command + */ + RemoveSecretCmd removeSecretCmd(String secretId); + + + /** + * Command to list all configs. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_30} + * @return command + */ + ListConfigsCmd listConfigsCmd(); + + /** + * Command to create a config in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_30} + * @return command + */ + CreateConfigCmd createConfigCmd(); + + /** + * Command to inspect a service + * + * @since {@link RemoteApiVersion#VERSION_1_30} + * @param configId config id or config name + * @return command + */ + InspectConfigCmd inspectConfigCmd(String configId); + + /** + * Command to remove a config + * @since {@link RemoteApiVersion#VERSION_1_30} + * @param configId config id or config name + * @return command + */ + RemoveConfigCmd removeConfigCmd(String configId); + + + @Override + void close() throws IOException; + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/DockerClientDelegate.java b/docker-java-api/src/main/java/com/github/dockerjava/api/DockerClientDelegate.java new file mode 100644 index 000000000..5de64641f --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/DockerClientDelegate.java @@ -0,0 +1,527 @@ +package com.github.dockerjava.api; + +import com.github.dockerjava.api.command.AttachContainerCmd; +import com.github.dockerjava.api.command.AuthCmd; +import com.github.dockerjava.api.command.BuildImageCmd; +import com.github.dockerjava.api.command.CommitCmd; +import com.github.dockerjava.api.command.ConnectToNetworkCmd; +import com.github.dockerjava.api.command.ContainerDiffCmd; +import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; +import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; +import com.github.dockerjava.api.command.CopyFileFromContainerCmd; +import com.github.dockerjava.api.command.CreateConfigCmd; +import com.github.dockerjava.api.command.CreateContainerCmd; +import com.github.dockerjava.api.command.CreateImageCmd; +import com.github.dockerjava.api.command.CreateNetworkCmd; +import com.github.dockerjava.api.command.CreateSecretCmd; +import com.github.dockerjava.api.command.CreateServiceCmd; +import com.github.dockerjava.api.command.CreateVolumeCmd; +import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; +import com.github.dockerjava.api.command.EventsCmd; +import com.github.dockerjava.api.command.ExecCreateCmd; +import com.github.dockerjava.api.command.ExecStartCmd; +import com.github.dockerjava.api.command.InfoCmd; +import com.github.dockerjava.api.command.InitializeSwarmCmd; +import com.github.dockerjava.api.command.InspectConfigCmd; +import com.github.dockerjava.api.command.InspectContainerCmd; +import com.github.dockerjava.api.command.InspectExecCmd; +import com.github.dockerjava.api.command.InspectImageCmd; +import com.github.dockerjava.api.command.InspectNetworkCmd; +import com.github.dockerjava.api.command.InspectServiceCmd; +import com.github.dockerjava.api.command.InspectSwarmCmd; +import com.github.dockerjava.api.command.InspectVolumeCmd; +import com.github.dockerjava.api.command.JoinSwarmCmd; +import com.github.dockerjava.api.command.KillContainerCmd; +import com.github.dockerjava.api.command.LeaveSwarmCmd; +import com.github.dockerjava.api.command.ListConfigsCmd; +import com.github.dockerjava.api.command.ListContainersCmd; +import com.github.dockerjava.api.command.ListImagesCmd; +import com.github.dockerjava.api.command.ListNetworksCmd; +import com.github.dockerjava.api.command.ListSecretsCmd; +import com.github.dockerjava.api.command.ListServicesCmd; +import com.github.dockerjava.api.command.ListSwarmNodesCmd; +import com.github.dockerjava.api.command.ListTasksCmd; +import com.github.dockerjava.api.command.ListVolumesCmd; +import com.github.dockerjava.api.command.LoadImageAsyncCmd; +import com.github.dockerjava.api.command.LoadImageCmd; +import com.github.dockerjava.api.command.LogContainerCmd; +import com.github.dockerjava.api.command.LogSwarmObjectCmd; +import com.github.dockerjava.api.command.PauseContainerCmd; +import com.github.dockerjava.api.command.PingCmd; +import com.github.dockerjava.api.command.PruneCmd; +import com.github.dockerjava.api.command.PullImageCmd; +import com.github.dockerjava.api.command.PushImageCmd; +import com.github.dockerjava.api.command.RemoveConfigCmd; +import com.github.dockerjava.api.command.RemoveContainerCmd; +import com.github.dockerjava.api.command.RemoveImageCmd; +import com.github.dockerjava.api.command.RemoveNetworkCmd; +import com.github.dockerjava.api.command.RemoveSecretCmd; +import com.github.dockerjava.api.command.RemoveServiceCmd; +import com.github.dockerjava.api.command.RemoveSwarmNodeCmd; +import com.github.dockerjava.api.command.RemoveVolumeCmd; +import com.github.dockerjava.api.command.RenameContainerCmd; +import com.github.dockerjava.api.command.ResizeContainerCmd; +import com.github.dockerjava.api.command.ResizeExecCmd; +import com.github.dockerjava.api.command.RestartContainerCmd; +import com.github.dockerjava.api.command.SaveImageCmd; +import com.github.dockerjava.api.command.SaveImagesCmd; +import com.github.dockerjava.api.command.SearchImagesCmd; +import com.github.dockerjava.api.command.StartContainerCmd; +import com.github.dockerjava.api.command.StatsCmd; +import com.github.dockerjava.api.command.StopContainerCmd; +import com.github.dockerjava.api.command.TagImageCmd; +import com.github.dockerjava.api.command.TopContainerCmd; +import com.github.dockerjava.api.command.UnpauseContainerCmd; +import com.github.dockerjava.api.command.UpdateContainerCmd; +import com.github.dockerjava.api.command.UpdateServiceCmd; +import com.github.dockerjava.api.command.UpdateSwarmCmd; +import com.github.dockerjava.api.command.UpdateSwarmNodeCmd; +import com.github.dockerjava.api.command.VersionCmd; +import com.github.dockerjava.api.command.WaitContainerCmd; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.Identifier; +import com.github.dockerjava.api.model.PruneType; +import com.github.dockerjava.api.model.SecretSpec; +import com.github.dockerjava.api.model.ServiceSpec; +import com.github.dockerjava.api.model.SwarmSpec; + +import javax.annotation.Nonnull; +import java.io.File; +import java.io.IOException; +import java.io.InputStream; + +/** + * @apiNote implementations MUST override {{@link #getDockerClient()}} + * @implNote We're not using an abstract class here because we want + * Java compiler to force us to implement every {@link DockerClient}'s method, + * especially when new methods are added + */ +@SuppressWarnings("unused") +public class DockerClientDelegate implements DockerClient { + + protected DockerClient getDockerClient() { + throw new IllegalStateException("Implement me!"); + } + + @Override + public AuthConfig authConfig() throws DockerException { + return getDockerClient().authConfig(); + } + + @Override + public AuthCmd authCmd() { + return getDockerClient().authCmd(); + } + + @Override + public InfoCmd infoCmd() { + return getDockerClient().infoCmd(); + } + + @Override + public PingCmd pingCmd() { + return getDockerClient().pingCmd(); + } + + @Override + public VersionCmd versionCmd() { + return getDockerClient().versionCmd(); + } + + @Override + public PullImageCmd pullImageCmd(@Nonnull String repository) { + return getDockerClient().pullImageCmd(repository); + } + + @Override + public PushImageCmd pushImageCmd(@Nonnull String name) { + return getDockerClient().pushImageCmd(name); + } + + @Override + public PushImageCmd pushImageCmd(@Nonnull Identifier identifier) { + return getDockerClient().pushImageCmd(identifier); + } + + @Override + public CreateImageCmd createImageCmd(@Nonnull String repository, @Nonnull InputStream imageStream) { + return getDockerClient().createImageCmd(repository, imageStream); + } + + @Override + public LoadImageCmd loadImageCmd(@Nonnull InputStream imageStream) { + return getDockerClient().loadImageCmd(imageStream); + } + + @Override + public LoadImageAsyncCmd loadImageAsyncCmd(@Nonnull InputStream imageStream) { + return getDockerClient().loadImageAsyncCmd(imageStream); + } + + @Override + public SearchImagesCmd searchImagesCmd(@Nonnull String term) { + return getDockerClient().searchImagesCmd(term); + } + + @Override + public RemoveImageCmd removeImageCmd(@Nonnull String imageId) { + return getDockerClient().removeImageCmd(imageId); + } + + @Override + public ListImagesCmd listImagesCmd() { + return getDockerClient().listImagesCmd(); + } + + @Override + public InspectImageCmd inspectImageCmd(@Nonnull String imageId) { + return getDockerClient().inspectImageCmd(imageId); + } + + @Override + public SaveImageCmd saveImageCmd(@Nonnull String name) { + return getDockerClient().saveImageCmd(name); + } + + @Override + public SaveImagesCmd saveImagesCmd() { + return getDockerClient().saveImagesCmd(); + } + + @Override + public ListContainersCmd listContainersCmd() { + return getDockerClient().listContainersCmd(); + } + + @Override + public CreateContainerCmd createContainerCmd(@Nonnull String image) { + return getDockerClient().createContainerCmd(image); + } + + @Override + public StartContainerCmd startContainerCmd(@Nonnull String containerId) { + return getDockerClient().startContainerCmd(containerId); + } + + @Override + public ExecCreateCmd execCreateCmd(@Nonnull String containerId) { + return getDockerClient().execCreateCmd(containerId); + } + + @Override + public ResizeExecCmd resizeExecCmd(@Nonnull String execId) { + return getDockerClient().resizeExecCmd(execId); + } + + @Override + public InspectContainerCmd inspectContainerCmd(@Nonnull String containerId) { + return getDockerClient().inspectContainerCmd(containerId); + } + + @Override + public RemoveContainerCmd removeContainerCmd(@Nonnull String containerId) { + return getDockerClient().removeContainerCmd(containerId); + } + + @Override + public WaitContainerCmd waitContainerCmd(@Nonnull String containerId) { + return getDockerClient().waitContainerCmd(containerId); + } + + @Override + public AttachContainerCmd attachContainerCmd(@Nonnull String containerId) { + return getDockerClient().attachContainerCmd(containerId); + } + + @Override + public ExecStartCmd execStartCmd(@Nonnull String execId) { + return getDockerClient().execStartCmd(execId); + } + + @Override + public InspectExecCmd inspectExecCmd(@Nonnull String execId) { + return getDockerClient().inspectExecCmd(execId); + } + + @Override + public LogContainerCmd logContainerCmd(@Nonnull String containerId) { + return getDockerClient().logContainerCmd(containerId); + } + + @Override + public CopyArchiveFromContainerCmd copyArchiveFromContainerCmd(@Nonnull String containerId, @Nonnull String resource) { + return getDockerClient().copyArchiveFromContainerCmd(containerId, resource); + } + + @Override + @Deprecated + public CopyFileFromContainerCmd copyFileFromContainerCmd(@Nonnull String containerId, @Nonnull String resource) { + return getDockerClient().copyFileFromContainerCmd(containerId, resource); + } + + @Override + public CopyArchiveToContainerCmd copyArchiveToContainerCmd(@Nonnull String containerId) { + return getDockerClient().copyArchiveToContainerCmd(containerId); + } + + @Override + public ContainerDiffCmd containerDiffCmd(@Nonnull String containerId) { + return getDockerClient().containerDiffCmd(containerId); + } + + @Override + public StopContainerCmd stopContainerCmd(@Nonnull String containerId) { + return getDockerClient().stopContainerCmd(containerId); + } + + @Override + public KillContainerCmd killContainerCmd(@Nonnull String containerId) { + return getDockerClient().killContainerCmd(containerId); + } + + @Override + public UpdateContainerCmd updateContainerCmd(@Nonnull String containerId) { + return getDockerClient().updateContainerCmd(containerId); + } + + @Override + public RenameContainerCmd renameContainerCmd(@Nonnull String containerId) { + return getDockerClient().renameContainerCmd(containerId); + } + + @Override + public RestartContainerCmd restartContainerCmd(@Nonnull String containerId) { + return getDockerClient().restartContainerCmd(containerId); + } + + @Override + public ResizeContainerCmd resizeContainerCmd(@Nonnull String containerId) { + return getDockerClient().resizeContainerCmd(containerId); + } + + @Override + public CommitCmd commitCmd(@Nonnull String containerId) { + return getDockerClient().commitCmd(containerId); + } + + @Override + public BuildImageCmd buildImageCmd() { + return getDockerClient().buildImageCmd(); + } + + @Override + public BuildImageCmd buildImageCmd(File dockerFileOrFolder) { + return getDockerClient().buildImageCmd(dockerFileOrFolder); + } + + @Override + public BuildImageCmd buildImageCmd(InputStream tarInputStream) { + return getDockerClient().buildImageCmd(tarInputStream); + } + + @Override + public TopContainerCmd topContainerCmd(String containerId) { + return getDockerClient().topContainerCmd(containerId); + } + + @Override + public TagImageCmd tagImageCmd(String imageId, String imageNameWithRepository, String tag) { + return getDockerClient().tagImageCmd(imageId, imageNameWithRepository, tag); + } + + @Override + public PauseContainerCmd pauseContainerCmd(String containerId) { + return getDockerClient().pauseContainerCmd(containerId); + } + + @Override + public UnpauseContainerCmd unpauseContainerCmd(String containerId) { + return getDockerClient().unpauseContainerCmd(containerId); + } + + @Override + public EventsCmd eventsCmd() { + return getDockerClient().eventsCmd(); + } + + @Override + public StatsCmd statsCmd(String containerId) { + return getDockerClient().statsCmd(containerId); + } + + @Override + public CreateVolumeCmd createVolumeCmd() { + return getDockerClient().createVolumeCmd(); + } + + @Override + public InspectVolumeCmd inspectVolumeCmd(String name) { + return getDockerClient().inspectVolumeCmd(name); + } + + @Override + public RemoveVolumeCmd removeVolumeCmd(String name) { + return getDockerClient().removeVolumeCmd(name); + } + + @Override + public ListVolumesCmd listVolumesCmd() { + return getDockerClient().listVolumesCmd(); + } + + @Override + public ListNetworksCmd listNetworksCmd() { + return getDockerClient().listNetworksCmd(); + } + + @Override + public InspectNetworkCmd inspectNetworkCmd() { + return getDockerClient().inspectNetworkCmd(); + } + + @Override + public CreateNetworkCmd createNetworkCmd() { + return getDockerClient().createNetworkCmd(); + } + + @Override + public RemoveNetworkCmd removeNetworkCmd(@Nonnull String networkId) { + return getDockerClient().removeNetworkCmd(networkId); + } + + @Override + public ConnectToNetworkCmd connectToNetworkCmd() { + return getDockerClient().connectToNetworkCmd(); + } + + @Override + public DisconnectFromNetworkCmd disconnectFromNetworkCmd() { + return getDockerClient().disconnectFromNetworkCmd(); + } + + @Override + public InitializeSwarmCmd initializeSwarmCmd(SwarmSpec swarmSpec) { + return getDockerClient().initializeSwarmCmd(swarmSpec); + } + + @Override + public InspectSwarmCmd inspectSwarmCmd() { + return getDockerClient().inspectSwarmCmd(); + } + + @Override + public JoinSwarmCmd joinSwarmCmd() { + return getDockerClient().joinSwarmCmd(); + } + + @Override + public LeaveSwarmCmd leaveSwarmCmd() { + return getDockerClient().leaveSwarmCmd(); + } + + @Override + public UpdateSwarmCmd updateSwarmCmd(SwarmSpec swarmSpec) { + return getDockerClient().updateSwarmCmd(swarmSpec); + } + + @Override + public UpdateSwarmNodeCmd updateSwarmNodeCmd() { + return getDockerClient().updateSwarmNodeCmd(); + } + + @Override + public RemoveSwarmNodeCmd removeSwarmNodeCmd(String swarmNodeId) { + return getDockerClient().removeSwarmNodeCmd(swarmNodeId); + } + + @Override + public ListSwarmNodesCmd listSwarmNodesCmd() { + return getDockerClient().listSwarmNodesCmd(); + } + + @Override + public ListServicesCmd listServicesCmd() { + return getDockerClient().listServicesCmd(); + } + + @Override + public CreateServiceCmd createServiceCmd(ServiceSpec serviceSpec) { + return getDockerClient().createServiceCmd(serviceSpec); + } + + @Override + public InspectServiceCmd inspectServiceCmd(String serviceId) { + return getDockerClient().inspectServiceCmd(serviceId); + } + + @Override + public UpdateServiceCmd updateServiceCmd(String serviceId, ServiceSpec serviceSpec) { + return getDockerClient().updateServiceCmd(serviceId, serviceSpec); + } + + @Override + public RemoveServiceCmd removeServiceCmd(String serviceId) { + return getDockerClient().removeServiceCmd(serviceId); + } + + @Override + public ListTasksCmd listTasksCmd() { + return getDockerClient().listTasksCmd(); + } + + @Override + public LogSwarmObjectCmd logServiceCmd(String serviceId) { + return getDockerClient().logServiceCmd(serviceId); + } + + @Override + public LogSwarmObjectCmd logTaskCmd(String taskId) { + return getDockerClient().logTaskCmd(taskId); + } + + @Override + public PruneCmd pruneCmd(PruneType pruneType) { + return getDockerClient().pruneCmd(pruneType); + } + + @Override + public ListSecretsCmd listSecretsCmd() { + return getDockerClient().listSecretsCmd(); + } + + @Override + public CreateSecretCmd createSecretCmd(SecretSpec secretSpec) { + return getDockerClient().createSecretCmd(secretSpec); + } + + @Override + public RemoveSecretCmd removeSecretCmd(String secretId) { + return getDockerClient().removeSecretCmd(secretId); + } + + @Override + public ListConfigsCmd listConfigsCmd() { + return getDockerClient().listConfigsCmd(); + } + + @Override + public CreateConfigCmd createConfigCmd() { + return getDockerClient().createConfigCmd(); + } + + @Override + public InspectConfigCmd inspectConfigCmd(String configId) { + return getDockerClient().inspectConfigCmd(configId); + } + + @Override + public RemoveConfigCmd removeConfigCmd(String configId) { + return getDockerClient().removeConfigCmd(configId); + } + + @Override + public void close() throws IOException { + getDockerClient().close(); + } +} diff --git a/src/main/java/com/github/dockerjava/api/async/ResultCallback.java b/docker-java-api/src/main/java/com/github/dockerjava/api/async/ResultCallback.java similarity index 80% rename from src/main/java/com/github/dockerjava/api/async/ResultCallback.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/async/ResultCallback.java index 5b9fdb81c..6a244d620 100644 --- a/src/main/java/com/github/dockerjava/api/async/ResultCallback.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/async/ResultCallback.java @@ -6,6 +6,14 @@ * Result callback */ public interface ResultCallback extends Closeable { + + class Adapter extends ResultCallbackTemplate, A_RES_T> { + @Override + public void onNext(A_RES_T object) { + + } + } + /** * Called when the async processing starts respectively when the response arrives from the server. The passed {@link Closeable} can be * used to close/interrupt the processing. diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/async/ResultCallbackTemplate.java b/docker-java-api/src/main/java/com/github/dockerjava/api/async/ResultCallbackTemplate.java new file mode 100644 index 000000000..911e67826 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/async/ResultCallbackTemplate.java @@ -0,0 +1,161 @@ +/* + * Created on 16.06.2015 + */ +package com.github.dockerjava.api.async; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.Closeable; +import java.io.IOException; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.TimeUnit; + +/** + * Abstract template implementation of {@link ResultCallback} + * + * @author Marcus Linke + * + */ +public abstract class ResultCallbackTemplate, A_RES_T> implements + ResultCallback { + + private static final Logger LOGGER = LoggerFactory.getLogger(ResultCallbackTemplate.class); + + private final CountDownLatch started = new CountDownLatch(1); + + private final CountDownLatch completed = new CountDownLatch(1); + + private Closeable stream; + + private boolean closed = false; + + private Throwable firstError = null; + + @Override + public void onStart(Closeable stream) { + this.stream = stream; + this.closed = false; + started.countDown(); + } + + @Override + public void onError(Throwable throwable) { + + if (closed) return; + + if (this.firstError == null) { + this.firstError = throwable; + } + + try { + LOGGER.error("Error during callback", throwable); + } finally { + try { + close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + } + + @Override + public void onComplete() { + try { + close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Override + public void close() throws IOException { + if (!closed) { + closed = true; + try { + if (stream != null) { + stream.close(); + } + } finally { + completed.countDown(); + } + } + } + + /** + * Blocks until {@link ResultCallback#onComplete()} was called + */ + @SuppressWarnings("unchecked") + public RC_T awaitCompletion() throws InterruptedException { + try { + completed.await(); + // eventually (re)throws RuntimeException + throwFirstError(); + return (RC_T) this; + } finally { + try { + close(); + } catch (IOException e) { + LOGGER.debug("Failed to close", e); + } + } + } + + /** + * Blocks until {@link ResultCallback#onComplete()} was called or the given timeout occurs + * @return {@code true} if completed and {@code false} if the waiting time elapsed + * before {@link ResultCallback#onComplete()} was called. + */ + public boolean awaitCompletion(long timeout, TimeUnit timeUnit) throws InterruptedException { + try { + boolean result = completed.await(timeout, timeUnit); + throwFirstError(); + return result; + } finally { + try { + close(); + } catch (IOException e) { + LOGGER.debug("Failed to close", e); + } + } + } + + /** + * Blocks until {@link ResultCallback#onStart(Closeable)} was called. + * {@link ResultCallback#onStart(Closeable)} is called when the request was processed on the server + * side and the response is incoming. + */ + @SuppressWarnings("unchecked") + public RC_T awaitStarted() throws InterruptedException { + started.await(); + return (RC_T) this; + } + + /** + * Blocks until {@link ResultCallback#onStart(Closeable)} was called or the given timeout occurs. + * {@link ResultCallback#onStart(Closeable)} is called when the request was processed on the server side + * and the response is incoming. + * @return {@code true} if started and {@code false} if the waiting time elapsed + * before {@link ResultCallback#onStart(Closeable)} was called. + */ + public boolean awaitStarted(long timeout, TimeUnit timeUnit) throws InterruptedException { + return started.await(timeout, timeUnit); + } + + /** + * Throws the first occurred error as a runtime exception + * @throws com.github.dockerjava.api.exception.DockerException The first docker based Error + * @throws RuntimeException on any other occurred error + */ + protected void throwFirstError() { + if (firstError != null) { + if (firstError instanceof Error) { + throw (Error) firstError; + } + if (firstError instanceof RuntimeException) { + throw (RuntimeException) firstError; + } + throw new RuntimeException(firstError); + } + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/AsyncDockerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/AsyncDockerCmd.java new file mode 100644 index 000000000..b2f287cc2 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/AsyncDockerCmd.java @@ -0,0 +1,22 @@ +/* + * Created on 17.06.2015 + */ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.async.ResultCallbackTemplate; + +/** + * + * + * @author Marcus Linke + * + */ +public interface AsyncDockerCmd, A_RES_T> extends DockerCmd { + + > T exec(T resultCallback); + + default ResultCallbackTemplate start() { + return exec(new ResultCallback.Adapter<>()); + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/AttachContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/AttachContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/AttachContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/AttachContainerCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/AuthCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/AuthCmd.java similarity index 91% rename from src/main/java/com/github/dockerjava/api/command/AuthCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/AuthCmd.java index a92b1593b..f9ffbf859 100644 --- a/src/main/java/com/github/dockerjava/api/command/AuthCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/AuthCmd.java @@ -1,12 +1,11 @@ package com.github.dockerjava.api.command; -import javax.annotation.CheckForNull; -import javax.annotation.Nonnull; - import com.github.dockerjava.api.exception.UnauthorizedException; import com.github.dockerjava.api.model.AuthConfig; import com.github.dockerjava.api.model.AuthResponse; +import javax.annotation.CheckForNull; + /** * * Authenticate with the server, useful for checking authentication. @@ -17,7 +16,7 @@ public interface AuthCmd extends SyncDockerCmd { @CheckForNull AuthConfig getAuthConfig(); - AuthCmd withAuthConfig(@Nonnull AuthConfig authConfig); + AuthCmd withAuthConfig(AuthConfig authConfig); /** * @return The status. Based on it's value you may mean you need to authorise your account, e.g.: "Account created. Please see the diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/BuildImageCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/BuildImageCmd.java new file mode 100644 index 000000000..1b7b76a67 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/BuildImageCmd.java @@ -0,0 +1,244 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.AuthConfigurations; +import com.github.dockerjava.api.model.BuildResponseItem; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import java.io.File; +import java.io.InputStream; +import java.net.URI; +import java.util.Map; +import java.util.Set; + +/** + * Build an image from Dockerfile. + *

+ * TODO: http://docs.docker.com/reference/builder/#dockerignore + * + * @see build-image-from-a-dockerfile + */ +public interface BuildImageCmd extends AsyncDockerCmd { + + // lib specific + + @CheckForNull + InputStream getTarInputStream(); + + @CheckForNull + AuthConfigurations getBuildAuthConfigs(); + + // getters + + /** + * "t" in API + * + * @deprecated since docker API version 1.21 there can be multiple tags + * specified so use {@link #getTags()} + */ + @CheckForNull + @Deprecated + String getTag(); + + /** + * Multple "t" tags. + * + * @since {@link RemoteApiVersion#VERSION_1_21} + */ + @CheckForNull + Set getTags(); + + /** + * "Cache-from" in API + */ + @CheckForNull + Set getCacheFrom(); + + /** + * "remote" in API + */ + @CheckForNull + URI getRemote(); + + /** + * "nocache" in API + */ + @CheckForNull + Boolean hasNoCacheEnabled(); + + /** + * "rm" in API + */ + @CheckForNull + Boolean hasRemoveEnabled(); + + /** + * "forcerm" in API + */ + @CheckForNull + Boolean isForcerm(); + + /** + * "q" in API + */ + @CheckForNull + Boolean isQuiet(); + + /** + * "pull" in API + */ + @CheckForNull + Boolean hasPullEnabled(); + + @CheckForNull + String getPathToDockerfile(); + + @CheckForNull + Long getMemory(); + + @CheckForNull + Long getMemswap(); + + @CheckForNull + String getCpushares(); + + @CheckForNull + String getCpusetcpus(); + + /** + * @since {@link RemoteApiVersion#VERSION_1_21} + */ + @CheckForNull + Map getBuildArgs(); + + /** + * @since {@link RemoteApiVersion#VERSION_1_22} + */ + @CheckForNull + Long getShmsize(); + + /** + * @since {@link RemoteApiVersion#VERSION_1_23} + */ + @CheckForNull + Map getLabels(); + + /** + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + @CheckForNull + String getNetworkMode(); + + /** + * "platform" in API + * + * @since {@link RemoteApiVersion#VERSION_1_32} + */ + @CheckForNull + String getPlatform(); + + /** + * @since {@link RemoteApiVersion#VERSION_1_38} + */ + @CheckForNull + String getTarget(); + + /** + * @since {@link RemoteApiVersion#VERSION_1_28} + */ + @CheckForNull + Set getExtraHosts(); + + // setters + + /** + * @deprecated since docker API version 1.21 there can be multiple tags + * specified so use {@link BuildImageCmd#withTags(java.util.Set)} + */ + @Deprecated + BuildImageCmd withTag(String tag); + + BuildImageCmd withTags(Set tags); + + /* + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + BuildImageCmd withCacheFrom(Set cacheFrom); + + BuildImageCmd withRemote(URI remote); + + BuildImageCmd withBaseDirectory(File baseDirectory); + + BuildImageCmd withDockerfile(File dockerfile); + + BuildImageCmd withDockerfilePath(String dockerfilePath); + + BuildImageCmd withNoCache(Boolean noCache); + + BuildImageCmd withRemove(Boolean rm); + + BuildImageCmd withForcerm(Boolean forcerm); + + BuildImageCmd withQuiet(Boolean quiet); + + BuildImageCmd withPull(Boolean pull); + + BuildImageCmd withMemory(Long memory); + + BuildImageCmd withMemswap(Long memswap); + + BuildImageCmd withCpushares(String cpushares); + + BuildImageCmd withCpusetcpus(String cpusetcpus); + + /** + * @since {@link RemoteApiVersion#VERSION_1_21} + */ + BuildImageCmd withBuildArg(String key, String value); + + // setters lib specific + + BuildImageCmd withBuildAuthConfigs(AuthConfigurations authConfig); + + BuildImageCmd withTarInputStream(@Nonnull InputStream tarInputStream); + + /** + * @since {@link RemoteApiVersion#VERSION_1_22} + */ + BuildImageCmd withShmsize(Long shmsize); + + /** + * @since {@link RemoteApiVersion#VERSION_1_23} + */ + BuildImageCmd withLabels(Map labels); + + /** + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + BuildImageCmd withNetworkMode(String networkMode); + + /** + *@since {@link RemoteApiVersion#VERSION_1_32} + */ + BuildImageCmd withPlatform(String platform); + + /** + * @since {@link RemoteApiVersion#VERSION_1_38} + */ + BuildImageCmd withTarget(String target); + + /** + * @since {@link RemoteApiVersion#VERSION_1_28} + */ + BuildImageCmd withExtraHosts(Set extraHosts); + + @Override + default BuildImageResultCallback start() { + return exec(new BuildImageResultCallback()); + } + + interface Exec extends DockerCmdAsyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/BuildImageResultCallback.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/BuildImageResultCallback.java new file mode 100644 index 000000000..9db21a6c4 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/BuildImageResultCallback.java @@ -0,0 +1,80 @@ +/* + * Created on 21.07.2015 + */ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.async.ResultCallbackTemplate; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.model.BuildResponseItem; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.concurrent.TimeUnit; + +/** + * + * @author Marcus Linke + * + */ +public class BuildImageResultCallback extends ResultCallbackTemplate { + + private static final Logger LOGGER = LoggerFactory.getLogger(BuildImageResultCallback.class); + + private String imageId; + + private String error; + + @Override + public void onNext(BuildResponseItem item) { + if (item.isBuildSuccessIndicated()) { + this.imageId = item.getImageId(); + } else if (item.isErrorIndicated()) { + this.error = item.getError(); + } + LOGGER.debug("{}", item); + } + + /** + * Awaits the image id from the response stream. + * + * @throws DockerClientException + * if the build fails. + */ + public String awaitImageId() { + try { + awaitCompletion(); + } catch (InterruptedException e) { + throw new DockerClientException("", e); + } + + return getImageId(); + } + + /** + * Awaits the image id from the response stream. + * + * @throws DockerClientException + * if the build fails or the timeout occurs. + */ + public String awaitImageId(long timeout, TimeUnit timeUnit) { + try { + awaitCompletion(timeout, timeUnit); + } catch (InterruptedException e) { + throw new DockerClientException("Awaiting image id interrupted: ", e); + } + + return getImageId(); + } + + private String getImageId() { + if (error != null) { + throw new DockerClientException("Could not build image: " + error); + } + + if (imageId != null) { + return imageId; + } + + throw new DockerClientException("Could not build image"); + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/CommitCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CommitCmd.java similarity index 93% rename from src/main/java/com/github/dockerjava/api/command/CommitCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/CommitCmd.java index c5da5bab6..a182751b8 100644 --- a/src/main/java/com/github/dockerjava/api/command/CommitCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CommitCmd.java @@ -3,6 +3,8 @@ import javax.annotation.CheckForNull; import javax.annotation.Nonnull; +import java.util.Map; + import com.github.dockerjava.api.exception.NotFoundException; import com.github.dockerjava.api.model.ExposedPorts; import com.github.dockerjava.api.model.Volumes; @@ -29,6 +31,9 @@ public interface CommitCmd extends SyncDockerCmd { @CheckForNull String getHostname(); + @CheckForNull + Map getLabels(); + @CheckForNull Integer getMemory(); @@ -88,6 +93,11 @@ public interface CommitCmd extends SyncDockerCmd { CommitCmd withHostname(String hostname); + /** + * @since 1.19 + */ + CommitCmd withLabels(Map labels); + CommitCmd withMemory(Integer memory); CommitCmd withMemorySwap(Integer memorySwap); diff --git a/src/main/java/com/github/dockerjava/api/command/ConnectToNetworkCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ConnectToNetworkCmd.java similarity index 93% rename from src/main/java/com/github/dockerjava/api/command/ConnectToNetworkCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/ConnectToNetworkCmd.java index 0f384e043..e6868fd7d 100644 --- a/src/main/java/com/github/dockerjava/api/command/ConnectToNetworkCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ConnectToNetworkCmd.java @@ -1,7 +1,6 @@ package com.github.dockerjava.api.command; import com.github.dockerjava.api.model.ContainerNetwork; -import com.github.dockerjava.core.RemoteApiVersion; import javax.annotation.CheckForNull; import javax.annotation.Nonnull; diff --git a/src/main/java/com/github/dockerjava/api/command/ContainerDiffCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ContainerDiffCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/ContainerDiffCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/ContainerDiffCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/CopyArchiveFromContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CopyArchiveFromContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/CopyArchiveFromContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/CopyArchiveFromContainerCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/CopyArchiveToContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CopyArchiveToContainerCmd.java similarity index 87% rename from src/main/java/com/github/dockerjava/api/command/CopyArchiveToContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/CopyArchiveToContainerCmd.java index a4dfb5c03..19b3c3843 100644 --- a/src/main/java/com/github/dockerjava/api/command/CopyArchiveToContainerCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CopyArchiveToContainerCmd.java @@ -16,6 +16,7 @@ public interface CopyArchiveToContainerCmd extends SyncDockerCmd { boolean isDirChildrenOnly(); + boolean isCopyUIDGID(); /** * Set container's id * @@ -49,6 +50,14 @@ public interface CopyArchiveToContainerCmd extends SyncDockerCmd { */ CopyArchiveToContainerCmd withNoOverwriteDirNonDir(boolean noOverwriteDirNonDir); + /** + * If set to true then ownership is set to the user and primary group at the destination + * + * @param copyUIDGID + * flag to know if ownership should be set to the user and primary group at the destination + */ + CopyArchiveToContainerCmd withCopyUIDGID(boolean copyUIDGID); + /** * If this flag is set to true, all children of the local directory will be copied to the remote without the root directory. For ex: if * I have root/titi and root/tata and the remote path is /var/data. dirChildrenOnly = true will create /var/data/titi and /var/data/tata diff --git a/src/main/java/com/github/dockerjava/api/command/CopyFileFromContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CopyFileFromContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/CopyFileFromContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/CopyFileFromContainerCmd.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateConfigCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateConfigCmd.java new file mode 100644 index 000000000..205bc7a7d --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateConfigCmd.java @@ -0,0 +1,51 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.ConflictException; + +import javax.annotation.CheckForNull; +import java.util.Map; + +/** + * Command to create a new config + * + * @since {@link RemoteApiVersion#VERSION_1_30} + */ +public interface CreateConfigCmd extends SyncDockerCmd { + + @CheckForNull + String getName(); + + @CheckForNull + String getData(); + + @CheckForNull + Map getLabels(); + + /** + * @param name + * - The new config name. + */ + CreateConfigCmd withName(String name); + + /** + * @param data + * - The new config data. + */ + CreateConfigCmd withData(byte[] data); + + /** + * @param labels + * - A mapping of labels keys and values. Labels are a mechanism for applying metadata to Docker objects. + */ + CreateConfigCmd withLabels(Map labels); + + /** + * @throws ConflictException Named config already exists + */ + @Override + CreateConfigResponse exec() throws ConflictException; + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateConfigResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateConfigResponse.java new file mode 100644 index 000000000..5836275ff --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateConfigResponse.java @@ -0,0 +1,20 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +/** + * The response of a {@link CreateConfigCmd} + */ +@EqualsAndHashCode +@ToString +public class CreateConfigResponse extends DockerObject { + @JsonProperty("ID") + private String id; + + public String getId() { + return id; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateContainerCmd.java new file mode 100644 index 000000000..fba83f50c --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateContainerCmd.java @@ -0,0 +1,1025 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonIgnore; +import com.github.dockerjava.api.exception.ConflictException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.Bind; +import com.github.dockerjava.api.model.Capability; +import com.github.dockerjava.api.model.Device; +import com.github.dockerjava.api.model.ExposedPort; +import com.github.dockerjava.api.model.HealthCheck; +import com.github.dockerjava.api.model.HostConfig; +import com.github.dockerjava.api.model.Link; +import com.github.dockerjava.api.model.LogConfig; +import com.github.dockerjava.api.model.LxcConf; +import com.github.dockerjava.api.model.PortBinding; +import com.github.dockerjava.api.model.Ports; +import com.github.dockerjava.api.model.RestartPolicy; +import com.github.dockerjava.api.model.Ulimit; +import com.github.dockerjava.api.model.Volume; +import com.github.dockerjava.api.model.VolumesFrom; + +import javax.annotation.CheckForNull; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +import static java.util.Objects.requireNonNull; + +public interface CreateContainerCmd extends SyncDockerCmd { + + @CheckForNull + AuthConfig getAuthConfig(); + + /** + * While using swarm classic, you can provide an optional auth config which will be used to pull images from a private registry, + * if the swarm node does not already have the docker image. + * Note: This option does not have any effect in normal docker + * + * @param authConfig The optional auth config + */ + CreateContainerCmd withAuthConfig(AuthConfig authConfig); + + @CheckForNull + List getAliases(); + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Bind[] getBinds() { + return getHostConfig().getBinds(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withBinds(Bind... binds) { + Objects.requireNonNull(binds, "binds was not specified"); + getHostConfig().setBinds(binds); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withBinds(List binds) { + Objects.requireNonNull(binds, "binds was not specified"); + return withBinds(binds.toArray(new Bind[binds.size()])); + } + + /** + * Add network-scoped alias for the container + * + * @param aliases on ore more aliases + */ + CreateContainerCmd withAliases(List aliases); + + /** + * Add network-scoped alias for the container + * + * @param aliases on ore more aliases + */ + CreateContainerCmd withAliases(String... aliases); + + @CheckForNull + String[] getCmd(); + + CreateContainerCmd withCmd(String... cmd); + + CreateContainerCmd withCmd(List cmd); + + @CheckForNull + HealthCheck getHealthcheck(); + + CreateContainerCmd withHealthcheck(HealthCheck healthCheck); + + @CheckForNull + Boolean getArgsEscaped(); + + CreateContainerCmd withArgsEscaped(Boolean argsEscaped); + + @CheckForNull + String getDomainName(); + + CreateContainerCmd withDomainName(String domainName); + + @CheckForNull + String[] getEntrypoint(); + + CreateContainerCmd withEntrypoint(String... entrypoint); + + CreateContainerCmd withEntrypoint(List entrypoint); + + @CheckForNull + String[] getEnv(); + + /** + * Adds environment-variables. NB: Not additive, i.e. in case of multiple calls to the method, only the most recent + * values will be injected. Prior env-variables will be deleted. + * + * @param env the String(s) to set as ENV in the container + */ + CreateContainerCmd withEnv(String... env); + + /** + * Adds environment-variables. NB: Not additive, i.e. in case of multiple calls to the method, only the most recent + * values will be injected. Prior env-variables will be deleted. + * + * @param env the list of Strings to set as ENV in the container + */ + CreateContainerCmd withEnv(List env); + + @CheckForNull + ExposedPort[] getExposedPorts(); + + CreateContainerCmd withExposedPorts(List exposedPorts); + + CreateContainerCmd withExposedPorts(ExposedPort... exposedPorts); + + @CheckForNull + String getStopSignal(); + + CreateContainerCmd withStopSignal(String stopSignal); + + @CheckForNull + Integer getStopTimeout(); + + CreateContainerCmd withStopTimeout(Integer stopTimeout); + + @CheckForNull + String getHostName(); + + CreateContainerCmd withHostName(String hostName); + + @CheckForNull + String getImage(); + + CreateContainerCmd withImage(String image); + + @CheckForNull + String getIpv4Address(); + + CreateContainerCmd withIpv4Address(String ipv4Address); + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Link[] getLinks() { + return getHostConfig().getLinks(); + } + + /** + * Add link to another container. + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withLinks(Link... links) { + requireNonNull(links, "links was not specified"); + getHostConfig().setLinks(links); + return this; + } + + /** + * Add link to another container. + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withLinks(List links) { + requireNonNull(links, "links was not specified"); + return withLinks(links.toArray(new Link[links.size()])); + } + + @CheckForNull + String getIpv6Address(); + + CreateContainerCmd withIpv6Address(String ipv6Address); + + @CheckForNull + Map getLabels(); + + CreateContainerCmd withLabels(Map labels); + + @CheckForNull + String getMacAddress(); + + CreateContainerCmd withMacAddress(String macAddress); + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Long getMemory() { + return getHostConfig().getMemory(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withMemory(Long memory) { + Objects.requireNonNull(memory, "memory was not specified"); + getHostConfig().withMemory(memory); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Long getMemorySwap() { + return getHostConfig().getMemorySwap(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withMemorySwap(Long memorySwap) { + Objects.requireNonNull(memorySwap, "memorySwap was not specified"); + getHostConfig().withMemorySwap(memorySwap); + return this; + } + + @CheckForNull + String getName(); + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default String getNetworkMode() { + return getHostConfig().getNetworkMode(); + } + + /** + * Set the Network mode for the container + *

+ * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withNetworkMode(String networkMode) { + Objects.requireNonNull(networkMode, "networkMode was not specified"); + getHostConfig().withNetworkMode(networkMode); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Ports getPortBindings() { + return getHostConfig().getPortBindings(); + } + + /** + * Add one or more {@link PortBinding}s. This corresponds to the --publish (-p) option of the + * docker run CLI command. + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withPortBindings(PortBinding... portBindings) { + Objects.requireNonNull(portBindings, "portBindings was not specified"); + getHostConfig().withPortBindings(new Ports(portBindings)); + return this; + } + + /** + * Add one or more {@link PortBinding}s. This corresponds to the --publish (-p) option of the + * docker run CLI command. + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withPortBindings(List portBindings) { + Objects.requireNonNull(portBindings, "portBindings was not specified"); + return withPortBindings(portBindings.toArray(new PortBinding[0])); + } + + /** + * Add the port bindings that are contained in the given {@link Ports} object. + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withPortBindings(Ports portBindings) { + Objects.requireNonNull(portBindings, "portBindings was not specified"); + getHostConfig().withPortBindings(portBindings); + return this; + } + + CreateContainerCmd withName(String name); + + @CheckForNull + String[] getPortSpecs(); + + CreateContainerCmd withPortSpecs(String... portSpecs); + + CreateContainerCmd withPortSpecs(List portSpecs); + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Boolean getPrivileged() { + return getHostConfig().getPrivileged(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withPrivileged(Boolean privileged) { + Objects.requireNonNull(privileged, "no privileged was specified"); + getHostConfig().withPrivileged(privileged); + return this; + } + + @CheckForNull + String getUser(); + + CreateContainerCmd withUser(String user); + + @CheckForNull + Volume[] getVolumes(); + + CreateContainerCmd withVolumes(Volume... volumes); + + CreateContainerCmd withVolumes(List volumes); + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default VolumesFrom[] getVolumesFrom() { + return getHostConfig().getVolumesFrom(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withVolumesFrom(VolumesFrom... volumesFrom) { + Objects.requireNonNull(volumesFrom, "volumesFrom was not specified"); + getHostConfig().withVolumesFrom(volumesFrom); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withVolumesFrom(List volumesFrom) { + requireNonNull(volumesFrom, "volumesFrom was not specified"); + return withVolumesFrom(volumesFrom.toArray(new VolumesFrom[volumesFrom.size()])); + } + + @CheckForNull + String getWorkingDir(); + + CreateContainerCmd withWorkingDir(String workingDir); + + @CheckForNull + Boolean isAttachStderr(); + + CreateContainerCmd withAttachStderr(Boolean attachStderr); + + @CheckForNull + Boolean isAttachStdin(); + + CreateContainerCmd withAttachStdin(Boolean attachStdin); + + @CheckForNull + Boolean isAttachStdout(); + + CreateContainerCmd withAttachStdout(Boolean attachStdout); + + @CheckForNull + Boolean isNetworkDisabled(); + + CreateContainerCmd withNetworkDisabled(Boolean disableNetwork); + + @CheckForNull + Boolean isStdInOnce(); + + CreateContainerCmd withStdInOnce(Boolean stdInOnce); + + @CheckForNull + Boolean isStdinOpen(); + + CreateContainerCmd withStdinOpen(Boolean stdinOpen); + + @CheckForNull + Boolean isTty(); + + CreateContainerCmd withTty(Boolean tty); + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Boolean getPublishAllPorts() { + return getHostConfig().getPublishAllPorts(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withPublishAllPorts(Boolean publishAllPorts) { + requireNonNull(publishAllPorts, "no publishAllPorts was specified"); + getHostConfig().withPublishAllPorts(publishAllPorts); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @CheckForNull + @Deprecated + @JsonIgnore + default String[] getExtraHosts() { + return getHostConfig().getExtraHosts(); + } + + /** + * Add hostnames to /etc/hosts in the container + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withExtraHosts(String... extraHosts) { + requireNonNull(extraHosts, "extraHosts was not specified"); + getHostConfig().withExtraHosts(extraHosts); + return this; + } + + /** + * Add hostnames to /etc/hosts in the container + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withExtraHosts(List extraHosts) { + requireNonNull(extraHosts, "extraHosts was not specified"); + return withExtraHosts(extraHosts.toArray(new String[extraHosts.size()])); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @CheckForNull + @Deprecated + @JsonIgnore + default Capability[] getCapAdd() { + return getHostConfig().getCapAdd(); + } + + /** + * Add linux kernel capability to the container. For example: + * adding {@link Capability#MKNOD} allows the container to create special files using the 'mknod' command. + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withCapAdd(Capability... capAdd) { + requireNonNull(capAdd, "capAdd was not specified"); + getHostConfig().withCapAdd(capAdd); + return this; + } + + /** + * Add linux kernel capability to the container. For example: + * adding {@link Capability#MKNOD} allows the container to create special files using the 'mknod' command. + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withCapAdd(List capAdd) { + requireNonNull(capAdd, "capAdd was not specified"); + return withCapAdd(capAdd.toArray(new Capability[capAdd.size()])); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @CheckForNull + @Deprecated + @JsonIgnore + default Capability[] getCapDrop() { + return getHostConfig().getCapDrop(); + } + + /** + * Drop linux kernel capability from the container. For example: + * dropping {@link Capability#CHOWN} prevents the container from changing the owner of any files. + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withCapDrop(Capability... capDrop) { + requireNonNull(capDrop, "capDrop was not specified"); + getHostConfig().withCapDrop(capDrop); + return this; + } + + /** + * Drop linux kernel capability from the container. For example: + * dropping {@link Capability#CHOWN} prevents the container from changing the owner of any files. + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withCapDrop(List capDrop) { + requireNonNull(capDrop, "capDrop was not specified"); + return withCapDrop(capDrop.toArray(new Capability[capDrop.size()])); + } + + + @CheckForNull + List getOnBuild(); + + CreateContainerCmd withOnBuild(List onBuild); + + @CheckForNull + HostConfig getHostConfig(); + + CreateContainerCmd withHostConfig(HostConfig hostConfig); + + // The following methods are deprecated and should be set on {@link #getHostConfig()} instead. + // TODO remove in the next big release + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Integer getBlkioWeight() { + return getHostConfig().getBlkioWeight(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @CheckForNull + @Deprecated + @JsonIgnore + default String getCgroupParent() { + return getHostConfig().getCgroupParent(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Integer getCpuPeriod() { + Long result = getHostConfig().getCpuPeriod(); + return result != null ? result.intValue() : null; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Integer getCpuShares() { + return getHostConfig().getCpuShares(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default String getCpusetCpus() { + return getHostConfig().getCpusetCpus(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default String getCpusetMems() { + return getHostConfig().getCpusetMems(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Device[] getDevices() { + return getHostConfig().getDevices(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default String[] getDns() { + return getHostConfig().getDns(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default String[] getDnsSearch() { + return getHostConfig().getDnsSearch(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default LogConfig getLogConfig() { + return getHostConfig().getLogConfig(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default LxcConf[] getLxcConf() { + return getHostConfig().getLxcConf(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Boolean getOomKillDisable() { + return getHostConfig().getOomKillDisable(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default String getPidMode() { + return getHostConfig().getPidMode(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Boolean getReadonlyRootfs() { + return getHostConfig().getReadonlyRootfs(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default RestartPolicy getRestartPolicy() { + return getHostConfig().getRestartPolicy(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @CheckForNull + @JsonIgnore + default Ulimit[] getUlimits() { + return getHostConfig().getUlimits(); + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withBlkioWeight(Integer blkioWeight) { + getHostConfig().withBlkioWeight(blkioWeight); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withCgroupParent(String cgroupParent) { + getHostConfig().withCgroupParent(cgroupParent); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withContainerIDFile(String containerIDFile) { + getHostConfig().withContainerIDFile(containerIDFile); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withCpuPeriod(Integer cpuPeriod) { + getHostConfig().withCpuPeriod(cpuPeriod != null ? cpuPeriod.longValue() : null); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withCpuShares(Integer cpuShares) { + getHostConfig().withCpuShares(cpuShares); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withCpusetCpus(String cpusetCpus) { + getHostConfig().withCpusetCpus(cpusetCpus); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withCpusetMems(String cpusetMems) { + getHostConfig().withCpusetMems(cpusetMems); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withDevices(Device... devices) { + getHostConfig().withDevices(devices); + return this; + } + + /** + * Add host devices to the container + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withDevices(List devices) { + getHostConfig().withDevices(devices); + return this; + } + + /** + * Set custom DNS servers + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withDns(String... dns) { + getHostConfig().withDns(dns); + return this; + } + + /** + * Set custom DNS servers + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withDns(List dns) { + getHostConfig().withDns(dns); + return this; + } + + /** + * Set custom DNS search domains + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withDnsSearch(String... dnsSearch) { + getHostConfig().withDnsSearch(dnsSearch); + return this; + } + + /** + * Set custom DNS search domains + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withDnsSearch(List dnsSearch) { + getHostConfig().withDnsSearch(dnsSearch); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withLogConfig(LogConfig logConfig) { + getHostConfig().withLogConfig(logConfig); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withLxcConf(LxcConf... lxcConf) { + getHostConfig().withLxcConf(lxcConf); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withLxcConf(List lxcConf) { + getHostConfig().withLxcConf(lxcConf.toArray(new LxcConf[0])); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withOomKillDisable(Boolean oomKillDisable) { + getHostConfig().withOomKillDisable(oomKillDisable); + return this; + } + + /** + * Set the PID (Process) Namespace mode for the container, 'host': use the host's PID namespace inside the container + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withPidMode(String pidMode) { + getHostConfig().withPidMode(pidMode); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withReadonlyRootfs(Boolean readonlyRootfs) { + getHostConfig().withReadonlyRootfs(readonlyRootfs); + return this; + } + + /** + * Set custom {@link RestartPolicy} for the container. Defaults to {@link RestartPolicy#noRestart()} + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withRestartPolicy(RestartPolicy restartPolicy) { + getHostConfig().withRestartPolicy(restartPolicy); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + @JsonIgnore + default CreateContainerCmd withUlimits(Ulimit... ulimits) { + getHostConfig().withUlimits(ulimits); + return this; + } + + /** + * + * @deprecated see {@link #getHostConfig()} + */ + @Deprecated + default CreateContainerCmd withUlimits(List ulimits) { + getHostConfig().withUlimits(ulimits); + return this; + } + + @CheckForNull + default String getPlatform() { + return null; + } + + CreateContainerCmd withPlatform(String platform); + + /** + * @throws NotFoundException No such container + * @throws ConflictException Named container already exists + */ + @Override + CreateContainerResponse exec() throws NotFoundException, ConflictException; + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateContainerResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateContainerResponse.java new file mode 100644 index 000000000..ad24d7ec8 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateContainerResponse.java @@ -0,0 +1,38 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +/** + * + * @author Konstantin Pelykh (kpelykh@gmail.com) + * + */ +@EqualsAndHashCode +@ToString +public class CreateContainerResponse extends DockerObject { + + @JsonProperty("Id") + private String id; + + @JsonProperty("Warnings") + private String[] warnings; + + public String getId() { + return id; + } + + public String[] getWarnings() { + return warnings; + } + + public void setId(String id) { + this.id = id; + } + + public void setWarnings(String[] warnings) { + this.warnings = warnings; + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/CreateImageCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateImageCmd.java similarity index 83% rename from src/main/java/com/github/dockerjava/api/command/CreateImageCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateImageCmd.java index 86935cd10..4e78dd5e0 100644 --- a/src/main/java/com/github/dockerjava/api/command/CreateImageCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateImageCmd.java @@ -13,6 +13,9 @@ public interface CreateImageCmd extends SyncDockerCmd { @CheckForNull String getTag(); + @CheckForNull + String getPlatform(); + @CheckForNull InputStream getImageStream(); @@ -35,6 +38,12 @@ public interface CreateImageCmd extends SyncDockerCmd { */ CreateImageCmd withTag(String tag); + /** + * @param platform + * the platform for this image + */ + CreateImageCmd withPlatform(String platform); + interface Exec extends DockerCmdSyncExec { } diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateImageResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateImageResponse.java new file mode 100644 index 000000000..53b2b5367 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateImageResponse.java @@ -0,0 +1,24 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +/** + * Parse reponses from /images/create + * + * @author Ryan Campbell (ryan.campbell@gmail.com) + * + */ +@EqualsAndHashCode +@ToString +public class CreateImageResponse extends DockerObject { + + @JsonProperty("status") + private String id; + + public String getId() { + return id; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateNetworkCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateNetworkCmd.java new file mode 100644 index 000000000..56b9df17a --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateNetworkCmd.java @@ -0,0 +1,78 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.Network; +import com.github.dockerjava.api.model.Network.Ipam; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import java.util.Map; + +/** + * Create a network. + * + * @since {@link RemoteApiVersion#VERSION_1_21} + */ +public interface CreateNetworkCmd extends SyncDockerCmd { + + @CheckForNull + String getName(); + + @CheckForNull + String getDriver(); + + @CheckForNull + Network.Ipam getIpam(); + + @CheckForNull + Map getOptions(); + + @CheckForNull + Boolean getCheckDuplicate(); + + @CheckForNull + Boolean getInternal(); + + @CheckForNull + Boolean getEnableIPv6(); + + @CheckForNull + Boolean getAttachable(); + + @CheckForNull + Map getLabels(); + + /** The new network's name. Required. */ + CreateNetworkCmd withName(@Nonnull String name); + + /** Name of the network driver to use. Defaults to bridge. */ + CreateNetworkCmd withDriver(String driver); + + /** Ipam config, such as subnet, gateway and ip range of the network */ + CreateNetworkCmd withIpam(Ipam ipam); + + /** Driver specific options */ + CreateNetworkCmd withOptions(Map options); + + CreateNetworkCmd withCheckDuplicate(boolean checkForDuplicate); + + CreateNetworkCmd withInternal(boolean internal); + + CreateNetworkCmd withEnableIpv6(boolean enableIpv6); + + /** + * If enabled, and the network is in the global scope, non-service containers on worker nodes will be able to connect to the network. + * + * @since {@link RemoteApiVersion#VERSION_1_21} + */ + CreateNetworkCmd withAttachable(Boolean attachable); + + /** + * Add label for network + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + CreateNetworkCmd withLabels(Map labels); + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateNetworkResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateNetworkResponse.java new file mode 100644 index 000000000..3f6f219e1 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateNetworkResponse.java @@ -0,0 +1,33 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@EqualsAndHashCode +@ToString +public class CreateNetworkResponse extends DockerObject { + + @JsonProperty("Id") + private String id; + + @JsonProperty("Warnings") + private String[] warnings; + + public String getId() { + return id; + } + + public String[] getWarnings() { + return warnings; + } + + public void setId(String id) { + this.id = id; + } + + public void setWarnings(String[] warnings) { + this.warnings = warnings; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateSecretCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateSecretCmd.java new file mode 100644 index 000000000..c27e42317 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateSecretCmd.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.ConflictException; +import com.github.dockerjava.api.model.SecretSpec; + +import javax.annotation.CheckForNull; + +/** + * Command to create a new secret + * + * @since {@link RemoteApiVersion#VERSION_1_25} + */ +public interface CreateSecretCmd extends SyncDockerCmd { + + @CheckForNull + SecretSpec getSecretSpec(); + + CreateSecretCmd withSecretSpec(SecretSpec secretSpec); + + /** + * @throws ConflictException Named secret already exists + */ + @Override + CreateSecretResponse exec() throws ConflictException; + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateSecretResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateSecretResponse.java new file mode 100644 index 000000000..2c1b6f11b --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateSecretResponse.java @@ -0,0 +1,20 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +/** + * The response of a {@link CreateSecretCmd} + */ +@EqualsAndHashCode +@ToString +public class CreateSecretResponse extends DockerObject { + @JsonProperty("ID") + private String id; + + public String getId() { + return id; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateServiceCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateServiceCmd.java new file mode 100644 index 000000000..bfcce27ad --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateServiceCmd.java @@ -0,0 +1,36 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.ConflictException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.ServiceSpec; + +import javax.annotation.CheckForNull; + +/** + * Command to create a new service + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public interface CreateServiceCmd extends SyncDockerCmd { + + @CheckForNull + ServiceSpec getServiceSpec(); + + @CheckForNull + AuthConfig getAuthConfig(); + + CreateServiceCmd withServiceSpec(ServiceSpec serviceSpec); + + CreateServiceCmd withAuthConfig(AuthConfig authConfig); + + /** + * @throws ConflictException + * Named service already exists + */ + @Override + CreateServiceResponse exec() throws ConflictException; + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateServiceResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateServiceResponse.java new file mode 100644 index 000000000..b68343f55 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateServiceResponse.java @@ -0,0 +1,20 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +/** + * The response of a {@link CreateServiceCmd} + */ +@EqualsAndHashCode +@ToString +public class CreateServiceResponse extends DockerObject { + @JsonProperty("ID") + private String id; + + public String getId() { + return id; + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/CreateVolumeCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateVolumeCmd.java similarity index 78% rename from src/main/java/com/github/dockerjava/api/command/CreateVolumeCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateVolumeCmd.java index 3d052033f..4a3ef9819 100644 --- a/src/main/java/com/github/dockerjava/api/command/CreateVolumeCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateVolumeCmd.java @@ -9,6 +9,9 @@ public interface CreateVolumeCmd extends SyncDockerCmd { @CheckForNull String getName(); + @CheckForNull + Map getLabels(); + @CheckForNull String getDriver(); @@ -21,6 +24,12 @@ public interface CreateVolumeCmd extends SyncDockerCmd { */ CreateVolumeCmd withName(String name); + /** + * @param labels + * - A mapping of labels keys and values. Labels are a mechanism for applying metadata to Docker objects. + */ + CreateVolumeCmd withLabels(Map labels); + /** * @param driver * - Name of the volume driver to use. Defaults to local for the name. diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateVolumeResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateVolumeResponse.java new file mode 100644 index 000000000..4afc6f6ba --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/CreateVolumeResponse.java @@ -0,0 +1,45 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.util.Map; + +/** + * + * @author Marcus Linke + */ +@EqualsAndHashCode +@ToString +public class CreateVolumeResponse extends DockerObject { + + @JsonProperty("Name") + private String name; + + @JsonProperty("Labels") + private Map labels; + + @JsonProperty("Driver") + private String driver; + + @JsonProperty("Mountpoint") + private String mountpoint; + + public String getName() { + return name; + } + + public Map getLabels() { + return labels; + } + + public String getDriver() { + return driver; + } + + public String getMountpoint() { + return mountpoint; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/DelegatingDockerCmdExecFactory.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/DelegatingDockerCmdExecFactory.java new file mode 100644 index 000000000..161ff2c29 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/DelegatingDockerCmdExecFactory.java @@ -0,0 +1,412 @@ +package com.github.dockerjava.api.command; + +import java.io.IOException; + +public class DelegatingDockerCmdExecFactory implements DockerCmdExecFactory { + + // We're not using abstract class because we want + // the compiler to force us to implement new DockerCmdExecFactory when added + public DockerCmdExecFactory getDockerCmdExecFactory() { + throw new IllegalStateException("Implement me!"); + } + + @Override + public AuthCmd.Exec createAuthCmdExec() { + return getDockerCmdExecFactory().createAuthCmdExec(); + } + + @Override + public InfoCmd.Exec createInfoCmdExec() { + return getDockerCmdExecFactory().createInfoCmdExec(); + } + + @Override + public PingCmd.Exec createPingCmdExec() { + return getDockerCmdExecFactory().createPingCmdExec(); + } + + @Override + public ResizeContainerCmd.Exec createResizeContainerCmdExec() { + return getDockerCmdExecFactory().createResizeContainerCmdExec(); + } + + @Override + public ExecCreateCmd.Exec createExecCmdExec() { + return getDockerCmdExecFactory().createExecCmdExec(); + } + + @Override + public ResizeExecCmd.Exec createResizeExecCmdExec() { + return getDockerCmdExecFactory().createResizeExecCmdExec(); + } + + @Override + public VersionCmd.Exec createVersionCmdExec() { + return getDockerCmdExecFactory().createVersionCmdExec(); + } + + @Override + public PullImageCmd.Exec createPullImageCmdExec() { + return getDockerCmdExecFactory().createPullImageCmdExec(); + } + + @Override + public PushImageCmd.Exec createPushImageCmdExec() { + return getDockerCmdExecFactory().createPushImageCmdExec(); + } + + @Override + public SaveImageCmd.Exec createSaveImageCmdExec() { + return getDockerCmdExecFactory().createSaveImageCmdExec(); + } + + @Override + public SaveImagesCmd.Exec createSaveImagesCmdExec() { + return getDockerCmdExecFactory().createSaveImagesCmdExec(); + } + + @Override + public CreateImageCmd.Exec createCreateImageCmdExec() { + return getDockerCmdExecFactory().createCreateImageCmdExec(); + } + + @Override + public LoadImageCmd.Exec createLoadImageCmdExec() { + return getDockerCmdExecFactory().createLoadImageCmdExec(); + } + + @Override + public LoadImageAsyncCmd.Exec createLoadImageAsyncCmdExec() { + return getDockerCmdExecFactory().createLoadImageAsyncCmdExec(); + } + + @Override + public SearchImagesCmd.Exec createSearchImagesCmdExec() { + return getDockerCmdExecFactory().createSearchImagesCmdExec(); + } + + @Override + public RemoveImageCmd.Exec createRemoveImageCmdExec() { + return getDockerCmdExecFactory().createRemoveImageCmdExec(); + } + + @Override + public ListImagesCmd.Exec createListImagesCmdExec() { + return getDockerCmdExecFactory().createListImagesCmdExec(); + } + + @Override + public InspectImageCmd.Exec createInspectImageCmdExec() { + return getDockerCmdExecFactory().createInspectImageCmdExec(); + } + + @Override + public ListContainersCmd.Exec createListContainersCmdExec() { + return getDockerCmdExecFactory().createListContainersCmdExec(); + } + + @Override + public CreateContainerCmd.Exec createCreateContainerCmdExec() { + return getDockerCmdExecFactory().createCreateContainerCmdExec(); + } + + @Override + public StartContainerCmd.Exec createStartContainerCmdExec() { + return getDockerCmdExecFactory().createStartContainerCmdExec(); + } + + @Override + public InspectContainerCmd.Exec createInspectContainerCmdExec() { + return getDockerCmdExecFactory().createInspectContainerCmdExec(); + } + + @Override + public RemoveContainerCmd.Exec createRemoveContainerCmdExec() { + return getDockerCmdExecFactory().createRemoveContainerCmdExec(); + } + + @Override + public WaitContainerCmd.Exec createWaitContainerCmdExec() { + return getDockerCmdExecFactory().createWaitContainerCmdExec(); + } + + @Override + public AttachContainerCmd.Exec createAttachContainerCmdExec() { + return getDockerCmdExecFactory().createAttachContainerCmdExec(); + } + + @Override + public ExecStartCmd.Exec createExecStartCmdExec() { + return getDockerCmdExecFactory().createExecStartCmdExec(); + } + + @Override + public InspectExecCmd.Exec createInspectExecCmdExec() { + return getDockerCmdExecFactory().createInspectExecCmdExec(); + } + + @Override + public LogContainerCmd.Exec createLogContainerCmdExec() { + return getDockerCmdExecFactory().createLogContainerCmdExec(); + } + + @Override + public CopyFileFromContainerCmd.Exec createCopyFileFromContainerCmdExec() { + return getDockerCmdExecFactory().createCopyFileFromContainerCmdExec(); + } + + @Override + public CopyArchiveFromContainerCmd.Exec createCopyArchiveFromContainerCmdExec() { + return getDockerCmdExecFactory().createCopyArchiveFromContainerCmdExec(); + } + + @Override + public CopyArchiveToContainerCmd.Exec createCopyArchiveToContainerCmdExec() { + return getDockerCmdExecFactory().createCopyArchiveToContainerCmdExec(); + } + + @Override + public StopContainerCmd.Exec createStopContainerCmdExec() { + return getDockerCmdExecFactory().createStopContainerCmdExec(); + } + + @Override + public ContainerDiffCmd.Exec createContainerDiffCmdExec() { + return getDockerCmdExecFactory().createContainerDiffCmdExec(); + } + + @Override + public KillContainerCmd.Exec createKillContainerCmdExec() { + return getDockerCmdExecFactory().createKillContainerCmdExec(); + } + + @Override + public UpdateContainerCmd.Exec createUpdateContainerCmdExec() { + return getDockerCmdExecFactory().createUpdateContainerCmdExec(); + } + + @Override + public RenameContainerCmd.Exec createRenameContainerCmdExec() { + return getDockerCmdExecFactory().createRenameContainerCmdExec(); + } + + @Override + public RestartContainerCmd.Exec createRestartContainerCmdExec() { + return getDockerCmdExecFactory().createRestartContainerCmdExec(); + } + + @Override + public CommitCmd.Exec createCommitCmdExec() { + return getDockerCmdExecFactory().createCommitCmdExec(); + } + + @Override + public BuildImageCmd.Exec createBuildImageCmdExec() { + return getDockerCmdExecFactory().createBuildImageCmdExec(); + } + + @Override + public TopContainerCmd.Exec createTopContainerCmdExec() { + return getDockerCmdExecFactory().createTopContainerCmdExec(); + } + + @Override + public TagImageCmd.Exec createTagImageCmdExec() { + return getDockerCmdExecFactory().createTagImageCmdExec(); + } + + @Override + public PauseContainerCmd.Exec createPauseContainerCmdExec() { + return getDockerCmdExecFactory().createPauseContainerCmdExec(); + } + + @Override + public UnpauseContainerCmd.Exec createUnpauseContainerCmdExec() { + return getDockerCmdExecFactory().createUnpauseContainerCmdExec(); + } + + @Override + public EventsCmd.Exec createEventsCmdExec() { + return getDockerCmdExecFactory().createEventsCmdExec(); + } + + @Override + public StatsCmd.Exec createStatsCmdExec() { + return getDockerCmdExecFactory().createStatsCmdExec(); + } + + @Override + public CreateVolumeCmd.Exec createCreateVolumeCmdExec() { + return getDockerCmdExecFactory().createCreateVolumeCmdExec(); + } + + @Override + public InspectVolumeCmd.Exec createInspectVolumeCmdExec() { + return getDockerCmdExecFactory().createInspectVolumeCmdExec(); + } + + @Override + public RemoveVolumeCmd.Exec createRemoveVolumeCmdExec() { + return getDockerCmdExecFactory().createRemoveVolumeCmdExec(); + } + + @Override + public ListVolumesCmd.Exec createListVolumesCmdExec() { + return getDockerCmdExecFactory().createListVolumesCmdExec(); + } + + @Override + public ListNetworksCmd.Exec createListNetworksCmdExec() { + return getDockerCmdExecFactory().createListNetworksCmdExec(); + } + + @Override + public InspectNetworkCmd.Exec createInspectNetworkCmdExec() { + return getDockerCmdExecFactory().createInspectNetworkCmdExec(); + } + + @Override + public CreateNetworkCmd.Exec createCreateNetworkCmdExec() { + return getDockerCmdExecFactory().createCreateNetworkCmdExec(); + } + + @Override + public RemoveNetworkCmd.Exec createRemoveNetworkCmdExec() { + return getDockerCmdExecFactory().createRemoveNetworkCmdExec(); + } + + @Override + public ConnectToNetworkCmd.Exec createConnectToNetworkCmdExec() { + return getDockerCmdExecFactory().createConnectToNetworkCmdExec(); + } + + @Override + public DisconnectFromNetworkCmd.Exec createDisconnectFromNetworkCmdExec() { + return getDockerCmdExecFactory().createDisconnectFromNetworkCmdExec(); + } + + @Override + public InitializeSwarmCmd.Exec createInitializeSwarmCmdExec() { + return getDockerCmdExecFactory().createInitializeSwarmCmdExec(); + } + + @Override + public InspectSwarmCmd.Exec createInspectSwarmCmdExec() { + return getDockerCmdExecFactory().createInspectSwarmCmdExec(); + } + + @Override + public JoinSwarmCmd.Exec createJoinSwarmCmdExec() { + return getDockerCmdExecFactory().createJoinSwarmCmdExec(); + } + + @Override + public LeaveSwarmCmd.Exec createLeaveSwarmCmdExec() { + return getDockerCmdExecFactory().createLeaveSwarmCmdExec(); + } + + @Override + public UpdateSwarmCmd.Exec createUpdateSwarmCmdExec() { + return getDockerCmdExecFactory().createUpdateSwarmCmdExec(); + } + + @Override + public ListServicesCmd.Exec createListServicesCmdExec() { + return getDockerCmdExecFactory().createListServicesCmdExec(); + } + + @Override + public CreateServiceCmd.Exec createCreateServiceCmdExec() { + return getDockerCmdExecFactory().createCreateServiceCmdExec(); + } + + @Override + public InspectServiceCmd.Exec createInspectServiceCmdExec() { + return getDockerCmdExecFactory().createInspectServiceCmdExec(); + } + + @Override + public UpdateServiceCmd.Exec createUpdateServiceCmdExec() { + return getDockerCmdExecFactory().createUpdateServiceCmdExec(); + } + + @Override + public RemoveServiceCmd.Exec createRemoveServiceCmdExec() { + return getDockerCmdExecFactory().createRemoveServiceCmdExec(); + } + + @Override + public LogSwarmObjectCmd.Exec logSwarmObjectExec(String endpoint) { + return getDockerCmdExecFactory().logSwarmObjectExec(endpoint); + } + + @Override + public ListSwarmNodesCmd.Exec listSwarmNodeCmdExec() { + return getDockerCmdExecFactory().listSwarmNodeCmdExec(); + } + + @Override + public InspectSwarmNodeCmd.Exec inspectSwarmNodeCmdExec() { + return getDockerCmdExecFactory().inspectSwarmNodeCmdExec(); + } + + @Override + public RemoveSwarmNodeCmd.Exec removeSwarmNodeCmdExec() { + return getDockerCmdExecFactory().removeSwarmNodeCmdExec(); + } + + @Override + public UpdateSwarmNodeCmd.Exec updateSwarmNodeCmdExec() { + return getDockerCmdExecFactory().updateSwarmNodeCmdExec(); + } + + @Override + public ListTasksCmd.Exec listTasksCmdExec() { + return getDockerCmdExecFactory().listTasksCmdExec(); + } + + @Override + public PruneCmd.Exec pruneCmdExec() { + return getDockerCmdExecFactory().pruneCmdExec(); + } + + @Override + public ListSecretsCmd.Exec createListSecretsCmdExec() { + return getDockerCmdExecFactory().createListSecretsCmdExec(); + } + + @Override + public CreateSecretCmd.Exec createCreateSecretCmdExec() { + return getDockerCmdExecFactory().createCreateSecretCmdExec(); + } + + @Override + public RemoveSecretCmd.Exec createRemoveSecretCmdExec() { + return getDockerCmdExecFactory().createRemoveSecretCmdExec(); + } + + @Override + public ListConfigsCmd.Exec createListConfigsCmdExec() { + return getDockerCmdExecFactory().createListConfigsCmdExec(); + } + + @Override + public CreateConfigCmd.Exec createCreateConfigCmdExec() { + return getDockerCmdExecFactory().createCreateConfigCmdExec(); + } + + @Override + public InspectConfigCmd.Exec createInspectConfigCmdExec() { + return getDockerCmdExecFactory().createInspectConfigCmdExec(); + } + + @Override + public RemoveConfigCmd.Exec createRemoveConfigCmdExec() { + return getDockerCmdExecFactory().createRemoveConfigCmdExec(); + } + + @Override + public void close() throws IOException { + getDockerCmdExecFactory().close(); + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/DisconnectFromNetworkCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/DisconnectFromNetworkCmd.java similarity index 93% rename from src/main/java/com/github/dockerjava/api/command/DisconnectFromNetworkCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/DisconnectFromNetworkCmd.java index ce75172b4..07dadec1e 100644 --- a/src/main/java/com/github/dockerjava/api/command/DisconnectFromNetworkCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/DisconnectFromNetworkCmd.java @@ -1,7 +1,5 @@ package com.github.dockerjava.api.command; -import com.github.dockerjava.core.RemoteApiVersion; - import javax.annotation.CheckForNull; import javax.annotation.Nonnull; diff --git a/src/main/java/com/github/dockerjava/api/command/DockerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/DockerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/DockerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/DockerCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/DockerCmdAsyncExec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/DockerCmdAsyncExec.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/DockerCmdAsyncExec.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/DockerCmdAsyncExec.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/DockerCmdExecFactory.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/DockerCmdExecFactory.java new file mode 100644 index 000000000..cedf6d40d --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/DockerCmdExecFactory.java @@ -0,0 +1,273 @@ +package com.github.dockerjava.api.command; + +import java.io.Closeable; +import java.io.IOException; + +public interface DockerCmdExecFactory extends Closeable { + + AuthCmd.Exec createAuthCmdExec(); + + InfoCmd.Exec createInfoCmdExec(); + + PingCmd.Exec createPingCmdExec(); + + ExecCreateCmd.Exec createExecCmdExec(); + + VersionCmd.Exec createVersionCmdExec(); + + PullImageCmd.Exec createPullImageCmdExec(); + + PushImageCmd.Exec createPushImageCmdExec(); + + SaveImageCmd.Exec createSaveImageCmdExec(); + + SaveImagesCmd.Exec createSaveImagesCmdExec(); + + CreateImageCmd.Exec createCreateImageCmdExec(); + + LoadImageCmd.Exec createLoadImageCmdExec(); + + LoadImageAsyncCmd.Exec createLoadImageAsyncCmdExec(); + + SearchImagesCmd.Exec createSearchImagesCmdExec(); + + RemoveImageCmd.Exec createRemoveImageCmdExec(); + + ListImagesCmd.Exec createListImagesCmdExec(); + + InspectImageCmd.Exec createInspectImageCmdExec(); + + ListContainersCmd.Exec createListContainersCmdExec(); + + CreateContainerCmd.Exec createCreateContainerCmdExec(); + + StartContainerCmd.Exec createStartContainerCmdExec(); + + InspectContainerCmd.Exec createInspectContainerCmdExec(); + + RemoveContainerCmd.Exec createRemoveContainerCmdExec(); + + WaitContainerCmd.Exec createWaitContainerCmdExec(); + + AttachContainerCmd.Exec createAttachContainerCmdExec(); + + ResizeContainerCmd.Exec createResizeContainerCmdExec(); + + ExecStartCmd.Exec createExecStartCmdExec(); + + ResizeExecCmd.Exec createResizeExecCmdExec(); + + InspectExecCmd.Exec createInspectExecCmdExec(); + + LogContainerCmd.Exec createLogContainerCmdExec(); + + CopyFileFromContainerCmd.Exec createCopyFileFromContainerCmdExec(); + + CopyArchiveFromContainerCmd.Exec createCopyArchiveFromContainerCmdExec(); + + CopyArchiveToContainerCmd.Exec createCopyArchiveToContainerCmdExec(); + + StopContainerCmd.Exec createStopContainerCmdExec(); + + ContainerDiffCmd.Exec createContainerDiffCmdExec(); + + KillContainerCmd.Exec createKillContainerCmdExec(); + + UpdateContainerCmd.Exec createUpdateContainerCmdExec(); + + /** + * Rename container. + * + * @since {@link RemoteApiVersion#VERSION_1_17} + */ + RenameContainerCmd.Exec createRenameContainerCmdExec(); + + RestartContainerCmd.Exec createRestartContainerCmdExec(); + + CommitCmd.Exec createCommitCmdExec(); + + BuildImageCmd.Exec createBuildImageCmdExec(); + + TopContainerCmd.Exec createTopContainerCmdExec(); + + TagImageCmd.Exec createTagImageCmdExec(); + + PauseContainerCmd.Exec createPauseContainerCmdExec(); + + UnpauseContainerCmd.Exec createUnpauseContainerCmdExec(); + + EventsCmd.Exec createEventsCmdExec(); + + StatsCmd.Exec createStatsCmdExec(); + + CreateVolumeCmd.Exec createCreateVolumeCmdExec(); + + InspectVolumeCmd.Exec createInspectVolumeCmdExec(); + + RemoveVolumeCmd.Exec createRemoveVolumeCmdExec(); + + ListVolumesCmd.Exec createListVolumesCmdExec(); + + ListNetworksCmd.Exec createListNetworksCmdExec(); + + InspectNetworkCmd.Exec createInspectNetworkCmdExec(); + + CreateNetworkCmd.Exec createCreateNetworkCmdExec(); + + RemoveNetworkCmd.Exec createRemoveNetworkCmdExec(); + + ConnectToNetworkCmd.Exec createConnectToNetworkCmdExec(); + + DisconnectFromNetworkCmd.Exec createDisconnectFromNetworkCmdExec(); + + // swarm + InitializeSwarmCmd.Exec createInitializeSwarmCmdExec(); + + InspectSwarmCmd.Exec createInspectSwarmCmdExec(); + + JoinSwarmCmd.Exec createJoinSwarmCmdExec(); + + LeaveSwarmCmd.Exec createLeaveSwarmCmdExec(); + + UpdateSwarmCmd.Exec createUpdateSwarmCmdExec(); + + /** + * Command to list all services in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + ListServicesCmd.Exec createListServicesCmdExec(); + + /** + * Command to create a new service in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + CreateServiceCmd.Exec createCreateServiceCmdExec(); + + /** + * Command to inspect a service in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + InspectServiceCmd.Exec createInspectServiceCmdExec(); + + /** + * Command to update a service specification in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + UpdateServiceCmd.Exec createUpdateServiceCmdExec(); + + /** + * Command to remove a service in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + RemoveServiceCmd.Exec createRemoveServiceCmdExec(); + + /** + * @param endpoint endpoint name to tail logs + * @return + * @since {@link RemoteApiVersion#VERSION_1_29} + */ + LogSwarmObjectCmd.Exec logSwarmObjectExec(String endpoint); + + // nodes + + /** + * List all nodes. Node operations require the engine to be part of a swarm + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + ListSwarmNodesCmd.Exec listSwarmNodeCmdExec(); + + /** + * Return low-level information on the node. Node operations require the engine to be part of a swarm + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + InspectSwarmNodeCmd.Exec inspectSwarmNodeCmdExec(); + + /** + * Remove a node from the swarm. Node operations require the engine to be part of a swarm + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + RemoveSwarmNodeCmd.Exec removeSwarmNodeCmdExec(); + + /** + * Update a node. Node operations require the engine to be part of a swarm + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + UpdateSwarmNodeCmd.Exec updateSwarmNodeCmdExec(); + + /** + * Update a node. Node operations require the engine to be part of a swarm + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ + ListTasksCmd.Exec listTasksCmdExec(); + + /** + * Delete unused content (containers, images, volumes, networks, build relicts) + * + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + PruneCmd.Exec pruneCmdExec(); + + /** + * Command to list all secrets. + * + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + ListSecretsCmd.Exec createListSecretsCmdExec(); + + /** + * Command to create a new secret in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + CreateSecretCmd.Exec createCreateSecretCmdExec(); + + /** + * Command to remove a secret in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + RemoveSecretCmd.Exec createRemoveSecretCmdExec(); + + /** + * Command to list all configs. + * + * @since {@link RemoteApiVersion#VERSION_1_30} + */ + ListConfigsCmd.Exec createListConfigsCmdExec(); + + /** + * Command to inspect a config in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_30} + */ + InspectConfigCmd.Exec createInspectConfigCmdExec(); + + /** + * Command to create a new config in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_30} + */ + CreateConfigCmd.Exec createCreateConfigCmdExec(); + + /** + * Command to remove a config in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_30} + */ + RemoveConfigCmd.Exec createRemoveConfigCmdExec(); + + + @Override + void close() throws IOException; + +} diff --git a/src/main/java/com/github/dockerjava/api/command/DockerCmdSyncExec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/DockerCmdSyncExec.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/DockerCmdSyncExec.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/DockerCmdSyncExec.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/EventsCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/EventsCmd.java new file mode 100644 index 000000000..34a0c5ad5 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/EventsCmd.java @@ -0,0 +1,88 @@ +package com.github.dockerjava.api.command; + +import java.util.List; +import java.util.Map; +import java.util.stream.Stream; + +import javax.annotation.CheckForNull; + +import com.github.dockerjava.api.model.Event; +import com.github.dockerjava.api.model.EventType; + +/** + * Get events + */ +public interface EventsCmd extends AsyncDockerCmd { + + @CheckForNull + Map> getFilters(); + + @CheckForNull + String getSince(); + + @CheckForNull + String getUntil(); + + /** + * @param container + * - container to filter + */ + EventsCmd withContainerFilter(String... container); + + /** + * @param event + * - event to filter (pull | create | attach | start | stop | kill) + */ + EventsCmd withEventFilter(String... event); + + /** + * @param eventTypes event types to filter + */ + EventsCmd withEventTypeFilter(String... eventTypes); + + /** + * This provides a type safe version of {@link #withEventTypeFilter(String...)}. + * + * @param eventTypes event types to filter + */ + default EventsCmd withEventTypeFilter(EventType... eventTypes) { + return withEventTypeFilter( + Stream.of(eventTypes) + .map(EventType::getValue) + .toArray(String[]::new) + ); + } + + /** + * @param image + * - image to filter + */ + EventsCmd withImageFilter(String... image); + + /** + * @param label + * - label to filter + */ + EventsCmd withLabelFilter(String... label); + + /** + * @param labels + * - labels to filter (map of names and values) + */ + EventsCmd withLabelFilter(Map labels); + + /** + * @param since + * - Show all events created since timestamp + */ + EventsCmd withSince(String since); + + /** + * @param until + * - Show all events created until timestamp + */ + EventsCmd withUntil(String until); + + interface Exec extends DockerCmdAsyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ExecCreateCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ExecCreateCmd.java new file mode 100644 index 000000000..c03a6334a --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ExecCreateCmd.java @@ -0,0 +1,60 @@ +package com.github.dockerjava.api.command; + +import java.util.List; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +public interface ExecCreateCmd extends SyncDockerCmd { + + @CheckForNull + String getContainerId(); + + @CheckForNull + Boolean hasAttachStderrEnabled(); + + @CheckForNull + Boolean hasAttachStdinEnabled(); + + @CheckForNull + Boolean hasAttachStdoutEnabled(); + + @CheckForNull + Boolean hasTtyEnabled(); + + @CheckForNull + List getEnv(); + + @CheckForNull + String getUser(); + + @CheckForNull + Boolean getPrivileged(); + + @CheckForNull + String getWorkingDir(); + + ExecCreateCmd withAttachStderr(Boolean attachStderr); + + ExecCreateCmd withAttachStdin(Boolean attachStdin); + + ExecCreateCmd withAttachStdout(Boolean attachStdout); + + ExecCreateCmd withCmd(String... cmd); + + ExecCreateCmd withEnv(List env); + + ExecCreateCmd withContainerId(@Nonnull String containerId); + + ExecCreateCmd withTty(Boolean tty); + + ExecCreateCmd withUser(String user); + + ExecCreateCmd withPrivileged(Boolean isPrivileged); + + ExecCreateCmd withWorkingDir(String workingDir); + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ExecCreateCmdResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ExecCreateCmdResponse.java new file mode 100644 index 000000000..449803236 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ExecCreateCmdResponse.java @@ -0,0 +1,18 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@EqualsAndHashCode +@ToString +public class ExecCreateCmdResponse extends DockerObject { + + @JsonProperty("Id") + private String id; + + public String getId() { + return id; + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/ExecStartCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ExecStartCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/ExecStartCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/ExecStartCmd.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/GraphData.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/GraphData.java new file mode 100644 index 000000000..44abc176d --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/GraphData.java @@ -0,0 +1,112 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; + +/** + * part of {@link GraphDriver} + * @author Kanstantsin Shautsou + */ +@EqualsAndHashCode +@ToString +public class GraphData extends DockerObject { + + @JsonProperty("RootDir") + private String rootDir; + + @JsonProperty("DeviceId") + private String deviceId; + + @JsonProperty("DeviceName") + private String deviceName; + + @JsonProperty("DeviceSize") + private String deviceSize; + + @JsonProperty("dir") + private String dir; + + /** + * @see #rootDir + */ + @CheckForNull + public String getRootDir() { + return rootDir; + } + + /** + * @see #rootDir + */ + public GraphData withRootDir(String rootDir) { + this.rootDir = rootDir; + return this; + } + + /** + * @see #deviceId + */ + @CheckForNull + public String getDeviceId() { + return deviceId; + } + + /** + * @see #deviceId + */ + public GraphData withDeviceId(String deviceId) { + this.deviceId = deviceId; + return this; + } + + /** + * @see #deviceName + */ + @CheckForNull + public String getDeviceName() { + return deviceName; + } + + /** + * @see #deviceName + */ + public GraphData withDeviceName(String deviceName) { + this.deviceName = deviceName; + return this; + } + + /** + * @see #deviceSize + */ + @CheckForNull + public String getDeviceSize() { + return deviceSize; + } + + /** + * @see #deviceSize + */ + public GraphData withDeviceSize(String deviceSize) { + this.deviceSize = deviceSize; + return this; + } + + /** + * @see #dir + */ + @CheckForNull + public String getDir() { + return dir; + } + + /** + * @see #dir + */ + public GraphData withDir(String dir) { + this.dir = dir; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/GraphDriver.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/GraphDriver.java new file mode 100644 index 000000000..4d6679416 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/GraphDriver.java @@ -0,0 +1,63 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; + +/** + * Part of {@link InspectImageResponse} and {@link InspectContainerResponse} + * + * @author Kanstantsin Shautsou + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_21} + */ +@EqualsAndHashCode +@ToString +public class GraphDriver extends DockerObject { + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_21} + */ + @JsonProperty("Name") + private String name; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_21} + */ + @JsonProperty("Data") + private GraphData data; + + + /** + * @see #data + */ + @CheckForNull + public GraphData getData() { + return data; + } + + /** + * @see #data + */ + public GraphDriver withData(GraphData data) { + this.data = data; + return this; + } + + /** + * @see #name + */ + @CheckForNull + public String getName() { + return name; + } + + /** + * @see #name + */ + public GraphDriver withName(String name) { + this.name = name; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/HealthState.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/HealthState.java new file mode 100644 index 000000000..0d8e399c1 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/HealthState.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.util.List; + +@EqualsAndHashCode +@ToString +public class HealthState extends DockerObject { + + @JsonProperty("Status") + private String status; + + @JsonProperty("FailingStreak") + private Integer failingStreak; + + @JsonProperty("Log") + private List log; + + public String getStatus() { + return status; + } + + public Integer getFailingStreak() { + return failingStreak; + } + + public List getLog() { + return log; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/HealthStateLog.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/HealthStateLog.java new file mode 100644 index 000000000..71939f872 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/HealthStateLog.java @@ -0,0 +1,48 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@EqualsAndHashCode +@ToString +public class HealthStateLog extends DockerObject { + + @JsonProperty("Start") + private String start; + + @JsonProperty("End") + private String end; + + @JsonProperty("ExitCode") + private Long exitCode; + + @JsonProperty("Output") + private String output; + + public String getStart() { + return start; + } + + public String getEnd() { + return end; + } + + /** + * + * @deprecated use {@link #getExitCodeLong()} + */ + @Deprecated + public Integer getExitCode() { + return exitCode != null ? exitCode.intValue() : null; + } + + public Long getExitCodeLong() { + return exitCode; + } + + public String getOutput() { + return output; + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/InfoCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InfoCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/InfoCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/InfoCmd.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/InitializeSwarmCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InitializeSwarmCmd.java new file mode 100644 index 000000000..ff0fabf99 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InitializeSwarmCmd.java @@ -0,0 +1,36 @@ +package com.github.dockerjava.api.command; + + +import com.github.dockerjava.api.model.SwarmSpec; + +import javax.annotation.CheckForNull; + +public interface InitializeSwarmCmd extends SyncDockerCmd { + + @CheckForNull + String getListenAddr(); + + InitializeSwarmCmd withListenAddr(String listenAddr); + + @CheckForNull + String getAdvertiseAddr(); + + InitializeSwarmCmd withAdvertiseAddr(String advertiseAddr); + + @CheckForNull + Boolean isForceNewCluster(); + + InitializeSwarmCmd withForceNewCluster(Boolean forceNewCluster); + + @CheckForNull + SwarmSpec getSwarmSpec(); + + InitializeSwarmCmd withSwarmSpec(SwarmSpec swarmSpec); + + @Override + Void exec(); + + interface Exec extends DockerCmdSyncExec { + + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectConfigCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectConfigCmd.java new file mode 100644 index 000000000..96374d795 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectConfigCmd.java @@ -0,0 +1,25 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Config; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +public interface InspectConfigCmd extends SyncDockerCmd { + + @CheckForNull + String getConfigId(); + + InspectConfigCmd withConfigId(@Nonnull String configId); + + /** + * @throws NotFoundException + * No such config + */ + @Override + Config exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/InspectContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/InspectContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectContainerCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/InspectContainerResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectContainerResponse.java similarity index 75% rename from src/main/java/com/github/dockerjava/api/command/InspectContainerResponse.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectContainerResponse.java index 59a267e9d..f06bd4ed9 100644 --- a/src/main/java/com/github/dockerjava/api/command/InspectContainerResponse.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectContainerResponse.java @@ -1,17 +1,9 @@ package com.github.dockerjava.api.command; -import java.util.List; - -import javax.annotation.CheckForNull; - -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonProperty; import com.github.dockerjava.api.model.ContainerConfig; +import com.github.dockerjava.api.model.DockerObject; import com.github.dockerjava.api.model.HostConfig; import com.github.dockerjava.api.model.NetworkSettings; import com.github.dockerjava.api.model.Volume; @@ -19,15 +11,21 @@ import com.github.dockerjava.api.model.VolumeBinds; import com.github.dockerjava.api.model.VolumeRW; import com.github.dockerjava.api.model.VolumesRW; -import com.github.dockerjava.core.RemoteApiVersion; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.util.List; +import java.util.Map; /** * * @author Konstantin Pelykh (kpelykh@gmail.com) * */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class InspectContainerResponse { +@EqualsAndHashCode +@ToString +public class InspectContainerResponse extends DockerObject { @JsonProperty("Args") private String[] args; @@ -53,11 +51,20 @@ public class InspectContainerResponse { @JsonProperty("HostsPath") private String hostsPath; + /** + * @since {@link RemoteApiVersion#VERSION_1_17} + */ + @JsonProperty("LogPath") + private String logPath; + @JsonProperty("Id") private String id; @JsonProperty("SizeRootFs") - private Integer sizeRootFs; + private Long sizeRootFs; + + @JsonProperty("SizeRw") + private Long sizeRw; @JsonProperty("Image") private String imageId; @@ -98,15 +105,31 @@ public class InspectContainerResponse { @JsonProperty("VolumesRW") private VolumesRW volumesRW; + @JsonProperty("Node") + private Node node; + @JsonProperty("Mounts") private List mounts; + @JsonProperty("GraphDriver") + private GraphDriver graphDriver; + + /** + * @since {@link RemoteApiVersion#VERSION_1_30} + */ + @JsonProperty("Platform") + private String platform; + public String getId() { return id; } - public Integer getSizeRootFs() { - return sizeRootFs; + public Long getSizeRootFs() { + return sizeRootFs; + } + + public Long getSizeRw() { + return sizeRw; } public String getCreated() { @@ -168,6 +191,11 @@ public String getHostsPath() { return hostsPath; } + @CheckForNull + public String getLogPath() { + return logPath; + } + public String getName() { return name; } @@ -204,13 +232,34 @@ public List getExecIds() { return execIds; } - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); + /** + * Get the underlying swarm node info. This property does only contains a value in swarm-classic + * @return The underlying swarm-classic node info + * @CheckForNull + */ + public Node getNode() { + return node; } - @JsonIgnoreProperties(ignoreUnknown = true) - public class ContainerState { + /** + * @see #graphDriver + */ + @CheckForNull + public GraphDriver getGraphDriver() { + return graphDriver; + } + + /** + * @see #platform + */ + @CheckForNull + public String getPlatform() { + return platform; + } + + @EqualsAndHashCode + @ToString + public class ContainerState extends DockerObject { /** * @since {@link RemoteApiVersion#VERSION_1_20} @@ -261,14 +310,14 @@ public class ContainerState { */ @CheckForNull @JsonProperty("Pid") - private Integer pid; + private Long pid; /** * @since < {@link RemoteApiVersion#VERSION_1_16} */ @CheckForNull @JsonProperty("ExitCode") - private Integer exitCode; + private Long exitCode; /** * @since {@link RemoteApiVersion#VERSION_1_17} @@ -291,6 +340,13 @@ public class ContainerState { @JsonProperty("FinishedAt") private String finishedAt; + + /** + * @since Docker version 1.12 + */ + @JsonProperty("Health") + private HealthState health; + /** * See {@link #status} */ @@ -341,17 +397,39 @@ public Boolean getDead() { /** * See {@link #pid} + * + * @deprecated use {@link #getPidLong()} */ + @Deprecated @CheckForNull public Integer getPid() { + return pid != null ? pid.intValue() : null; + } + + /** + * See {@link #pid} + */ + @CheckForNull + public Long getPidLong() { return pid; } /** * See {@link #exitCode} + * + * @deprecated use {@link #getExitCodeLong()} */ + @Deprecated @CheckForNull public Integer getExitCode() { + return exitCode != null ? exitCode.intValue() : null; + } + + /** + * See {@link #exitCode} + */ + @CheckForNull + public Long getExitCodeLong() { return exitCode; } @@ -379,24 +457,14 @@ public String getFinishedAt() { return finishedAt; } - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); + public HealthState getHealth() { + return health; } } - @JsonIgnoreProperties(ignoreUnknown = true) - public static class Mount { + @EqualsAndHashCode + @ToString + public static class Mount extends DockerObject { /** * @since {@link RemoteApiVersion#VERSION_1_20} @@ -517,20 +585,59 @@ public Mount withSource(String source) { this.source = source; return this; } + } + + @EqualsAndHashCode + @ToString + public class Node extends DockerObject { + + @JsonProperty("ID") + private String id; + + @JsonProperty("IP") + private String ip; + + @JsonProperty("Addr") + private String addr; + + @JsonProperty("Name") + private String name; + + @JsonProperty("Cpus") + private Integer cpus; + + @JsonProperty("Memory") + private Long memory; + + @JsonProperty("Labels") + private Map labels; + + public String getId() { + return id; + } + + public String getIp() { + return ip; + } + + public String getAddr() { + return addr; + } + + public String getName() { + return name; + } - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); + public Integer getCpus() { + return cpus; } - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); + public Long getMemory() { + return memory; } - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); + public Map getLabels() { + return labels; } } } diff --git a/src/main/java/com/github/dockerjava/api/command/InspectExecCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectExecCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/InspectExecCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectExecCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/InspectExecResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectExecResponse.java similarity index 76% rename from src/main/java/com/github/dockerjava/api/command/InspectExecResponse.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectExecResponse.java index 2833a5ade..307fdd873 100644 --- a/src/main/java/com/github/dockerjava/api/command/InspectExecResponse.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectExecResponse.java @@ -2,17 +2,17 @@ import java.util.List; -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; import com.github.dockerjava.api.model.NetworkSettings; -import com.github.dockerjava.core.RemoteApiVersion; +import lombok.EqualsAndHashCode; +import lombok.ToString; import javax.annotation.CheckForNull; -@JsonIgnoreProperties(ignoreUnknown = true) -public class InspectExecResponse { +@EqualsAndHashCode +@ToString +public class InspectExecResponse extends DockerObject { @JsonProperty("ID") private String id; @@ -35,7 +35,7 @@ public class InspectExecResponse { private Boolean canRemove; @JsonProperty("ExitCode") - private Integer exitCode; + private Long exitCode; @JsonProperty("ProcessConfig") private ProcessConfig processConfig; @@ -59,6 +59,12 @@ public class InspectExecResponse { @JsonProperty("DetachKeys") private String detachKeys; + /** + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + @JsonProperty("Pid") + private Long pid; + public String getId() { return id; } @@ -79,7 +85,15 @@ public Boolean isRunning() { return running; } + /** + * @deprecated use {@link #getExitCodeLong()} + */ + @Deprecated public Integer getExitCode() { + return exitCode != null ? exitCode.intValue() : null; + } + + public Long getExitCodeLong() { return exitCode; } @@ -119,13 +133,27 @@ public String getDetachKeys() { return detachKeys; } - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); + /** + * @see #pid + * @deprecated use {@link #getPidLong()} + */ + @CheckForNull + @Deprecated + public Integer getPid() { + return pid != null ? pid.intValue() : null; + } + + /** + * @see #pid + */ + @CheckForNull + public Long getPidLong() { + return pid; } - @JsonIgnoreProperties(ignoreUnknown = true) - public class ProcessConfig { + @EqualsAndHashCode + @ToString + public class ProcessConfig extends DockerObject { @JsonProperty("arguments") private List arguments; @@ -161,15 +189,11 @@ public Boolean isTty() { public String getUser() { return user; } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } } - @JsonIgnoreProperties(ignoreUnknown = true) - public class Container { + @EqualsAndHashCode + @ToString + public class Container extends DockerObject { @JsonProperty("NetworkSettings") private NetworkSettings networkSettings; diff --git a/src/main/java/com/github/dockerjava/api/command/InspectImageCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectImageCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/InspectImageCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectImageCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/InspectImageResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectImageResponse.java similarity index 84% rename from src/main/java/com/github/dockerjava/api/command/InspectImageResponse.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectImageResponse.java index 740a1b832..bf48ba8f0 100644 --- a/src/main/java/com/github/dockerjava/api/command/InspectImageResponse.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectImageResponse.java @@ -1,12 +1,10 @@ package com.github.dockerjava.api.command; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonProperty; import com.github.dockerjava.api.model.ContainerConfig; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; import javax.annotation.CheckForNull; import java.util.List; @@ -16,8 +14,9 @@ * @author Konstantin Pelykh (kpelykh@gmail.com) * */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class InspectImageResponse { +@EqualsAndHashCode +@ToString +public class InspectImageResponse extends DockerObject { @JsonProperty("Architecture") private String arch; @@ -49,6 +48,12 @@ public class InspectImageResponse { @JsonProperty("Os") private String os; + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_25} + */ + @JsonProperty("OsVersion") + private String osVersion; + @JsonProperty("Parent") private String parent; @@ -73,6 +78,9 @@ public class InspectImageResponse { @JsonProperty("GraphDriver") private GraphDriver graphDriver; + @JsonProperty("RootFS") + private RootFS rootFS; + /** * @see #arch */ @@ -122,6 +130,7 @@ public InspectImageResponse withComment(String comment) { } /** + * Get the image commit configuration * @see #config */ @CheckForNull @@ -154,6 +163,8 @@ public InspectImageResponse withContainer(String container) { } /** + * If the image was created from a container, this config contains the configuration of the container + * which was committed in this image * @see #containerConfig */ @CheckForNull @@ -233,6 +244,22 @@ public InspectImageResponse withOs(String os) { return this; } + /** + * @see #osVersion + */ + @CheckForNull + public String getOsVersion() { + return osVersion; + } + + /** + * @see #osVersion + */ + public InspectImageResponse withOsVersion(String osVersion) { + this.osVersion = osVersion; + return this; + } + /** * @see #parent */ @@ -329,18 +356,19 @@ public InspectImageResponse withVirtualSize(Long virtualSize) { return this; } - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); + /** + * @see #rootFS + */ + @CheckForNull + public RootFS getRootFS() { + return rootFS; } - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); + /** + * @see #rootFS + */ + public InspectImageResponse withRootFS(RootFS rootFS) { + this.rootFS = rootFS; + return this; } } diff --git a/src/main/java/com/github/dockerjava/api/command/InspectNetworkCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectNetworkCmd.java similarity index 93% rename from src/main/java/com/github/dockerjava/api/command/InspectNetworkCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectNetworkCmd.java index 2e5536731..9dd462ac4 100644 --- a/src/main/java/com/github/dockerjava/api/command/InspectNetworkCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectNetworkCmd.java @@ -2,7 +2,6 @@ import com.github.dockerjava.api.exception.NotFoundException; import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.core.RemoteApiVersion; import javax.annotation.CheckForNull; import javax.annotation.Nonnull; diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectServiceCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectServiceCmd.java new file mode 100644 index 000000000..74c52c1f3 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectServiceCmd.java @@ -0,0 +1,25 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Service; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +public interface InspectServiceCmd extends SyncDockerCmd { + + @CheckForNull + String getServiceId(); + + InspectServiceCmd withServiceId(@Nonnull String serviceId); + + /** + * @throws NotFoundException + * No such service + */ + @Override + Service exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectSwarmCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectSwarmCmd.java new file mode 100644 index 000000000..9342d9969 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectSwarmCmd.java @@ -0,0 +1,18 @@ +package com.github.dockerjava.api.command; + + +import com.github.dockerjava.api.model.Swarm; + +/** + * Inspect a swarm. + */ +public interface InspectSwarmCmd extends SyncDockerCmd { + + @Override + Swarm exec(); + + interface Exec extends DockerCmdSyncExec { + } + + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectSwarmNodeCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectSwarmNodeCmd.java new file mode 100644 index 000000000..07750f09e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectSwarmNodeCmd.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.api.command; + + +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.SwarmNode; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * Inspect a swarmNode. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public interface InspectSwarmNodeCmd extends SyncDockerCmd { + + @CheckForNull + String getSwarmNodeId(); + + InspectSwarmNodeCmd withSwarmNodeId(@Nonnull String swarmNodeId); + + /** + * @throws NotFoundException No such swarmNode + */ + @Override + SwarmNode exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectTaskCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectTaskCmd.java new file mode 100644 index 000000000..118910eee --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectTaskCmd.java @@ -0,0 +1,19 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Task; + +import javax.annotation.CheckForNull; + +public interface InspectTaskCmd extends SyncDockerCmd { + @CheckForNull + String getTaskId(); + + InspectTaskCmd withTaskId(); + + @Override + Task exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/InspectVolumeCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectVolumeCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/InspectVolumeCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectVolumeCmd.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectVolumeResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectVolumeResponse.java new file mode 100644 index 000000000..bc0008817 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/InspectVolumeResponse.java @@ -0,0 +1,53 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.util.Map; + +/** + * + * @author Marcus Linke + */ +@EqualsAndHashCode +@ToString +public class InspectVolumeResponse extends DockerObject { + + @JsonProperty("Name") + private String name; + + @JsonProperty("Labels") + private Map labels; + + @JsonProperty("Driver") + private String driver; + + @JsonProperty("Mountpoint") + private String mountpoint; + + @JsonProperty("Options") + private Map options; + + public String getName() { + return name; + } + + public Map getLabels() { + return labels; + } + + public String getDriver() { + return driver; + } + + public String getMountpoint() { + return mountpoint; + } + + public Map getOptions() { + return options; + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/JoinSwarmCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/JoinSwarmCmd.java new file mode 100644 index 000000000..cc2412a58 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/JoinSwarmCmd.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.api.command; + + +import javax.annotation.CheckForNull; +import java.util.List; + +public interface JoinSwarmCmd extends SyncDockerCmd { + + @CheckForNull + String getListenAddr(); + + JoinSwarmCmd withListenAddr(String listenAddr); + + @CheckForNull + String getAdvertiseAddr(); + + JoinSwarmCmd withAdvertiseAddr(String advertiseAddr); + + @CheckForNull + List getRemoteAddrs(); + + JoinSwarmCmd withRemoteAddrs(List remoteAddrs); + + @CheckForNull + String getJoinToken(); + + JoinSwarmCmd withJoinToken(String joinToken); + + @Override + Void exec(); + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/KillContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/KillContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/KillContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/KillContainerCmd.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/LeaveSwarmCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LeaveSwarmCmd.java new file mode 100644 index 000000000..11a5b80d9 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LeaveSwarmCmd.java @@ -0,0 +1,18 @@ +package com.github.dockerjava.api.command; + + +import javax.annotation.CheckForNull; + +public interface LeaveSwarmCmd extends SyncDockerCmd { + + @CheckForNull + Boolean hasForceEnabled(); + + LeaveSwarmCmd withForceEnabled(Boolean force); + + @Override + Void exec(); + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListConfigsCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListConfigsCmd.java new file mode 100644 index 000000000..38d34816a --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListConfigsCmd.java @@ -0,0 +1,21 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.Config; + +import java.util.List; +import java.util.Map; + +/** + * Command to list all configs in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_30} + */ +public interface ListConfigsCmd extends SyncDockerCmd> { + + Map> getFilters(); + + ListConfigsCmd withFilters(Map> filters); + + interface Exec extends DockerCmdSyncExec> { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListContainersCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListContainersCmd.java new file mode 100644 index 000000000..879ec711c --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListContainersCmd.java @@ -0,0 +1,128 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.Container; + +import javax.annotation.CheckForNull; +import java.util.Collection; +import java.util.List; +import java.util.Map; + +/** + * List containers + * + */ +public interface ListContainersCmd extends SyncDockerCmd> { + + @CheckForNull + String getBeforeId(); + + @CheckForNull + Map> getFilters(); + + @CheckForNull + Integer getLimit(); + + @CheckForNull + String getSinceId(); + + @CheckForNull + Boolean hasShowAllEnabled(); + + @CheckForNull + Boolean hasShowSizeEnabled(); + + /** + * @param before + * - Show only containers created before Id, include non-running ones. + */ + ListContainersCmd withBefore(String before); + + /** + * @param name + * - Show only containers that has the container's name + */ + ListContainersCmd withNameFilter(Collection name); + + /** + * @param id + * - Show only containers that has the container's id + */ + ListContainersCmd withIdFilter(Collection id); + + /** + * @param ancestor + * - Show only containers created from an image or a descendant. + */ + ListContainersCmd withAncestorFilter(Collection ancestor); + + /** + * @param volume + * - Show only containers with volume name or mount point destination + */ + ListContainersCmd withVolumeFilter(Collection volume); + + /** + * @param network + * - Show only containers with network id or network name + */ + ListContainersCmd withNetworkFilter(Collection network); + + /** + * @param exited + * - Show only containers that exited with the passed exitcode. + */ + ListContainersCmd withExitedFilter(Integer exited); + + /** + * @param status + * - Show only containers with the passed status (created|restarting|running|paused|exited). + */ + ListContainersCmd withStatusFilter(Collection status); + + /** + * @param labels + * - Show only containers with the passed labels. + */ + ListContainersCmd withLabelFilter(Collection labels); + + /** + * @param labels + * - Show only containers with the passed labels. Labels is a {@link Map} that contains label keys and values + */ + ListContainersCmd withLabelFilter(Map labels); + + /** + * @param limit + * - Show `limit` last created containers, include non-running ones. There is no limit by default. + */ + ListContainersCmd withLimit(Integer limit); + + /** + * @param showAll + * - Show all containers. Only running containers are shown by default. + */ + ListContainersCmd withShowAll(Boolean showAll); + + /** + * @param showSize + * - Show the containers sizes. This is false by default. + */ + ListContainersCmd withShowSize(Boolean showSize); + + /** + * @param since + * - Show only containers created since Id, include non-running ones. + */ + ListContainersCmd withSince(String since); + + /** + * @param filterName + * @param filterValues + * - Show only containers where the filter matches the given values + */ + ListContainersCmd withFilter(String filterName, Collection filterValues); + + interface Exec extends DockerCmdSyncExec> { + } + +} diff --git a/src/main/java/com/github/dockerjava/api/command/ListImagesCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListImagesCmd.java similarity index 80% rename from src/main/java/com/github/dockerjava/api/command/ListImagesCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/ListImagesCmd.java index 7741df743..cc60a5bcc 100644 --- a/src/main/java/com/github/dockerjava/api/command/ListImagesCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListImagesCmd.java @@ -1,5 +1,6 @@ package com.github.dockerjava.api.command; +import java.util.Collection; import java.util.List; import java.util.Map; @@ -44,6 +45,15 @@ public interface ListImagesCmd extends SyncDockerCmd> { */ ListImagesCmd withLabelFilter(Map labels); + /** + * Filter images by reference + * + * @param reference string in the form {@code [:]} + */ + ListImagesCmd withReferenceFilter(String reference); + + ListImagesCmd withFilter(String key, Collection values); + interface Exec extends DockerCmdSyncExec> { } diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListNetworksCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListNetworksCmd.java new file mode 100644 index 000000000..a7921824d --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListNetworksCmd.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.Network; + +import javax.annotation.CheckForNull; + +import java.util.Collection; +import java.util.List; +import java.util.Map; + +/** + * List networks. + * + * @since {@link RemoteApiVersion#VERSION_1_21} + */ +public interface ListNetworksCmd extends SyncDockerCmd> { + + @CheckForNull + Map> getFilters(); + + ListNetworksCmd withNameFilter(String... networkName); + + ListNetworksCmd withIdFilter(String... networkId); + + /** + * @param filterName + * @param filterValues + * - Show only networks where the filter matches the given values + */ + ListNetworksCmd withFilter(String filterName, Collection filterValues); + + interface Exec extends DockerCmdSyncExec> { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListSecretsCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListSecretsCmd.java new file mode 100644 index 000000000..959d68831 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListSecretsCmd.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.Secret; + +import javax.annotation.CheckForNull; +import java.util.List; +import java.util.Map; + +/** + * List secrets + */ +public interface ListSecretsCmd extends SyncDockerCmd> { + + @CheckForNull + Map> getFilters(); + + /** + * @param ids - Show only secrets with the given ids + */ + ListSecretsCmd withIdFilter(List ids); + + /** + * @param names - Show only secrets with the given names + */ + ListSecretsCmd withNameFilter(List names); + + /** + * @param labels - Show only secrets with the passed labels. Labels is a {@link Map} that contains label keys and values + */ + ListSecretsCmd withLabelFilter(Map labels); + + interface Exec extends DockerCmdSyncExec> { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListServicesCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListServicesCmd.java new file mode 100644 index 000000000..5926add1e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListServicesCmd.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.Service; + +import javax.annotation.CheckForNull; +import java.util.List; +import java.util.Map; + +/** + * Command to list all services in a docker swarm. Only applicable if docker runs in swarm mode. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public interface ListServicesCmd extends SyncDockerCmd> { + + @CheckForNull + Map> getFilters(); + + /** + * @param ids - Show only services with the given ids + */ + ListServicesCmd withIdFilter(List ids); + + /** + * @param names - Show only services with the given names + */ + ListServicesCmd withNameFilter(List names); + + /** + * @param labels - Show only services with the passed labels. Labels is a {@link Map} that contains label keys and values + */ + ListServicesCmd withLabelFilter(Map labels); + + interface Exec extends DockerCmdSyncExec> { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListSwarmNodesCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListSwarmNodesCmd.java new file mode 100644 index 000000000..6ad0205d3 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListSwarmNodesCmd.java @@ -0,0 +1,42 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.SwarmNode; + +import javax.annotation.CheckForNull; +import java.util.List; +import java.util.Map; + +/** + * List SwarmNodes + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public interface ListSwarmNodesCmd extends SyncDockerCmd> { + + @CheckForNull + Map> getFilters(); + + /** + * @param ids - Show only swarmNodes with the given ids + */ + ListSwarmNodesCmd withIdFilter(List ids); + + /** + * @param names - Show only swarmNodes with the given names + */ + ListSwarmNodesCmd withNameFilter(List names); + + /** + * @param memberships - Show only swarmNodes with the given memberships + */ + ListSwarmNodesCmd withMembershipFilter(List memberships); + + /** + * @param roles - Show only swarmNodes with the given roles + */ + ListSwarmNodesCmd withRoleFilter(List roles); + + interface Exec extends DockerCmdSyncExec> { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListTasksCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListTasksCmd.java new file mode 100644 index 000000000..8037ded5d --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListTasksCmd.java @@ -0,0 +1,56 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Task; +import com.github.dockerjava.api.model.TaskState; + +import javax.annotation.CheckForNull; +import java.util.List; +import java.util.Map; + +public interface ListTasksCmd extends SyncDockerCmd> { + @CheckForNull + Map> getFilters(); + + /** + * @param labels - Show only tasks with the passed labels. + * Labels is a {@link Map} that contains label keys and values + */ + ListTasksCmd withLabelFilter(Map labels); + + /** + * @param labels - Show only tasks with the passed labels. + */ + ListTasksCmd withLabelFilter(String... labels); + + /** + * @param ids Task id(s) + */ + ListTasksCmd withIdFilter(String... ids); + + /** + * @param names Task name(s) + */ + ListTasksCmd withNameFilter(String... names); + + /** + * @param nodeNames Node id(s) or name(s) + */ + ListTasksCmd withNodeFilter(String... nodeNames); + + /** + * @param serviceNames Service name(s) + */ + ListTasksCmd withServiceFilter(String... serviceNames); + + /** + * @param desiredState The desired-state filter can take the values running, shutdown, or accepted. + */ + ListTasksCmd withStateFilter(TaskState... desiredState); + + @Override + List exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec> { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListVolumesCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListVolumesCmd.java new file mode 100644 index 000000000..bd66653eb --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListVolumesCmd.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.api.command; + +import java.util.Collection; +import java.util.List; +import java.util.Map; + +import javax.annotation.CheckForNull; + +/** + * List volumes. + * + * @author Marcus Linke + */ +public interface ListVolumesCmd extends SyncDockerCmd { + + @CheckForNull + Map> getFilters(); + + /** + * @param dangling + * - Show dangling volumes filter + */ + ListVolumesCmd withDanglingFilter(Boolean dangling); + + /** + * @param filterName + * @param filterValues + * - Show only volumes where the filter matches the given values + */ + ListVolumesCmd withFilter(String filterName, Collection filterValues); + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListVolumesResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListVolumesResponse.java new file mode 100644 index 000000000..7c434a48d --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ListVolumesResponse.java @@ -0,0 +1,24 @@ +package com.github.dockerjava.api.command; + +import java.util.List; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +/** + * + * @author Marcus Linke + */ +@EqualsAndHashCode +@ToString +public class ListVolumesResponse extends DockerObject { + + @JsonProperty("Volumes") + private List volumes; + + public List getVolumes() { + return volumes; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/LoadImageAsyncCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LoadImageAsyncCmd.java new file mode 100644 index 000000000..4f054db22 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LoadImageAsyncCmd.java @@ -0,0 +1,22 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.LoadResponseItem; + +import java.io.InputStream; + +public interface LoadImageAsyncCmd extends AsyncDockerCmd { + InputStream getImageStream(); + + /** + * @param imageStream the InputStream of the tar file + */ + LoadImageAsyncCmd withImageStream(InputStream imageStream); + + @Override + default LoadImageCallback start() { + return exec(new LoadImageCallback()); + } + + interface Exec extends DockerCmdAsyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/LoadImageCallback.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LoadImageCallback.java new file mode 100644 index 000000000..80cca18de --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LoadImageCallback.java @@ -0,0 +1,49 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.async.ResultCallbackTemplate; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.model.LoadResponseItem; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class LoadImageCallback extends ResultCallbackTemplate { + + private static final Logger LOGGER = LoggerFactory.getLogger(LoadImageCallback.class); + + private String message; + + private String error; + + @Override + public void onNext(LoadResponseItem item) { + if (item.isBuildSuccessIndicated()) { + this.message = item.getMessage(); + } else if (item.isErrorIndicated()) { + this.error = item.getError(); + } + + LOGGER.debug("{}", item); + } + + public String awaitMessage() { + try { + awaitCompletion(); + } catch (InterruptedException e) { + throw new DockerClientException("", e); + } + + return getMessage(); + } + + private String getMessage() { + if (this.message != null) { + return this.message; + } + + if (this.error == null) { + throw new DockerClientException("Could not build image"); + } + + throw new DockerClientException("Could not build image: " + this.error); + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/LoadImageCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LoadImageCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/LoadImageCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/LoadImageCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/LogContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LogContainerCmd.java similarity index 92% rename from src/main/java/com/github/dockerjava/api/command/LogContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/LogContainerCmd.java index 53674cbaa..29c4516cd 100644 --- a/src/main/java/com/github/dockerjava/api/command/LogContainerCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LogContainerCmd.java @@ -25,6 +25,8 @@ * @param since * - UNIX timestamp (integer) to filter logs. Specifying a timestamp will only output log-entries since that timestamp. Default: * 0 (unfiltered) + * @param until + * - Only return logs before this time, as a UNIX timestamp. Default: 0 */ public interface LogContainerCmd extends AsyncDockerCmd { @@ -49,6 +51,9 @@ public interface LogContainerCmd extends AsyncDockerCmd @CheckForNull Integer getSince(); + @CheckForNull + Integer getUntil(); + LogContainerCmd withContainerId(@Nonnull String containerId); /** @@ -69,6 +74,8 @@ public interface LogContainerCmd extends AsyncDockerCmd LogContainerCmd withSince(Integer since); + LogContainerCmd withUntil(Integer until); + /** * @throws com.github.dockerjava.api.NotFoundException * No such container diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/LogSwarmObjectCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LogSwarmObjectCmd.java new file mode 100644 index 000000000..f5e738518 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/LogSwarmObjectCmd.java @@ -0,0 +1,94 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.model.Frame; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * Get docker service/task stdout/stderr logs + */ +public interface LogSwarmObjectCmd extends AsyncDockerCmd { + /** + * @param id ID or name of the service + * @return self + */ + LogSwarmObjectCmd withId(@Nonnull String id); + + /** + * @param follow Return the logs as a raw stream. + * @return self + */ + LogSwarmObjectCmd withFollow(Boolean follow); + + /** + * @param timestamps Add timestamps to every log line + * @return self + */ + LogSwarmObjectCmd withTimestamps(Boolean timestamps); + + /** + * @param stdout Return logs from stdout + * @return self + */ + LogSwarmObjectCmd withStdout(Boolean stdout); + + /** + * @param stderr Return logs from stderr + * @return self + */ + LogSwarmObjectCmd withStderr(Boolean stderr); + + /** + * @param tail only return this number of log lines from the end of the logs. + * @return self + */ + LogSwarmObjectCmd withTail(Integer tail); + + /** + * @param since Only return logs since this time, as a UNIX timestamp + * @return self + */ + LogSwarmObjectCmd withSince(Integer since); + + /** + * @param details Show service context and extra details provided to logs. + * @return + */ + LogSwarmObjectCmd withDetails(Boolean details); + + @CheckForNull + String getId(); + + @CheckForNull + Integer getTail(); + + @CheckForNull + Boolean getFollow(); + + @CheckForNull + Boolean getTimestamps(); + + @CheckForNull + Boolean getStdout(); + + @CheckForNull + Boolean getStderr(); + + @CheckForNull + Integer getSince(); + + @CheckForNull + Boolean getDetails(); + + /** + * @throws com.github.dockerjava.api.exception.NotFoundException no such service + */ + @Override + > T exec(T resultCallback); + + interface Exec extends DockerCmdAsyncExec { + } + +} diff --git a/src/main/java/com/github/dockerjava/api/command/PauseContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/PauseContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/PauseContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/PauseContainerCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/PingCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/PingCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/PingCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/PingCmd.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/PruneCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/PruneCmd.java new file mode 100644 index 000000000..89a082eaf --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/PruneCmd.java @@ -0,0 +1,53 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.PruneResponse; +import com.github.dockerjava.api.model.PruneType; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import java.util.List; +import java.util.Map; + +/** + * Delete unused content (containers, images, volumes, networks, build relicts) + * + * @since {@link RemoteApiVersion#VERSION_1_25} + */ +public interface PruneCmd extends SyncDockerCmd { + + @Nonnull + PruneType getPruneType(); + + @Nonnull + String getApiPath(); + + @CheckForNull + Map> getFilters(); + + PruneCmd withPruneType(final PruneType pruneType); + /** + * Prune containers created before this timestamp + * Meaningful only for CONTAINERS and IMAGES prune type + * @param until Can be Unix timestamps, date formatted timestamps, + * or Go duration strings (e.g. 10m, 1h30m) computed relative to the daemon machine’s time. + */ + PruneCmd withUntilFilter(String until); + + /** + * When set to true, prune only unused and untagged images. When set to false, all unused images are pruned. + * Meaningful only for IMAGES prune type + */ + PruneCmd withDangling(Boolean dangling); + + /** + * Prune containers with the specified labels + */ + PruneCmd withLabelFilter(String... label); + + @Override + PruneResponse exec(); + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/PullImageCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/PullImageCmd.java new file mode 100644 index 000000000..d86bddfe3 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/PullImageCmd.java @@ -0,0 +1,52 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.PullResponseItem; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * + * Pull image from repository. + * + */ +public interface PullImageCmd extends AsyncDockerCmd { + + @CheckForNull + String getRepository(); + + @CheckForNull + String getTag(); + + @CheckForNull + String getPlatform(); + + @CheckForNull + String getRegistry(); + + @CheckForNull + AuthConfig getAuthConfig(); + + PullImageCmd withRepository(@Nonnull String repository); + + PullImageCmd withTag(String tag); + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_32} + */ + PullImageCmd withPlatform(String tag); + + PullImageCmd withRegistry(String registry); + + PullImageCmd withAuthConfig(AuthConfig authConfig); + + @Override + default ResultCallback.Adapter start() { + return exec(new PullImageResultCallback()); + } + + interface Exec extends DockerCmdAsyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/PullImageResultCallback.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/PullImageResultCallback.java new file mode 100644 index 000000000..5980ce3df --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/PullImageResultCallback.java @@ -0,0 +1,115 @@ +/* + * Created on 21.07.2015 + */ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.model.PullResponseItem; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.annotation.CheckForNull; +import java.util.HashMap; +import java.util.Map; + +/** + * + * @author Marcus Linke + * + */ +public class PullImageResultCallback extends ResultCallback.Adapter { + + private static final Logger LOGGER = LoggerFactory.getLogger(PullImageResultCallback.class); + + private boolean isSwarm = false; + private Map results = null; + + @CheckForNull + private PullResponseItem latestItem = null; + + @Override + public void onNext(PullResponseItem item) { + // only do it once + if (results == null && latestItem == null) { + checkForDockerSwarmResponse(item); + } + + if (isSwarm) { + handleDockerSwarmResponse(item); + } else { + handleDockerClientResponse(item); + } + + LOGGER.debug("{}", item); + } + + private void checkForDockerSwarmResponse(PullResponseItem item) { + if (item.getStatus().matches("Pulling\\s.+\\.{3}$")) { + isSwarm = true; + LOGGER.debug("Communicating with Docker Swarm."); + } + } + + private void handleDockerSwarmResponse(final PullResponseItem item) { + if (results == null) { + results = new HashMap<>(); + } + + // Swarm terminates a pull sometimes with an empty line. + // Therefore keep first success message + PullResponseItem currentItem = results.get(item.getId()); + if (currentItem == null || !currentItem.isPullSuccessIndicated()) { + results.put(item.getId(), item); + } + } + + private void handleDockerClientResponse(PullResponseItem item) { + latestItem = item; + } + + private void checkDockerSwarmPullSuccessful() { + if (results.isEmpty()) { + throw new DockerClientException("Could not pull image through Docker Swarm"); + } else { + boolean pullFailed = false; + StringBuilder sb = new StringBuilder(); + + for (PullResponseItem pullResponseItem : results.values()) { + if (!pullResponseItem.isPullSuccessIndicated()) { + pullFailed = true; + sb.append("[" + pullResponseItem.getId() + ":" + messageFromPullResult(pullResponseItem) + "]"); + } + } + + if (pullFailed) { + throw new DockerClientException("Could not pull image: " + sb.toString()); + } + } + } + + private void checkDockerClientPullSuccessful() { + if (latestItem == null) { + return; + } + + if (!latestItem.isPullSuccessIndicated()) { + throw new DockerClientException("Could not pull image: " + messageFromPullResult(latestItem)); + } + } + + private String messageFromPullResult(PullResponseItem pullResponseItem) { + return (pullResponseItem.getError() != null) ? pullResponseItem.getError() : pullResponseItem.getStatus(); + } + + @Override + protected void throwFirstError() { + super.throwFirstError(); + + if (isSwarm) { + checkDockerSwarmPullSuccessful(); + } else { + checkDockerClientPullSuccessful(); + } + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/PushImageCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/PushImageCmd.java new file mode 100644 index 000000000..01f9d0567 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/PushImageCmd.java @@ -0,0 +1,78 @@ +package com.github.dockerjava.api.command; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import javax.annotation.Nullable; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.PushResponseItem; + +/** + * Push the latest image to the repository. + * + * @param name + * The name, e.g. "alexec/busybox" or just "busybox" if you want to default. Not null. + */ +public interface PushImageCmd extends AsyncDockerCmd { + + @CheckForNull + AuthConfig getAuthConfig(); + + @CheckForNull + String getName(); + + @CheckForNull + String getTag(); + + /** + * @param name + * The name, e.g. "alexec/busybox" or just "busybox" if you want to default. Not null. + */ + PushImageCmd withName(@Nonnull String name); + + /** + * @param tag + * The image's tag. Not null. + */ + PushImageCmd withTag(String tag); + + PushImageCmd withAuthConfig(AuthConfig authConfig); + + /** + * @throws NotFoundException + * No such image + */ + @Override + > T exec(T resultCallback); + + @Override + default ResultCallback.Adapter start() { + return exec(new ResultCallback.Adapter() { + + @Nullable + private PushResponseItem latestItem = null; + + @Override + public void onNext(PushResponseItem item) { + this.latestItem = item; + } + + @Override + protected void throwFirstError() { + super.throwFirstError(); + + if (latestItem == null) { + throw new DockerClientException("Could not push image"); + } else if (latestItem.isErrorIndicated()) { + throw new DockerClientException("Could not push image: " + latestItem.getError()); + } + } + }); + } + + interface Exec extends DockerCmdAsyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveConfigCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveConfigCmd.java new file mode 100644 index 000000000..741fe32c0 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveConfigCmd.java @@ -0,0 +1,28 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * Remove a config. + */ +public interface RemoveConfigCmd extends SyncDockerCmd { + + @CheckForNull + String getConfigId(); + + RemoveConfigCmd withConfigId(@Nonnull String secretId); + + /** + * @throws NotFoundException + * No such config + */ + @Override + Void exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/src/main/java/com/github/dockerjava/api/command/RemoveContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/RemoveContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveContainerCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/RemoveImageCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveImageCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/RemoveImageCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveImageCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/RemoveNetworkCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveNetworkCmd.java similarity index 92% rename from src/main/java/com/github/dockerjava/api/command/RemoveNetworkCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveNetworkCmd.java index b6ad0e652..1466fd3e4 100644 --- a/src/main/java/com/github/dockerjava/api/command/RemoveNetworkCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveNetworkCmd.java @@ -1,7 +1,6 @@ package com.github.dockerjava.api.command; import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.core.RemoteApiVersion; import javax.annotation.CheckForNull; import javax.annotation.Nonnull; diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveSecretCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveSecretCmd.java new file mode 100644 index 000000000..28476da1d --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveSecretCmd.java @@ -0,0 +1,28 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * Remove a secret. + */ +public interface RemoveSecretCmd extends SyncDockerCmd { + + @CheckForNull + String getSecretId(); + + RemoveSecretCmd withSecretId(@Nonnull String secretId); + + /** + * @throws NotFoundException + * No such secret + */ + @Override + Void exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveServiceCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveServiceCmd.java new file mode 100644 index 000000000..d8a58126a --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveServiceCmd.java @@ -0,0 +1,28 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * Remove a service. + */ +public interface RemoveServiceCmd extends SyncDockerCmd { + + @CheckForNull + String getServiceId(); + + RemoveServiceCmd withServiceId(@Nonnull String serviceId); + + /** + * @throws NotFoundException + * No such service + */ + @Override + Void exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveSwarmNodeCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveSwarmNodeCmd.java new file mode 100644 index 000000000..603c610b3 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveSwarmNodeCmd.java @@ -0,0 +1,33 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * Remove a swarmNode. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public interface RemoveSwarmNodeCmd extends SyncDockerCmd { + + @CheckForNull + String getSwarmNodeId(); + + @CheckForNull + Boolean hasForceEnabled(); + + RemoveSwarmNodeCmd withSwarmNodeId(@Nonnull String swarmNodeId); + + RemoveSwarmNodeCmd withForce(Boolean force); + + /** + * @throws NotFoundException No such swarmNode + */ + @Override + Void exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/RemoveVolumeCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveVolumeCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/RemoveVolumeCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/RemoveVolumeCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/RenameContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RenameContainerCmd.java similarity index 93% rename from src/main/java/com/github/dockerjava/api/command/RenameContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/RenameContainerCmd.java index bce20251f..a69ba2a3f 100644 --- a/src/main/java/com/github/dockerjava/api/command/RenameContainerCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RenameContainerCmd.java @@ -1,7 +1,6 @@ package com.github.dockerjava.api.command; import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.core.RemoteApiVersion; import javax.annotation.CheckForNull; import javax.annotation.Nonnull; diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ResizeContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ResizeContainerCmd.java new file mode 100644 index 000000000..fef0087ed --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ResizeContainerCmd.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +public interface ResizeContainerCmd extends SyncDockerCmd { + + @CheckForNull + String getContainerId(); + + Integer getHeight(); + + Integer getWidth(); + + ResizeContainerCmd withContainerId(@Nonnull String execId); + + ResizeContainerCmd withSize(int height, int width); + + /** + * @throws NotFoundException no such container instance + */ + @Override + Void exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/ResizeExecCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ResizeExecCmd.java new file mode 100644 index 000000000..5910705e0 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/ResizeExecCmd.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +public interface ResizeExecCmd extends SyncDockerCmd { + @CheckForNull + String getExecId(); + + Integer getHeight(); + + Integer getWidth(); + + ResizeExecCmd withExecId(@Nonnull String execId); + + ResizeExecCmd withSize(int height, int width); + + /** + * @throws NotFoundException no such exec instance + */ + @Override + Void exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/RestartContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RestartContainerCmd.java new file mode 100644 index 000000000..372456813 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RestartContainerCmd.java @@ -0,0 +1,51 @@ +package com.github.dockerjava.api.command; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +import com.github.dockerjava.api.exception.NotFoundException; + +/** + * Restart a running container. + * + * @param signal - Signal to send to the container as an integer or string (e.g. SIGINT). + * @param timeout - Timeout in seconds before killing the container. Defaults to 10 seconds. + */ +public interface RestartContainerCmd extends SyncDockerCmd { + + @CheckForNull + String getContainerId(); + + @CheckForNull + Integer getTimeout(); + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_42} + */ + @CheckForNull + String getSignal(); + + RestartContainerCmd withContainerId(@Nonnull String containerId); + + /** + * @deprecated wrong name, use {@link #withTimeout(Integer)} + */ + @Deprecated + default RestartContainerCmd withtTimeout(Integer timeout) { + return withTimeout(timeout); + } + + RestartContainerCmd withTimeout(Integer timeout); + + RestartContainerCmd withSignal(String signal); + + /** + * @throws NotFoundException No such container + */ + @Override + Void exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/RootFS.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RootFS.java new file mode 100644 index 000000000..c190852af --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/RootFS.java @@ -0,0 +1,57 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.util.List; + +/** + * Part of {@link InspectImageResponse} + * + * @author Dmitry Tretyakov + */ +@EqualsAndHashCode +@ToString +public class RootFS extends DockerObject { + + @JsonProperty("Type") + private String type; + + @JsonProperty("Layers") + private List layers; + + /** + * @see #type + */ + @CheckForNull + public String getType() { + return type; + } + + /** + * @see #type + */ + public RootFS withType(String type) { + this.type = type; + return this; + } + + /** + * @see #layers + */ + @CheckForNull + public List getLayers() { + return layers; + } + + /** + * @see #layers + */ + public RootFS withLayers(List layers) { + this.layers = layers; + return this; + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/SaveImageCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/SaveImageCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/SaveImageCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/SaveImageCmd.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/SaveImagesCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/SaveImagesCmd.java new file mode 100644 index 000000000..1dd504434 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/SaveImagesCmd.java @@ -0,0 +1,47 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.exception.NotFoundException; + +import javax.annotation.Nonnull; +import java.io.InputStream; +import java.util.List; + +/** Command for downloading multiple images at once. */ +public interface SaveImagesCmd extends SyncDockerCmd { + + /** Image name and tag. */ + interface TaggedImage { + + /** + * The (tagged) image name. + * @return "name:tag" if a tag was specified, otherwise "name" + */ + String asString(); + } + + /** + * Adds an image to the list of images to download. + * @param name image name (not null) + * @param tag tag + * @return this + */ + SaveImagesCmd withImage(@Nonnull String name, @Nonnull String tag); + + + /** + * Gets the images that were added by {@link #withImage(String, String)}. + * @return images to be downloaded + */ + List getImages(); + + /** + * Its the responsibility of the caller to consume and/or close the {@link InputStream} to prevent connection leaks. + * + * @throws NotFoundException no such image + */ + InputStream exec() throws NotFoundException; + + interface Exec extends DockerCmdSyncExec { + } + +} diff --git a/src/main/java/com/github/dockerjava/api/command/SearchImagesCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/SearchImagesCmd.java similarity index 86% rename from src/main/java/com/github/dockerjava/api/command/SearchImagesCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/SearchImagesCmd.java index 2bc87cc73..8dc38ee3e 100644 --- a/src/main/java/com/github/dockerjava/api/command/SearchImagesCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/SearchImagesCmd.java @@ -18,8 +18,10 @@ public interface SearchImagesCmd extends SyncDockerCmd> { @CheckForNull String getTerm(); + Integer getLimit(); SearchImagesCmd withTerm(@Nonnull String term); + SearchImagesCmd withLimit(@Nonnull Integer limit); interface Exec extends DockerCmdSyncExec> { } diff --git a/src/main/java/com/github/dockerjava/api/command/StartContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/StartContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/StartContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/StartContainerCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/StatsCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/StatsCmd.java similarity index 87% rename from src/main/java/com/github/dockerjava/api/command/StatsCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/StatsCmd.java index 6085a810b..c85ca2b2f 100644 --- a/src/main/java/com/github/dockerjava/api/command/StatsCmd.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/StatsCmd.java @@ -16,6 +16,11 @@ public interface StatsCmd extends AsyncDockerCmd { StatsCmd withContainerId(@Nonnull String containerId); + @CheckForNull + Boolean hasNoStream(); + + StatsCmd withNoStream(boolean noStream); + interface Exec extends DockerCmdAsyncExec { } } diff --git a/src/main/java/com/github/dockerjava/api/command/StopContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/StopContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/StopContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/StopContainerCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/SyncDockerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/SyncDockerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/SyncDockerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/SyncDockerCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/TagImageCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/TagImageCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/TagImageCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/TagImageCmd.java diff --git a/src/main/java/com/github/dockerjava/api/command/TopContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/TopContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/TopContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/TopContainerCmd.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/TopContainerResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/TopContainerResponse.java new file mode 100644 index 000000000..e604c20ae --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/TopContainerResponse.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.model.DockerObject; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +/** + * + * @author Marcus Linke + * + */ +@EqualsAndHashCode +@ToString +public class TopContainerResponse extends DockerObject { + + @JsonProperty("Titles") + private String[] titles; + + @JsonProperty("Processes") + private String[][] processes; + + public String[] getTitles() { + return titles; + } + + public String[][] getProcesses() { + return processes; + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/UnpauseContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/UnpauseContainerCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/UnpauseContainerCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/UnpauseContainerCmd.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateContainerCmd.java new file mode 100644 index 000000000..d53bcdcdf --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateContainerCmd.java @@ -0,0 +1,157 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.BlkioRateDevice; +import com.github.dockerjava.api.model.BlkioWeightDevice; +import com.github.dockerjava.api.model.Device; +import com.github.dockerjava.api.model.DeviceRequest; +import com.github.dockerjava.api.model.RestartPolicy; +import com.github.dockerjava.api.model.Ulimit; +import com.github.dockerjava.api.model.UpdateContainerResponse; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import java.util.List; + +/** + * @author Kanstantsin Shautsou + * @since {@link RemoteApiVersion#VERSION_1_22} + */ +public interface UpdateContainerCmd extends SyncDockerCmd { + @CheckForNull + String getContainerId(); + + UpdateContainerCmd withContainerId(@Nonnull String containerId); + + @CheckForNull + Integer getBlkioWeight(); + + UpdateContainerCmd withBlkioWeight(Integer blkioWeight); + + @CheckForNull + List getBlkioWeightDevice(); + + UpdateContainerCmd withBlkioWeightDevice(List blkioWeightDevice); + + @CheckForNull + List getBlkioDeviceReadBps(); + + UpdateContainerCmd withBlkioDeviceReadBps(List blkioDeviceReadBps); + + @CheckForNull + List getBlkioDeviceWriteBps(); + + UpdateContainerCmd withBlkioDeviceWriteBps(List blkioDeviceWriteBps); + + @CheckForNull + List getBlkioDeviceReadIOps(); + + UpdateContainerCmd withBlkioDeviceReadIOps(List blkioDeviceReadIOps); + + @CheckForNull + List getBlkioDeviceWriteIOps(); + + UpdateContainerCmd withBlkioDeviceWriteIOps(List blkioDeviceWriteIOps); + + @CheckForNull + Long getCpuPeriod(); + + UpdateContainerCmd withCpuPeriod(Long cpuPeriod); + + @CheckForNull + Long getCpuQuota(); + + UpdateContainerCmd withCpuQuota(Long cpuQuota); + + @CheckForNull + String getCpusetCpus(); + + UpdateContainerCmd withCpusetCpus(String cpusetCpus); + + @CheckForNull + String getCpusetMems(); + + UpdateContainerCmd withCpusetMems(String cpusetMems); + + @CheckForNull + Integer getCpuShares(); + + UpdateContainerCmd withCpuShares(Integer cpuShares); + + @CheckForNull + Long getCpuRealtimePeriod(); + + UpdateContainerCmd withCpuRealtimePeriod(Long cpuRealtimePeriod); + + @CheckForNull + Long getCpuRealtimeRuntime(); + + UpdateContainerCmd withCpuRealtimeRuntime(Long cpuRealtimeRuntime); + + @CheckForNull + List getDevices(); + + UpdateContainerCmd withDevices(List devices); + + @CheckForNull + List getDeviceCgroupRules(); + + UpdateContainerCmd withDeviceCgroupRules(List deviceCgroupRules); + + @CheckForNull + List getDeviceRequests(); + + UpdateContainerCmd withDeviceRequests(List deviceRequests); + + @CheckForNull + Long getKernelMemory(); + + UpdateContainerCmd withKernelMemory(Long kernelMemory); + + @CheckForNull + Long getMemory(); + + UpdateContainerCmd withMemory(Long memory); + + @CheckForNull + Long getMemoryReservation(); + + UpdateContainerCmd withMemoryReservation(Long memoryReservation); + + @CheckForNull + Long getMemorySwap(); + + UpdateContainerCmd withMemorySwap(Long memorySwap); + + @CheckForNull + Long getNanoCPUs(); + + UpdateContainerCmd withNanoCPUs(Long nanoCPUs); + + @CheckForNull + Boolean getOomKillDisable(); + + UpdateContainerCmd withOomKillDisable(Boolean oomKillDisable); + + @CheckForNull + Boolean getInit(); + + UpdateContainerCmd withInit(Boolean init); + + @CheckForNull + Long getPidsLimit(); + + UpdateContainerCmd withPidsLimit(Long pidsLimit); + + @CheckForNull + List getUlimits(); + + UpdateContainerCmd withUlimits(List ulimits); + + @CheckForNull + RestartPolicy getRestartPolicy(); + + UpdateContainerCmd withRestartPolicy(RestartPolicy restartPolicy); + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateServiceCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateServiceCmd.java new file mode 100644 index 000000000..da4b78387 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateServiceCmd.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.ServiceSpec; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public interface UpdateServiceCmd extends SyncDockerCmd { + @CheckForNull + String getServiceId(); + + UpdateServiceCmd withServiceId(@Nonnull String serviceId); + + @CheckForNull + ServiceSpec getServiceSpec(); + + UpdateServiceCmd withServiceSpec(ServiceSpec serviceSpec); + + @CheckForNull + Long getVersion(); + + UpdateServiceCmd withVersion(Long version); + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateSwarmCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateSwarmCmd.java new file mode 100644 index 000000000..60b69aa4e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateSwarmCmd.java @@ -0,0 +1,35 @@ +package com.github.dockerjava.api.command; + + +import com.github.dockerjava.api.model.SwarmSpec; + +import javax.annotation.CheckForNull; + +public interface UpdateSwarmCmd extends SyncDockerCmd { + + @CheckForNull + Long getVersion(); + + UpdateSwarmCmd withVersion(Long version); + + @CheckForNull + Boolean getRotateWorkerToken(); + + UpdateSwarmCmd withRotateWorkerToken(Boolean rotateWorkerToken); + + @CheckForNull + Boolean getRotateManagerToken(); + + UpdateSwarmCmd withRotateManagerToken(Boolean rotateManagerToken); + + @CheckForNull + SwarmSpec getSwarmSpec(); + + UpdateSwarmCmd withSwarmSpec(SwarmSpec swarmSpec); + + @Override + Void exec(); + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateSwarmNodeCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateSwarmNodeCmd.java new file mode 100644 index 000000000..f535d2b7f --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/UpdateSwarmNodeCmd.java @@ -0,0 +1,32 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.SwarmNodeSpec; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * Update swarmNode spec + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public interface UpdateSwarmNodeCmd extends SyncDockerCmd { + + @CheckForNull + String getSwarmNodeId(); + + UpdateSwarmNodeCmd withSwarmNodeId(@Nonnull String swarmNodeId); + + @CheckForNull + SwarmNodeSpec getSwarmNodeSpec(); + + UpdateSwarmNodeCmd withSwarmNodeSpec(SwarmNodeSpec swarmNodeSpec); + + UpdateSwarmNodeCmd withVersion(@Nonnull Long versionId); + + @CheckForNull + Long getVersion(); + + interface Exec extends DockerCmdSyncExec { + } +} diff --git a/src/main/java/com/github/dockerjava/api/command/VersionCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/VersionCmd.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/command/VersionCmd.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/command/VersionCmd.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/WaitContainerCmd.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/WaitContainerCmd.java new file mode 100644 index 000000000..7b910cd69 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/WaitContainerCmd.java @@ -0,0 +1,51 @@ +package com.github.dockerjava.api.command; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import javax.annotation.Nullable; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.WaitContainerCondition; +import com.github.dockerjava.api.model.WaitResponse; + +/** + * Wait a container + * + * Block until container stops, then returns its exit code + */ +public interface WaitContainerCmd extends AsyncDockerCmd { + + @CheckForNull + String getContainerId(); + + WaitContainerCmd withContainerId(@Nonnull String containerId); + + /** + * Defaults to {@link WaitContainerCondition#NOT_RUNNING} if omitted or empty. + * + * @since {@link RemoteApiVersion#VERSION_1_30} + */ + @Nullable + WaitContainerCondition getCondition(); + + /** + * @since {@link RemoteApiVersion#VERSION_1_30} + */ + WaitContainerCmd withCondition(@Nullable WaitContainerCondition condition); + + /** + * @throws NotFoundException container not found + */ + @Override + > T exec(T resultCallback); + + @Override + default WaitContainerResultCallback start() { + return exec(new WaitContainerResultCallback()); + } + + interface Exec extends DockerCmdAsyncExec { + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/command/WaitContainerResultCallback.java b/docker-java-api/src/main/java/com/github/dockerjava/api/command/WaitContainerResultCallback.java new file mode 100644 index 000000000..6cb160151 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/command/WaitContainerResultCallback.java @@ -0,0 +1,74 @@ +/* + * Created on 21.07.2015 + */ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.async.ResultCallbackTemplate; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.model.WaitResponse; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.annotation.CheckForNull; +import java.util.concurrent.TimeUnit; + +/** + * + * @author Marcus Linke + * + */ +public class WaitContainerResultCallback extends ResultCallbackTemplate { + + private static final Logger LOGGER = LoggerFactory.getLogger(WaitContainerResultCallback.class); + + @CheckForNull + private WaitResponse waitResponse = null; + + @Override + public void onNext(WaitResponse waitResponse) { + this.waitResponse = waitResponse; + LOGGER.debug("{}", waitResponse); + } + + /** + * Awaits the status code from the container. + * + * @throws DockerClientException + * if the wait operation fails. + */ + public Integer awaitStatusCode() { + try { + awaitCompletion(); + } catch (InterruptedException e) { + throw new DockerClientException("", e); + } + + return getStatusCode(); + } + + /** + * Awaits the status code from the container. + * + * @throws DockerClientException + * if the wait operation fails. + */ + public Integer awaitStatusCode(long timeout, TimeUnit timeUnit) { + try { + if (!awaitCompletion(timeout, timeUnit)) { + throw new DockerClientException("Awaiting status code timeout."); + } + } catch (InterruptedException e) { + throw new DockerClientException("Awaiting status code interrupted: ", e); + } + + return getStatusCode(); + } + + private Integer getStatusCode() { + if (waitResponse == null) { + throw new DockerClientException("Error while wait container"); + } else { + return waitResponse.getStatusCode(); + } + } +} diff --git a/src/main/java/com/github/dockerjava/api/exception/BadRequestException.java b/docker-java-api/src/main/java/com/github/dockerjava/api/exception/BadRequestException.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/exception/BadRequestException.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/exception/BadRequestException.java diff --git a/src/main/java/com/github/dockerjava/api/exception/ConflictException.java b/docker-java-api/src/main/java/com/github/dockerjava/api/exception/ConflictException.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/exception/ConflictException.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/exception/ConflictException.java diff --git a/src/main/java/com/github/dockerjava/api/exception/DockerClientException.java b/docker-java-api/src/main/java/com/github/dockerjava/api/exception/DockerClientException.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/exception/DockerClientException.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/exception/DockerClientException.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/exception/DockerException.java b/docker-java-api/src/main/java/com/github/dockerjava/api/exception/DockerException.java new file mode 100644 index 000000000..69baf047e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/exception/DockerException.java @@ -0,0 +1,28 @@ +package com.github.dockerjava.api.exception; + +/** + * + * @author Konstantin Pelykh (kpelykh@gmail.com) + * + */ + +public class DockerException extends RuntimeException { + + private static final long serialVersionUID = 7667768099261650608L; + + private int httpStatus = 0; + + public DockerException(String message, int httpStatus) { + super(String.format("Status %d: %s", httpStatus, message)); + this.httpStatus = httpStatus; + } + + public DockerException(String message, int httpStatus, Throwable cause) { + super(String.format("Status %d: %s", httpStatus, message), cause); + this.httpStatus = httpStatus; + } + + public int getHttpStatus() { + return httpStatus; + } +} diff --git a/src/main/java/com/github/dockerjava/api/exception/InternalServerErrorException.java b/docker-java-api/src/main/java/com/github/dockerjava/api/exception/InternalServerErrorException.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/exception/InternalServerErrorException.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/exception/InternalServerErrorException.java diff --git a/src/main/java/com/github/dockerjava/api/exception/NotAcceptableException.java b/docker-java-api/src/main/java/com/github/dockerjava/api/exception/NotAcceptableException.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/exception/NotAcceptableException.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/exception/NotAcceptableException.java diff --git a/src/main/java/com/github/dockerjava/api/exception/NotFoundException.java b/docker-java-api/src/main/java/com/github/dockerjava/api/exception/NotFoundException.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/exception/NotFoundException.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/exception/NotFoundException.java diff --git a/src/main/java/com/github/dockerjava/api/exception/NotModifiedException.java b/docker-java-api/src/main/java/com/github/dockerjava/api/exception/NotModifiedException.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/exception/NotModifiedException.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/exception/NotModifiedException.java diff --git a/src/main/java/com/github/dockerjava/api/exception/UnauthorizedException.java b/docker-java-api/src/main/java/com/github/dockerjava/api/exception/UnauthorizedException.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/exception/UnauthorizedException.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/exception/UnauthorizedException.java diff --git a/src/main/java/com/github/dockerjava/api/model/AccessMode.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/AccessMode.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/model/AccessMode.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/AccessMode.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/AuthConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/AuthConfig.java new file mode 100644 index 000000000..cbb108571 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/AuthConfig.java @@ -0,0 +1,147 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +@EqualsAndHashCode +@ToString(onlyExplicitlyIncluded = true) +public class AuthConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * For backwards compatibility. Make sure you update the properties if you change this. + * + * @see "/docker.io.properties" + */ + public static final String DEFAULT_SERVER_ADDRESS = "https://index.docker.io/v1/"; + + @JsonProperty("username") + @ToString.Include + private String username; + + @JsonProperty("password") + private String password; + + @JsonProperty("email") + @ToString.Include + private String email; + + @JsonProperty("serveraddress") + @ToString.Include + private String registryAddress = DEFAULT_SERVER_ADDRESS; + + @JsonProperty("auth") + private String auth; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("registrytoken") + private String registrytoken; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_23} + */ + @JsonProperty("identitytoken") + private String identitytoken; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_25} + */ + @JsonProperty("stackOrchestrator") + @ToString.Include + private String stackOrchestrator; + + public String getUsername() { + return username; + } + + public AuthConfig withUsername(String username) { + this.username = username; + return this; + } + + public String getPassword() { + return password; + } + + public AuthConfig withPassword(String password) { + this.password = password; + return this; + } + + public String getEmail() { + return email; + } + + public AuthConfig withEmail(String email) { + this.email = email; + return this; + } + + public String getRegistryAddress() { + return registryAddress; + } + + public AuthConfig withRegistryAddress(String registryAddress) { + this.registryAddress = registryAddress; + return this; + } + + public String getAuth() { + return auth; + } + + public AuthConfig withAuth(String auth) { + this.auth = auth; + return this; + } + + /** + * @see #identitytoken + */ + public String getIdentitytoken() { + return identitytoken; + } + /** + * @see #identitytoken + */ + public AuthConfig withIdentityToken(String identitytoken) { + this.identitytoken = identitytoken; + return this; + } + + /** + * @see #registrytoken + */ + @CheckForNull + public String getRegistrytoken() { + return registrytoken; + } + + /** + * @see #registrytoken + */ + public AuthConfig withRegistrytoken(String registrytoken) { + this.registrytoken = registrytoken; + return this; + } + + /** + * @see #stackOrchestrator + */ + public String getStackOrchestrator() { + return stackOrchestrator; + } + + /** + * @see #stackOrchestrator + */ + public void setStackOrchestrator(String stackOrchestrator) { + this.stackOrchestrator = stackOrchestrator; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/AuthConfigurations.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/AuthConfigurations.java new file mode 100644 index 000000000..cbb7240f4 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/AuthConfigurations.java @@ -0,0 +1,27 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; +import java.util.Map; +import java.util.TreeMap; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@EqualsAndHashCode +@ToString +public class AuthConfigurations extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("configs") + private Map configs = new TreeMap<>(); + + public void addConfig(AuthConfig authConfig) { + configs.put(authConfig.getRegistryAddress(), authConfig); + } + + public Map getConfigs() { + return this.configs; + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/AuthResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/AuthResponse.java new file mode 100644 index 000000000..0703cab68 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/AuthResponse.java @@ -0,0 +1,44 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import java.io.Serializable; + +@EqualsAndHashCode +@ToString(onlyExplicitlyIncluded = true) +public class AuthResponse extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * @since 1.23 + */ + @JsonProperty("Status") + @ToString.Include + private String status; + + /** + * @since 1.23 + */ + @JsonProperty("IdentityToken") + private String identityToken; + + /** + * @see #status + */ + @Nonnull + public String getStatus() { + return status; + } + + /** + * @see #identityToken + */ + @CheckForNull + public String getIdentityToken() { + return identityToken; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Bind.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Bind.java new file mode 100644 index 000000000..ea7c22e69 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Bind.java @@ -0,0 +1,164 @@ +package com.github.dockerjava.api.model; + +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * Represents a host path being bind mounted as a {@link Volume} in a Docker container. + * The Bind can be in read only or read write access mode. + */ +@EqualsAndHashCode +@ToString +public class Bind extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + private String path; + + private Volume volume; + + private AccessMode accessMode; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_23} + */ + private Boolean noCopy; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_17} + */ + private SELContext secMode; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} + */ + private PropagationMode propagationMode; + + public Bind(String path, Volume volume) { + this(path, volume, AccessMode.DEFAULT, SELContext.DEFAULT); + } + + public Bind(String path, Volume volume, Boolean noCopy) { + this(path, volume, AccessMode.DEFAULT, SELContext.DEFAULT, noCopy); + } + + public Bind(String path, Volume volume, AccessMode accessMode) { + this(path, volume, accessMode, SELContext.DEFAULT); + } + + public Bind(String path, Volume volume, AccessMode accessMode, SELContext secMode) { + this(path, volume, accessMode, secMode, null); + } + + public Bind(String path, Volume volume, AccessMode accessMode, SELContext secMode, Boolean noCopy) { + this(path, volume, accessMode, secMode, noCopy, PropagationMode.DEFAULT_MODE); + } + + public Bind(String path, Volume volume, AccessMode accessMode, SELContext secMode, Boolean noCopy, PropagationMode propagationMode) { + this.path = path; + this.volume = volume; + this.accessMode = accessMode; + this.secMode = secMode; + this.noCopy = noCopy; + this.propagationMode = propagationMode; + } + + public String getPath() { + return path; + } + + public Volume getVolume() { + return volume; + } + + public AccessMode getAccessMode() { + return accessMode; + } + + public SELContext getSecMode() { + return secMode; + } + + public Boolean getNoCopy() { + return noCopy; + } + + public PropagationMode getPropagationMode() { + return propagationMode; + } + + /** + * Parses a bind mount specification to a {@link Bind}. + * + * @param serialized + * the specification, e.g. /host:/container:ro + * @return a {@link Bind} matching the specification + * @throws IllegalArgumentException + * if the specification cannot be parsed + */ + public static Bind parse(String serialized) { + try { + // Split by ':' but not ':\' (Windows-style path) + String[] parts = serialized.split(":(?!\\\\)"); + switch (parts.length) { + case 2: { + return new Bind(parts[0], new Volume(parts[1])); + } + case 3: { + String[] flags = parts[2].split(","); + AccessMode accessMode = AccessMode.DEFAULT; + SELContext seMode = SELContext.DEFAULT; + Boolean nocopy = null; + PropagationMode propagationMode = PropagationMode.DEFAULT_MODE; + for (String p : flags) { + if (p.length() == 2) { + accessMode = AccessMode.valueOf(p.toLowerCase()); + } else if ("nocopy".equals(p)) { + nocopy = true; + } else if (PropagationMode.SHARED.toString().equals(p)) { + propagationMode = PropagationMode.SHARED; + } else if (PropagationMode.RSHARED.toString().equals(p)) { + propagationMode = PropagationMode.RSHARED; + } else if (PropagationMode.SLAVE.toString().equals(p)) { + propagationMode = PropagationMode.SLAVE; + } else if (PropagationMode.RSLAVE.toString().equals(p)) { + propagationMode = PropagationMode.RSLAVE; + } else if (PropagationMode.PRIVATE.toString().equals(p)) { + propagationMode = PropagationMode.PRIVATE; + } else if (PropagationMode.RPRIVATE.toString().equals(p)) { + propagationMode = PropagationMode.RPRIVATE; + } else { + seMode = SELContext.fromString(p); + } + } + + return new Bind(parts[0], new Volume(parts[1]), accessMode, seMode, nocopy, propagationMode); + } + default: { + throw new IllegalArgumentException(); + } + } + } catch (Exception e) { + throw new IllegalArgumentException("Error parsing Bind '" + serialized + "'", e); + } + } + + /** + * Returns a string representation of this {@link Bind} suitable for inclusion in a JSON message. + * The format is <host path>:<container path>:<access mode>, + * like the argument in {@link #parse(String)}. + * + * @return a string representation of this {@link Bind} + */ + @Override + public String toString() { + return String.format("%s:%s:%s%s%s%s", + path, + volume.getPath(), + accessMode.toString(), + secMode != SELContext.none ? "," + secMode.toString() : "", + noCopy != null ? ",nocopy" : "", + propagationMode != PropagationMode.DEFAULT_MODE ? "," + propagationMode.toString() : ""); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/BindOptions.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BindOptions.java new file mode 100644 index 000000000..801e7719a --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BindOptions.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class BindOptions extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Propagation") + BindPropagation propagation; + + /** + * @see #propagation + */ + public BindPropagation getPropagation() { + return propagation; + } + + /** + * @see #propagation + */ + public BindOptions withPropagation(BindPropagation propagation) { + this.propagation = propagation; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/BindPropagation.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BindPropagation.java new file mode 100644 index 000000000..81a189e77 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BindPropagation.java @@ -0,0 +1,26 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum BindPropagation { + @JsonProperty("rprivate") + R_PRIVATE, + + @JsonProperty("private") + PRIVATE, + + @JsonProperty("rshared") + R_SHARED, + + @JsonProperty("shared") + SHARED, + + @JsonProperty("rslave") + R_SLAVE, + + @JsonProperty("slave") + SLAVE +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Binds.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Binds.java new file mode 100644 index 000000000..06aff4ecf --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Binds.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; +import java.util.stream.Stream; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; + +public class Binds implements Serializable { + private static final long serialVersionUID = 1L; + + private Bind[] binds; + + public Binds(Bind... binds) { + this.binds = binds; + } + + public Bind[] getBinds() { + return binds; + } + + @JsonValue + public String[] toPrimitive() { + return Stream.of(binds).map(Bind::toString).toArray(String[]::new); + } + + @JsonCreator + public static Binds fromPrimitive(String[] binds) { + return new Binds( + Stream.of(binds).map(Bind::parse).toArray(Bind[]::new) + ); + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioRateDevice.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioRateDevice.java new file mode 100644 index 000000000..300bcbf24 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioRateDevice.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +@EqualsAndHashCode +@ToString +public class BlkioRateDevice extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + @JsonProperty("Path") + private String path; + + @JsonProperty("Rate") + private Long rate; + + public String getPath() { + return path; + } + + public BlkioRateDevice withPath(String path) { + this.path = path; + return this; + } + + public Long getRate() { + return rate; + } + + public BlkioRateDevice withRate(Long rate) { + this.rate = rate; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioStatEntry.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioStatEntry.java new file mode 100644 index 000000000..2fccabaa2 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioStatEntry.java @@ -0,0 +1,61 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * BlkioStat is not documented in pubic docker swapper.yaml yet, reference: + * https://github.com/moby/moby/blob/master/api/types/stats.go + */ +@EqualsAndHashCode +@ToString +public class BlkioStatEntry extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + @JsonProperty("major") + Long major; + @JsonProperty("minor") + Long minor; + @JsonProperty("op") + String op; + @JsonProperty("value") + Long value; + + public Long getMajor() { + return major; + } + + public BlkioStatEntry withMajor(Long major) { + this.major = major; + return this; + } + + public Long getMinor() { + return minor; + } + + public BlkioStatEntry withMinor(Long minor) { + this.minor = minor; + return this; + } + + public String getOp() { + return op; + } + + public BlkioStatEntry withOp(String op) { + this.op = op; + return this; + } + + public Long getValue() { + return value; + } + + public BlkioStatEntry withValue(Long value) { + this.value = value; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioStatsConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioStatsConfig.java new file mode 100644 index 000000000..5a7db4d8b --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioStatsConfig.java @@ -0,0 +1,108 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; + +/** + * Used in {@link Statistics} + * + * @author Yuting Liu + */ +@EqualsAndHashCode +@ToString +public class BlkioStatsConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("io_service_bytes_recursive") + private List ioServiceBytesRecursive; + + @JsonProperty("io_serviced_recursive") + private List ioServicedRecursive; + + @JsonProperty("io_queue_recursive") + private List ioQueueRecursive; + + @JsonProperty("io_service_time_recursive") + private List ioServiceTimeRecursive; + + @JsonProperty("io_wait_time_recursive") + private List ioWaitTimeRecursive; + + @JsonProperty("io_merged_recursive") + private List ioMergedRecursive; + + @JsonProperty("io_time_recursive") + private List ioTimeRecursive; + + @JsonProperty("sectors_recursive") + private List sectorsRecursive; + + /** + * @see #ioServiceBytesRecursive + */ + @CheckForNull + public List getIoServiceBytesRecursive() { + return ioServiceBytesRecursive; + } + + /** + * @see #ioServicedRecursive + */ + @CheckForNull + public List getIoServicedRecursive() { + return ioServicedRecursive; + } + + /** + * @see #ioQueueRecursive + */ + @CheckForNull + public List getIoQueueRecursive() { + return ioQueueRecursive; + } + + /** + * @see #ioServiceTimeRecursive + */ + @CheckForNull + public List getIoServiceTimeRecursive() { + return ioServiceTimeRecursive; + } + + /** + * @see #ioWaitTimeRecursive + */ + @CheckForNull + public List getIoWaitTimeRecursive() { + return ioWaitTimeRecursive; + } + + /** + * @see #ioMergedRecursive + */ + @CheckForNull + public List getIoMergedRecursive() { + return ioMergedRecursive; + } + + /** + * @see #ioTimeRecursive + */ + @CheckForNull + public List getIoTimeRecursive() { + return ioTimeRecursive; + } + + /** + * @see #sectorsRecursive + */ + @CheckForNull + public List getSectorsRecursive() { + return sectorsRecursive; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioWeightDevice.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioWeightDevice.java new file mode 100644 index 000000000..3fd9704d8 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BlkioWeightDevice.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +@EqualsAndHashCode +@ToString +public class BlkioWeightDevice extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + @JsonProperty("Path") + private String path; + + @JsonProperty("Weight") + private Integer weight; + + public String getPath() { + return path; + } + + public BlkioWeightDevice withPath(String path) { + this.path = path; + return this; + } + + public Integer getWeight() { + return weight; + } + + public BlkioWeightDevice withWeight(Integer weight) { + this.weight = weight; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/BuildResponseItem.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BuildResponseItem.java new file mode 100644 index 000000000..80356e55c --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/BuildResponseItem.java @@ -0,0 +1,42 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonIgnore; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +/** + * Represents a build response stream item + */ +@EqualsAndHashCode +@ToString +public class BuildResponseItem extends ResponseItem { + private static final long serialVersionUID = -1252904184236343612L; + + private static final String BUILD_SUCCESS = "Successfully built"; + private static final String SHA256 = "sha256:"; + + /** + * Returns whether the stream field indicates a successful build operation + */ + @JsonIgnore + public boolean isBuildSuccessIndicated() { + if (isErrorIndicated() || getStream() == null) { + return false; + } + + return getStream().contains(BUILD_SUCCESS) || getStream().startsWith(SHA256); + } + + @JsonIgnore + public String getImageId() { + if (!isBuildSuccessIndicated()) { + return null; + } + + if (getStream().startsWith(SHA256)) { + return getStream().replaceFirst(SHA256, "").trim(); + } + + return getStream().replaceFirst(BUILD_SUCCESS, "").trim(); + } +} diff --git a/src/main/java/com/github/dockerjava/api/model/Capability.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Capability.java similarity index 93% rename from src/main/java/com/github/dockerjava/api/model/Capability.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/Capability.java index 6237a65ae..57fb97c7b 100644 --- a/src/main/java/com/github/dockerjava/api/model/Capability.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Capability.java @@ -1,5 +1,7 @@ package com.github.dockerjava.api.model; +import com.fasterxml.jackson.annotation.JsonCreator; + /** * The Linux capabilities supported by Docker. The list of capabilities is defined in Docker's types.go, {@link #ALL} was added manually. * @@ -18,6 +20,10 @@ public enum Capability { * */ AUDIT_CONTROL, + /** + * Allow reading the audit log via multicast netlink socket. + */ + AUDIT_READ, /** * Write records to kernel auditing log. */ @@ -26,6 +32,14 @@ public enum Capability { * Employ features that can block system suspend. */ BLOCK_SUSPEND, + /** + * Allow creating BPF maps, loading BPF Type Format (BTF) data, retrieve JITed code of BPF programs, and more. + */ + BPF, + /** + * Allow checkpoint/restore related operations. Introduced in kernel 5.9. + */ + CHECKPOINT_RESTORE, /** * Make arbitrary changes to file UIDs and GIDs (see chown(2)). */ @@ -120,6 +134,10 @@ public enum Capability { * */ NET_RAW, + /** + * Allow system performance and observability privileged operations using perf_events, i915_perf and other kernel subsystems + */ + PERFMON, /** * Set file capabilities. */ @@ -283,5 +301,11 @@ public enum Capability { /** * Trigger something that will wake up the system (set CLOCK_REALTIME_ALARM and CLOCK_BOOTTIME_ALARM timers). */ - WAKE_ALARM + WAKE_ALARM; + + @JsonCreator + public static Capability fromValue(String cap) { + String result = !cap.startsWith("CAP_") ? cap : cap.split("_", 2)[1]; + return Capability.valueOf(result); + } } diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ChangeLog.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ChangeLog.java new file mode 100644 index 000000000..c8a5be890 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ChangeLog.java @@ -0,0 +1,32 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * + * @author Konstantin Pelykh (kpelykh@gmail.com) + * + */ +@EqualsAndHashCode +@ToString +public class ChangeLog extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Path") + private String path; + + @JsonProperty("Kind") + private Integer kind; + + public String getPath() { + return path; + } + + public Integer getKind() { + return kind; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ClusterInfo.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ClusterInfo.java new file mode 100644 index 000000000..b6e1e5566 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ClusterInfo.java @@ -0,0 +1,131 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Date; + + +/** + * Used for creating swarms. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ClusterInfo extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("CreatedAt") + private Date createdAt; + + /** + * @since 1.24 + */ + @JsonProperty("Spec") + private SwarmSpec spec; + + /** + * @since 1.24 + */ + @JsonProperty("ID") + private String id; + + /** + * @since 1.24 + */ + @JsonProperty("UpdatedAt") + private Date updatedAt; + + /** + * @since 1.24 + */ + @JsonProperty("Version") + private ResourceVersion version; + + /** + * @see #createdAt + */ + @CheckForNull + public Date getCreatedAt() { + return createdAt; + } + + /** + * @see #createdAt + */ + public ClusterInfo withCreatedAt(Date createdAt) { + this.createdAt = createdAt; + return this; + } + + /** + * @see #spec + */ + @CheckForNull + public SwarmSpec getSpec() { + return spec; + } + + /** + * @see #spec + */ + public ClusterInfo withSpec(SwarmSpec spec) { + this.spec = spec; + return this; + } + /** + * @see #id + */ + @CheckForNull + public String getId() { + return id; + } + + /** + * @see #id + */ + public ClusterInfo withId(String id) { + this.id = id; + return this; + } + + /** + * @see #updatedAt + */ + @CheckForNull + public Date getUpdatedAt() { + return updatedAt; + } + + /** + * @see #updatedAt + */ + public ClusterInfo withUpdatedAt(Date updatedAt) { + this.updatedAt = updatedAt; + return this; + } + + /** + * @see #version + */ + @CheckForNull + public ResourceVersion getVersion() { + return version; + } + + /** + * @see #version + */ + public ClusterInfo withVersion(ResourceVersion version) { + this.version = version; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Config.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Config.java new file mode 100644 index 000000000..2c5b87aa8 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Config.java @@ -0,0 +1,90 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.Date; + +/** + * Used for Listing config. + * + * @since {@link RemoteApiVersion#VERSION_1_30} + */ +@ToString +@EqualsAndHashCode +public class Config extends DockerObject implements Serializable { + + private static final long serialVersionUID = 1L; + + /** + * @since 1.30 + */ + @JsonProperty("ID") + private String id; + + /** + * @since 1.30 + */ + @JsonProperty("CreatedAt") + private Date createdAt; + + /** + * @since 1.30 + */ + @JsonProperty("UpdatedAt") + private Date updatedAt; + + /** + * @since 1.30 + */ + @JsonProperty("Spec") + private ConfigSpec spec; + + /** + * @since 1.30 + */ + @JsonProperty("Version") + private ResourceVersion version; + + public String getId() { + return id; + } + + public void setId(String id) { + this.id = id; + } + + public Date getCreatedAt() { + return createdAt; + } + + public void setCreatedAt(Date createdAt) { + this.createdAt = createdAt; + } + + public Date getUpdatedAt() { + return updatedAt; + } + + public void setUpdatedAt(Date updatedAt) { + this.updatedAt = updatedAt; + } + + public ConfigSpec getSpec() { + return spec; + } + + public void setSpec(ConfigSpec spec) { + this.spec = spec; + } + + public ResourceVersion getVersion() { + return version; + } + + public void setVersion(ResourceVersion version) { + this.version = version; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ConfigSpec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ConfigSpec.java new file mode 100644 index 000000000..62e525d0b --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ConfigSpec.java @@ -0,0 +1,24 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_30} + */ +@EqualsAndHashCode +@ToString +public class ConfigSpec extends DockerObject implements Serializable { + + private static final long serialVersionUID = 1L; + + @JsonProperty("Name") + private String name; + + public String getName() { + return name; + } +} diff --git a/src/main/java/com/github/dockerjava/api/model/Container.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Container.java similarity index 77% rename from src/main/java/com/github/dockerjava/api/model/Container.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/Container.java index 47492920d..3b4bdf394 100644 --- a/src/main/java/com/github/dockerjava/api/model/Container.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Container.java @@ -1,16 +1,13 @@ package com.github.dockerjava.api.model; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; import com.fasterxml.jackson.annotation.JsonProperty; import com.github.dockerjava.api.command.ListContainersCmd; -import com.github.dockerjava.core.RemoteApiVersion; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; +import lombok.EqualsAndHashCode; +import lombok.ToString; import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; import java.util.Map; /** @@ -18,9 +15,10 @@ * * @author Konstantin Pelykh (kpelykh@gmail.com) */ -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(Include.NON_NULL) -public class Container { +@EqualsAndHashCode +@ToString +public class Container extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; @JsonProperty("Command") private String command; @@ -52,6 +50,12 @@ public class Container { @JsonProperty("Status") private String status; + /** + * @since ~{@link RemoteApiVersion#VERSION_1_23} + */ + @JsonProperty("State") + private String state; + /** * @since ~{@link RemoteApiVersion#VERSION_1_19} */ @@ -81,6 +85,12 @@ public class Container { @JsonProperty("NetworkSettings") private ContainerNetworkSettings networkSettings; + /** + * @since ~{@link RemoteApiVersion#VERSION_1_23} + */ + @JsonProperty("Mounts") + private List mounts; + public String getId() { return id; } @@ -106,6 +116,10 @@ public String getStatus() { return status; } + public String getState() { + return state; + } + public ContainerPort[] getPorts() { return ports; } @@ -150,18 +164,7 @@ public ContainerHostConfig getHostConfig() { return hostConfig; } - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); + public List getMounts() { + return mounts; } } diff --git a/src/main/java/com/github/dockerjava/api/model/ContainerConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerConfig.java similarity index 90% rename from src/main/java/com/github/dockerjava/api/model/ContainerConfig.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerConfig.java index 224da3272..db5437220 100644 --- a/src/main/java/com/github/dockerjava/api/model/ContainerConfig.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerConfig.java @@ -1,15 +1,12 @@ package com.github.dockerjava.api.model; import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; import com.fasterxml.jackson.annotation.JsonProperty; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; +import lombok.EqualsAndHashCode; +import lombok.ToString; import javax.annotation.CheckForNull; +import java.io.Serializable; import java.util.Map; /** @@ -17,9 +14,10 @@ * * @author Konstantin Pelykh (kpelykh@gmail.com) */ -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(Include.NON_NULL) -public class ContainerConfig { +@EqualsAndHashCode +@ToString +public class ContainerConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; @JsonProperty("AttachStderr") private Boolean attachStderr; @@ -84,6 +82,9 @@ public class ContainerConfig { @JsonProperty("WorkingDir") private String workingDir; + @JsonProperty("Healthcheck") + private HealthCheck healthCheck; + @JsonIgnore public ExposedPort[] getExposedPorts() { return exposedPorts != null ? exposedPorts.getExposedPorts() : null; @@ -409,6 +410,14 @@ public String getWorkingDir() { return workingDir; } + /** + * @see #healthCheck + */ + @CheckForNull + public HealthCheck getHealthcheck() { + return healthCheck; + } + /** * @see #workingDir */ @@ -416,19 +425,4 @@ public ContainerConfig withWorkingDir(String workingDir) { this.workingDir = workingDir; return this; } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } } diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerDNSConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerDNSConfig.java new file mode 100644 index 000000000..63d3cae11 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerDNSConfig.java @@ -0,0 +1,53 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.List; + +/** + * Specification for DNS related configurations in resolver configuration file (`resolv.conf`). + * + * @since {@link RemoteApiVersion#VERSION_1_25} + */ +@EqualsAndHashCode +@ToString +public class ContainerDNSConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Nameservers") + private List nameservers; + @JsonProperty("Search") + private List search; + @JsonProperty("Options") + private List options; + + public List getNameservers() { + return nameservers; + } + + public ContainerDNSConfig withNameservers(List nameservers) { + this.nameservers = nameservers; + return this; + } + + public List getSearch() { + return search; + } + + public ContainerDNSConfig withSearch(List search) { + this.search = search; + return this; + } + + public List getOptions() { + return options; + } + + public ContainerDNSConfig withOptions(List options) { + this.options = options; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerHostConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerHostConfig.java new file mode 100644 index 000000000..cdc446282 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerHostConfig.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * Used in {@link Container} + * + * @see Container + * @author Kanstantsin Shautsou + */ +@EqualsAndHashCode +@ToString +public class ContainerHostConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("NetworkMode") + private String networkMode; + + public String getNetworkMode() { + return networkMode; + } + + /** + * @see #networkMode + */ + public ContainerHostConfig withNetworkMode(String networkMode) { + this.networkMode = networkMode; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerMount.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerMount.java new file mode 100644 index 000000000..a08a6ea3f --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerMount.java @@ -0,0 +1,151 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @author Yuting Liu + * @see Container + */ +@EqualsAndHashCode +@ToString +public class ContainerMount extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Name") + String name; + + @JsonProperty("Source") + String source; + + @JsonProperty("Destination") + String destination; + + @JsonProperty("Driver") + String driver; + + @JsonProperty("Mode") + String mode; + + @JsonProperty("RW") + boolean rw; + + @JsonProperty("Propagation") + String propagation; + + /** + * @see #name + */ + @CheckForNull + public String getName() { + return name; + } + + /** + * @see #name + */ + public ContainerMount withName(String name) { + this.name = name; + return this; + } + + /** + * @see #source + */ + @CheckForNull + public String getSource() { + return source; + } + + /** + * @see #source + */ + public ContainerMount withSource(String source) { + this.source = source; + return this; + } + + /** + * @see #destination + */ + @CheckForNull + public String getDestination() { + return destination; + } + + /** + * @see #destination + */ + public ContainerMount withDestination(String destination) { + this.destination = destination; + return this; + } + + /** + * @see #driver + */ + @CheckForNull + public String getDriver() { + return driver; + } + + /** + * @see #driver + */ + public ContainerMount withDriver(String driver) { + this.driver = driver; + return this; + } + + /** + * @see #mode + */ + @CheckForNull + public String getMode() { + return mode; + } + + /** + * @see #mode + */ + public ContainerMount withMode(String mode) { + this.mode = mode; + return this; + } + + /** + * @see #rw + */ + @CheckForNull + public Boolean getRw() { + return rw; + } + + /** + * @see #rw + */ + public ContainerMount withRw(Boolean rw) { + this.rw = rw; + return this; + } + + /** + * @see #propagation + */ + @CheckForNull + public String getPropagation() { + return propagation; + } + + /** + * @see #propagation + */ + public ContainerMount withPropagation(String propagation) { + this.propagation = propagation; + return this; + } +} diff --git a/src/main/java/com/github/dockerjava/api/model/ContainerNetwork.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerNetwork.java similarity index 89% rename from src/main/java/com/github/dockerjava/api/model/ContainerNetwork.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerNetwork.java index 9b2160e98..823828900 100644 --- a/src/main/java/com/github/dockerjava/api/model/ContainerNetwork.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerNetwork.java @@ -1,14 +1,12 @@ package com.github.dockerjava.api.model; import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.core.RemoteApiVersion; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; +import lombok.EqualsAndHashCode; +import lombok.ToString; import javax.annotation.CheckForNull; +import java.io.Serializable; import java.util.Arrays; import java.util.List; @@ -20,8 +18,11 @@ * @see ContainerNetworkSettings * @author Kanstantsin Shautsou */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class ContainerNetwork { +@EqualsAndHashCode +@ToString +public class ContainerNetwork extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + /** * FIXME verify */ @@ -279,25 +280,13 @@ public ContainerNetwork withNetworkID(String networkID) { return this; } - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } - /** * Docker named it EndpointIPAMConfig */ - public static class Ipam { + @EqualsAndHashCode + @ToString + public static class Ipam extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; @JsonProperty("IPv4Address") private String ipv4Address; diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerNetworkSettings.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerNetworkSettings.java new file mode 100644 index 000000000..9e8381500 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerNetworkSettings.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.Map; + +/** + * Sub-object in {@link Container} + * + * @see Container + * @since {@link RemoteApiVersion#VERSION_1_22} + */ +@EqualsAndHashCode +@ToString +public class ContainerNetworkSettings extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * @since {@link RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("Networks") + private Map networks; + + /** + * @see #networks + */ + public Map getNetworks() { + return networks; + } + + /** + * @see #networks + */ + public ContainerNetworkSettings withNetworks(Map networks) { + this.networks = networks; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerPort.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerPort.java new file mode 100644 index 000000000..35f9f6ab9 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerPort.java @@ -0,0 +1,94 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @author Kanstantsin Shautsou + * @see Container + */ +@EqualsAndHashCode +@ToString +public class ContainerPort extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("IP") + private String ip; + + @JsonProperty("PrivatePort") + private Integer privatePort; + + @JsonProperty("PublicPort") + private Integer publicPort; + + @JsonProperty("Type") + private String type; + + /** + * @see #ip + */ + @CheckForNull + public String getIp() { + return ip; + } + + /** + * @see #ip + */ + public ContainerPort withIp(String ip) { + this.ip = ip; + return this; + } + + /** + * @see #privatePort + */ + @CheckForNull + public Integer getPrivatePort() { + return privatePort; + } + + /** + * @see #privatePort + */ + public ContainerPort withPrivatePort(Integer privatePort) { + this.privatePort = privatePort; + return this; + } + + /** + * @see #publicPort + */ + @CheckForNull + public Integer getPublicPort() { + return publicPort; + } + + /** + * @see #publicPort + */ + public ContainerPort withPublicPort(Integer publicPort) { + this.publicPort = publicPort; + return this; + } + + /** + * @see #type + */ + @CheckForNull + public String getType() { + return type; + } + + /** + * @see #type + */ + public ContainerPort withType(String type) { + this.type = type; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpec.java new file mode 100644 index 000000000..0a26e54fd --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpec.java @@ -0,0 +1,455 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; +import java.util.Map; + +/** + * The specification for containers as used in {@link TaskSpec} + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ContainerSpec extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Image") + private String image; + + /** + * @since 1.24 + */ + @JsonProperty("Labels") + private Map labels; + + /** + * @since 1.24 + */ + @JsonProperty("Command") + private List command; + + /** + * @since 1.24 + */ + @JsonProperty("Args") + private List args; + + /** + * @since 1.24 + */ + @JsonProperty("Env") + private List env; + + /** + * @since 1.24 + */ + @JsonProperty("Dir") + private String dir; + + /** + * @since 1.24 + */ + @JsonProperty("User") + private String user; + + /** + * @since 1.24 + */ + @JsonProperty("Groups") + private String groups; + + /** + * @since 1.24 + */ + @JsonProperty("TTY") + private Boolean tty; + + /** + * @since 1.24 + */ + @JsonProperty("Mounts") + private List mounts; + + /** + * @since 1.24 + */ + @JsonProperty("Duration") + private Long duration; + + /** + * @since 1.24 + */ + @JsonProperty("StopGracePeriod") + private Long stopGracePeriod; + + /** + * @since 1.25 + * Specification for DNS related configurations in resolver configuration file + */ + @JsonProperty("DNSConfig") + private ContainerDNSConfig dnsConfig; + + /** + * @since 1.26 + * Open stdin + */ + @JsonProperty("OpenStdin") + private Boolean openStdin; + + /** + * @since 1.26 + * Mount the container's root filesystem as read only. + */ + @JsonProperty("ReadOnly") + private Boolean readOnly; + + /** + * @since 1.26 + * A list of hostnames/IP mappings to add to the container's /etc/hosts file. + */ + @JsonProperty("Hosts") + private List hosts; + + /** + * @since 1.26 + * The hostname to use for the container, as a valid RFC 1123 hostname + */ + @JsonProperty("Hostname") + private String hostname; + + /** + * @since 1.26 + * Secrets contains references to zero or more secrets that will be exposed to the service. + */ + @JsonProperty("Secrets") + private List secrets; + + /** + * @since 1.26 + * A test to perform to check that the container is healthy. + */ + @JsonProperty("Healthcheck") + private HealthCheck healthCheck; + + /** + * @since 1.28 + * Signal to stop the container. + */ + @JsonProperty("StopSignal") + private String stopSignal; + + /** + * @since 1.29 + * Security options for the container + */ + @JsonProperty("Privileges") + private ContainerSpecPrivileges privileges; + + /** + * @since 1.29 + * Configs contains references to zero or more configs that will be exposed to the service. + */ + @JsonProperty("Configs") + private List configs; + + /** + * @since 1.38 + * Run an init inside the container that forwards signals and reaps processes. + * This field is omitted if empty, and the default (as configured on the daemon) is used. + */ + @JsonProperty("Init") + private Boolean init; + + /** + * @see #image + */ + @CheckForNull + public String getImage() { + return image; + } + + /** + * @see #image + */ + public ContainerSpec withImage(String image) { + this.image = image; + return this; + } + + /** + * @see #labels + */ + @CheckForNull + public Map getLabels() { + return labels; + } + + /** + * @see #labels + */ + public ContainerSpec withLabels(Map labels) { + this.labels = labels; + return this; + } + + /** + * @see #command + */ + @CheckForNull + public List getCommand() { + return command; + } + + /** + * @see #command + */ + public ContainerSpec withCommand(List command) { + this.command = command; + return this; + } + + /** + * @see #args + */ + @CheckForNull + public List getArgs() { + return args; + } + + /** + * @see #args + */ + public ContainerSpec withArgs(List args) { + this.args = args; + return this; + } + + /** + * @see #env + */ + @CheckForNull + public List getEnv() { + return env; + } + + /** + * @see #env + */ + public ContainerSpec withEnv(List env) { + this.env = env; + return this; + } + + /** + * @see #dir + */ + @CheckForNull + public String getDir() { + return dir; + } + + /** + * @see #dir + */ + public ContainerSpec withDir(String dir) { + this.dir = dir; + return this; + } + + /** + * @see #user + */ + @CheckForNull + public String getUser() { + return user; + } + + /** + * @see #user + */ + public ContainerSpec withUser(String user) { + this.user = user; + return this; + } + + /** + * @see #groups + */ + @CheckForNull + public String getGroups() { + return groups; + } + + /** + * @see #groups + */ + public ContainerSpec withGroups(String groups) { + this.groups = groups; + return this; + } + + /** + * @see #tty + */ + @CheckForNull + public Boolean getTty() { + return tty; + } + + /** + * @see #tty + */ + public ContainerSpec withTty(Boolean tty) { + this.tty = tty; + return this; + } + + /** + * @see #mounts + */ + @CheckForNull + public List getMounts() { + return mounts; + } + + /** + * @see #mounts + */ + public ContainerSpec withMounts(List mounts) { + this.mounts = mounts; + return this; + } + + /** + * @see #duration + */ + @CheckForNull + public Long getDuration() { + return duration; + } + + /** + * @see #duration + */ + public ContainerSpec withDuration(Long duration) { + this.duration = duration; + return this; + } + + public ContainerDNSConfig getDnsConfig() { + return dnsConfig; + } + + public ContainerSpec withDnsConfig(ContainerDNSConfig dnsConfig) { + this.dnsConfig = dnsConfig; + return this; + } + + public Boolean getOpenStdin() { + return openStdin; + } + + public ContainerSpec withOpenStdin(Boolean openStdin) { + this.openStdin = openStdin; + return this; + } + + public Boolean getReadOnly() { + return readOnly; + } + + public ContainerSpec withReadOnly(Boolean readOnly) { + this.readOnly = readOnly; + return this; + } + + public List getHosts() { + return hosts; + } + + public ContainerSpec withHosts(List hosts) { + this.hosts = hosts; + return this; + } + + public String getHostname() { + return hostname; + } + + public ContainerSpec withHostname(String hostname) { + this.hostname = hostname; + return this; + } + + public List getSecrets() { + return secrets; + } + + public ContainerSpec withSecrets(List secrets) { + this.secrets = secrets; + return this; + } + + public HealthCheck getHealthCheck() { + return healthCheck; + } + + public ContainerSpec withHealthCheck(HealthCheck healthCheck) { + this.healthCheck = healthCheck; + return this; + } + + public String getStopSignal() { + return stopSignal; + } + + public ContainerSpec withStopSignal(String stopSignal) { + this.stopSignal = stopSignal; + return this; + } + + public Long getStopGracePeriod() { + return stopGracePeriod; + } + + public ContainerSpec withStopGracePeriod(Long stopGracePeriod) { + this.stopGracePeriod = stopGracePeriod; + return this; + } + + public ContainerSpecPrivileges getPrivileges() { + return privileges; + } + + public ContainerSpec withPrivileges(ContainerSpecPrivileges privileges) { + this.privileges = privileges; + return this; + } + + public List getConfigs() { + return configs; + } + + public ContainerSpec withConfigs(List configs) { + this.configs = configs; + return this; + } + + public Boolean getInit() { + return init; + } + + public ContainerSpec withInit(Boolean init) { + this.init = init; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecConfig.java new file mode 100644 index 000000000..fbd93b606 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecConfig.java @@ -0,0 +1,53 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * docker configs that will be exposed to the service + * + * @since {@link RemoteApiVersion#VERSION_1_29} + */ +@EqualsAndHashCode +@ToString +public class ContainerSpecConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + @JsonProperty("File") + private ContainerSpecFile file; + + @JsonProperty("ConfigID") + private String configID; + + @JsonProperty("ConfigName") + private String configName; + + public ContainerSpecFile getFile() { + return file; + } + + public ContainerSpecConfig withFile(ContainerSpecFile file) { + this.file = file; + return this; + } + + public String getConfigID() { + return configID; + } + + public ContainerSpecConfig withConfigID(String configID) { + this.configID = configID; + return this; + } + + public String getConfigName() { + return configName; + } + + public ContainerSpecConfig withConfigName(String configName) { + this.configName = configName; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecFile.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecFile.java new file mode 100644 index 000000000..ac9ef4d81 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecFile.java @@ -0,0 +1,72 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * File represents a specific target that is backed by a file. + * + * @since {@link RemoteApiVersion#VERSION_1_26} + */ +@EqualsAndHashCode +@ToString +public class ContainerSpecFile extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Name") + private String name; + /** + * UID represents the file UID. + */ + @JsonProperty("UID") + private String uid; + /** + * GID represents the file GID. + */ + @JsonProperty("GID") + private String gid; + /** + * Mode represents the FileMode of the file. + */ + @JsonProperty("Mode") + private Long mode; + + public String getName() { + return name; + } + + public ContainerSpecFile withName(String name) { + this.name = name; + return this; + } + + public String getUid() { + return uid; + } + + public ContainerSpecFile withUid(String uid) { + this.uid = uid; + return this; + } + + public String getGid() { + return gid; + } + + public ContainerSpecFile withGid(String gid) { + this.gid = gid; + return this; + } + + public Long getMode() { + return mode; + } + + public ContainerSpecFile withMode(Long mode) { + this.mode = mode; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecPrivileges.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecPrivileges.java new file mode 100644 index 000000000..5d8d7cd55 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecPrivileges.java @@ -0,0 +1,42 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * Security options for the container + * + * @since {@link RemoteApiVersion#VERSION_1_29} + */ +@EqualsAndHashCode +@ToString +public class ContainerSpecPrivileges extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("CredentialSpec") + private ContainerSpecPrivilegesCredential credentialSpec; + + @JsonProperty("SELinuxContext") + private ContainerSpecPrivilegesSELinuxContext seLinuxContext; + + public ContainerSpecPrivilegesCredential getCredentialSpec() { + return credentialSpec; + } + + public ContainerSpecPrivileges withCredentialSpec(ContainerSpecPrivilegesCredential credentialSpec) { + this.credentialSpec = credentialSpec; + return this; + } + + public ContainerSpecPrivilegesSELinuxContext getSeLinuxContext() { + return seLinuxContext; + } + + public ContainerSpecPrivileges withSeLinuxContext(ContainerSpecPrivilegesSELinuxContext seLinuxContext) { + this.seLinuxContext = seLinuxContext; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecPrivilegesCredential.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecPrivilegesCredential.java new file mode 100644 index 000000000..e6ca62fd4 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecPrivilegesCredential.java @@ -0,0 +1,54 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * Credential for managed service account (Windows only) + * + * @since {@link RemoteApiVersion#VERSION_1_29} + */ +@EqualsAndHashCode +@ToString +public class ContainerSpecPrivilegesCredential extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * Load credential spec from this file. The file is read by the daemon, and must be present in the + * `CredentialSpecs` subdirectory in the docker data directory, which defaults to + * C:\ProgramData\Docker\ on Windows. + */ + @JsonProperty("File") + private String file; + + /** + * Load credential spec from this value in the Windows registry. The specified registry value must be + * located in: + * + * HKLM\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Virtualization\Containers\CredentialSpecs + * + */ + @JsonProperty("Registry") + private String registry; + + public String getFile() { + return file; + } + + public ContainerSpecPrivilegesCredential withFile(String file) { + this.file = file; + return this; + } + + public String getRegistry() { + return registry; + } + + public ContainerSpecPrivilegesCredential withRegistry(String registry) { + this.registry = registry; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecPrivilegesSELinuxContext.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecPrivilegesSELinuxContext.java new file mode 100644 index 000000000..d1b2cc15b --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecPrivilegesSELinuxContext.java @@ -0,0 +1,77 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * SELinux labels of the container + * + * @since {@link RemoteApiVersion#VERSION_1_29} + */ +@EqualsAndHashCode +@ToString +public class ContainerSpecPrivilegesSELinuxContext extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + @JsonProperty("Disable") + private Boolean disable; + + @JsonProperty("User") + private String user; + + @JsonProperty("Role") + private String role; + + @JsonProperty("Type") + private String type; + + @JsonProperty("Level") + private String level; + + public Boolean getDisable() { + return disable; + } + + public ContainerSpecPrivilegesSELinuxContext withDisable(Boolean disable) { + this.disable = disable; + return this; + } + + public String getUser() { + return user; + } + + public ContainerSpecPrivilegesSELinuxContext withUser(String user) { + this.user = user; + return this; + } + + public String getRole() { + return role; + } + + public ContainerSpecPrivilegesSELinuxContext withRole(String role) { + this.role = role; + return this; + } + + public String getType() { + return type; + } + + public ContainerSpecPrivilegesSELinuxContext withType(String type) { + this.type = type; + return this; + } + + public String getLevel() { + return level; + } + + public ContainerSpecPrivilegesSELinuxContext withLevel(String level) { + this.level = level; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecSecret.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecSecret.java new file mode 100644 index 000000000..742272e16 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ContainerSpecSecret.java @@ -0,0 +1,54 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * docker secrets that will be exposed to the service + * + * @since {@link RemoteApiVersion#VERSION_1_26} + */ +@EqualsAndHashCode +@ToString +public class ContainerSpecSecret extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("File") + private ContainerSpecFile file; + + @JsonProperty("SecretID") + private String secretId; + + @JsonProperty("SecretName") + private String secretName; + + public ContainerSpecFile getFile() { + return file; + } + + public ContainerSpecSecret withFile(ContainerSpecFile file) { + this.file = file; + return this; + } + + public String getSecretId() { + return secretId; + } + + public ContainerSpecSecret withSecretId(String secretId) { + this.secretId = secretId; + return this; + } + + public String getSecretName() { + return secretName; + } + + public ContainerSpecSecret withSecretName(String secretName) { + this.secretName = secretName; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/CpuStatsConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/CpuStatsConfig.java new file mode 100644 index 000000000..04d91c826 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/CpuStatsConfig.java @@ -0,0 +1,63 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * Used in {@link Statistics} + * + * @author Yuting Liu + */ +@EqualsAndHashCode +@ToString +public class CpuStatsConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("cpu_usage") + private CpuUsageConfig cpuUsage; + + @JsonProperty("system_cpu_usage") + private Long systemCpuUsage; + + @JsonProperty("online_cpus") + private Long onlineCpus; + + @JsonProperty("throttling_data") + private ThrottlingDataConfig throttlingData; + + /** + * @see #cpuUsage + */ + @CheckForNull + public CpuUsageConfig getCpuUsage() { + return cpuUsage; + } + + /** + * @see #systemCpuUsage + */ + @CheckForNull + public Long getSystemCpuUsage() { + return systemCpuUsage; + } + + /** + * @see #onlineCpus + */ + @CheckForNull + public Long getOnlineCpus() { + return onlineCpus; + } + + /** + * @see #throttlingData + */ + @CheckForNull + public ThrottlingDataConfig getThrottlingData() { + return throttlingData; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/CpuUsageConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/CpuUsageConfig.java new file mode 100644 index 000000000..f87afeec8 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/CpuUsageConfig.java @@ -0,0 +1,64 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; + +/** + * Used in {@link Statistics} + * + * @author Yuting Liu + */ +@EqualsAndHashCode +@ToString +public class CpuUsageConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("total_usage") + private Long totalUsage; + + @JsonProperty("percpu_usage") + private List percpuUsage; + + @JsonProperty("usage_in_kernelmode") + private Long usageInKernelmode; + + @JsonProperty("usage_in_usermode") + private Long usageInUsermode; + + /** + * @see #totalUsage + */ + @CheckForNull + public Long getTotalUsage() { + return totalUsage; + } + + /** + * @see #percpuUsage + */ + @CheckForNull + public List getPercpuUsage() { + return percpuUsage; + } + + /** + * @see #usageInKernelmode + */ + @CheckForNull + public Long getUsageInKernelmode() { + return usageInKernelmode; + } + + /** + * @see #usageInUsermode + */ + @CheckForNull + public Long getUsageInUsermode() { + return usageInUsermode; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Device.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Device.java new file mode 100644 index 000000000..b6f16029e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Device.java @@ -0,0 +1,121 @@ +package com.github.dockerjava.api.model; + +import static java.util.Objects.requireNonNull; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.Nonnull; +import java.io.Serializable; +import java.util.HashMap; +import java.util.Map; +import java.util.StringTokenizer; + +@EqualsAndHashCode +@ToString +public class Device extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("CgroupPermissions") + private String cGroupPermissions = ""; + + @JsonProperty("PathOnHost") + private String pathOnHost = null; + + @JsonProperty("PathInContainer") + private String pathInContainer = null; + + public Device() { + } + + public Device(String cGroupPermissions, String pathInContainer, String pathOnHost) { + requireNonNull(cGroupPermissions, "cGroupPermissions is null"); + requireNonNull(pathInContainer, "pathInContainer is null"); + requireNonNull(pathOnHost, "pathOnHost is null"); + this.cGroupPermissions = cGroupPermissions; + this.pathInContainer = pathInContainer; + this.pathOnHost = pathOnHost; + } + + public String getcGroupPermissions() { + return cGroupPermissions; + } + + public String getPathInContainer() { + return pathInContainer; + } + + public String getPathOnHost() { + return pathOnHost; + } + + /** + * @link https://github.com/docker/docker/blob/6b4a46f28266031ce1a1315f17fb69113a06efe1/runconfig/opts/parse_test.go#L468 + */ + @Nonnull + public static Device parse(@Nonnull String deviceStr) { + String src = ""; + String dst = ""; + String permissions = "rwm"; + final String[] arr = deviceStr.trim().split(":"); + // java String.split() returns wrong length, use tokenizer instead + switch (new StringTokenizer(deviceStr, ":").countTokens()) { + case 3: { + // Mismatches docker code logic. While there is no validations after parsing, checking heregit + if (validDeviceMode(arr[2])) { + permissions = arr[2]; + } else { + throw new IllegalArgumentException("Invalid device specification: " + deviceStr); + } + } + case 2: { + if (validDeviceMode(arr[1])) { + permissions = arr[1]; + } else { + dst = arr[1]; + } + } + case 1: { + src = arr[0]; + break; + } + default: { + throw new IllegalArgumentException("Invalid device specification: " + deviceStr); + } + } + + if (dst == null || dst.length() == 0) { + dst = src; + } + + return new Device(permissions, dst, src); + } + + /** + * ValidDeviceMode checks if the mode for device is valid or not. + * Valid mode is a composition of r (read), w (write), and m (mknod). + * + * @link https://github.com/docker/docker/blob/6b4a46f28266031ce1a1315f17fb69113a06efe1/runconfig/opts/parse.go#L796 + */ + private static boolean validDeviceMode(String deviceMode) { + Map validModes = new HashMap<>(3); + validModes.put("r", true); + validModes.put("w", true); + validModes.put("m", true); + + if (deviceMode == null || deviceMode.length() == 0) { + return false; + } + + for (char ch : deviceMode.toCharArray()) { + final String mode = String.valueOf(ch); + if (!Boolean.TRUE.equals(validModes.get(mode))) { + return false; // wrong mode + } + validModes.put(mode, false); + } + + return true; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/DeviceRequest.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/DeviceRequest.java new file mode 100644 index 000000000..549d51b57 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/DeviceRequest.java @@ -0,0 +1,75 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.List; +import java.util.Map; + +@EqualsAndHashCode +@ToString +public class DeviceRequest extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + @JsonProperty("Driver") + private String driver; + + @JsonProperty("Count") + private Integer count; + + @JsonProperty("DeviceIDs") + private List deviceIds; + + @JsonProperty("Capabilities") + private List> capabilities; + + @JsonProperty("Options") + private Map options; + + public String getDriver() { + return driver; + } + + public DeviceRequest withDriver(String driver) { + this.driver = driver; + return this; + } + + public Integer getCount() { + return count; + } + + public DeviceRequest withCount(Integer count) { + this.count = count; + return this; + } + + public List getDeviceIds() { + return deviceIds; + } + + public DeviceRequest withDeviceIds(List deviceIds) { + this.deviceIds = deviceIds; + return this; + } + + public List> getCapabilities() { + return capabilities; + } + + public DeviceRequest withCapabilities(List> capabilities) { + this.capabilities = capabilities; + return this; + } + + public Map getOptions() { + return options; + } + + public DeviceRequest withOptions(Map options) { + this.options = options; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/DiscreteResourceSpec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/DiscreteResourceSpec.java new file mode 100644 index 000000000..80feee509 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/DiscreteResourceSpec.java @@ -0,0 +1,8 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; + +@Deprecated +public class DiscreteResourceSpec extends GenericResource implements Serializable { + private static final long serialVersionUID = 1L; +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/DockerObject.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/DockerObject.java new file mode 100644 index 000000000..463dc15a1 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/DockerObject.java @@ -0,0 +1,20 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonAnyGetter; + +import java.util.Collections; +import java.util.HashMap; +import java.util.Map; + +/** + * @see DockerObjectAccessor + */ +public abstract class DockerObject { + + HashMap rawValues = new HashMap<>(); + + @JsonAnyGetter + public Map getRawValues() { + return Collections.unmodifiableMap(this.rawValues); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/DockerObjectAccessor.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/DockerObjectAccessor.java new file mode 100644 index 000000000..0827c4a34 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/DockerObjectAccessor.java @@ -0,0 +1,27 @@ +package com.github.dockerjava.api.model; + +import java.util.HashMap; + +public final class DockerObjectAccessor { + + /** + * @deprecated not for public usage, unless you _really_ understand what you're doing + */ + @Deprecated + public static void overrideRawValues(DockerObject o, HashMap rawValues) { + o.rawValues = rawValues != null ? rawValues : new HashMap<>(); + } + + /** + * This is an advanced method for setting raw values on the resulting object + * that will fully overwrite any previously set value for given key. + * + * Make sure to check Docker's API before using it. + */ + public static void overrideRawValue(DockerObject o, String key, Object value) { + o.rawValues.put(key, value); + } + + private DockerObjectAccessor() { + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Driver.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Driver.java new file mode 100644 index 000000000..bdc05e53b --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Driver.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Map; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class Driver extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Name") + private String name; + + /** + * @since 1.24 + */ + @JsonProperty("Options") + private Map options; + + /** + * @see #name + */ + @CheckForNull + public String getName() { + return name; + } + + /** + * @see #name + */ + public Driver withName(String name) { + this.name = name; + return this; + } + + /** + * @see #options + */ + @CheckForNull + public Map getOptions() { + return options; + } + + /** + * @see #options + */ + public Driver withOptions(Map options) { + this.options = options; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/DriverStatus.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/DriverStatus.java new file mode 100644 index 000000000..57fe32247 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/DriverStatus.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * @author ben + */ +@EqualsAndHashCode +@ToString +public class DriverStatus extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Root Dir") + private String rootDir; + + @JsonProperty("Dirs") + private Integer dirs; + + public String getRootDir() { + return rootDir; + } + + public Integer getDirs() { + return dirs; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Endpoint.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Endpoint.java new file mode 100644 index 000000000..cebbfea1c --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Endpoint.java @@ -0,0 +1,83 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class Endpoint extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Spec") + private EndpointSpec spec; + + /** + * @since 1.24 + */ + @JsonProperty("Ports") + private PortConfig[] ports; + + /** + * @since 1.24 + */ + @JsonProperty("VirtualIPs") + private EndpointVirtualIP[] virtualIPs; + + /** + * @see #spec + */ + @CheckForNull + public EndpointSpec getSpec() { + return spec; + } + + /** + * @see #spec + */ + public Endpoint withSpec(EndpointSpec spec) { + this.spec = spec; + return this; + } + + /** + * @see #ports + */ + @CheckForNull + public PortConfig[] getPorts() { + return ports; + } + + /** + * @see #ports + */ + public Endpoint withPorts(PortConfig[] ports) { + this.ports = ports; + return this; + } + + /** + * @see #virtualIPs + */ + @CheckForNull + public EndpointVirtualIP[] getVirtualIPs() { + return virtualIPs; + } + + /** + * @see #virtualIPs + */ + public Endpoint withVirtualIPs(EndpointVirtualIP[] virtualIPs) { + this.virtualIPs = virtualIPs; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/EndpointResolutionMode.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/EndpointResolutionMode.java new file mode 100644 index 000000000..b67cea38b --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/EndpointResolutionMode.java @@ -0,0 +1,16 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum EndpointResolutionMode { + + @JsonProperty("vip") + VIP, + + @JsonProperty("dnsrr") + DNSRR + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/EndpointSpec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/EndpointSpec.java new file mode 100644 index 000000000..c0ce386fa --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/EndpointSpec.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class EndpointSpec extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Mode") + EndpointResolutionMode mode; + + /** + * @since 1.24 + */ + @JsonProperty("Ports") + List ports; + + /** + * @see #mode + */ + @CheckForNull + public EndpointResolutionMode getMode() { + return mode; + } + + /** + * @see #mode + */ + public EndpointSpec withMode(EndpointResolutionMode mode) { + this.mode = mode; + return this; + } + + /** + * @see #ports + */ + @CheckForNull + public List getPorts() { + return ports; + } + + /** + * @see #ports + */ + public EndpointSpec withPorts(List ports) { + this.ports = ports; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/EndpointVirtualIP.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/EndpointVirtualIP.java new file mode 100644 index 000000000..0babfba4c --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/EndpointVirtualIP.java @@ -0,0 +1,61 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class EndpointVirtualIP extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("NetworkID") + private String networkID; + + /** + * @since 1.24 + */ + @JsonProperty("Addr") + private String addr; + + /** + * @see #networkID + */ + @CheckForNull + public String getNetworkID() { + return networkID; + } + + /** + * @see #networkID + */ + public EndpointVirtualIP withNetworkID(String networkID) { + this.networkID = networkID; + return this; + } + + /** + * @see #addr + */ + @CheckForNull + public String getAddr() { + return addr; + } + + /** + * @see #addr + */ + public EndpointVirtualIP withAddr(String addr) { + this.addr = addr; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ErrorDetail.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ErrorDetail.java new file mode 100644 index 000000000..63e670772 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ErrorDetail.java @@ -0,0 +1,20 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +@EqualsAndHashCode +@ToString +public class ErrorDetail extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty + private String message; + + public String getMessage() { + return message; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ErrorResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ErrorResponse.java new file mode 100644 index 000000000..523a35729 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ErrorResponse.java @@ -0,0 +1,24 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +import java.io.Serializable; + +@Deprecated +public class ErrorResponse implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty + private ErrorDetail errorDetail; + + @JsonProperty + private String error; + + public ErrorDetail getErrorDetail() { + return errorDetail; + } + + public String getError() { + return error; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Event.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Event.java new file mode 100644 index 000000000..0eedbc553 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Event.java @@ -0,0 +1,240 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; + +import java.io.Serializable; + +/** + * Representation of a Docker event. + */ +@EqualsAndHashCode +@ToString +public class Event extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * @since 1.16 + */ + @JsonProperty("status") + private String status; + + /** + * @since 1.16 + */ + @JsonProperty("id") + private String id; + + /** + * @since 1.16 + */ + @JsonProperty("from") + private String from; + + /** + * ?? + */ + @JsonProperty("node") + private Node node; + + /* + * @since 1. + */ + @JsonProperty("Type") + private EventType type; + + /** + * @since 1.22 + */ + @JsonProperty("Action") + private String action; + + /** + * @since 1.22 + */ + @JsonProperty("Actor") + private EventActor actor; + + /** + * @since 1.16 + */ + @JsonProperty("time") + private Long time; + + /** + * @since 1.21 + */ + @JsonProperty("timeNano") + private Long timeNano; + + /** + * Default constructor for the deserialization. + */ + public Event() { + } + + /** + * Constructor. + * + * @param id Container ID + * @param status Status string. List of statuses is available in Docker API v.1.16 + * @param from Image, from which the container has been created + * @param time Event time The time is specified in milliseconds since January 1, 1970, 00:00:00 GMT + * @since 1.16 + */ + public Event(String status, String id, String from, Long time) { + this.status = status; + this.id = id; + this.from = from; + this.time = time; + } + + /** + * Status of docker image or container. List of statuses is available in Docker API v.1.16 + * + * @return Status string + */ + public String getStatus() { + return status; + } + + /** + * @see #status + */ + public Event withStatus(String status) { + this.status = status; + return this; + } + + /** + * Get ID of docker container. + * + * @return Container ID + */ + public String getId() { + return id; + } + + /** + * @see #id + */ + public Event withId(String id) { + this.id = id; + return this; + } + + /** + * Get source image of the container. + * + * @return Name of the parent container + */ + public String getFrom() { + return from; + } + + /** + * @see #from + */ + public Event withFrom(String from) { + this.from = from; + return this; + } + + /** + * Get the event time. The time is specified in milliseconds since January 1, 1970, 00:00:00 GMT + * + * @return Event time in the specified format. + */ + public Long getTime() { + return time; + } + + /** + * @see #time + */ + public Event withTime(Long time) { + this.time = time; + return this; + } + + /** + * @see #timeNano + */ + @CheckForNull + public Long getTimeNano() { + return timeNano; + } + + /** + * @see #timeNano + */ + public Event withTimenano(Long timenano) { + this.timeNano = timenano; + return this; + } + + /** + * Returns the node when working against docker swarm + */ + public Node getNode() { + return node; + } + + /** + * @see #node + */ + public Event withNode(Node node) { + this.node = node; + return this; + } + + @CheckForNull + public EventType getType() { + return type; + } + + /** + * @see #type + */ + public Event withType(EventType type) { + this.type = type; + return this; + } + + /** + * @see #action + */ + @CheckForNull + public String getAction() { + return action; + } + + /** + * @see #action + */ + public Event withAction(String action) { + this.action = action; + return this; + } + + /** + * @see #actor + */ + @CheckForNull + public EventActor getActor() { + return actor; + } + + /** + * @see #actor + */ + public Event withEventActor(EventActor actor) { + this.actor = actor; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/EventActor.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/EventActor.java new file mode 100644 index 000000000..fbcf088f7 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/EventActor.java @@ -0,0 +1,63 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Map; + +/** + * @author Kanstantsin Shautsou + * @since 1.22 + */ +@EqualsAndHashCode +@ToString +public class EventActor extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * @since 1.22 + */ + @JsonProperty("ID") + private String id; + + /** + * @since 1.22 + */ + @JsonProperty("Attributes") + private Map attributes; + + /** + * @see #id + */ + @CheckForNull + public String getId() { + return id; + } + + /** + * @see #id + */ + public EventActor withId(String id) { + this.id = id; + return this; + } + + /** + * @see #attributes + */ + @CheckForNull + public Map getAttributes() { + return attributes; + } + + /** + * @see #attributes + */ + public EventActor withAttributes(Map attributes) { + this.attributes = attributes; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/EventType.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/EventType.java new file mode 100644 index 000000000..b7c64ecc5 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/EventType.java @@ -0,0 +1,59 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; + +import javax.annotation.Nonnull; +import java.util.HashMap; +import java.util.Map; + +/** + * @since 1.24 + */ +public enum EventType { + CONFIG("config"), + /** + * @since 1.24 + */ + CONTAINER("container"), + + /** + * @since 1.24 + */ + DAEMON("daemon"), + + /** + * @since 1.24 + */ + IMAGE("image"), + NETWORK("network"), + NODE("node"), + PLUGIN("plugin"), + SECRET("secret"), + SERVICE("service"), + VOLUME("volume"); + + private static final Map EVENT_TYPES = new HashMap<>(); + + static { + for (EventType t : values()) { + EVENT_TYPES.put(t.name().toLowerCase(), t); + } + } + + private String value; + + EventType(@Nonnull String value) { + this.value = value; + } + + @JsonValue + public String getValue() { + return value; + } + + @JsonCreator + public static EventType forValue(String s) { + return EVENT_TYPES.get(s); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExposedPort.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExposedPort.java new file mode 100644 index 000000000..4226fd94b --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExposedPort.java @@ -0,0 +1,146 @@ +package com.github.dockerjava.api.model; + +import static com.github.dockerjava.api.model.InternetProtocol.TCP; +import static com.github.dockerjava.api.model.InternetProtocol.UDP; +import static com.github.dockerjava.api.model.InternetProtocol.SCTP; + +import java.io.Serializable; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; +import com.github.dockerjava.api.model.Ports.Binding; +import lombok.EqualsAndHashCode; + +/** + * Represents a container port that Docker exposes to external clients. The port is defined by its {@link #getPort() port number} and an + * {@link InternetProtocol}. It can be published by Docker by {@link Ports#bind(ExposedPort, Binding) binding} it to a host port, + * represented by a {@link Binding}. + */ +@EqualsAndHashCode +public class ExposedPort implements Serializable { + private static final long serialVersionUID = 1L; + + private final InternetProtocol protocol; + + private final int port; + + /** + * Creates an {@link ExposedPort} for the given parameters. + * + * @param port + * the {@link #getPort() port number} + * @param protocol + * the {@link InternetProtocol} + */ + public ExposedPort(int port, InternetProtocol protocol) { + this.port = port; + this.protocol = protocol; + } + + /** + * Creates an {@link ExposedPort} for the given {@link #getPort() port number} and {@link InternetProtocol#DEFAULT}. + * + * @param port + * the {@link #getPort() port number} + */ + public ExposedPort(int port) { + this(port, InternetProtocol.DEFAULT); + } + + /** + * Creates an {@link ExposedPort} for the given parameters. + * + * @param scheme + * the {@link #getScheme() scheme}, tcp, udp or sctp + * @param port + * the {@link #getPort() port number} + * @deprecated use {@link #ExposedPort(int, InternetProtocol)} + */ + @Deprecated + public ExposedPort(String scheme, int port) { + this(port, InternetProtocol.valueOf(scheme)); + } + + /** + * @return the {@link InternetProtocol} of the {@link #getPort() port} that the container exposes + */ + public InternetProtocol getProtocol() { + return protocol; + } + + /** + * @return the scheme (internet protocol), tcp, udp or sctp + * @deprecated use {@link #getProtocol()} + */ + @Deprecated + public String getScheme() { + return protocol.toString(); + } + + /** @return the port number that the container exposes */ + public int getPort() { + return port; + } + + /** + * Creates an {@link ExposedPort} for {@link InternetProtocol#TCP}. This is a shortcut for + * new ExposedPort(port, {@link InternetProtocol#TCP}) + */ + public static ExposedPort tcp(int port) { + return new ExposedPort(port, TCP); + } + + /** + * Creates an {@link ExposedPort} for {@link InternetProtocol#UDP}. This is a shortcut for + * new ExposedPort(port, {@link InternetProtocol#UDP}) + */ + public static ExposedPort udp(int port) { + return new ExposedPort(port, UDP); + } + + /** + * Creates an {@link ExposedPort} for {@link InternetProtocol#SCTP}. This is a shortcut for + * new ExposedPort(port, {@link InternetProtocol#SCTP}) + */ + public static ExposedPort sctp(int port) { + return new ExposedPort(port, SCTP); + } + + /** + * Parses a textual port specification (as used by the Docker CLI) to an {@link ExposedPort}. + * + * @param serialized + * the specification, e.g. 80/tcp + * @return an {@link ExposedPort} matching the specification + * @throws IllegalArgumentException + * if the specification cannot be parsed + */ + @JsonCreator + public static ExposedPort parse(String serialized) throws IllegalArgumentException { + try { + String[] parts = serialized.split("/"); + switch (parts.length) { + case 1: + return new ExposedPort(Integer.parseInt(parts[0])); + case 2: + return new ExposedPort(Integer.parseInt(parts[0]), InternetProtocol.parse(parts[1])); + default: + throw new IllegalArgumentException(); + } + } catch (Exception e) { + throw new IllegalArgumentException("Error parsing ExposedPort '" + serialized + "'"); + } + } + + /** + * Returns a string representation of this {@link ExposedPort} suitable for inclusion in a JSON message. The format is + * port/protocol, like the argument in {@link #parse(String)}. + * + * @return a string representation of this {@link ExposedPort} + */ + @Override + @JsonValue + public String toString() { + return port + "/" + protocol.toString(); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExposedPorts.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExposedPorts.java new file mode 100644 index 000000000..6f5ae9ebd --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExposedPorts.java @@ -0,0 +1,47 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; +import lombok.ToString; + +import java.io.Serializable; +import java.util.List; +import java.util.Map; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +@ToString +public class ExposedPorts implements Serializable { + private static final long serialVersionUID = 1L; + + private ExposedPort[] exposedPorts; + + public ExposedPorts(ExposedPort... exposedPorts) { + this.exposedPorts = exposedPorts; + } + + public ExposedPorts(List exposedPorts) { + this.exposedPorts = exposedPorts.toArray(new ExposedPort[exposedPorts.size()]); + } + + public ExposedPort[] getExposedPorts() { + return exposedPorts; + } + + @JsonCreator + public static ExposedPorts fromPrimitive(Map object) { + return new ExposedPorts( + object.keySet().stream().map(ExposedPort::parse).toArray(ExposedPort[]::new) + ); + } + + @JsonValue + public Map toPrimitive() { + return Stream.of(exposedPorts).collect(Collectors.toMap( + ExposedPort::toString, + __ -> new Object(), + (a, b) -> a + )); + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExternalCA.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExternalCA.java new file mode 100644 index 000000000..3a68410d8 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExternalCA.java @@ -0,0 +1,85 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Map; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ExternalCA extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Protocol") + private ExternalCAProtocol protocol; + + /** + * @since 1.24 + */ + @JsonProperty("URL") + private String url; + + /** + * @since 1.24 + */ + @JsonProperty("Options") + private Map options; + + /** + * @see #protocol + */ + @CheckForNull + public ExternalCAProtocol getProtocol() { + return protocol; + } + + /** + * @see #protocol + */ + public ExternalCA withProtocol(ExternalCAProtocol protocol) { + this.protocol = protocol; + return this; + } + + /** + * @see #url + */ + @CheckForNull + public String getUrl() { + return url; + } + + /** + * @see #url + */ + public ExternalCA withUrl(String url) { + this.url = url; + return this; + } + + /** + * @see #options + */ + @CheckForNull + public Map getOptions() { + return options; + } + + /** + * @see #options + */ + public ExternalCA withOptions(Map options) { + this.options = options; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExternalCAProtocol.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExternalCAProtocol.java new file mode 100644 index 000000000..b4268a3eb --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ExternalCAProtocol.java @@ -0,0 +1,12 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum ExternalCAProtocol { + + @JsonProperty("cfssl") + CFSSL +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Frame.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Frame.java new file mode 100644 index 000000000..fdd5dd62e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Frame.java @@ -0,0 +1,35 @@ +package com.github.dockerjava.api.model; + +import lombok.EqualsAndHashCode; + +import java.io.Serializable; + +/** + * Represents a logging frame. + */ +@EqualsAndHashCode +public class Frame extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + private final StreamType streamType; + + private final byte[] payload; + + public Frame(StreamType streamType, byte[] payload) { + this.streamType = streamType; + this.payload = payload; + } + + public StreamType getStreamType() { + return streamType; + } + + public byte[] getPayload() { + return payload; + } + + @Override + public String toString() { + return String.format("%s: %s", streamType, new String(payload).trim()); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/GenericResource.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/GenericResource.java new file mode 100644 index 000000000..f6ddfabe5 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/GenericResource.java @@ -0,0 +1,36 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +@EqualsAndHashCode +@ToString +public abstract class GenericResource extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Kind") + String kind; + @JsonProperty("Value") + T value = null; + + public String getKind() { + return kind; + } + + public GenericResource withKind(String kind) { + this.kind = kind; + return this; + } + + public T getValue() { + return value; + } + + public GenericResource withValue(T value) { + this.value = value; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/HealthCheck.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/HealthCheck.java new file mode 100644 index 000000000..0e41b873f --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/HealthCheck.java @@ -0,0 +1,133 @@ +/* + * Copyright 2017 cdancy. + * + * 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.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.List; + +/** + * + * @author cdancy + */ +@EqualsAndHashCode +@ToString +public class HealthCheck extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Interval") + private Long interval; + + @JsonProperty("Timeout") + private Long timeout; + + /** + * @since 1.26 + */ + @JsonProperty("Test") + private List test; + + /** + * @since 1.26 + */ + @JsonProperty("Retries") + private Integer retries; + + /** + * @since 1.26 + */ + @JsonProperty("StartPeriod") + private Long startPeriod; + + /** + * @since 1.44 + */ + @JsonProperty("StartInterval") + private Long startInterval; + + public Long getInterval() { + return interval; + } + + public Long getTimeout() { + return timeout; + } + + /** + * Set interval in nanoseconds + * @return this {@link HealthCheck} instance + */ + public HealthCheck withInterval(Long interval) { + this.interval = interval; + return this; + } + + /** + * Set timeout in nanoseconds + * @return this {@link HealthCheck} instance + */ + public HealthCheck withTimeout(Long timeout) { + this.timeout = timeout; + return this; + } + + public List getTest() { + return test; + } + + public HealthCheck withTest(List test) { + this.test = test; + return this; + } + + public Integer getRetries() { + return retries; + } + + public HealthCheck withRetries(Integer retries) { + this.retries = retries; + return this; + } + + public Long getStartPeriod() { + return startPeriod; + } + + /** + * Set startPeriod in nanoseconds + * @return this {@link HealthCheck} instance + */ + public HealthCheck withStartPeriod(Long startPeriod) { + this.startPeriod = startPeriod; + return this; + } + + public Long getStartInterval() { + return startInterval; + } + + /** + * Set startInterval in nanoseconds + * @return this {@link HealthCheck} instance + */ + public HealthCheck withStartInterval(Long startInterval) { + this.startInterval = startInterval; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/HostConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/HostConfig.java new file mode 100644 index 000000000..603bc6347 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/HostConfig.java @@ -0,0 +1,1272 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonIgnore; +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +import static java.util.Objects.requireNonNull; + +/** + * Used in `/containers/create`, and in inspect container. + * TODO exclude usage for 2 different models. + */ +@EqualsAndHashCode +@ToString +public class HostConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + private static final List PREDEFINED_NETWORKS = Arrays.asList("bridge", "host", "none"); + + public static HostConfig newHostConfig() { + return new HostConfig(); + } + + @JsonProperty("Binds") + private Binds binds; + + @JsonProperty("BlkioWeight") + private Integer blkioWeight; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("BlkioWeightDevice") + private List blkioWeightDevice; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("BlkioDeviceReadBps") + private List blkioDeviceReadBps; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("BlkioDeviceWriteBps") + private List blkioDeviceWriteBps; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("BlkioDeviceReadIOps") + private List blkioDeviceReadIOps; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("BlkioDeviceWriteIOps") + private List blkioDeviceWriteIOps; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_20} + */ + @JsonProperty("MemorySwappiness") + private Long memorySwappiness; + + @JsonProperty("NanoCpus") + private Long nanoCPUs; + + @JsonProperty("Annotations") + private Map annotations; + + @JsonProperty("CapAdd") + private Capability[] capAdd; + + @JsonProperty("CapDrop") + private Capability[] capDrop; + + @JsonProperty("ContainerIDFile") + private String containerIDFile; + + @JsonProperty("CpuPeriod") + private Long cpuPeriod; + + @JsonProperty("CpuRealtimePeriod") + private Long cpuRealtimePeriod; + + @JsonProperty("CpuRealtimeRuntime") + private Long cpuRealtimeRuntime; + + @JsonProperty("CpuShares") + private Integer cpuShares; + + /** + * @since ~{@link RemoteApiVersion#VERSION_1_20} + */ + @JsonProperty("CpuQuota") + private Long cpuQuota; + + @JsonProperty("CpusetCpus") + private String cpusetCpus; + + @JsonProperty("CpusetMems") + private String cpusetMems; + + @JsonProperty("Devices") + private Device[] devices; + + @JsonProperty("DeviceCgroupRules") + private List deviceCgroupRules; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_40} + */ + @JsonProperty("DeviceRequests") + private List deviceRequests; + + /** + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + @JsonProperty("DiskQuota") + private Long diskQuota; + + @JsonProperty("Dns") + private String[] dns; + + @JsonProperty("DnsOptions") + private List dnsOptions; + + @JsonProperty("DnsSearch") + private String[] dnsSearch; + + @JsonProperty("ExtraHosts") + private String[] extraHosts; + + @JsonProperty("GroupAdd") + private List groupAdd; + + @JsonProperty("IpcMode") + private String ipcMode; + + @JsonProperty("Cgroup") + private String cgroup; + + @JsonProperty("Links") + private Links links; + + @JsonProperty("LogConfig") + private LogConfig logConfig; + + @JsonProperty("LxcConf") + private LxcConf[] lxcConf; + + @JsonProperty("Memory") + private Long memory; + + @JsonProperty("MemorySwap") + private Long memorySwap; + + /** + * @since {@link RemoteApiVersion#VERSION_1_21} + */ + @JsonProperty("MemoryReservation") + private Long memoryReservation; + + /** + * @since {@link RemoteApiVersion#VERSION_1_21} + */ + @JsonProperty("KernelMemory") + private Long kernelMemory; + + @JsonProperty("NetworkMode") + private String networkMode; + + @JsonProperty("OomKillDisable") + private Boolean oomKillDisable; + + @JsonProperty("Init") + private Boolean init; + + /** + * @since {@link RemoteApiVersion#VERSION_1_25} + */ + @JsonProperty("AutoRemove") + private Boolean autoRemove; + + /** + * @since {@link RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("OomScoreAdj") + private Integer oomScoreAdj; + + @JsonProperty("PortBindings") + private Ports portBindings; + + @JsonProperty("Privileged") + private Boolean privileged; + + @JsonProperty("PublishAllPorts") + private Boolean publishAllPorts; + + @JsonProperty("ReadonlyRootfs") + private Boolean readonlyRootfs; + + @JsonProperty("RestartPolicy") + private RestartPolicy restartPolicy; + + @JsonProperty("Ulimits") + private Ulimit[] ulimits; + + @JsonProperty("CpuCount") + private Long cpuCount; + + @JsonProperty("CpuPercent") + private Long cpuPercent; + + @JsonProperty("IOMaximumIOps") + private Long ioMaximumIOps; + + @JsonProperty("IOMaximumBandwidth") + private Long ioMaximumBandwidth; + + @JsonProperty("VolumesFrom") + private VolumesFrom[] volumesFrom; + + @JsonProperty("Mounts") + private List mounts; + + @JsonProperty("PidMode") + private String pidMode; + + @JsonProperty("Isolation") + private Isolation isolation; + + /** + * @since {@link RemoteApiVersion#VERSION_1_20} + */ + @JsonProperty("SecurityOpt") + private List securityOpts; + + @JsonProperty("StorageOpt") + private Map storageOpt; + + /** + * @since {@link RemoteApiVersion#VERSION_1_20} + */ + @JsonProperty("CgroupParent") + private String cgroupParent; + + /** + * @since {@link RemoteApiVersion#VERSION_1_21} + */ + @JsonProperty("VolumeDriver") + private String volumeDriver; + + /** + * @since {@link RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("ShmSize") + private Long shmSize; + + /** + * @since ~{@link RemoteApiVersion#VERSION_1_23} + */ + @JsonProperty("PidsLimit") + private Long pidsLimit; + + /** + * @since ~{@link RemoteApiVersion#VERSION_1_30} + */ + @JsonProperty("Runtime") + private String runtime; + + /** + * @since ~{@link RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("Tmpfs") + private Map tmpFs; + + @JsonProperty("UTSMode") + private String utSMode; + + @JsonProperty("UsernsMode") + private String usernsMode; + + @JsonProperty("Sysctls") + private Map sysctls; + + @JsonProperty("ConsoleSize") + private List consoleSize; + + @JsonProperty("CgroupnsMode") + private String cgroupnsMode; + + @JsonIgnore + public Bind[] getBinds() { + return (binds == null) ? new Bind[0] : binds.getBinds(); + } + + public Integer getBlkioWeight() { + return blkioWeight; + } + + @CheckForNull + public Map getAnnotations() { + return annotations; + } + + public Capability[] getCapAdd() { + return capAdd; + } + + public Capability[] getCapDrop() { + return capDrop; + } + + public String getContainerIDFile() { + return containerIDFile; + } + + public Long getCpuPeriod() { + return cpuPeriod; + } + + + public Integer getCpuShares() { + return cpuShares; + } + + public String getCpusetCpus() { + return cpusetCpus; + } + + public String getCpusetMems() { + return cpusetMems; + } + + public Device[] getDevices() { + return devices; + } + + public Long getDiskQuota() { + return diskQuota; + } + + public String[] getDns() { + return dns; + } + + public String[] getDnsSearch() { + return dnsSearch; + } + + public String[] getExtraHosts() { + return extraHosts; + } + + @JsonIgnore + public Link[] getLinks() { + return (links == null) ? new Link[0] : links.getLinks(); + } + + @JsonIgnore + public LogConfig getLogConfig() { + return (logConfig == null) ? new LogConfig() : logConfig; + } + + public LxcConf[] getLxcConf() { + return lxcConf; + } + + public Long getMemory() { + return memory; + } + + public Long getMemorySwap() { + return memorySwap; + } + + public String getNetworkMode() { + return networkMode; + } + + public Ports getPortBindings() { + return portBindings; + } + + public RestartPolicy getRestartPolicy() { + return restartPolicy; + } + + public Ulimit[] getUlimits() { + return ulimits; + } + + public VolumesFrom[] getVolumesFrom() { + return volumesFrom; + } + + @CheckForNull + public String getPidMode() { + return pidMode; + } + + /** + * @see #blkioDeviceReadBps + */ + @CheckForNull + public List getBlkioDeviceReadBps() { + return blkioDeviceReadBps; + } + + /** + * @see #blkioDeviceReadIOps + */ + @CheckForNull + public List getBlkioDeviceReadIOps() { + return blkioDeviceReadIOps; + } + + /** + * @see #blkioDeviceWriteBps + */ + @CheckForNull + public List getBlkioDeviceWriteBps() { + return blkioDeviceWriteBps; + } + + /** + * @see #blkioDeviceWriteIOps + */ + @CheckForNull + public List getBlkioDeviceWriteIOps() { + return blkioDeviceWriteIOps; + } + + /** + * @see #blkioWeightDevice + */ + @CheckForNull + public List getBlkioWeightDevice() { + return blkioWeightDevice; + } + + /** + * @see #oomScoreAdj + */ + @CheckForNull + public Integer getOomScoreAdj() { + return oomScoreAdj; + } + + /** + * @see #cpuQuota + */ + @CheckForNull + public Long getCpuQuota() { + return cpuQuota; + } + + /** + * @see #kernelMemory + */ + @CheckForNull + public Long getKernelMemory() { + return kernelMemory; + } + + /** + * @see #memoryReservation + */ + @CheckForNull + public Long getMemoryReservation() { + return memoryReservation; + } + + /** + * @see #memorySwappiness + */ + @CheckForNull + public Long getMemorySwappiness() { + return memorySwappiness; + } + + /** + * @see #oomKillDisable + */ + @CheckForNull + public Boolean getOomKillDisable() { + return oomKillDisable; + } + + /** + * @see #autoRemove + */ + @CheckForNull + public Boolean getAutoRemove() { + return autoRemove; + } + + /** + * @see #securityOpts + */ + @CheckForNull + public List getSecurityOpts() { + return securityOpts; + } + + /** + * @see #cgroupParent + */ + @CheckForNull + public String getCgroupParent() { + return cgroupParent; + } + + /** + * @see #shmSize + */ + @CheckForNull + public Long getShmSize() { + return shmSize; + } + + /** + * @see #volumeDriver + */ + @CheckForNull + public String getVolumeDriver() { + return volumeDriver; + } + + /** + * @see #pidsLimit + */ + @CheckForNull + public Long getPidsLimit() { + return pidsLimit; + } + + /** + * @see #tmpFs + */ + @CheckForNull + public Map getTmpFs() { + return tmpFs; + } + + /** + * Parse the network mode as specified at + * {@see https://github.com/docker/engine-api/blob/master/types/container/hostconfig_unix.go} + */ + @JsonIgnore + public boolean isUserDefinedNetwork() { + return networkMode != null && !PREDEFINED_NETWORKS.contains(networkMode) && !networkMode.startsWith("container:"); + } + + public String getRuntime() { + return runtime; + } + + @JsonIgnore + public void setBinds(Bind... binds) { + this.binds = new Binds(binds); + } + + @JsonIgnore + public void setLinks(Link... links) { + this.links = new Links(links); + } + + // auto-generated builder setters + + /** + * @see #binds + */ + public HostConfig withBinds(Binds binds) { + this.binds = binds; + return this; + } + + public HostConfig withBinds(Bind... binds) { + requireNonNull(binds, "binds was not specified"); + setBinds(binds); + return this; + } + + public HostConfig withBinds(List binds) { + requireNonNull(binds, "binds was not specified"); + return withBinds(binds.toArray(new Bind[binds.size()])); + } + + /** + * @see #blkioDeviceReadBps + */ + public HostConfig withBlkioDeviceReadBps(List blkioDeviceReadBps) { + this.blkioDeviceReadBps = blkioDeviceReadBps; + return this; + } + + /** + * @see #blkioDeviceReadIOps + */ + public HostConfig withBlkioDeviceReadIOps(List blkioDeviceReadIOps) { + this.blkioDeviceReadIOps = blkioDeviceReadIOps; + return this; + } + + /** + * @see #blkioDeviceWriteBps + */ + public HostConfig withBlkioDeviceWriteBps(List blkioDeviceWriteBps) { + this.blkioDeviceWriteBps = blkioDeviceWriteBps; + return this; + } + + /** + * @see #blkioDeviceWriteIOps + */ + public HostConfig withBlkioDeviceWriteIOps(List blkioDeviceWriteIOps) { + this.blkioDeviceWriteIOps = blkioDeviceWriteIOps; + return this; + } + + /** + * @see #blkioWeight + */ + public HostConfig withBlkioWeight(Integer blkioWeight) { + this.blkioWeight = blkioWeight; + return this; + } + + /** + * @see #blkioWeightDevice + */ + public HostConfig withBlkioWeightDevice(List blkioWeightDevice) { + this.blkioWeightDevice = blkioWeightDevice; + return this; + } + + public HostConfig withAnnotations(Map annotations) { + this.annotations = annotations; + return this; + } + + /** + * @see #capAdd + */ + public HostConfig withCapAdd(Capability... capAdd) { + this.capAdd = capAdd; + return this; + } + + /** + * @see #capDrop + */ + public HostConfig withCapDrop(Capability... capDrop) { + this.capDrop = capDrop; + return this; + } + + /** + * @see #cgroupParent + */ + public HostConfig withCgroupParent(String cgroupParent) { + this.cgroupParent = cgroupParent; + return this; + } + + /** + * @see #containerIDFile + */ + public HostConfig withContainerIDFile(String containerIDFile) { + this.containerIDFile = containerIDFile; + return this; + } + + /** + * @see #cpuPeriod + */ + public HostConfig withCpuPeriod(Long cpuPeriod) { + this.cpuPeriod = cpuPeriod; + return this; + } + + /** + * @see #cpuQuota + */ + public HostConfig withCpuQuota(Long cpuQuota) { + this.cpuQuota = cpuQuota; + return this; + } + + /** + * @see #cpusetCpus + */ + public HostConfig withCpusetCpus(String cpusetCpus) { + this.cpusetCpus = cpusetCpus; + return this; + } + + /** + * @see #cpusetMems + */ + public HostConfig withCpusetMems(String cpusetMems) { + this.cpusetMems = cpusetMems; + return this; + } + + /** + * @see #cpuShares + */ + public HostConfig withCpuShares(Integer cpuShares) { + this.cpuShares = cpuShares; + return this; + } + + /** + * @see #devices + */ + public HostConfig withDevices(Device... devices) { + this.devices = devices; + return this; + } + + public HostConfig withDevices(List devices) { + requireNonNull(devices, "devices was not specified"); + return withDevices(devices.toArray(new Device[0])); + } + + /** + * @see #diskQuota + */ + public HostConfig withDiskQuota(Long diskQuota) { + this.diskQuota = diskQuota; + return this; + } + + /** + * @see #dns + */ + public HostConfig withDns(String... dns) { + this.dns = dns; + return this; + } + + public HostConfig withDns(List dns) { + requireNonNull(dns, "dns was not specified"); + return withDns(dns.toArray(new String[0])); + } + + /** + * @see #dnsSearch + */ + public HostConfig withDnsSearch(String... dnsSearch) { + this.dnsSearch = dnsSearch; + return this; + } + + public HostConfig withDnsSearch(List dnsSearch) { + requireNonNull(dnsSearch, "dnsSearch was not specified"); + return withDnsSearch(dnsSearch.toArray(new String[0])); + } + + /** + * @see #extraHosts + */ + public HostConfig withExtraHosts(String... extraHosts) { + this.extraHosts = extraHosts; + return this; + } + + /** + * @see #kernelMemory + */ + public HostConfig withKernelMemory(Long kernelMemory) { + this.kernelMemory = kernelMemory; + return this; + } + + /** + * @see #links + */ + public HostConfig withLinks(Links links) { + this.links = links; + return this; + } + + public HostConfig withLinks(Link... links) { + requireNonNull(links, "links was not specified"); + setLinks(links); + return this; + } + + public HostConfig withLinks(List links) { + requireNonNull(links, "links was not specified"); + return withLinks(links.toArray(new Link[0])); + } + + + /** + * @see #logConfig + */ + public HostConfig withLogConfig(LogConfig logConfig) { + this.logConfig = logConfig; + return this; + } + + /** + * @see #lxcConf + */ + public HostConfig withLxcConf(LxcConf[] lxcConf) { + this.lxcConf = lxcConf; + return this; + } + + /** + * @see #memory + */ + public HostConfig withMemory(Long memory) { + this.memory = memory; + return this; + } + + /** + * @see #memoryReservation + */ + public HostConfig withMemoryReservation(Long memoryReservation) { + this.memoryReservation = memoryReservation; + return this; + } + + /** + * @see #memorySwap + */ + public HostConfig withMemorySwap(Long memorySwap) { + this.memorySwap = memorySwap; + return this; + } + + /** + * @see #memorySwappiness + */ + public HostConfig withMemorySwappiness(Long memorySwappiness) { + this.memorySwappiness = memorySwappiness; + return this; + } + + /** + * Set the Network mode for the container + *
    + *
  • 'bridge': creates a new network stack for the container on the docker bridge
  • + *
  • 'none': no networking for this container
  • + *
  • 'container:': reuses another container network stack
  • + *
  • 'host': use the host network stack inside the container. Note: the host mode gives the container full access to local system + * services such as D-bus and is therefore considered insecure.
  • + *
+ * Any other value is interpreted as a custom network's name for this container to connect to. + */ + public HostConfig withNetworkMode(String networkMode) { + this.networkMode = networkMode; + return this; + } + + /** + * @see #oomKillDisable + * @since 1.19 + */ + public HostConfig withOomKillDisable(Boolean oomKillDisable) { + this.oomKillDisable = oomKillDisable; + return this; + } + + /** + * @see #autoRemove + */ + public HostConfig withAutoRemove(Boolean autoRemove) { + this.autoRemove = autoRemove; + return this; + } + + /** + * @see #oomScoreAdj + */ + public HostConfig withOomScoreAdj(Integer oomScoreAdj) { + this.oomScoreAdj = oomScoreAdj; + return this; + } + + /** + * Set the PID (Process) Namespace mode for the container, 'host': use the host's PID namespace inside the container + * + * @see #pidMode + */ + public HostConfig withPidMode(String pidMode) { + this.pidMode = pidMode; + return this; + } + + /** + * Add one or more {@link PortBinding}s. This corresponds to the --publish (-p) option of the + * docker run CLI command. + */ + public HostConfig withPortBindings(Ports portBindings) { + this.portBindings = portBindings; + return this; + } + + public HostConfig withPortBindings(PortBinding... portBindings) { + requireNonNull(portBindings, "portBindings was not specified"); + withPortBindings(new Ports(portBindings)); + return this; + } + + public HostConfig withPortBindings(List portBindings) { + requireNonNull(portBindings, "portBindings was not specified"); + return withPortBindings(portBindings.toArray(new PortBinding[0])); + } + + /** + * @see #privileged + */ + @CheckForNull + public Boolean getPrivileged() { + return privileged; + } + + /** + * @see #privileged + */ + public HostConfig withPrivileged(Boolean privileged) { + this.privileged = privileged; + return this; + } + + /** + * @see #publishAllPorts + */ + @CheckForNull + public Boolean getPublishAllPorts() { + return publishAllPorts; + } + + /** + * @see #publishAllPorts + */ + public HostConfig withPublishAllPorts(Boolean publishAllPorts) { + this.publishAllPorts = publishAllPorts; + return this; + } + + /** + * @see #readonlyRootfs + */ + @CheckForNull + public Boolean getReadonlyRootfs() { + return readonlyRootfs; + } + + /** + * @see #readonlyRootfs + */ + public HostConfig withReadonlyRootfs(Boolean readonlyRootfs) { + this.readonlyRootfs = readonlyRootfs; + return this; + } + + /** + * Set custom {@link RestartPolicy} for the container. Defaults to {@link RestartPolicy#noRestart()} + */ + public HostConfig withRestartPolicy(RestartPolicy restartPolicy) { + this.restartPolicy = restartPolicy; + return this; + } + + /** + * @see #securityOpts + */ + public HostConfig withSecurityOpts(List securityOpts) { + this.securityOpts = securityOpts; + return this; + } + + /** + * @see #shmSize + */ + public HostConfig withShmSize(Long shmSize) { + this.shmSize = shmSize; + return this; + } + + /** + * @see #ulimits + */ + public HostConfig withUlimits(Ulimit[] ulimits) { + this.ulimits = ulimits; + return this; + } + + public HostConfig withUlimits(List ulimits) { + requireNonNull(ulimits, "no ulimits was specified"); + return withUlimits(ulimits.toArray(new Ulimit[0])); + } + + /** + * @see #volumeDriver + */ + public HostConfig withVolumeDriver(String volumeDriver) { + this.volumeDriver = volumeDriver; + return this; + } + + /** + * @see #volumesFrom + */ + public HostConfig withVolumesFrom(VolumesFrom... volumesFrom) { + this.volumesFrom = volumesFrom; + return this; + } + + public HostConfig withVolumesFrom(List volumesFrom) { + requireNonNull(volumesFrom, "volumesFrom was not specified"); + return withVolumesFrom(volumesFrom.toArray(new VolumesFrom[0])); + } + + /** + * @see #pidsLimit + */ + public HostConfig withPidsLimit(Long pidsLimit) { + this.pidsLimit = pidsLimit; + return this; + } + + public HostConfig withRuntime(String runtime) { + this.runtime = runtime; + return this; + } + + /** + * @see #tmpFs + */ + public HostConfig withTmpFs(Map tmpFs) { + this.tmpFs = tmpFs; + return this; + } + + @CheckForNull + public List getDeviceCgroupRules() { + return deviceCgroupRules; + } + + public HostConfig withDeviceCgroupRules(List deviceCgroupRules) { + this.deviceCgroupRules = deviceCgroupRules; + return this; + } + + @CheckForNull + public List getDeviceRequests() { + return deviceRequests; + } + + public HostConfig withDeviceRequests(List deviceRequests) { + this.deviceRequests = deviceRequests; + return this; + } + + @CheckForNull + public Long getNanoCPUs() { + return nanoCPUs; + } + + public HostConfig withNanoCPUs(Long nanoCPUs) { + this.nanoCPUs = nanoCPUs; + return this; + } + + @CheckForNull + public Boolean getInit() { + return init; + } + + public HostConfig withInit(Boolean init) { + this.init = init; + return this; + } + + @CheckForNull + public Long getCpuCount() { + return cpuCount; + } + + public HostConfig withCpuCount(Long cpuCount) { + this.cpuCount = cpuCount; + return this; + } + + @CheckForNull + public Long getCpuPercent() { + return cpuPercent; + } + + public HostConfig withCpuPercent(Long cpuPercent) { + this.cpuPercent = cpuPercent; + return this; + } + + @CheckForNull + public Long getIoMaximumIOps() { + return ioMaximumIOps; + } + + public HostConfig withIoMaximumIOps(Long ioMaximumIOps) { + this.ioMaximumIOps = ioMaximumIOps; + return this; + } + + @CheckForNull + public Long getIoMaximumBandwidth() { + return ioMaximumBandwidth; + } + + public HostConfig withIoMaximumBandwidth(Long ioMaximumBandwidth) { + this.ioMaximumBandwidth = ioMaximumBandwidth; + return this; + } + + @CheckForNull + public List getMounts() { + return mounts; + } + + public HostConfig withMounts(List mounts) { + this.mounts = mounts; + return this; + } + + @CheckForNull + public List getDnsOptions() { + return dnsOptions; + } + + public HostConfig withDnsOptions(List dnsOptions) { + this.dnsOptions = dnsOptions; + return this; + } + + @CheckForNull + public List getGroupAdd() { + return groupAdd; + } + + public HostConfig withGroupAdd(List groupAdd) { + this.groupAdd = groupAdd; + return this; + } + + @CheckForNull + public String getIpcMode() { + return ipcMode; + } + + public HostConfig withIpcMode(String ipcMode) { + this.ipcMode = ipcMode; + return this; + } + + @CheckForNull + public String getCgroup() { + return cgroup; + } + + public HostConfig withCgroup(String cgroup) { + this.cgroup = cgroup; + return this; + } + + @CheckForNull + public Map getStorageOpt() { + return storageOpt; + } + + public HostConfig withStorageOpt(Map storageOpt) { + this.storageOpt = storageOpt; + return this; + } + + @CheckForNull + public String getUtSMode() { + return utSMode; + } + + public HostConfig withUtSMode(String utSMode) { + this.utSMode = utSMode; + return this; + } + + @CheckForNull + public String getUsernsMode() { + return usernsMode; + } + + public HostConfig withUsernsMode(String usernsMode) { + this.usernsMode = usernsMode; + return this; + } + + @CheckForNull + public String getCgroupnsMode() { + return cgroupnsMode; + } + + public HostConfig withCgroupnsMode(String cgroupnsMode) { + this.cgroupnsMode = cgroupnsMode; + return this; + } + + @CheckForNull + public Map getSysctls() { + return sysctls; + } + + public HostConfig withSysctls(Map sysctls) { + this.sysctls = sysctls; + return this; + } + + @CheckForNull + public List getConsoleSize() { + return consoleSize; + } + + public HostConfig withConsoleSize(List consoleSize) { + this.consoleSize = consoleSize; + return this; + } + + @CheckForNull + public Isolation getIsolation() { + return isolation; + } + + public HostConfig withIsolation(Isolation isolation) { + this.isolation = isolation; + return this; + } + + @CheckForNull + public Long getCpuRealtimePeriod() { + return cpuRealtimePeriod; + } + + public HostConfig withCpuRealtimePeriod(Long cpuRealtimePeriod) { + this.cpuRealtimePeriod = cpuRealtimePeriod; + return this; + } + + @CheckForNull + public Long getCpuRealtimeRuntime() { + return cpuRealtimeRuntime; + } + + public HostConfig withCpuRealtimeRuntime(Long cpuRealtimeRuntime) { + this.cpuRealtimeRuntime = cpuRealtimeRuntime; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Identifier.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Identifier.java new file mode 100644 index 000000000..a690548b4 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Identifier.java @@ -0,0 +1,54 @@ +package com.github.dockerjava.api.model; + +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.Optional; + +/** + * @author magnayn + */ +@EqualsAndHashCode +@ToString +public class Identifier extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + public final Repository repository; + + public final Optional tag; + + public Identifier(Repository repository, String tag) { + this.repository = repository; + + this.tag = Optional.ofNullable(tag); + } + + /** + * Return an identifier that correctly splits up the repository and tag. There can be > 1 ":" fred/jim --> fred/jim, [] + * fred/jim:123 --> fred/jim, 123 fred:123/jim:123 --> fred:123/jim, 123 + * + * + * @param identifier + * as a string + * @return parsed identifier. + */ + public static Identifier fromCompoundString(String identifier) { + String[] parts = identifier.split("/"); + if (parts.length != 2) { + String[] rhs = identifier.split(":"); + if (rhs.length != 2) { + return new Identifier(new Repository(identifier), null); + } else { + return new Identifier(new Repository(rhs[0]), rhs[1]); + } + } + + String[] rhs = parts[1].split(":"); + if (rhs.length != 2) { + return new Identifier(new Repository(identifier), null); + } + + return new Identifier(new Repository(parts[0] + "/" + rhs[0]), rhs[1]); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Image.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Image.java new file mode 100644 index 000000000..732dcfe4f --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Image.java @@ -0,0 +1,91 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.Map; + +/** + * + * @author Konstantin Pelykh (kpelykh@gmail.com) + * + */ +@EqualsAndHashCode +@ToString +public class Image extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Created") + private Long created; + + @JsonProperty("Id") + private String id; + + @JsonProperty("ParentId") + private String parentId; + + @JsonProperty("RepoTags") + private String[] repoTags; + + @JsonProperty("RepoDigests") + private String[] repoDigests; + + @JsonProperty("Size") + private Long size; + + @JsonProperty("VirtualSize") + private Long virtualSize; + + @JsonProperty("SharedSize") + private Long sharedSize; + + @JsonProperty("Labels") + public Map labels; + + @JsonProperty("Containers") + private Integer containers; + + public String getId() { + return id; + } + + public String[] getRepoTags() { + return repoTags; + } + + public String[] getRepoDigests() { + return repoDigests; + } + + public String getParentId() { + return parentId; + } + + public Long getCreated() { + return created; + } + + public Long getSize() { + return size; + } + + public Long getVirtualSize() { + return virtualSize; + } + + + public Long getSharedSize() { + return sharedSize; + } + + public Map getLabels() { + return labels; + } + + public Integer getContainers() { + return containers; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ImageOptions.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ImageOptions.java new file mode 100644 index 000000000..bc8b89acb --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ImageOptions.java @@ -0,0 +1,27 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_48} + */ +@EqualsAndHashCode +@ToString +public class ImageOptions extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + @JsonProperty("Subpath") + private String subpath; + + public String getSubpath() { + return subpath; + } + + public ImageOptions withSubpath(String subpath) { + this.subpath = subpath; + return this; + } +} diff --git a/src/main/java/com/github/dockerjava/api/model/Info.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Info.java similarity index 88% rename from src/main/java/com/github/dockerjava/api/model/Info.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/Info.java index 7270eceb8..67348b86b 100644 --- a/src/main/java/com/github/dockerjava/api/model/Info.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Info.java @@ -1,14 +1,11 @@ package com.github.dockerjava.api.model; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; import com.fasterxml.jackson.annotation.JsonProperty; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; +import lombok.EqualsAndHashCode; +import lombok.ToString; import javax.annotation.CheckForNull; +import java.io.Serializable; import java.util.List; import java.util.Map; @@ -17,9 +14,10 @@ * * @author Konstantin Pelykh (kpelykh@gmail.com) */ -@JsonInclude(Include.NON_NULL) -@JsonIgnoreProperties(ignoreUnknown = true) -public class Info { +@EqualsAndHashCode +@ToString +public class Info extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; /** * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} @@ -99,6 +97,12 @@ public class Info { @JsonProperty("LoggingDriver") private String loggingDriver; + @JsonProperty("CgroupDriver") + private String cGroupDriver; + + @JsonProperty("CgroupVersion") + private String cGroupVersion; + /** * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_20} */ @@ -222,6 +226,24 @@ public class Info { @JsonProperty("ClusterAdvertise") private String clusterAdvertise; + /** + * @since 1.24 + */ + @JsonProperty("Swarm") + private SwarmInfo swarm; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_25} + */ + @JsonProperty("Isolation") + private String isolation; + + @JsonProperty("SecurityOptions") + private List securityOptions; + + @JsonProperty("Runtimes") + private Map runtimes; + /** * @see #architecture */ @@ -470,6 +492,22 @@ public String getLoggingDriver() { return loggingDriver; } + /** + * @see #cGroupDriver + */ + @CheckForNull + public String getCGroupDriver() { + return cGroupDriver; + } + + /** + * @see #cGroupVersion + */ + @CheckForNull + public String getCGroupVersion() { + return cGroupVersion; + } + /** * @see #loggingDriver */ @@ -478,6 +516,22 @@ public Info withLoggingDriver(String loggingDriver) { return this; } + /** + * @see #cGroupDriver + */ + public Info withCGroupDriver(String cGroupDriver) { + this.cGroupDriver = cGroupDriver; + return this; + } + + /** + * @see #cGroupVersion + */ + public Info withCGroupVersion(String cGroupVersion) { + this.cGroupVersion = cGroupVersion; + return this; + } + /** * @see #experimentalBuild */ @@ -1022,18 +1076,65 @@ public Info withSystemTime(String systemTime) { return this; } - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); + /** + * @see #swarm + */ + @CheckForNull + public SwarmInfo getSwarm() { + return swarm; + } + + /** + * @see #swarm + */ + public Info withSwarm(SwarmInfo swarm) { + this.swarm = swarm; + return this; + } + + /** + * @see #isolation + */ + @CheckForNull + public String getIsolation() { + return isolation; } - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); + /** + * @see #isolation + */ + public Info withIsolation(String isolation) { + this.isolation = isolation; + return this; } - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); + /** + * @see #securityOptions + */ + public List getSecurityOptions() { + return securityOptions; + } + + /** + * @see #securityOptions + */ + public Info withSecurityOptions(List securityOptions) { + this.securityOptions = securityOptions; + return this; + } + + /** + * @see #runtimes + */ + public Map getRuntimes() { + return runtimes; + } + + /** + * @see #runtimes + */ + public Info withRuntimes(Map runtimes) { + this.runtimes = runtimes; + return this; } } diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/InfoRegistryConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/InfoRegistryConfig.java new file mode 100644 index 000000000..80bf803d8 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/InfoRegistryConfig.java @@ -0,0 +1,164 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; +import java.util.Map; + +/** + * @since ~{@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_20} + */ +@EqualsAndHashCode +@ToString +public final class InfoRegistryConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("IndexConfigs") + private Map indexConfigs; + + @JsonProperty("InsecureRegistryCIDRs") + private List insecureRegistryCIDRs; + + /** + * //FIXME unknown field + */ + @JsonProperty("Mirrors") + private Object mirrors; + + /** + * @see #indexConfigs + */ + @CheckForNull + public Map getIndexConfigs() { + return indexConfigs; + } + + /** + * @see #indexConfigs + */ + public InfoRegistryConfig withIndexConfigs(Map indexConfigs) { + this.indexConfigs = indexConfigs; + return this; + } + + /** + * @see #insecureRegistryCIDRs + */ + @CheckForNull + public List getInsecureRegistryCIDRs() { + return insecureRegistryCIDRs; + } + + /** + * @see #insecureRegistryCIDRs + */ + public InfoRegistryConfig withInsecureRegistryCIDRs(List insecureRegistryCIDRs) { + this.insecureRegistryCIDRs = insecureRegistryCIDRs; + return this; + } + + /** + * @see #mirrors + */ + @CheckForNull + public Object getMirrors() { + return mirrors; + } + + /** + * @see #mirrors + */ + public InfoRegistryConfig withMirrors(Object mirrors) { + this.mirrors = mirrors; + return this; + } + + /** + * @since ~{@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_20} + */ + @EqualsAndHashCode + @ToString + public static final class IndexConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Mirrors") + private List mirrors; + + @JsonProperty("Name") + private String name; + + @JsonProperty("Official") + private Boolean official; + + @JsonProperty("Secure") + private Boolean secure; + + /** + * @see #mirrors + */ + @CheckForNull + public List getMirrors() { + return mirrors; + } + + /** + * @see #mirrors + */ + public IndexConfig withMirrors(List mirrors) { + this.mirrors = mirrors; + return this; + } + + /** + * @see #name + */ + @CheckForNull + public String getName() { + return name; + } + + /** + * @see #name + */ + public IndexConfig withName(String name) { + this.name = name; + return this; + } + + /** + * @see #official + */ + @CheckForNull + public Boolean getOfficial() { + return official; + } + + /** + * @see #official + */ + public IndexConfig withOfficial(Boolean official) { + this.official = official; + return this; + } + + /** + * @see #secure + */ + @CheckForNull + public Boolean getSecure() { + return secure; + } + + /** + * @see #secure + */ + public IndexConfig withSecure(Boolean secure) { + this.secure = secure; + return this; + } + } +} diff --git a/src/main/java/com/github/dockerjava/api/model/InternetProtocol.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/InternetProtocol.java similarity index 93% rename from src/main/java/com/github/dockerjava/api/model/InternetProtocol.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/InternetProtocol.java index 2bb4db285..ab400fcc8 100644 --- a/src/main/java/com/github/dockerjava/api/model/InternetProtocol.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/InternetProtocol.java @@ -5,6 +5,7 @@ * * @see #TCP * @see #UDP + * @see #SCTP */ public enum InternetProtocol { /** @@ -15,7 +16,12 @@ public enum InternetProtocol { /** * The User Datagram Protocol */ - UDP; + UDP, + + /** + * The Stream Control Transmission Protocol + */ + SCTP; /** * The default {@link InternetProtocol}: {@link #TCP} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Isolation.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Isolation.java new file mode 100644 index 000000000..c59c8848f --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Isolation.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; + +import java.io.Serializable; + +public enum Isolation implements Serializable { + DEFAULT("default"), + + PROCESS("process"), + + HYPERV("hyperv"); + + private String value; + + Isolation(String value) { + this.value = value; + } + + @JsonValue + public String getValue() { + return value; + } + + @JsonCreator + public static Isolation fromValue(String text) { + for (Isolation b : Isolation.values()) { + if (String.valueOf(b.value).equals(text)) { + return b; + } + } + return null; + } + + @Override + public String toString() { + return String.valueOf(value); + } + +} diff --git a/src/main/java/com/github/dockerjava/api/model/Link.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Link.java similarity index 83% rename from src/main/java/com/github/dockerjava/api/model/Link.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/Link.java index 89647306c..4b9b27acf 100644 --- a/src/main/java/com/github/dockerjava/api/model/Link.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Link.java @@ -1,14 +1,19 @@ package com.github.dockerjava.api.model; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; /** * Represents a network link between two Docker containers. The container with the name {@link #getName()} is made available in the target * container with the aliased name {@link #getAlias()}. This involves creating an entry in /etc/hosts and some environment * variables in the target container as well as creating a network bridge between both containers. */ -public class Link { +@EqualsAndHashCode +@ToString +public class Link extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; private final String name; @@ -68,21 +73,6 @@ public static Link parse(final String serialized) throws IllegalArgumentExceptio } } - @Override - public boolean equals(final Object obj) { - if (obj instanceof Link) { - final Link other = (Link) obj; - return new EqualsBuilder().append(name, other.getName()).append(alias, other.getAlias()).isEquals(); - } else { - return super.equals(obj); - } - } - - @Override - public int hashCode() { - return new HashCodeBuilder().append(name).append(alias).toHashCode(); - } - /** * Returns a string representation of this {@link Link} suitable for inclusion in a JSON message. The format is name:alias, * like the argument in {@link #parse(String)}. diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Links.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Links.java new file mode 100644 index 000000000..18bfc75ba --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Links.java @@ -0,0 +1,38 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; +import java.util.List; +import java.util.stream.Stream; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; + +public class Links implements Serializable { + private static final long serialVersionUID = 1L; + + private final Link[] links; + + public Links(final Link... links) { + this.links = links; + } + + public Links(final List links) { + this.links = links.toArray(new Link[links.size()]); + } + + public Link[] getLinks() { + return links; + } + + @JsonCreator + public static Links fromPrimitive(String[] links) { + return new Links( + Stream.of(links).map(Link::parse).toArray(Link[]::new) + ); + } + + @JsonValue + public String[] toPrimitive() { + return Stream.of(links).map(Link::toString).toArray(String[]::new); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/LoadResponseItem.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/LoadResponseItem.java new file mode 100644 index 000000000..bf90c69bf --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/LoadResponseItem.java @@ -0,0 +1,33 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonIgnore; + +public class LoadResponseItem extends ResponseItem { + + private static final long serialVersionUID = 1L; + + private static final String IMPORT_SUCCESS = "Loaded image:"; + + /** + * Returns whether the stream field indicates a successful build operation + */ + @JsonIgnore + public boolean isBuildSuccessIndicated() { + if (isErrorIndicated() || getStream() == null) { + return false; + } + + return getStream().contains(IMPORT_SUCCESS); + } + + @JsonIgnore + public String getMessage() { + if (!isBuildSuccessIndicated()) { + return null; + } else if (getStream().contains(IMPORT_SUCCESS)) { + return getStream(); + } + + return null; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/LocalNodeState.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/LocalNodeState.java new file mode 100644 index 000000000..07e2e4158 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/LocalNodeState.java @@ -0,0 +1,53 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; + +import javax.annotation.Nonnull; +import java.util.HashMap; +import java.util.Map; + +/** + * https://github.com/moby/moby/blob/master/api/types/swarm/swarm.go#L174-L188 + * + * @since 1.24 + */ +public enum LocalNodeState { + + INACTIVE("inactive"), + PENDING("pending"), + ACTIVE("active"), + ERROR("error"), + LOCKED("locked"), + + /** + * Can not construct instance of com.github.dockerjava.api.model.LocalNodeState + * from String value '': value not one of declared Enum instance names: [error, locked, inactive, active, pending] + */ + EMPTY(""); + + private static final Map TYPES = new HashMap<>(); + + static { + for (LocalNodeState t : values()) { + TYPES.put(t.name().toLowerCase(), t); + } + } + + private String value; + + LocalNodeState(@Nonnull String value) { + this.value = value; + } + + @JsonValue + public String getValue() { + return value; + } + + @JsonCreator + public static LocalNodeState forValue(String s) { + return TYPES.get(s); + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/LogConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/LogConfig.java new file mode 100644 index 000000000..218cdd827 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/LogConfig.java @@ -0,0 +1,109 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonIgnore; +import com.fasterxml.jackson.annotation.JsonProperty; +import com.fasterxml.jackson.annotation.JsonValue; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Map; + +/** + * Log driver to use for a created/running container. The available types are: + *

+ * json-file (default) syslog journald none + *

+ * If a driver is specified that is NOT supported,docker will default to null. If configs are supplied that are not supported by the type + * docker will ignore them. In most cases setting the config option to null will suffice. Consult the docker remote API for a more detailed + * and up-to-date explanation of the available types and their options. + */ +@EqualsAndHashCode +@ToString +public class LogConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Type") + public LoggingType type = null; + + @JsonProperty("Config") + public Map config; + + public LogConfig(LoggingType type, Map config) { + this.type = type; + this.config = config; + } + + public LogConfig(LoggingType type) { + this(type, null); + } + + public LogConfig() { + } + + public LoggingType getType() { + return type; + } + + public LogConfig setType(LoggingType type) { + this.type = type; + return this; + } + + @JsonIgnore + public Map getConfig() { + return config; + } + + @JsonIgnore + public LogConfig setConfig(Map config) { + this.config = config; + return this; + } + + public enum LoggingType { + NONE("none"), + DEFAULT("json-file"), + LOCAL("local"), + ETWLOGS("etwlogs"), + JSON_FILE("json-file"), + SYSLOG("syslog"), + JOURNALD("journald"), + GELF("gelf"), + FLUENTD("fluentd"), + AWSLOGS("awslogs"), + DB("db"), // Synology specific driver + SPLUNK("splunk"), + GCPLOGS("gcplogs"), + LOKI("loki"); + + private String type; + + LoggingType(String type) { + this.type = type; + } + + @JsonValue + public String getType() { + return type; + } + + @JsonCreator + @CheckForNull + public static LoggingType fromValue(String text) { + for (LoggingType b : LoggingType.values()) { + if (String.valueOf(b.type).equals(text)) { + return b; + } + } + return null; + } + + @Override + public String toString() { + return String.valueOf(type); + } + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/LxcConf.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/LxcConf.java new file mode 100644 index 000000000..36fc1a9cb --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/LxcConf.java @@ -0,0 +1,46 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +@EqualsAndHashCode +@ToString +public class LxcConf extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Key") + public String key; + + @JsonProperty("Value") + public String value; + + public LxcConf(String key, String value) { + this.key = key; + this.value = value; + } + + public LxcConf() { + } + + public String getKey() { + return key; + } + + public LxcConf setKey(String key) { + this.key = key; + return this; + } + + public String getValue() { + return value; + } + + public LxcConf setValue(String value) { + this.value = value; + return this; + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/MemoryStatsConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/MemoryStatsConfig.java new file mode 100644 index 000000000..b57f05135 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/MemoryStatsConfig.java @@ -0,0 +1,73 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * Used in {@link Statistics} + * + * @author Yuting Liu + */ +@EqualsAndHashCode +@ToString +public class MemoryStatsConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("stats") + private StatsConfig stats; + + @JsonProperty("usage") + private Long usage; + + @JsonProperty("max_usage") + private Long maxUsage; + + @JsonProperty("failcnt") + private Long failcnt; + + @JsonProperty("limit") + private Long limit; + + /** + * @see #stats + */ + @CheckForNull + public StatsConfig getStats() { + return stats; + } + + /** + * @see #usage + */ + @CheckForNull + public Long getUsage() { + return usage; + } + + /** + * @see #maxUsage + */ + @CheckForNull + public Long getMaxUsage() { + return maxUsage; + } + + /** + * @see #failcnt + */ + public Long getFailcnt() { + return failcnt; + } + + /** + * @see #limit + */ + @CheckForNull + public Long getLimit() { + return limit; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Mount.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Mount.java new file mode 100644 index 000000000..3f17343c3 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Mount.java @@ -0,0 +1,205 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class Mount extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Type") + private MountType type; + + /** + * @since 1.24 + */ + @JsonProperty("Source") + private String source; + + /** + * @since 1.24 + */ + @JsonProperty("Target") + private String target; + + /** + * @since 1.24 + */ + @JsonProperty("ReadOnly") + private Boolean readOnly; + + /** + * @since 1.24 + */ + @JsonProperty("BindOptions") + private BindOptions bindOptions; + + /** + * @since 1.24 + */ + @JsonProperty("VolumeOptions") + private VolumeOptions volumeOptions; + + /** + * @since 1.29 + */ + @JsonProperty("TmpfsOptions") + private TmpfsOptions tmpfsOptions; + + /** + * @since 1.48 + */ + @JsonProperty("ImageOptions") + private ImageOptions imageOptions; + + /** + * @see #type + */ + @CheckForNull + public MountType getType() { + return type; + } + + /** + * @see #type + */ + public Mount withType(MountType type) { + this.type = type; + return this; + } + + /** + * @see #source + */ + @CheckForNull + public String getSource() { + return source; + } + + /** + * @see #source + */ + public Mount withSource(String source) { + this.source = source; + return this; + } + + /** + * @see #target + */ + @CheckForNull + public String getTarget() { + return target; + } + + /** + * @see #target + */ + public Mount withTarget(String target) { + this.target = target; + return this; + } + + /** + * @see #readOnly + */ + @CheckForNull + public Boolean getReadOnly() { + return readOnly; + } + + /** + * @see #readOnly + */ + public Mount withReadOnly(Boolean readOnly) { + this.readOnly = readOnly; + return this; + } + + /** + * @see #bindOptions + */ + @CheckForNull + public BindOptions getBindOptions() { + return bindOptions; + } + + /** + * @see #bindOptions + */ + public Mount withBindOptions(BindOptions bindOptions) { + this.bindOptions = bindOptions; + if (bindOptions != null) { + this.type = MountType.BIND; + } + return this; + } + + /** + * @see #volumeOptions + */ + @CheckForNull + public VolumeOptions getVolumeOptions() { + return volumeOptions; + } + + /** + * @see #volumeOptions + */ + public Mount withVolumeOptions(VolumeOptions volumeOptions) { + this.volumeOptions = volumeOptions; + if (volumeOptions != null) { + this.type = MountType.VOLUME; + } + return this; + } + + /** + * @see #tmpfsOptions + */ + @CheckForNull + public TmpfsOptions getTmpfsOptions() { + return tmpfsOptions; + } + + /** + * @see #tmpfsOptions + */ + public Mount withTmpfsOptions(TmpfsOptions tmpfsOptions) { + this.tmpfsOptions = tmpfsOptions; + if (tmpfsOptions != null) { + this.type = MountType.TMPFS; + } + return this; + } + + /** + * @see #imageOptions + */ + @CheckForNull + public ImageOptions getImageOptions() { + return imageOptions; + } + + /** + * @see #imageOptions + */ + public Mount withImageOptions(ImageOptions imageOptions) { + this.imageOptions = imageOptions; + if (imageOptions != null) { + this.type = MountType.IMAGE; + } + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/MountType.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/MountType.java new file mode 100644 index 000000000..b522c9612 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/MountType.java @@ -0,0 +1,27 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum MountType { + @JsonProperty("bind") + BIND, + + @JsonProperty("volume") + VOLUME, + + //@since 1.29 + @JsonProperty("tmpfs") + TMPFS, + + //@since 1.40 + @JsonProperty("npipe") + NPIPE, + + //@since 1.48 + @JsonProperty("image") + IMAGE, + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/NamedResourceSpec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/NamedResourceSpec.java new file mode 100644 index 000000000..198c75543 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/NamedResourceSpec.java @@ -0,0 +1,11 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@Deprecated +public class NamedResourceSpec extends GenericResource implements Serializable { + private static final long serialVersionUID = 1L; +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Network.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Network.java new file mode 100644 index 000000000..7e9d3b2fd --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Network.java @@ -0,0 +1,241 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Date; +import java.util.List; +import java.util.Map; + +@EqualsAndHashCode +@ToString +public class Network extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Id") + private String id; + + @JsonProperty("Created") + private Date created; + + @JsonProperty("Name") + private String name; + + @JsonProperty("Scope") + private String scope; + + @JsonProperty("Driver") + private String driver; + + @JsonProperty("EnableIPv6") + private Boolean enableIPv6; + + @JsonProperty("Internal") + private Boolean internal; + + @JsonProperty("IPAM") + private Ipam ipam; + + @JsonProperty("Containers") + private Map containers; + + @JsonProperty("Options") + private Map options; + + @JsonProperty("Attachable") + private Boolean attachable; + + @JsonProperty("Labels") + public Map labels; + + public String getId() { + return id; + } + + public Date getCreated() { + return created; + } + + public String getName() { + return name; + } + + public String getScope() { + return scope; + } + + public String getDriver() { + return driver; + } + + public Boolean getEnableIPv6() { + return enableIPv6; + } + + public Boolean getInternal() { + return internal; + } + + public Ipam getIpam() { + return ipam; + } + + public Map getContainers() { + return containers; + } + + public Map getOptions() { + return options; + } + + public Boolean isAttachable() { + return attachable; + } + + public Map getLabels() { + return labels; + } + + @EqualsAndHashCode + @ToString + public static class ContainerNetworkConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * @since {@link RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("Name") + private String name; + + @JsonProperty("EndpointID") + private String endpointId; + + @JsonProperty("MacAddress") + private String macAddress; + + @JsonProperty("IPv4Address") + private String ipv4Address; + + @JsonProperty("IPv6Address") + private String ipv6Address; + + public String getName() { + return name; + } + + public String getEndpointId() { + return endpointId; + } + + public String getMacAddress() { + return macAddress; + } + + public String getIpv4Address() { + return ipv4Address; + } + + public String getIpv6Address() { + return ipv6Address; + } + } + + @EqualsAndHashCode + @ToString + public static class Ipam extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Driver") + private String driver; + + @JsonProperty("Config") + private List config = new ArrayList<>(); + + @JsonProperty("Options") + private Map options = null; + + public String getDriver() { + return driver; + } + + public Map getOptions() { + return options; + } + + public List getConfig() { + return config; + } + + public Ipam withConfig(List ipamConfigs) { + config = ipamConfigs; + return this; + } + + public Ipam withConfig(Config... ipamConfigs) { + config = Arrays.asList(ipamConfigs); + return this; + } + + public Ipam withDriver(String driver) { + this.driver = driver; + return this; + } + + @EqualsAndHashCode + @ToString + public static class Config extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Subnet") + private String subnet; + + @JsonProperty("IPRange") + private String ipRange; + + @JsonProperty("Gateway") + private String gateway; + + @JsonProperty("NetworkID") + private String networkID; + + public String getSubnet() { + return subnet; + } + + public String getIpRange() { + return ipRange; + } + + public String getGateway() { + return gateway; + } + + public Config withSubnet(String subnet) { + this.subnet = subnet; + return this; + } + + public Config withIpRange(String ipRange) { + this.ipRange = ipRange; + return this; + } + + public Config withGateway(String gateway) { + this.gateway = gateway; + return this; + } + + public String getNetworkID() { + return networkID; + } + + public void setNetworkID(String networkID) { + this.networkID = networkID; + } + } + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/NetworkAttachmentConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/NetworkAttachmentConfig.java new file mode 100644 index 000000000..db0eb1ded --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/NetworkAttachmentConfig.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class NetworkAttachmentConfig extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Target") + private String target; + + /** + * @since 1.24 + */ + @JsonProperty("Aliases") + private List aliases; + + /** + * @see #target + */ + @CheckForNull + public String getTarget() { + return target; + } + + /** + * @see #target + */ + public NetworkAttachmentConfig withTarget(String target) { + this.target = target; + return this; + } + + /** + * @see #aliases + */ + @CheckForNull + public List getAliases() { + return aliases; + } + + /** + * @see #aliases + */ + public NetworkAttachmentConfig withAliases(List aliases) { + this.aliases = aliases; + return this; + } +} diff --git a/src/main/java/com/github/dockerjava/api/model/NetworkSettings.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/NetworkSettings.java similarity index 92% rename from src/main/java/com/github/dockerjava/api/model/NetworkSettings.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/NetworkSettings.java index 1d523bf4b..e28d8f52c 100644 --- a/src/main/java/com/github/dockerjava/api/model/NetworkSettings.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/NetworkSettings.java @@ -3,11 +3,11 @@ */ package com.github.dockerjava.api.model; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.core.RemoteApiVersion; -import org.apache.commons.lang.builder.ToStringBuilder; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import java.io.Serializable; import java.util.Map; /** @@ -15,8 +15,10 @@ * @author Marcus Linke * */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class NetworkSettings { +@EqualsAndHashCode +@ToString +public class NetworkSettings extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; @JsonProperty("Bridge") private String bridge; @@ -198,9 +200,4 @@ public String getGlobalIPv6Address() { public Integer getGlobalIPv6PrefixLen() { return globalIPv6PrefixLen; } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } } diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Node.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Node.java new file mode 100644 index 000000000..2bb832e48 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Node.java @@ -0,0 +1,60 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * A node as returned by the /events API, for instance, when Swarm is used. + */ +@EqualsAndHashCode +@ToString +public class Node extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Name") + private String name; + + @JsonProperty("Id") + private String id; + + @JsonProperty("Addr") + private String addr; + + @JsonProperty("Ip") + private String ip; + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } + + public String getId() { + return id; + } + + public void setId(String id) { + this.id = id; + } + + public String getAddr() { + return addr; + } + + public void setAddr(String addr) { + this.addr = addr; + } + + public String getIp() { + return ip; + } + + public void setIp(String ip) { + this.ip = ip; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ObjectVersion.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ObjectVersion.java new file mode 100644 index 000000000..5fa361977 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ObjectVersion.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; + +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + * The version number of the object such as node, service, etc. This is needed to avoid conflicting writes. + * The client must send the version number along with the modified specification when updating these objects. + * This approach ensures safe concurrency and determinism in that the change on the object may not be applied + * if the version number has changed from the last read. In other words, if two update requests specify the + * same base version, only one of the requests can succeed. As a result, two separate update requests that + * happen at the same time will not unintentionally overwrite each other. + */ +@EqualsAndHashCode +public class ObjectVersion extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Index") + private Long index = null; + + public Long getIndex() { + return index; + } + + public ObjectVersion withIndex(Long index) { + this.index = index; + return this; + } + + @Override + public String toString() { + return String.valueOf(index); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/PeerNode.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PeerNode.java new file mode 100644 index 000000000..8937b9593 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PeerNode.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since 1.24 + */ +@EqualsAndHashCode +@ToString +public class PeerNode extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("NodeID") + private String nodeID; + + /** + * @since 1.24 + */ + @JsonProperty("Addr") + private String addr; + + /** + * @see #nodeID + */ + @CheckForNull + public String getNodeID() { + return nodeID; + } + + /** + * @see #nodeID + */ + public PeerNode withNodeID(String nodeID) { + this.nodeID = nodeID; + return this; + } + + /** + * @see #addr + */ + @CheckForNull + public String getAddr() { + return addr; + } + + /** + * @see #addr + */ + public PeerNode withAddr(String addr) { + this.addr = addr; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/PidsStatsConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PidsStatsConfig.java new file mode 100644 index 000000000..df953e140 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PidsStatsConfig.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * Used in {@link Statistics} + * + * @author Yuting Liu + */ +@EqualsAndHashCode +@ToString +public class PidsStatsConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("current") + private Long current; + + /** + * @see #current + */ + @CheckForNull + public Long getCurrent() { + return current; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/PortBinding.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PortBinding.java new file mode 100644 index 000000000..2b7901e92 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PortBinding.java @@ -0,0 +1,64 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.api.model.Ports.Binding; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * In a {@link PortBinding}, a network socket on the Docker host, expressed as a {@link Binding}, is bound to an {@link ExposedPort} of a + * container. A {@link PortBinding} corresponds to the --publish (-p) option of the docker run (and + * similar) CLI command for adding port bindings to a container. + *

+ * Note: This is an abstraction used for creating new port bindings. It is not to be confused with the abstraction used for querying + * existing port bindings from a container configuration in {@link NetworkSettings#getPorts()} and {@link HostConfig#getPortBindings()}. In + * that context, a Map<ExposedPort, Binding[]> is used. + */ +@EqualsAndHashCode +@ToString +public class PortBinding extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + private final Binding binding; + + private final ExposedPort exposedPort; + + public PortBinding(Binding binding, ExposedPort exposedPort) { + this.binding = binding; + this.exposedPort = exposedPort; + } + + public Binding getBinding() { + return binding; + } + + public ExposedPort getExposedPort() { + return exposedPort; + } + + public static PortBinding parse(String serialized) throws IllegalArgumentException { + try { + String[] parts = serialized.split(":"); + switch (parts.length) { + case 3: + // 127.0.0.1:80:8080/tcp + return createFromSubstrings(parts[0] + ":" + parts[1], parts[2]); + case 2: + // 80:8080 // 127.0.0.1::8080 + return createFromSubstrings(parts[0], parts[1]); + case 1: + // 8080 + return createFromSubstrings("", parts[0]); + default: + throw new IllegalArgumentException(); + } + } catch (Exception e) { + throw new IllegalArgumentException("Error parsing PortBinding '" + serialized + "'", e); + } + } + + private static PortBinding createFromSubstrings(String binding, String exposedPort) throws IllegalArgumentException { + return new PortBinding(Binding.parse(binding), ExposedPort.parse(exposedPort)); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/PortConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PortConfig.java new file mode 100644 index 000000000..cec07d9cd --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PortConfig.java @@ -0,0 +1,131 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class PortConfig extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Name") + private String name; + + /** + * @since 1.24 + */ + @JsonProperty("Protocol") + private PortConfigProtocol protocol; + + /** + * @since 1.24 + */ + @JsonProperty("TargetPort") + private int targetPort; + + /** + * @since 1.24 + */ + @JsonProperty("PublishedPort") + private int publishedPort; + + /** + * @since 1.25 + * docker 1.13 + * https://github.com/mrjana/docker/blob/14ac9f60d0174256e0713701ebffaf5ca827da71/api/types/swarm/network.go + */ + @JsonProperty("PublishMode") + private PublishMode publishMode; + + /** + * @see #name + */ + public String getName() { + return name; + } + + /** + * @see #name + */ + public PortConfig withName(String name) { + this.name = name; + return this; + } + + /** + * @see #protocol + */ + @CheckForNull + public PortConfigProtocol getProtocol() { + return protocol; + } + + /** + * @see #protocol + */ + public PortConfig withProtocol(PortConfigProtocol protocol) { + this.protocol = protocol; + return this; + } + + /** + * @see #targetPort + */ + @CheckForNull + public int getTargetPort() { + return targetPort; + } + + /** + * @see #targetPort + */ + public PortConfig withTargetPort(int targetPort) { + this.targetPort = targetPort; + return this; + } + + /** + * @see #publishedPort + */ + @CheckForNull + public int getPublishedPort() { + return publishedPort; + } + + /** + * @see #publishedPort + */ + public PortConfig withPublishedPort(int publishedPort) { + this.publishedPort = publishedPort; + return this; + } + + @CheckForNull + public PublishMode getPublishMode() { + return publishMode; + } + + public PortConfig withPublishMode(PublishMode publishMode) { + this.publishMode = publishMode; + return this; + } + + public enum PublishMode { + //ingress load balancing using routing mesh. + @JsonProperty("ingress") + ingress, + //direct host level access on the host where the task is running. + @JsonProperty("host") + host + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/PortConfigProtocol.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PortConfigProtocol.java new file mode 100644 index 000000000..8af2fe02c --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PortConfigProtocol.java @@ -0,0 +1,19 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum PortConfigProtocol { + + @JsonProperty("tcp") + TCP, + + @JsonProperty("udp") + UDP, + + @JsonProperty("sctp") + SCTP + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Ports.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Ports.java new file mode 100644 index 000000000..0411ca218 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Ports.java @@ -0,0 +1,287 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; +import lombok.EqualsAndHashCode; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Map.Entry; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +/** + * A container for port bindings, made available as a {@link Map} via its {@link #getBindings()} method. + *

+ * Note: This is an abstraction used for querying existing port bindings from a container configuration. It is not to be confused with + * the {@link PortBinding} abstraction used for adding new port bindings to a container. + * + * @see HostConfig#getPortBindings() + * @see NetworkSettings#getPorts() + */ +public class Ports implements Serializable { + private static final long serialVersionUID = 1L; + + private final Map ports = new HashMap<>(); + + /** + * Creates a {@link Ports} object with no {@link PortBinding}s. Use {@link #bind(ExposedPort, Binding)} or {@link #add(PortBinding...)} + * to add {@link PortBinding}s. + */ + public Ports() { + } + + /** + * Creates a {@link Ports} object with an initial {@link PortBinding} for the specified {@link ExposedPort} and {@link Binding}. Use + * {@link #bind(ExposedPort, Binding)} or {@link #add(PortBinding...)} to add more {@link PortBinding}s. + */ + public Ports(ExposedPort exposedPort, Binding host) { + bind(exposedPort, host); + } + + public Ports(PortBinding... portBindings) { + add(portBindings); + } + + /** + * Adds a new {@link PortBinding} for the specified {@link ExposedPort} and {@link Binding} to the current bindings. + */ + public void bind(ExposedPort exposedPort, Binding binding) { + if (ports.containsKey(exposedPort)) { + Binding[] bindings = ports.get(exposedPort); + Binding[] newBindings = new Binding[bindings.length + 1]; + System.arraycopy(bindings, 0, newBindings, 0, bindings.length); + newBindings[newBindings.length - 1] = binding; + ports.put(exposedPort, newBindings); + } else { + if (binding == null) { + ports.put(exposedPort, null); + } else { + ports.put(exposedPort, new Binding[] {binding}); + } + } + } + + /** + * Adds the specified {@link PortBinding}(s) to the list of {@link PortBinding}s. + */ + public void add(PortBinding... portBindings) { + for (PortBinding binding : portBindings) { + bind(binding.getExposedPort(), binding.getBinding()); + } + } + + @Override + public String toString() { + return ports.toString(); + } + + /** + * Returns the port bindings in the format used by the Docker remote API, i.e. the {@link Binding}s grouped by {@link ExposedPort}. + * + * @return the port bindings as a {@link Map} that contains one or more {@link Binding}s per {@link ExposedPort}. + */ + public Map getBindings() { + return ports; + } + + // public PortBinding[] getBindingsAsArray() { + // List bindings = new ArrayList<>(); + // for(Map.Entry entry: ports.entrySet()) { + // for(Ports.Binding binding : entry.getValue()) { + // bindings.add(new PortBinding(binding, entry.getKey())); + // } + // } + // return bindings.toArray(new PortBinding[bindings.size()]); + // } + + /** + * A {@link Binding} represents a socket on the Docker host that is used in a {@link PortBinding}. It is characterized by an + * {@link #getHostIp() IP address} and a {@link #getHostPortSpec() port spec}. Both properties may be null in order to + * let Docker assign them dynamically/using defaults. + * + * @see Ports#bind(ExposedPort, Binding) + * @see ExposedPort + */ + @EqualsAndHashCode + public static class Binding extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * Creates a {@link Binding} for the given {@link #getHostPortSpec() port spec}, leaving the {@link #getHostIp() IP address} + * undefined. + * + * @see Ports#bind(ExposedPort, Binding) + * @see ExposedPort + */ + public static Binding bindPortSpec(String portSpec) { + return new Binding(null, portSpec); + } + + /** + * Creates a {@link Binding} for the given {@link #getHostIp() IP address}, leaving the {@link #getHostPortSpec() port spec} + * undefined. + */ + public static Binding bindIp(String hostIp) { + return new Binding(hostIp, null); + } + + /** + * Creates a {@link Binding} for the given {@link #getHostIp() IP address} and port number. + */ + public static Binding bindIpAndPort(String hostIp, int port) { + return new Binding(hostIp, "" + port); + } + + /** + * Creates a {@link Binding} for the given {@link #getHostIp() IP address} and port range. + */ + public static Binding bindIpAndPortRange(String hostIp, int lowPort, int highPort) { + return new Binding(hostIp, lowPort + "-" + highPort); + } + + /** + * Creates a {@link Binding} for the given port range, leaving the {@link #getHostIp() IP address} + * undefined. + */ + public static Binding bindPortRange(int lowPort, int highPort) { + return bindIpAndPortRange(null, lowPort, highPort); + } + + /** + * Creates a {@link Binding} for the given port leaving the {@link #getHostIp() IP address} + * undefined. + */ + public static Binding bindPort(int port) { + return bindIpAndPort(null, port); + } + + /** + * Creates an empty {@link Binding}. + */ + public static Binding empty() { + return new Binding(null, null); + } + + private final String hostIp; + + private final String hostPortSpec; + + /** + * Creates a {@link Binding} for the given {@link #getHostIp() host IP address} and {@link #getHostPortSpec() host port spec}. + * + * @see Ports#bind(ExposedPort, Binding) + * @see ExposedPort + */ + public Binding(String hostIp, String hostPortSpec) { + this.hostIp = hostIp == null || hostIp.length() == 0 ? null : hostIp; + this.hostPortSpec = hostPortSpec; + } + + /** + * @return the IP address on the Docker host. May be null, in which case Docker will bind the port to all interfaces ( + * 0.0.0.0). + */ + public String getHostIp() { + return hostIp; + } + + /** + * @return the port spec for the binding on the Docker host. May reference a single port ("1234"), a port range ("1234-2345") or + * null, in which case Docker will dynamically assign a port. + */ + public String getHostPortSpec() { + return hostPortSpec; + } + + /** + * Parses a textual host and port specification (as used by the Docker CLI) to a {@link Binding}. + *

+ * Legal syntax: IP|IP:portSpec|portSpec where portSpec is either a single port or a port range + * + * @param serialized + * serialized the specification, e.g. 127.0.0.1:80 + * @return a {@link Binding} matching the specification + * @throws IllegalArgumentException + * if the specification cannot be parsed + */ + public static Binding parse(String serialized) throws IllegalArgumentException { + try { + if (serialized.isEmpty()) { + return Binding.empty(); + } + + String[] parts = serialized.split(":"); + switch (parts.length) { + case 2: { + return new Binding(parts[0], parts[1]); + } + case 1: { + return parts[0].contains(".") ? Binding.bindIp(parts[0]) : Binding.bindPortSpec(parts[0]); + } + default: { + throw new IllegalArgumentException(); + } + } + } catch (Exception e) { + throw new IllegalArgumentException("Error parsing Binding '" + serialized + "'"); + } + } + + /** + * Returns a string representation of this {@link Binding} suitable for inclusion in a JSON message. The format is + * [IP:]Port, like the argument in {@link #parse(String)}. + * + * @return a string representation of this {@link Binding} + */ + @Override + public String toString() { + if (hostIp == null || hostIp.length() == 0) { + return hostPortSpec; + } else if (hostPortSpec == null) { + return hostIp; + } else { + return hostIp + ":" + hostPortSpec; + } + } + } + + @JsonCreator + public static Ports fromPrimitive(Map>> map) { + Ports out = new Ports(); + for (Entry>> entry : map.entrySet()) { + ExposedPort exposedPort = ExposedPort.parse(entry.getKey()); + + if (entry.getValue() == null) { + out.bind(exposedPort, null); + } else { + for (Map binding : entry.getValue()) { + out.bind(exposedPort, new Binding(binding.get("HostIp"), binding.get("HostPort"))); + } + } + } + return out; + } + + @JsonValue + public Map>> toPrimitive() { + // Use reduce-like collect to be able to put nulls into the values + return ports.entrySet().stream().collect( + HashMap::new, + (map, entry) -> { + List> value = entry.getValue() == null ? null : Stream.of(entry.getValue()) + .map(binding -> { + Map result = new HashMap<>(); + result.put("HostIp", binding.getHostIp() == null ? "" : binding.getHostIp()); + result.put("HostPort", binding.getHostPortSpec() == null ? "" : binding.getHostPortSpec()); + return result; + }) + .collect(Collectors.toList()); + map.put(entry.getKey().toString(), value); + }, + HashMap::putAll + ); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/PropagationMode.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PropagationMode.java new file mode 100644 index 000000000..3e1db4438 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PropagationMode.java @@ -0,0 +1,65 @@ +package com.github.dockerjava.api.model; + +/** + * The propagation mode of a file system or file: shared, slave or private. + * + * @see https://github.com/docker/docker/pull/17034 + * @since 1.22 + */ +public enum PropagationMode { + /** default */ + DEFAULT(""), + + /** shared */ + SHARED("shared"), + + /** rshared */ + RSHARED("rshared"), + + /** slave */ + SLAVE("slave"), + + /** rslave */ + RSLAVE("rslave"), + + /** private */ + PRIVATE("private"), + + /** rprivate */ + RPRIVATE("rprivate"); + + /** + * The default {@link PropagationMode}: {@link #DEFAULT} + */ + public static final PropagationMode DEFAULT_MODE = DEFAULT; + + private String value; + + PropagationMode(String v) { + value = v; + } + + @Override + public String toString() { + return value; + } + + public static PropagationMode fromString(String v) { + switch (v) { + case "shared": + return SHARED; + case "rshared": + return RSHARED; + case "slave": + return SLAVE; + case "rslave": + return RSLAVE; + case "private": + return PRIVATE; + case "rprivate": + return RPRIVATE; + default: + return DEFAULT; + } + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/PruneResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PruneResponse.java new file mode 100644 index 000000000..2ccdf72c4 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PruneResponse.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * Delete unused content (containers, images, volumes, networks, build relicts) + */ +@EqualsAndHashCode +@ToString +public class PruneResponse extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("SpaceReclaimed") + private Long spaceReclaimed; + + /** + * Default constructor for the deserialization. + */ + public PruneResponse() { + } + + /** + * Constructor. + * + * @param spaceReclaimed Space reclaimed after purification + */ + public PruneResponse(Long spaceReclaimed) { + this.spaceReclaimed = spaceReclaimed; + } + + /** + * Disk space reclaimed in bytes + */ + public Long getSpaceReclaimed() { + return spaceReclaimed; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/PruneType.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PruneType.java new file mode 100644 index 000000000..10d8704fd --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PruneType.java @@ -0,0 +1,9 @@ +package com.github.dockerjava.api.model; + +public enum PruneType { + BUILD, + CONTAINERS, + IMAGES, + NETWORKS, + VOLUMES +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/PullResponseItem.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PullResponseItem.java new file mode 100644 index 000000000..1d3f33c8e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PullResponseItem.java @@ -0,0 +1,43 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonIgnore; + +/** + * Represents a pull response stream item + */ +public class PullResponseItem extends ResponseItem { + + private static final long serialVersionUID = -2575482839766823293L; + + private static final String LEGACY_REGISTRY = "this image was pulled from a legacy registry"; + + private static final String DOWNLOADED_NEWER_IMAGE = "Downloaded newer image"; + + private static final String IMAGE_UP_TO_DATE = "Image is up to date"; + + private static final String DOWNLOAD_COMPLETE = "Download complete"; + + private static final String DOWNLOADED_SWARM = ": downloaded"; + + private static final String ALREADY_EXISTS = "Already exists"; + + /** + * Returns whether the status indicates a successful pull operation + * + * @returns true: status indicates that pull was successful, false: status doesn't indicate a successful pull + */ + @JsonIgnore + public boolean isPullSuccessIndicated() { + if (isErrorIndicated() || getStatus() == null) { + return false; + } + + return (getStatus().contains(DOWNLOAD_COMPLETE) || + getStatus().contains(IMAGE_UP_TO_DATE) || + getStatus().contains(DOWNLOADED_NEWER_IMAGE) || + getStatus().contains(LEGACY_REGISTRY) || + getStatus().contains(DOWNLOADED_SWARM) || + getStatus().contains(ALREADY_EXISTS) + ); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/PushResponseItem.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PushResponseItem.java new file mode 100644 index 000000000..0b28a46fb --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/PushResponseItem.java @@ -0,0 +1,10 @@ +package com.github.dockerjava.api.model; + +/** + * Represents a push response stream item + */ +public class PushResponseItem extends ResponseItem { + + private static final long serialVersionUID = 8256977108011295857L; + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Reachability.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Reachability.java new file mode 100644 index 000000000..a536080c3 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Reachability.java @@ -0,0 +1,18 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum Reachability { + + @JsonProperty("unknown") + UNKNOWN, + + @JsonProperty("unreachable") + UNREACHABLE, + + @JsonProperty("reachable") + REACHABLE +} diff --git a/src/main/java/com/github/dockerjava/api/model/Repository.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Repository.java similarity index 77% rename from src/main/java/com/github/dockerjava/api/model/Repository.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/Repository.java index 1e814a1b5..5dd636981 100644 --- a/src/main/java/com/github/dockerjava/api/model/Repository.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Repository.java @@ -1,14 +1,20 @@ package com.github.dockerjava.api.model; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; import java.net.MalformedURLException; import java.net.URL; -import com.google.common.base.Objects; - /** * A repository or image name. */ -public class Repository { +@EqualsAndHashCode +@ToString +public class Repository extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + public final String name; /** @@ -31,11 +37,6 @@ public URL getURL() throws MalformedURLException { return new URL("http://" + name); } - @Override - public String toString() { - return Objects.toStringHelper(this).add("name", name).toString(); - } - public String getPath() { if (!name.contains("/")) { return name; diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResourceRequirements.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResourceRequirements.java new file mode 100644 index 000000000..54e3001b8 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResourceRequirements.java @@ -0,0 +1,61 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ResourceRequirements extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Limits") + private ResourceSpecs limits; + + /** + * @since 1.24 + */ + @JsonProperty("Reservations") + private ResourceSpecs reservations; + + /** + * @see #limits + */ + @CheckForNull + public ResourceSpecs getLimits() { + return limits; + } + + /** + * @see #limits + */ + public ResourceRequirements withLimits(ResourceSpecs limits) { + this.limits = limits; + return this; + } + + /** + * @see #reservations + */ + @CheckForNull + public ResourceSpecs getReservations() { + return reservations; + } + + /** + * @see #reservations + */ + public ResourceRequirements withReservations(ResourceSpecs reservations) { + this.reservations = reservations; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResourceSpecs.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResourceSpecs.java new file mode 100644 index 000000000..00f2de7e1 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResourceSpecs.java @@ -0,0 +1,61 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ResourceSpecs extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("MemoryBytes") + private Long memoryBytes; + + /** + * @since 1.24 + */ + @JsonProperty("NanoCPUs") + private Long nanoCPUs; + + /** + * @see #memoryBytes + */ + @CheckForNull + public Long getMemoryBytes() { + return memoryBytes; + } + + /** + * @see #memoryBytes + */ + public ResourceSpecs withMemoryBytes(long memoryBytes) { + this.memoryBytes = memoryBytes; + return this; + } + + /** + * @see #nanoCPUs + */ + @CheckForNull + public Long getNanoCPUs() { + return nanoCPUs; + } + + /** + * @see #nanoCPUs + */ + public ResourceSpecs withNanoCPUs(long nanoCPUs) { + this.nanoCPUs = nanoCPUs; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResourceVersion.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResourceVersion.java new file mode 100644 index 000000000..babee6a50 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResourceVersion.java @@ -0,0 +1,40 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ResourceVersion extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Index") + private Long index; + + /** + * @see #index + */ + @CheckForNull + public Long getIndex() { + return index; + } + + /** + * @see #index + */ + public ResourceVersion withIndex(Long index) { + this.index = index; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResponseItem.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResponseItem.java new file mode 100644 index 000000000..cd90b78f3 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ResponseItem.java @@ -0,0 +1,200 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonIgnore; +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * Represents a pull response stream item + */ +@EqualsAndHashCode +@ToString +public class ResponseItem extends DockerObject implements Serializable { + private static final long serialVersionUID = -5187169652557467828L; + + @JsonProperty("stream") + private String stream; + + @JsonProperty("status") + private String status; + + @JsonProperty("progressDetail") + private ProgressDetail progressDetail; + + @Deprecated + @JsonProperty("progress") + private String progress; + + @JsonProperty("id") + private String id; + + @JsonProperty("from") + private String from; + + @JsonProperty("time") + private Long time; + + @JsonProperty("errorDetail") + private ErrorDetail errorDetail; + + @Deprecated + @JsonProperty("error") + private String error; + + /** + * @since {@link RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("aux") + private AuxDetail aux; + + @CheckForNull + public String getStream() { + return stream; + } + + @CheckForNull + public String getStatus() { + return status; + } + + @CheckForNull + public ProgressDetail getProgressDetail() { + return progressDetail; + } + + @CheckForNull + @Deprecated + public String getProgress() { + return progress; + } + + @CheckForNull + public String getId() { + return id; + } + + @CheckForNull + public String getFrom() { + return from; + } + + @CheckForNull + public Long getTime() { + return time; + } + + @CheckForNull + public ErrorDetail getErrorDetail() { + return errorDetail; + } + + @Deprecated + public String getError() { + return error; + } + + /** + * @see #aux + */ + @CheckForNull + public AuxDetail getAux() { + return aux; + } + + /** + * Returns whether the error field indicates an error + * + * @returns true: the error field indicates an error, false: the error field doesn't indicate an error + */ + @JsonIgnore + public boolean isErrorIndicated() { + // check both the deprecated and current error fields, just in case + return getError() != null || getErrorDetail() != null; + } + + @EqualsAndHashCode + @ToString + public static class ProgressDetail extends DockerObject implements Serializable { + private static final long serialVersionUID = -1954994695645715264L; + + @JsonProperty("current") + Long current; + + @JsonProperty("total") + Long total; + + @JsonProperty("start") + Long start; + + @CheckForNull + public Long getCurrent() { + return current; + } + + @CheckForNull + public Long getTotal() { + return total; + } + + @CheckForNull + public Long getStart() { + return start; + } + } + + @EqualsAndHashCode + @ToString + public static class ErrorDetail extends DockerObject implements Serializable { + private static final long serialVersionUID = -9136704865403084083L; + + @JsonProperty("code") + Integer code; + + @JsonProperty("message") + String message; + + @CheckForNull + public Integer getCode() { + return code; + } + + @CheckForNull + public String getMessage() { + return message; + } + } + + @EqualsAndHashCode + @ToString + public static class AuxDetail extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Size") + private Integer size; + + @JsonProperty("Tag") + private String tag; + + @JsonProperty("Digest") + private String digest; + + @CheckForNull + public Integer getSize() { + return size; + } + + @CheckForNull + public String getTag() { + return tag; + } + + @CheckForNull + public String getDigest() { + return digest; + } + } +} diff --git a/src/main/java/com/github/dockerjava/api/model/RestartPolicy.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/RestartPolicy.java similarity index 81% rename from src/main/java/com/github/dockerjava/api/model/RestartPolicy.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/RestartPolicy.java index ac6c4a318..53453915c 100644 --- a/src/main/java/com/github/dockerjava/api/model/RestartPolicy.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/RestartPolicy.java @@ -1,11 +1,12 @@ package com.github.dockerjava.api.model; -import static com.google.common.base.Preconditions.checkNotNull; +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; +import java.io.Serializable; -import com.fasterxml.jackson.annotation.JsonProperty; +import static java.util.Objects.requireNonNull; /** * Container restart policy @@ -26,7 +27,10 @@ * @author Marcus Linke * */ -public class RestartPolicy { +@EqualsAndHashCode +@ToString +public class RestartPolicy extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; @JsonProperty("MaximumRetryCount") private Integer maximumRetryCount = 0; @@ -38,7 +42,7 @@ public RestartPolicy() { } private RestartPolicy(int maximumRetryCount, String name) { - checkNotNull(name, "name is null"); + requireNonNull(name, "name is null"); this.maximumRetryCount = maximumRetryCount; this.name = name; } @@ -67,6 +71,13 @@ public static RestartPolicy onFailureRestart(int maximumRetryCount) { return new RestartPolicy(maximumRetryCount, "on-failure"); } + /** + * Restart the container unless it has been stopped + */ + public static RestartPolicy unlessStoppedRestart() { + return new RestartPolicy(0, "unless-stopped"); + } + public Integer getMaximumRetryCount() { return maximumRetryCount; } @@ -96,6 +107,10 @@ public static RestartPolicy parse(String serialized) throws IllegalArgumentExcep return alwaysRestart(); } + if ("unless-stopped".equals(name)) { + return unlessStoppedRestart(); + } + if ("on-failure".equals(name)) { int count = 0; if (parts.length == 2) { @@ -120,21 +135,4 @@ public String toString() { String result = name.isEmpty() ? "no" : name; return maximumRetryCount > 0 ? result + ":" + maximumRetryCount : result; } - - @Override - public boolean equals(Object obj) { - if (obj instanceof RestartPolicy) { - RestartPolicy other = (RestartPolicy) obj; - return new EqualsBuilder().append(maximumRetryCount, other.getMaximumRetryCount()) - .append(name, other.getName()).isEquals(); - } else { - return super.equals(obj); - } - } - - @Override - public int hashCode() { - return new HashCodeBuilder().append(maximumRetryCount).append(name).toHashCode(); - } - } diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/RuntimeInfo.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/RuntimeInfo.java new file mode 100644 index 000000000..c64511cda --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/RuntimeInfo.java @@ -0,0 +1,23 @@ +package com.github.dockerjava.api.model; + +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +@EqualsAndHashCode +@ToString +public class RuntimeInfo extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + private String path; + + public String getPath() { + return path; + } + + public void setPath(String path) { + this.path = path; + } + +} diff --git a/src/main/java/com/github/dockerjava/api/model/SELContext.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SELContext.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/model/SELContext.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/SELContext.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SearchItem.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SearchItem.java new file mode 100644 index 000000000..23a5c3bbf --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SearchItem.java @@ -0,0 +1,53 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * + * @author Konstantin Pelykh (kpelykh@gmail.com) + * + */ +@EqualsAndHashCode +@ToString +public class SearchItem extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("star_count") + private Integer starCount; + + @JsonProperty("is_official") + private Boolean isOfficial; + + @JsonProperty("is_trusted") + private Boolean isTrusted; + + @JsonProperty("name") + private String name; + + @JsonProperty("description") + private String description; + + public Integer getStarCount() { + return starCount; + } + + public Boolean isOfficial() { + return isOfficial; + } + + public Boolean isTrusted() { + return isTrusted; + } + + public String getName() { + return name; + } + + public String getDescription() { + return description; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Secret.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Secret.java new file mode 100644 index 000000000..bfbd7caf4 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Secret.java @@ -0,0 +1,89 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.Date; + +/** + * Used for Listing secret. + * + * @since {@link RemoteApiVersion#VERSION_1_25} + */ +@EqualsAndHashCode +@ToString +public class Secret extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("ID") + private String id; + + /** + * @since 1.24 + */ + @JsonProperty("CreatedAt") + private Date createdAt; + + /** + * @since 1.24 + */ + @JsonProperty("UpdatedAt") + private Date updatedAt; + + /** + * @since 1.24 + */ + @JsonProperty("Spec") + private ServiceSpec spec; + + /** + * @since 1.24 + */ + @JsonProperty("Version") + private ResourceVersion version; + + public String getId() { + return id; + } + + public void setId(String id) { + this.id = id; + } + + public Date getCreatedAt() { + return createdAt; + } + + public void setCreatedAt(Date createdAt) { + this.createdAt = createdAt; + } + + public Date getUpdatedAt() { + return updatedAt; + } + + public void setUpdatedAt(Date updatedAt) { + this.updatedAt = updatedAt; + } + + public ServiceSpec getSpec() { + return spec; + } + + public void setSpec(ServiceSpec spec) { + this.spec = spec; + } + + public ResourceVersion getVersion() { + return version; + } + + public void setVersion(ResourceVersion version) { + this.version = version; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SecretSpec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SecretSpec.java new file mode 100644 index 000000000..6b10239b9 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SecretSpec.java @@ -0,0 +1,83 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Base64; +import java.util.Map; + +/** + * @since {@link RemoteApiVersion#VERSION_1_25} + */ +@EqualsAndHashCode +@ToString +public class SecretSpec extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.25 + */ + @JsonProperty("Name") + private String name; + + /** + * @since 1.25 + */ + @JsonProperty("Data") + private String data; + + /** + * @since 1.25 + */ + @JsonProperty("Labels") + private Map labels; + + /** + * @see #name + */ + public String getName() { + return name; + } + + /** + * @see #name + */ + public SecretSpec withName(String name) { + this.name = name; + return this; + } + + /** + * @see #data + */ + public String getData() { + return data; + } + + /** + * @see #data + */ + public SecretSpec withData(String data) { + this.data = Base64.getEncoder().encodeToString(data.getBytes()); + return this; + } + + /** + * @see #labels + */ + public SecretSpec withLabels(Map labels) { + this.labels = labels; + return this; + } + + /** + * @see #labels + */ + @CheckForNull + public Map getLabels() { + return labels; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Service.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Service.java new file mode 100644 index 000000000..fd76be259 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Service.java @@ -0,0 +1,174 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Date; + +/** + * Used for Listing services. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class Service extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("ID") + private String id; + + /** + * @since 1.24 + */ + @JsonProperty("CreatedAt") + private Date createdAt; + + /** + * @since 1.24 + */ + @JsonProperty("UpdatedAt") + private Date updatedAt; + + /** + * @since 1.24 + */ + @JsonProperty("Spec") + private ServiceSpec spec; + + /** + * @since 1.24 + */ + @JsonProperty("Endpoint") + private Endpoint endpoint; + + /** + * @since 1.24 + */ + @JsonProperty("UpdateStatus") + private ServiceUpdateStatus updateStatus; + + /** + * @since 1.24 + */ + @JsonProperty("Version") + private ResourceVersion version; + + /** + * @see #id + */ + @CheckForNull + public String getId() { + return id; + } + + /** + * @see #id + */ + public Service withId(String id) { + this.id = id; + return this; + } + + /** + * @see #createdAt + */ + @CheckForNull + public Date getCreatedAt() { + return createdAt; + } + + /** + * @see #createdAt + */ + public Service withCreatedAt(Date createdAt) { + this.createdAt = createdAt; + return this; + } + + /** + * @see #updatedAt + */ + @CheckForNull + public Date getUpdatedAt() { + return updatedAt; + } + + /** + * @see #updatedAt + */ + public Service withUpdatedAt(Date updatedAt) { + this.updatedAt = updatedAt; + return this; + } + + /** + * @see #spec + */ + @CheckForNull + public ServiceSpec getSpec() { + return spec; + } + + /** + * @see #spec + */ + public Service withSpec(ServiceSpec spec) { + this.spec = spec; + return this; + } + + /** + * @see #endpoint + */ + @CheckForNull + public Endpoint getEndpoint() { + return endpoint; + } + + /** + * @see #endpoint + */ + public Service withEndpoint(Endpoint endpoint) { + this.endpoint = endpoint; + return this; + } + + /** + * @see #updateStatus + */ + @CheckForNull + public ServiceUpdateStatus getUpdateStatus() { + return updateStatus; + } + + /** + * @see #updateStatus + */ + public Service withUpdateStatus(ServiceUpdateStatus updateStatus) { + this.updateStatus = updateStatus; + return this; + } + + /** + * @see #version + */ + @CheckForNull + public ResourceVersion getVersion() { + return version; + } + + /** + * @see #version + */ + public Service withVersion(ResourceVersion version) { + this.version = version; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceGlobalModeOptions.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceGlobalModeOptions.java new file mode 100644 index 000000000..37feec292 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceGlobalModeOptions.java @@ -0,0 +1,18 @@ +package com.github.dockerjava.api.model; + +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +@SuppressWarnings("checkstyle:hideutilityclassconstructor") +public class ServiceGlobalModeOptions extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + // Intentionally left blank, there are no options for this mode +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceMode.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceMode.java new file mode 100644 index 000000000..bfc4a3de3 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceMode.java @@ -0,0 +1,12 @@ +package com.github.dockerjava.api.model; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum ServiceMode { + + REPLICATED, + + GLOBAL + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceModeConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceModeConfig.java new file mode 100644 index 000000000..82d1b3b20 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceModeConfig.java @@ -0,0 +1,84 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ServiceModeConfig extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Replicated") + private ServiceReplicatedModeOptions replicated; + + /** + * @since 1.24 + */ + @JsonProperty("Global") + private ServiceGlobalModeOptions global; + + /** + * @since 1.24 + */ + @CheckForNull + public ServiceMode getMode() { + if (replicated != null) { + return ServiceMode.REPLICATED; + } + if (global != null) { + return ServiceMode.GLOBAL; + } + + return null; + } + + /** + * @since 1.24 + */ + @CheckForNull + public ServiceReplicatedModeOptions getReplicated() { + return replicated; + } + + /** + * @since 1.24 + */ + public ServiceModeConfig withReplicated(ServiceReplicatedModeOptions replicated) { + if (replicated != null && this.global != null) { + throw new IllegalStateException("Cannot set both replicated and global mode"); + } + this.replicated = replicated; + + return this; + } + + /** + * @since 1.24 + */ + @CheckForNull + public ServiceGlobalModeOptions getGlobal() { + return global; + } + + /** + * @since 1.24 + */ + public ServiceModeConfig withGlobal(ServiceGlobalModeOptions global) { + if (global != null && this.replicated != null) { + throw new IllegalStateException("Cannot set both global and replicated mode"); + } + this.global = global; + + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServicePlacement.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServicePlacement.java new file mode 100644 index 000000000..4c0953508 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServicePlacement.java @@ -0,0 +1,90 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ServicePlacement extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Constraints") + private List constraints; + + /** + * @since 1.30 + */ + @JsonProperty("Platforms") + private List platforms; + + /** + * @since 1.40 + */ + @JsonProperty("MaxReplicas") + private Integer maxReplicas; + + /** + * @see #constraints + */ + @CheckForNull + public List getConstraints() { + return constraints; + } + + /** + * @see #constraints + */ + public ServicePlacement withConstraints(List constraints) { + this.constraints = constraints; + return this; + } + + /** + * @see #platforms + */ + public List getPlatforms() { + return platforms; + } + + public void setPlatforms(List platforms) { + this.platforms = platforms; + } + + /** + * Specifies the maximum amount of replicas / tasks that can run on one node. + * 0 means unlimited replicas per node. + * + * @param maxReplicas Max number of replicas + * @return This instance of ServicePlacement + * @throws IllegalArgumentException if maxReplicas is less than 0 + */ + public ServicePlacement withMaxReplicas(int maxReplicas) { + if (maxReplicas < 0) { + throw new IllegalArgumentException("The Value for MaxReplicas must be greater or equal to 0"); + } + + this.maxReplicas = maxReplicas; + return this; + } + + /** + * Getter for maxReplicas + * + * @return The maximum amount of replicas / tasks that can run on one node. + */ + public Integer getMaxReplicas() { + return this.maxReplicas; + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceReplicatedModeOptions.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceReplicatedModeOptions.java new file mode 100644 index 000000000..eea2a5211 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceReplicatedModeOptions.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ServiceReplicatedModeOptions extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Replicas") + private long replicas; + + /** + * @see #replicas + */ + public long getReplicas() { + return replicas; + } + + /** + * @see #replicas + */ + public ServiceReplicatedModeOptions withReplicas(int replicas) { + this.replicas = replicas; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceRestartCondition.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceRestartCondition.java new file mode 100644 index 000000000..639c84b2c --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceRestartCondition.java @@ -0,0 +1,19 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum ServiceRestartCondition { + + @JsonProperty("on-failure") + ON_FAILURE, + + @JsonProperty("any") + ANY, + + @JsonProperty("none") + NONE + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceRestartPolicy.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceRestartPolicy.java new file mode 100644 index 000000000..11b54f666 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceRestartPolicy.java @@ -0,0 +1,105 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ServiceRestartPolicy extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Condition") + private ServiceRestartCondition condition; + + /** + * @since 1.24 + */ + @JsonProperty("Delay") + private Long delay; + + /** + * @since 1.24 + */ + @JsonProperty("MaxAttempts") + private Long maxAttempts; + + /** + * @since 1.24 + */ + @JsonProperty("Window") + private Long window; + + /** + * @see #condition + */ + @CheckForNull + public ServiceRestartCondition getCondition() { + return condition; + } + + /** + * @see #condition + */ + public ServiceRestartPolicy withCondition(ServiceRestartCondition condition) { + this.condition = condition; + return this; + } + + /** + * @see #delay + */ + @CheckForNull + public Long getDelay() { + return delay; + } + + /** + * @see #delay + */ + public ServiceRestartPolicy withDelay(Long delay) { + this.delay = delay; + return this; + } + + /** + * @see #maxAttempts + */ + @CheckForNull + public Long getMaxAttempts() { + return maxAttempts; + } + + /** + * @see #maxAttempts + */ + public ServiceRestartPolicy withMaxAttempts(Long maxAttempts) { + this.maxAttempts = maxAttempts; + return this; + } + + /** + * @see #window + */ + @CheckForNull + public Long getWindow() { + return window; + } + + /** + * @see #window + */ + public ServiceRestartPolicy withWindow(Long window) { + this.window = window; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceSpec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceSpec.java new file mode 100644 index 000000000..a1fbec916 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceSpec.java @@ -0,0 +1,187 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; +import java.util.Map; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ServiceSpec extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Name") + private String name; + + /** + * @since 1.24 + */ + @JsonProperty("TaskTemplate") + private TaskSpec taskTemplate; + + /** + * @since 1.24 + */ + @JsonProperty("Mode") + private ServiceModeConfig mode; + + /** + * @since 1.24 + */ + @JsonProperty("UpdateConfig") + private UpdateConfig updateConfig; + + /** + * @since 1.24 + */ + @JsonProperty("Networks") + private List networks; + + /** + * @since 1.24 + */ + @JsonProperty("EndpointSpec") + private EndpointSpec endpointSpec; + + /** + * @since 1.24 + */ + @JsonProperty("Labels") + private Map labels; + + /** + * @since 1.28 + */ + @JsonProperty("RollbackConfig") + private UpdateConfig rollbackConfig; + + /** + * @see #name + */ + public String getName() { + return name; + } + + /** + * @see #name + */ + public ServiceSpec withName(String name) { + this.name = name; + return this; + } + + /** + * @see #taskTemplate + */ + @CheckForNull + public TaskSpec getTaskTemplate() { + return taskTemplate; + } + + /** + * @see #taskTemplate + */ + public ServiceSpec withTaskTemplate(TaskSpec taskTemplate) { + this.taskTemplate = taskTemplate; + return this; + } + + /** + * @see #mode + */ + @CheckForNull + public ServiceModeConfig getMode() { + return mode; + } + + /** + * @see #mode + */ + public ServiceSpec withMode(ServiceModeConfig mode) { + this.mode = mode; + return this; + } + + /** + * @see #updateConfig + */ + @CheckForNull + public UpdateConfig getUpdateConfig() { + return updateConfig; + } + + /** + * @see #updateConfig + */ + public ServiceSpec withUpdateConfig(UpdateConfig updateConfig) { + this.updateConfig = updateConfig; + return this; + } + + /** + * @see #networks + */ + @CheckForNull + public List getNetworks() { + return networks; + } + + /** + * @see #networks + */ + public ServiceSpec withNetworks(List networks) { + this.networks = networks; + return this; + } + + /** + * @see #endpointSpec + */ + @CheckForNull + public EndpointSpec getEndpointSpec() { + return endpointSpec; + } + + /** + * @see #endpointSpec + */ + public ServiceSpec withEndpointSpec(EndpointSpec endpointSpec) { + this.endpointSpec = endpointSpec; + return this; + } + + /** + * @see #labels + */ + public ServiceSpec withLabels(Map labels) { + this.labels = labels; + return this; + } + + public UpdateConfig getRollbackConfig() { + return rollbackConfig; + } + + public ServiceSpec withRollbackConfig(UpdateConfig rollbackConfig) { + this.rollbackConfig = rollbackConfig; + return this; + } + + /** + * @see #labels + */ + @CheckForNull + public Map getLabels() { + return labels; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceUpdateState.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceUpdateState.java new file mode 100644 index 000000000..d22f8999e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceUpdateState.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum ServiceUpdateState { + @JsonProperty("unknown") + UNKNOWN, + + @JsonProperty("updating") + UPDATING, + + @JsonProperty("paused") + PAUSED, + + @JsonProperty("completed") + COMPLETED, + + @JsonProperty("rollback_started") + ROLLBACK_STARTED, + + @JsonProperty("rollback_paused") + ROLLBACK_PAUSED, + + @JsonProperty("rollback_completed") + ROLLBACK_COMPLETED +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceUpdateStatus.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceUpdateStatus.java new file mode 100644 index 000000000..18cb54603 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ServiceUpdateStatus.java @@ -0,0 +1,106 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Date; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class ServiceUpdateStatus extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("State") + private ServiceUpdateState state; + + /** + * @since 1.24 + */ + @JsonProperty("StartedAt") + private Date startedAt; + + /** + * @since 1.24 + */ + @JsonProperty("CompletedAt") + private Date completedAt; + + /** + * @since 1.24 + */ + @JsonProperty("Message") + private String message; + + /** + * @see #state + */ + @CheckForNull + public ServiceUpdateState getState() { + return state; + } + + /** + * @see #state + */ + public ServiceUpdateStatus withState(ServiceUpdateState state) { + this.state = state; + return this; + } + + /** + * @see #startedAt + */ + @CheckForNull + public Date getStartedAt() { + return startedAt; + } + + /** + * @see #startedAt + */ + public ServiceUpdateStatus withStartedAt(Date startedAt) { + this.startedAt = startedAt; + return this; + } + + /** + * @see #completedAt + */ + @CheckForNull + public Date getCompletedAt() { + return completedAt; + } + + /** + * @see #completedAt + */ + public ServiceUpdateStatus withCompletedAt(Date completedAt) { + this.completedAt = completedAt; + return this; + } + + /** + * @see #message + */ + @CheckForNull + public String getMessage() { + return message; + } + + /** + * @see #message + */ + public ServiceUpdateStatus withMessage(String message) { + this.message = message; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/StatisticNetworksConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/StatisticNetworksConfig.java new file mode 100644 index 000000000..2ba57d76b --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/StatisticNetworksConfig.java @@ -0,0 +1,107 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * Used in {@link Statistics} + * + * @author Yuting Liu + */ +@EqualsAndHashCode +@ToString +public class StatisticNetworksConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("rx_bytes") + private Long rxBytes; + + @JsonProperty("rx_dropped") + private Long rxDropped; + + @JsonProperty("rx_errors") + private Long rxErrors; + + @JsonProperty("rx_packets") + private Long rxPackets; + + @JsonProperty("tx_bytes") + private Long txBytes; + + @JsonProperty("tx_dropped") + private Long txDropped; + + @JsonProperty("tx_errors") + private Long txErrors; + + @JsonProperty("tx_packets") + private Long txPackets; + + /** + * @see #rxBytes + */ + @CheckForNull + public Long getRxBytes() { + return rxBytes; + } + + /** + * @see #rxDropped + */ + @CheckForNull + public Long getRxDropped() { + return rxDropped; + } + + /** + * @see #rxErrors + */ + @CheckForNull + public Long getRxErrors() { + return rxErrors; + } + + /** + * @see #rxPackets + */ + @CheckForNull + public Long getRxPackets() { + return rxPackets; + } + + /** + * @see #txBytes + */ + @CheckForNull + public Long getTxBytes() { + return txBytes; + } + + /** + * @see #txDropped + */ + @CheckForNull + public Long getTxDropped() { + return txDropped; + } + + /** + * @see #txErrors + */ + @CheckForNull + public Long getTxErrors() { + return txErrors; + } + + /** + * @see #txPackets + */ + @CheckForNull + public Long getTxPackets() { + return txPackets; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Statistics.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Statistics.java new file mode 100644 index 000000000..3800363eb --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Statistics.java @@ -0,0 +1,115 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; +import java.util.Map; + +import javax.annotation.CheckForNull; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +/** + * Representation of a Docker statistics. + */ +@EqualsAndHashCode +@ToString +public class Statistics extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("read") + private String read; + + @JsonProperty("preread") + private String preread; + + /** + * @since Docker Remote API 1.21 + */ + @CheckForNull + @JsonProperty("networks") + private Map networks; + + /** + * @deprecated as of Docker Remote API 1.21, replaced by {@link #networks} + */ + @Deprecated + @JsonProperty("network") + private Map network; + + @JsonProperty("memory_stats") + private MemoryStatsConfig memoryStats; + + @JsonProperty("blkio_stats") + private BlkioStatsConfig blkioStats; + + @JsonProperty("cpu_stats") + private CpuStatsConfig cpuStats; + + @JsonProperty("num_procs") + private Long numProcs; + + /** + * @since Docker Remote API 1.19 + */ + @JsonProperty("precpu_stats") + private CpuStatsConfig preCpuStats; + + /** + * @since Docker Remote API 1.23 + */ + @JsonProperty("pids_stats") + private PidsStatsConfig pidsStats; + + public String getRead() { + return read; + } + + public String getPreread() { + return preread; + } + + /** + * @since Docker Remote API 1.21 + */ + @CheckForNull + public Map getNetworks() { + return networks; + } + + /** + * @deprecated as of Docker Remote API 1.21, replaced by {@link #getNetworks()} + */ + @Deprecated + public Map getNetwork() { + return network; + } + + public CpuStatsConfig getCpuStats() { + return cpuStats; + } + + public Long getNumProcs() { + return numProcs; + } + + /** + * The cpu statistic of last read, which is used for calculating the cpu usage percent. + * It is not the exact copy of the {@link #getCpuStats()}. + */ + public CpuStatsConfig getPreCpuStats() { + return preCpuStats; + } + + public MemoryStatsConfig getMemoryStats() { + return memoryStats; + } + + public BlkioStatsConfig getBlkioStats() { + return blkioStats; + } + + public PidsStatsConfig getPidsStats() { + return pidsStats; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/StatsConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/StatsConfig.java new file mode 100644 index 000000000..8afbc34d9 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/StatsConfig.java @@ -0,0 +1,388 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +@EqualsAndHashCode +@ToString +public class StatsConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("active_anon") + private Long activeAnon; + + @JsonProperty("active_file") + private Long activeFile; + + @JsonProperty("cache") + private Long cache; + + @JsonProperty("dirty") + private Long dirty; + + @JsonProperty("hierarchical_memory_limit") + private Long hierarchicalMemoryLimit; + + @JsonProperty("hierarchical_memsw_limit") + private Long hierarchicalMemswLimit; + + @JsonProperty("inactive_anon") + private Long inactiveAnon; + + @JsonProperty("inactive_file") + private Long inactiveFile; + + @JsonProperty("mapped_file") + private Long mappedFile; + + @JsonProperty("pgfault") + private Long pgfault; + + @JsonProperty("pgmajfault") + private Long pgmajfault; + + @JsonProperty("pgpgin") + private Long pgpgin; + + @JsonProperty("pgpgout") + private Long pgpgout; + + @JsonProperty("rss") + private Long rss; + + @JsonProperty("rss_huge") + private Long rssHuge; + + @JsonProperty("swap") + private Long swap; + + @JsonProperty("total_active_anon") + private Long totalActiveAnon; + + @JsonProperty("total_active_file") + private Long totalActiveFile; + + @JsonProperty("total_cache") + private Long totalCache; + + @JsonProperty("total_dirty") + private Long totalDirty; + + @JsonProperty("total_inactive_anon") + private Long totalInactiveAnon; + + @JsonProperty("total_inactive_file") + private Long totalInactiveFile; + + @JsonProperty("total_mapped_file") + private Long totalMappedFile; + + @JsonProperty("total_pgfault") + private Long totalPgfault; + + @JsonProperty("total_pgmajfault") + private Long totalPgmajfault; + + @JsonProperty("total_pgpgin") + private Long totalPgpgin; + + @JsonProperty("total_pgpgout") + private Long totalPgpgout; + + @JsonProperty("total_rss") + private Long totalRss; + + @JsonProperty("total_rss_huge") + private Long totalRssHuge; + + @JsonProperty("total_swap") + private Long totalSwap; + + @JsonProperty("total_unevictable") + private Long totalUnevictable; + + @JsonProperty("total_writeback") + private Long totalWriteback; + + @JsonProperty("unevictable") + private Long unevictable; + + @JsonProperty("writeback") + private Long writeback; + + /** + * @see #activeAnon + */ + @CheckForNull + public Long getActiveAnon() { + return activeAnon; + } + + /** + * @see #activeFile + */ + @CheckForNull + public Long getActiveFile() { + return activeFile; + } + + /** + * @see #cache + */ + @CheckForNull + public Long getCache() { + return cache; + } + + /** + * @see #dirty + */ + @CheckForNull + public Long getDirty() { + return dirty; + } + + /** + * @see #hierarchicalMemoryLimit + */ + @CheckForNull + public Long getHierarchicalMemoryLimit() { + return hierarchicalMemoryLimit; + } + + /** + * @see #hierarchicalMemswLimit + */ + @CheckForNull + public Long getHierarchicalMemswLimit() { + return hierarchicalMemswLimit; + } + + /** + * @see #inactiveAnon + */ + @CheckForNull + public Long getInactiveAnon() { + return inactiveAnon; + } + + /** + * @see #inactiveFile + */ + @CheckForNull + public Long getInactiveFile() { + return inactiveFile; + } + + /** + * @see #mappedFile + */ + @CheckForNull + public Long getMappedFile() { + return mappedFile; + } + + /** + * @see #pgfault + */ + @CheckForNull + public Long getPgfault() { + return pgfault; + } + + /** + * @see #pgmajfault + */ + @CheckForNull + public Long getPgmajfault() { + return pgmajfault; + } + + /** + * @see #pgpgin + */ + @CheckForNull + public Long getPgpgin() { + return pgpgin; + } + + /** + * @see #pgpgout + */ + @CheckForNull + public Long getPgpgout() { + return pgpgout; + } + + /** + * @see #rss + */ + @CheckForNull + public Long getRss() { + return rss; + } + + /** + * @see #rssHuge + */ + @CheckForNull + public Long getRssHuge() { + return rssHuge; + } + + /** + * @see #swap + */ + @CheckForNull + public Long getSwap() { + return swap; + } + + /** + * @see #totalActiveAnon + */ + @CheckForNull + public Long getTotalActiveAnon() { + return totalActiveAnon; + } + + /** + * @see #totalActiveFile + */ + @CheckForNull + public Long getTotalActiveFile() { + return totalActiveFile; + } + + /** + * @see #totalCache + */ + @CheckForNull + public Long getTotalCache() { + return totalCache; + } + + /** + * @see #totalDirty + */ + @CheckForNull + public Long getTotalDirty() { + return totalDirty; + } + + /** + * @see #totalInactiveAnon + */ + @CheckForNull + public Long getTotalInactiveAnon() { + return totalInactiveAnon; + } + + /** + * @see #totalInactiveFile + */ + @CheckForNull + public Long getTotalInactiveFile() { + return totalInactiveFile; + } + + /** + * @see #totalMappedFile + */ + @CheckForNull + public Long getTotalMappedFile() { + return totalMappedFile; + } + + /** + * @see #totalPgfault + */ + @CheckForNull + public Long getTotalPgfault() { + return totalPgfault; + } + + /** + * @see #totalPgmajfault + */ + @CheckForNull + public Long getTotalPgmajfault() { + return totalPgmajfault; + } + + /** + * @see #totalPgpgin + */ + @CheckForNull + public Long getTotalPgpgin() { + return totalPgpgin; + } + + /** + * @see #totalPgpgout + */ + @CheckForNull + public Long getTotalPgpgout() { + return totalPgpgout; + } + + /** + * @see #totalRss + */ + @CheckForNull + public Long getTotalRss() { + return totalRss; + } + + /** + * @see #totalRssHuge + */ + @CheckForNull + public Long getTotalRssHuge() { + return totalRssHuge; + } + + /** + * @see #totalSwap + */ + @CheckForNull + public Long getTotalSwap() { + return totalSwap; + } + + /** + * @see #totalUnevictable + */ + @CheckForNull + public Long getTotalUnevictable() { + return totalUnevictable; + } + + /** + * @see #totalWriteback + */ + @CheckForNull + public Long getTotalWriteback() { + return totalWriteback; + } + + /** + * @see #unevictable + */ + @CheckForNull + public Long getUnevictable() { + return unevictable; + } + + /** + * @see #writeback + */ + @CheckForNull + public Long getWriteback() { + return writeback; + } +} diff --git a/src/main/java/com/github/dockerjava/api/model/StreamType.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/StreamType.java similarity index 100% rename from src/main/java/com/github/dockerjava/api/model/StreamType.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/StreamType.java diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Swarm.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Swarm.java new file mode 100644 index 000000000..0554c53f1 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Swarm.java @@ -0,0 +1,65 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +import javax.annotation.CheckForNull; +import java.util.Date; + +/** + * @since 1.24 + */ +public class Swarm extends ClusterInfo { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("JoinTokens") + private SwarmJoinTokens joinTokens; + + /** + * @see #joinTokens + */ + @CheckForNull + public SwarmJoinTokens getJoinTokens() { + return joinTokens; + } + + /** + * @see #joinTokens + */ + public Swarm withJoinTokens(SwarmJoinTokens joinTokens) { + this.joinTokens = joinTokens; + return this; + } + + @Override + public Swarm withCreatedAt(Date createdAt) { + super.withCreatedAt(createdAt); + return this; + } + + @Override + public Swarm withId(String id) { + super.withId(id); + return this; + } + + @Override + public Swarm withSpec(SwarmSpec spec) { + super.withSpec(spec); + return this; + } + + @Override + public Swarm withUpdatedAt(Date updatedAt) { + super.withUpdatedAt(updatedAt); + return this; + } + + @Override + public Swarm withVersion(ResourceVersion version) { + super.withVersion(version); + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmCAConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmCAConfig.java new file mode 100644 index 000000000..8ebc97ffd --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmCAConfig.java @@ -0,0 +1,63 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmCAConfig extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("NodeCertExpiry") + private Long nodeCertExpiry; + + /** + * @since 1.24 + */ + @JsonProperty("ExternalCAs") + private List externalCA; + + /** + * @see #nodeCertExpiry + */ + @CheckForNull + public Long getNodeCertExpiry() { + return nodeCertExpiry; + } + + /** + * @see #nodeCertExpiry + */ + public SwarmCAConfig withNodeCertExpiry(Long nodeCertExpiry) { + this.nodeCertExpiry = nodeCertExpiry; + return this; + } + + /** + * @see #externalCA + */ + @CheckForNull + public List getExternalCA() { + return externalCA; + } + + /** + * @see #externalCA + */ + public SwarmCAConfig withExternalCA(List externalCA) { + this.externalCA = externalCA; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmDispatcherConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmDispatcherConfig.java new file mode 100644 index 000000000..2a45b84cd --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmDispatcherConfig.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.api.model; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmDispatcherConfig extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("HeartbeatPeriod") + private Long heartbeatPeriod; + + /** + * @see #heartbeatPeriod + */ + @CheckForNull + public Long getHeartbeatPeriod() { + return heartbeatPeriod; + } + + /** + * @see #heartbeatPeriod + */ + public SwarmDispatcherConfig withHeartbeatPeriod(Long heartbeatPeriod) { + this.heartbeatPeriod = heartbeatPeriod; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmInfo.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmInfo.java new file mode 100644 index 000000000..faed0fcf3 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmInfo.java @@ -0,0 +1,216 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; + +/** + * @since 1.24 + */ +@EqualsAndHashCode +@ToString +public class SwarmInfo extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("NodeID") + private String nodeID; + + /** + * @since 1.24 + */ + @JsonProperty("NodeAddr") + private String nodeAddr; + + /** + * @since 1.24 + */ + @JsonProperty("LocalNodeState") + private LocalNodeState localNodeState; + + /** + * @since 1.24 + */ + @JsonProperty("ControlAvailable") + private Boolean controlAvailable; + + /** + * @since 1.24 + */ + @JsonProperty("Error") + private String error; + + /** + * @since 1.24 + */ + @JsonProperty("RemoteManagers") + private List remoteManagers; + + /** + * @since 1.24 + */ + @JsonProperty("Nodes") + private Integer nodes; + + /** + * @since 1.24 + */ + @JsonProperty("Managers") + private Integer managers; + + /** + * @since 1.24 + */ + @JsonProperty("ClusterInfo") + private ClusterInfo clusterInfo; + + /** + * @see #nodeID + */ + @CheckForNull + public String getNodeID() { + return nodeID; + } + + /** + * @see #nodeID + */ + public SwarmInfo withNodeID(String nodeID) { + this.nodeID = nodeID; + return this; + } + + /** + * @see #nodeAddr + */ + @CheckForNull + public String getNodeAddr() { + return nodeAddr; + } + + /** + * @see #nodeAddr + */ + public SwarmInfo withNodeAddr(String nodeAddr) { + this.nodeAddr = nodeAddr; + return this; + } + + /** + * @see #localNodeState + */ + @CheckForNull + public LocalNodeState getLocalNodeState() { + return localNodeState; + } + + /** + * @see #localNodeState + */ + public SwarmInfo withLocalNodeState(LocalNodeState localNodeState) { + this.localNodeState = localNodeState; + return this; + } + + /** + * @see #controlAvailable + */ + @CheckForNull + public Boolean getControlAvailable() { + return controlAvailable; + } + + /** + * @see #controlAvailable + */ + public SwarmInfo withControlAvailable(Boolean controlAvailable) { + this.controlAvailable = controlAvailable; + return this; + } + + /** + * @see #error + */ + @CheckForNull + public String getError() { + return error; + } + + /** + * @see #error + */ + public SwarmInfo withError(String error) { + this.error = error; + return this; + } + + /** + * @see #remoteManagers + */ + @CheckForNull + public List getRemoteManagers() { + return remoteManagers; + } + + /** + * @see #remoteManagers + */ + public SwarmInfo withRemoteManagers(List remoteManagers) { + this.remoteManagers = remoteManagers; + return this; + } + + /** + * @see #nodes + */ + @CheckForNull + public Integer getNodes() { + return nodes; + } + + /** + * @see #nodes + */ + public SwarmInfo withNodes(Integer nodes) { + this.nodes = nodes; + return this; + } + + /** + * @see #managers + */ + @CheckForNull + public Integer getManagers() { + return managers; + } + + /** + * @see #managers + */ + public SwarmInfo withManagers(Integer managers) { + this.managers = managers; + return this; + } + + /** + * @see #clusterInfo + */ + @CheckForNull + public ClusterInfo getClusterInfo() { + return clusterInfo; + } + + /** + * @see #clusterInfo + */ + public SwarmInfo withClusterInfo(ClusterInfo clusterInfo) { + this.clusterInfo = clusterInfo; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmJoinTokens.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmJoinTokens.java new file mode 100644 index 000000000..9e5f63aea --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmJoinTokens.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmJoinTokens extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Worker") + private String worker; + + /** + * @since 1.24 + */ + @JsonProperty("Manager") + private String manager; + + /** + * @see #worker + */ + @CheckForNull + public String getWorker() { + return worker; + } + + /** + * @see #worker + */ + public SwarmJoinTokens withWorker(String worker) { + this.worker = worker; + return this; + } + + /** + * @see #manager + */ + @CheckForNull + public String getManager() { + return manager; + } + + /** + * @see #manager + */ + public SwarmJoinTokens withManager(String manager) { + this.manager = manager; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNode.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNode.java new file mode 100644 index 000000000..9b5aff96c --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNode.java @@ -0,0 +1,197 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Date; + + +/** + * Used for Listing SwarmNodes. + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmNode extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("ID") + private String id; + + /** + * @since 1.24 + */ + @JsonProperty("Version") + private ObjectVersion version; + + /** + * @since 1.24 + */ + @JsonProperty("CreatedAt") + private Date createdAt; + + /** + * @since 1.24 + */ + @JsonProperty("UpdatedAt") + private Date updatedAt; + + /** + * @since 1.24 + */ + @JsonProperty("Spec") + private SwarmNodeSpec spec; + + /** + * @since 1.24 + */ + @JsonProperty("Description") + private SwarmNodeDescription description; + + /** + * @since 1.24 + */ + @JsonProperty("Status") + private SwarmNodeStatus status; + + /** + * @since 1.24 + */ + @JsonProperty("ManagerStatus") + private SwarmNodeManagerStatus managerStatus; + + /** + * @see #id + */ + @CheckForNull + public String getId() { + return id; + } + + /** + * @see #id + */ + public SwarmNode withId(String id) { + this.id = id; + return this; + } + + /** + * @see #version + */ + @CheckForNull + public ObjectVersion getVersion() { + return version; + } + + /** + * @see #version + */ + public SwarmNode withVersion(ObjectVersion version) { + this.version = version; + return this; + } + + /** + * @see #createdAt + */ + @CheckForNull + public Date getCreatedAt() { + return createdAt; + } + + /** + * @see #createdAt + */ + public SwarmNode withCreatedAt(Date createdAt) { + this.createdAt = createdAt; + return this; + } + + /** + * @see #updatedAt + */ + @CheckForNull + public Date getUpdatedAt() { + return updatedAt; + } + + /** + * @see #updatedAt + */ + public SwarmNode withUpdatedAt(Date updatedAt) { + this.updatedAt = updatedAt; + return this; + } + + /** + * @see #spec + */ + @CheckForNull + public SwarmNodeSpec getSpec() { + return spec; + } + + /** + * @see #spec + */ + public SwarmNode withSpec(SwarmNodeSpec spec) { + this.spec = spec; + return this; + } + + /** + * @see #description + */ + @CheckForNull + public SwarmNodeDescription getDescription() { + return description; + } + + /** + * @see #description + */ + public SwarmNode withDescription(SwarmNodeDescription description) { + this.description = description; + return this; + } + + /** + * @see #status + */ + @CheckForNull + public SwarmNodeStatus getStatus() { + return status; + } + + /** + * @see #status + */ + public SwarmNode withStatus(SwarmNodeStatus status) { + this.status = status; + return this; + } + + /** + * @see #managerStatus + */ + @CheckForNull + public SwarmNodeManagerStatus getManagerStatus() { + return managerStatus; + } + + /** + * @see #managerStatus + */ + public SwarmNode withManagerStatus(SwarmNodeManagerStatus managerStatus) { + this.managerStatus = managerStatus; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeAvailability.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeAvailability.java new file mode 100644 index 000000000..1ab24146f --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeAvailability.java @@ -0,0 +1,18 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum SwarmNodeAvailability { + + @JsonProperty("active") + ACTIVE, + + @JsonProperty("pause") + PAUSE, + + @JsonProperty("drain") + DRAIN +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeDescription.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeDescription.java new file mode 100644 index 000000000..f929327d1 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeDescription.java @@ -0,0 +1,105 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmNodeDescription extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Hostname") + private String hostname; + + /** + * @since 1.24 + */ + @JsonProperty("Platform") + private SwarmNodePlatform platform; + + /** + * @since 1.24 + */ + @JsonProperty("Resources") + private SwarmNodeResources resources; + + /** + * @since 1.24 + */ + @JsonProperty("Engine") + private SwarmNodeEngineDescription engine; + + /** + * @see #hostname + */ + @CheckForNull + public String getHostname() { + return hostname; + } + + /** + * @see #hostname + */ + public SwarmNodeDescription withHostname(String hostname) { + this.hostname = hostname; + return this; + } + + /** + * @see #platform + */ + @CheckForNull + public SwarmNodePlatform getPlatform() { + return platform; + } + + /** + * @see #platform + */ + public SwarmNodeDescription withPlatform(SwarmNodePlatform platform) { + this.platform = platform; + return this; + } + + /** + * @see #resources + */ + @CheckForNull + public SwarmNodeResources getResources() { + return resources; + } + + /** + * @see #resources + */ + public SwarmNodeDescription withResources(SwarmNodeResources resources) { + this.resources = resources; + return this; + } + + /** + * @see #engine + */ + @CheckForNull + public SwarmNodeEngineDescription getEngine() { + return engine; + } + + /** + * @see #engine + */ + public SwarmNodeDescription withEngine(SwarmNodeEngineDescription engine) { + this.engine = engine; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeEngineDescription.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeEngineDescription.java new file mode 100644 index 000000000..a2f38531e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeEngineDescription.java @@ -0,0 +1,85 @@ +package com.github.dockerjava.api.model; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Map; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmNodeEngineDescription extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("EngineVersion") + private String engineVersion; + + /** + * @since 1.24 + */ + @JsonProperty("Labels") + private Map labels; + + /** + * @since 1.24 + */ + @JsonProperty("Plugins") + private SwarmNodePluginDescription[] plugins; + + /** + * @see #engineVersion + */ + @CheckForNull + public String getEngineVersion() { + return engineVersion; + } + + /** + * @see #engineVersion + */ + public SwarmNodeEngineDescription withEngineVersion(String engineVersion) { + this.engineVersion = engineVersion; + return this; + } + + /** + * @see #labels + */ + @CheckForNull + public Map getLabels() { + return labels; + } + + /** + * @see #labels + */ + public SwarmNodeEngineDescription withLabels(Map labels) { + this.labels = labels; + return this; + } + + /** + * @see #plugins + */ + @CheckForNull + public SwarmNodePluginDescription[] getPlugins() { + return plugins; + } + + /** + * @see #plugins + */ + public SwarmNodeEngineDescription withPlugins(SwarmNodePluginDescription[] plugins) { + this.plugins = plugins; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeManagerStatus.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeManagerStatus.java new file mode 100644 index 000000000..0307d18e7 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeManagerStatus.java @@ -0,0 +1,84 @@ +package com.github.dockerjava.api.model; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmNodeManagerStatus extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Leader") + private boolean leader; + + /** + * @since 1.24 + */ + @JsonProperty("Reachability") + private Reachability reachability; + + /** + * @since 1.24 + */ + @JsonProperty("Addr") + private String addr; + + /** + * @see #leader + */ + @CheckForNull + public boolean isLeader() { + return leader; + } + + /** + * @see #leader + */ + public SwarmNodeManagerStatus withLeader(boolean leader) { + this.leader = leader; + return this; + } + + /** + * @see #reachability + */ + @CheckForNull + public Reachability getReachability() { + return reachability; + } + + /** + * @see #reachability + */ + public SwarmNodeManagerStatus withReachability(Reachability reachability) { + this.reachability = reachability; + return this; + } + + /** + * @see #addr + */ + @CheckForNull + public String getAddr() { + return addr; + } + + /** + * @see #addr + */ + public SwarmNodeManagerStatus withAddr(String addr) { + this.addr = addr; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodePlatform.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodePlatform.java new file mode 100644 index 000000000..9688f10b0 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodePlatform.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.api.model; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmNodePlatform extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Architecture") + private String architecture; + + /** + * @since 1.24 + */ + @JsonProperty("OS") + private String os; + + /** + * @see #architecture + */ + @CheckForNull + public String getArchitecture() { + return architecture; + } + + /** + * @see #architecture + */ + public SwarmNodePlatform withArchitecture(String architecture) { + this.architecture = architecture; + return this; + } + + /** + * @see #os + */ + @CheckForNull + public String getOs() { + return os; + } + + /** + * @see #os + */ + public SwarmNodePlatform withOs(String os) { + this.os = os; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodePluginDescription.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodePluginDescription.java new file mode 100644 index 000000000..aa051aaa3 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodePluginDescription.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.api.model; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmNodePluginDescription extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Type") + private String type; + + /** + * @since 1.24 + */ + @JsonProperty("Name") + private String name; + + /** + * @see #type + */ + @CheckForNull + public String getType() { + return type; + } + + /** + * @see #type + */ + public SwarmNodePluginDescription withType(String type) { + this.type = type; + return this; + } + + /** + * @see #name + */ + @CheckForNull + public String getName() { + return name; + } + + /** + * @see #name + */ + public SwarmNodePluginDescription withName(String name) { + this.name = name; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeResources.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeResources.java new file mode 100644 index 000000000..c9586e921 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeResources.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.api.model; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmNodeResources extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("NanoCPUs") + private Long nanoCPUs; + + /** + * @since 1.24 + */ + @JsonProperty("MemoryBytes") + private Long memoryBytes; + + /** + * @see #nanoCPUs + */ + @CheckForNull + public Long getNanoCPUs() { + return nanoCPUs; + } + + /** + * @see #nanoCPUs + */ + public SwarmNodeResources withNanoCPUs(Long nanoCPUs) { + this.nanoCPUs = nanoCPUs; + return this; + } + + /** + * @see #memoryBytes + */ + @CheckForNull + public Long getMemoryBytes() { + return memoryBytes; + } + + /** + * @see #memoryBytes + */ + public SwarmNodeResources withMemoryBytes(Long memoryBytes) { + this.memoryBytes = memoryBytes; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeRole.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeRole.java new file mode 100644 index 000000000..224fd70f5 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeRole.java @@ -0,0 +1,15 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum SwarmNodeRole { + + @JsonProperty("worker") + WORKER, + + @JsonProperty("manager") + MANAGER +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeSpec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeSpec.java new file mode 100644 index 000000000..241c2be58 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeSpec.java @@ -0,0 +1,107 @@ +package com.github.dockerjava.api.model; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Map; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmNodeSpec extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Name") + private String name; + + /** + * @since 1.24 + */ + @JsonProperty("Role") + private SwarmNodeRole role; + + /** + * @since 1.24 + */ + @JsonProperty("Availability") + private SwarmNodeAvailability availability; + + /** + * @since 1.24 + */ + @JsonProperty("Labels") + public Map labels; + + /** + * @see #name + */ + @CheckForNull + public String getName() { + return name; + } + + /** + * @see #name + */ + public SwarmNodeSpec withName(String name) { + this.name = name; + return this; + } + + /** + * @see #role + */ + @CheckForNull + public SwarmNodeRole getRole() { + return role; + } + + /** + * @see #role + */ + public SwarmNodeSpec withRole(SwarmNodeRole role) { + this.role = role; + return this; + } + + /** + * @see #availability + */ + @CheckForNull + public SwarmNodeAvailability getAvailability() { + return availability; + } + + /** + * @see #availability + */ + public SwarmNodeSpec withAvailability(SwarmNodeAvailability availability) { + this.availability = availability; + return this; + } + + /** + * @see #labels + */ + @CheckForNull + public Map getLabels() { + return labels; + } + + /** + * @see #labels + */ + public SwarmNodeSpec withLabels(Map labels) { + this.labels = labels; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeState.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeState.java new file mode 100644 index 000000000..6a2776716 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeState.java @@ -0,0 +1,21 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum SwarmNodeState { + + @JsonProperty("unknown") + UNKNOWN, + + @JsonProperty("down") + DOWN, + + @JsonProperty("ready") + READY, + + @JsonProperty("disconnected") + DISCONNECTED +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeStatus.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeStatus.java new file mode 100644 index 000000000..34f40e80b --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeStatus.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmNodeStatus extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("State") + private SwarmNodeState state; + + /** + * @since 1.25 + */ + @JsonProperty("Addr") + private String address; + + /** + * @see #state + */ + @CheckForNull + public SwarmNodeState getState() { + return state; + } + + /** + * @see #state + */ + public SwarmNodeStatus withState(SwarmNodeState state) { + this.state = state; + return this; + } + + /** + * @see #address + */ + @CheckForNull + public String getAddress() { + return address; + } + + /** + * @see #address + */ + public SwarmNodeStatus withAddress(String address) { + this.address = address; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeVersion.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeVersion.java new file mode 100644 index 000000000..4182c120e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmNodeVersion.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.api.model; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmNodeVersion extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Index") + private long index; + + /** + * @see #index + */ + @CheckForNull + public long getIndex() { + return index; + } + + /** + * @see #index + */ + public SwarmNodeVersion withIndex(long index) { + this.index = index; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmOrchestration.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmOrchestration.java new file mode 100644 index 000000000..0479a3a6c --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmOrchestration.java @@ -0,0 +1,40 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmOrchestration extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("TaskHistoryRetentionLimit") + private int taskHistoryRententionLimit; + + /** + * @see #taskHistoryRententionLimit + */ + @CheckForNull + public int getTaskHistoryRententionLimit() { + return taskHistoryRententionLimit; + } + + /** + * @see #taskHistoryRententionLimit + */ + public SwarmOrchestration withTaskHistoryRententionLimit(int taskHistoryRententionLimit) { + this.taskHistoryRententionLimit = taskHistoryRententionLimit; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmRaftConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmRaftConfig.java new file mode 100644 index 000000000..69138ed2d --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmRaftConfig.java @@ -0,0 +1,106 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmRaftConfig extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("LogEntriesForSlowFollowers") + private long logEntriesForSlowFollowers; + + /** + * @since 1.24 + */ + @JsonProperty("HeartbeatTick") + private int heartbeatTick; + + /** + * @since 1.24 + */ + @JsonProperty("SnapshotInterval") + private long snapshotInterval; + + /** + * @since 1.24 + */ + @JsonProperty("ElectionTick") + private int electionTick; + + /** + * @see #logEntriesForSlowFollowers + */ + @CheckForNull + public long getLogEntriesForSlowFollowers() { + return logEntriesForSlowFollowers; + } + + /** + * @see #logEntriesForSlowFollowers + */ + public SwarmRaftConfig withLogEntriesForSlowFollowers(long logEntriesForSlowFollowers) { + this.logEntriesForSlowFollowers = logEntriesForSlowFollowers; + return this; + } + + /** + * @see #heartbeatTick + */ + @CheckForNull + public int getHeartbeatTick() { + return heartbeatTick; + } + + /** + * @see #heartbeatTick + */ + public SwarmRaftConfig withHeartbeatTick(int heartbeatTick) { + this.heartbeatTick = heartbeatTick; + return this; + } + + /** + * @see #snapshotInterval + */ + @CheckForNull + public long getSnapshotInterval() { + return snapshotInterval; + } + + /** + * @see #snapshotInterval + */ + public SwarmRaftConfig withSnapshotInterval(long snapshotInterval) { + this.snapshotInterval = snapshotInterval; + return this; + } + + /** + * @see #electionTick + */ + @CheckForNull + public int getElectionTick() { + return electionTick; + } + + /** + * @see #electionTick + */ + public SwarmRaftConfig withElectionTick(int electionTick) { + this.electionTick = electionTick; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmSpec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmSpec.java new file mode 100644 index 000000000..ee041a2a7 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmSpec.java @@ -0,0 +1,152 @@ +package com.github.dockerjava.api.model; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmSpec extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Dispatcher") + private SwarmDispatcherConfig dispatcher; + + /** + * @since 1.24 + */ + @JsonProperty("Orchestration") + private SwarmOrchestration orchestration; + + + /** + * @since 1.24 + */ + @JsonProperty("CAConfig") + private SwarmCAConfig caConfig; + + /** + * @since 1.24 + */ + @JsonProperty("Raft") + private SwarmRaftConfig raft; + + /** + * @since 1.24 + */ + @JsonProperty("TaskDefaults") + private TaskDefaults taskDefaults; + + /** + * @since 1.24 + */ + @JsonProperty("Name") + private String name; + + /** + * @see #dispatcher + */ + @CheckForNull + public SwarmDispatcherConfig getDispatcher() { + return dispatcher; + } + + /** + * @see #dispatcher + */ + public SwarmSpec withDispatcher(SwarmDispatcherConfig dispatcher) { + this.dispatcher = dispatcher; + return this; + } + + /** + * @see #orchestration + */ + @CheckForNull + public SwarmOrchestration getOrchestration() { + return orchestration; + } + + /** + * @see #orchestration + */ + public SwarmSpec withOrchestration(SwarmOrchestration orchestration) { + this.orchestration = orchestration; + return this; + } + + /** + * @see #caConfig + */ + @CheckForNull + public SwarmCAConfig getCaConfig() { + return caConfig; + } + + /** + * @see #caConfig + */ + public SwarmSpec withCaConfig(SwarmCAConfig caConfig) { + this.caConfig = caConfig; + return this; + } + + /** + * @see #raft + */ + @CheckForNull + public SwarmRaftConfig getRaft() { + return raft; + } + + /** + * @see #raft + */ + public SwarmSpec withRaft(SwarmRaftConfig raft) { + this.raft = raft; + return this; + } + + /** + * @see #taskDefaults + */ + @CheckForNull + public TaskDefaults getTaskDefaults() { + return taskDefaults; + } + + /** + * @see #taskDefaults + */ + public SwarmSpec withTaskDefaults(TaskDefaults taskDefaults) { + this.taskDefaults = taskDefaults; + return this; + } + + /** + * @see #name + */ + @CheckForNull + public String getName() { + return name; + } + + /** + * @see #name + */ + public SwarmSpec withName(String name) { + this.name = name; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmVersion.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmVersion.java new file mode 100644 index 000000000..161c0d7e5 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/SwarmVersion.java @@ -0,0 +1,40 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class SwarmVersion extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Index") + private long index; + + /** + * @see #index + */ + @CheckForNull + public long getIndex() { + return index; + } + + /** + * @see #index + */ + public SwarmVersion withIndex(long index) { + this.index = index; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Task.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Task.java new file mode 100644 index 000000000..0f1e77a2a --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Task.java @@ -0,0 +1,243 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.List; +import java.util.Map; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class Task extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("ID") + private String id = null; + + @JsonProperty("Version") + private ObjectVersion version = null; + + @JsonProperty("CreatedAt") + private String createdAt = null; + + @JsonProperty("UpdatedAt") + private String updatedAt = null; + + @JsonProperty("Name") + private String name = null; + + @JsonProperty("Labels") + private Map labels = null; + + @JsonProperty("Spec") + private TaskSpec spec = null; + + @JsonProperty("ServiceID") + private String serviceId = null; + + @JsonProperty("Slot") + private Integer slot = null; + + @JsonProperty("NodeID") + private String nodeId = null; + + @JsonProperty("AssignedGenericResources") + private List assignedGenericResources = null; + + @JsonProperty("Status") + private TaskStatus status = null; + + @JsonProperty("DesiredState") + private TaskState desiredState = null; + + /** + * The ID of the task. + * + * @return ID + **/ + public String getId() { + return id; + } + + /** + * Get version + * + * @return version + **/ + public ObjectVersion getVersion() { + return version; + } + + /** + * Get createdAt + * + * @return createdAt + **/ + public String getCreatedAt() { + return createdAt; + } + + /** + * Get updatedAt + * + * @return updatedAt + **/ + public String getUpdatedAt() { + return updatedAt; + } + + /** + * Name of the task. + * + * @return name + **/ + + public String getName() { + return name; + } + + /** + * User-defined key/value metadata. + * + * @return labels + **/ + public Map getLabels() { + return labels; + } + + /** + * Get spec + * + * @return spec + **/ + public TaskSpec getSpec() { + return spec; + } + + /** + * Get status + * + * @return status + **/ + public TaskStatus getStatus() { + return status; + } + + /** + * Get desiredState + * + * @return desiredState + **/ + public TaskState getDesiredState() { + return desiredState; + } + + /** + * The ID of the service this task is part of. + * + * @return serviceId + **/ + public String getServiceId() { + return serviceId; + } + + /** + * Get slot + * + * @return slot + **/ + public Integer getSlot() { + return slot; + } + + /** + * The ID of the node that this task is on. + * + * @return nodeId + **/ + public String getNodeId() { + return nodeId; + } + + public Task withId(String id) { + this.id = id; + return this; + } + + public Task withVersion(ObjectVersion version) { + this.version = version; + return this; + } + + public Task withCreatedAt(String createdAt) { + this.createdAt = createdAt; + return this; + } + + public Task withUpdatedAt(String updatedAt) { + this.updatedAt = updatedAt; + return this; + } + + public Task withName(String name) { + this.name = name; + return this; + } + + public Task withLabels(Map labels) { + this.labels = labels; + return this; + } + + public Task withSpec(TaskSpec spec) { + this.spec = spec; + return this; + } + + public Task withServiceId(String serviceId) { + this.serviceId = serviceId; + return this; + } + + public Task withSlot(Integer slot) { + this.slot = slot; + return this; + } + + public Task withNodeId(String nodeId) { + this.nodeId = nodeId; + return this; + } + + public Task withAssignedGenericResources(List assignedGenericResources) { + this.assignedGenericResources = assignedGenericResources; + return this; + } + + public Task withStatus(TaskStatus status) { + this.status = status; + return this; + } + + public Task withDesiredState(TaskState desiredState) { + this.desiredState = desiredState; + return this; + } + + /** + * Convert the given object to string with each line indented by 4 spaces + * (except the first line). + */ + private String toIndentedString(java.lang.Object o) { + if (o == null) { + return "null"; + } + return o.toString().replace("\n", "\n "); + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskDefaults.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskDefaults.java new file mode 100644 index 000000000..ae03c2136 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskDefaults.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.api.model; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class TaskDefaults extends DockerObject implements Serializable { + + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("LogDriver") + private Driver logDriver; + + /** + * @see #logDriver + */ + @CheckForNull + public Driver getLogDriver() { + return logDriver; + } + + /** + * @see #logDriver + */ + public TaskDefaults withLogDriver(Driver logDriver) { + this.logDriver = logDriver; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskSpec.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskSpec.java new file mode 100644 index 000000000..c60ca3b8d --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskSpec.java @@ -0,0 +1,182 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class TaskSpec extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("ContainerSpec") + private ContainerSpec containerSpec; + + /** + * @since 1.24 + */ + @JsonProperty("Resources") + private ResourceRequirements resources; + + /** + * @since 1.24 + */ + @JsonProperty("RestartPolicy") + private ServiceRestartPolicy restartPolicy; + + /** + * @since 1.24 + */ + @JsonProperty("Placement") + private ServicePlacement placement; + + /** + * @since 1.24 + */ + @JsonProperty("LogDriver") + private Driver logDriver; + + /** + * @since 1.24 + * a counter that triggers an update even if no relevant parameters have been changed + * a random value will work if it is incrementing. + */ + @JsonProperty("ForceUpdate") + private Integer forceUpdate; + + /** + * @since 1.25 + */ + @JsonProperty("Networks") + private List networks; + + /** + * @since 1.30 + */ + @JsonProperty("Runtime") + private String runtime; + + /** + * @see #containerSpec + */ + @CheckForNull + public ContainerSpec getContainerSpec() { + return containerSpec; + } + + /** + * @see #containerSpec + */ + public TaskSpec withContainerSpec(ContainerSpec containerSpec) { + this.containerSpec = containerSpec; + return this; + } + + /** + * @see #resources + */ + @CheckForNull + public ResourceRequirements getResources() { + return resources; + } + + /** + * @see #resources + */ + public TaskSpec withResources(ResourceRequirements resources) { + this.resources = resources; + return this; + } + + /** + * @see #restartPolicy + */ + @CheckForNull + public ServiceRestartPolicy getRestartPolicy() { + return restartPolicy; + } + + /** + * @see #restartPolicy + */ + public TaskSpec withRestartPolicy(ServiceRestartPolicy restartPolicy) { + this.restartPolicy = restartPolicy; + return this; + } + + /** + * @see #placement + */ + @CheckForNull + public ServicePlacement getPlacement() { + return placement; + } + + /** + * @see #placement + */ + public TaskSpec withPlacement(ServicePlacement placement) { + this.placement = placement; + return this; + } + + public String getRuntime() { + return runtime; + } + + public TaskSpec withRuntime(String runtime) { + this.runtime = runtime; + return this; + } + + /** + * @see #logDriver + */ + @CheckForNull + public Driver getLogDriver() { + return logDriver; + } + + /** + * @see #logDriver + */ + public TaskSpec withLogDriver(Driver logDriver) { + this.logDriver = logDriver; + return this; + } + + /** + * @see #forceUpdate + */ + @CheckForNull + public Integer getForceUpdate() { + return forceUpdate; + } + + /** + * @see #forceUpdate + */ + public TaskSpec withForceUpdate(Integer forceUpdate) { + this.forceUpdate = forceUpdate; + return this; + } + + public List getNetworks() { + return networks; + } + + public TaskSpec withNetworks(List networks) { + this.networks = networks; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskState.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskState.java new file mode 100644 index 000000000..0a2cb55d8 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskState.java @@ -0,0 +1,59 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonCreator; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum TaskState { + + NEW("new"), + + ALLOCATED("allocated"), + + PENDING("pending"), + + ASSIGNED("assigned"), + + ACCEPTED("accepted"), + + PREPARING("preparing"), + + READY("ready"), + + STARTING("starting"), + + RUNNING("running"), + + COMPLETE("complete"), + + SHUTDOWN("shutdown"), + + FAILED("failed"), + + REJECTED("rejected"), + + REMOVE("remove"), + + ORPHANED("orphaned"); + + private String value; + + TaskState(String value) { + this.value = value; + } + + public String getValue() { + return value; + } + + @JsonCreator + public static TaskState forValue(String s) { + return TaskState.valueOf(s.toUpperCase()); + } + + @Override + public String toString() { + return value; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskStatus.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskStatus.java new file mode 100644 index 000000000..9ae2a72ad --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskStatus.java @@ -0,0 +1,76 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class TaskStatus extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Timestamp") + private String timestamp = null; + + @JsonProperty("State") + private TaskState state = null; + + @JsonProperty("Message") + private String message = null; + + @JsonProperty("Err") + private String err = null; + + @JsonProperty("ContainerStatus") + private TaskStatusContainerStatus containerStatus = null; + + public String getTimestamp() { + return timestamp; + } + + public TaskStatus withTimestamp(String timestamp) { + this.timestamp = timestamp; + return this; + } + + public TaskState getState() { + return state; + } + + public TaskStatus withState(TaskState state) { + this.state = state; + return this; + } + + public String getMessage() { + return message; + } + + public TaskStatus withMessage(String message) { + this.message = message; + return this; + } + + public String getErr() { + return err; + } + + public TaskStatus withErr(String err) { + this.err = err; + return this; + } + + public TaskStatusContainerStatus getContainerStatus() { + return containerStatus; + } + + public TaskStatus withContainerStatus(TaskStatusContainerStatus containerStatus) { + this.containerStatus = containerStatus; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskStatusContainerStatus.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskStatusContainerStatus.java new file mode 100644 index 000000000..1f6f61d1d --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TaskStatusContainerStatus.java @@ -0,0 +1,89 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class TaskStatusContainerStatus extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("ContainerID") + private String containerID = null; + + @JsonProperty("PID") + private Long pid = null; + + @JsonProperty("ExitCode") + private Long exitCode = null; + + public String getContainerID() { + return containerID; + } + + /** + * + * @deprecated use {@link #getPidLong()} + */ + @Deprecated + public Integer getPid() { + return pid != null ? pid.intValue() : null; + } + + public Long getPidLong() { + return pid; + } + + /** + * @deprecated use {@link #getExitCodeLong()} + */ + @Deprecated + public Integer getExitCode() { + return exitCode != null ? exitCode.intValue() : null; + } + + public Long getExitCodeLong() { + return exitCode; + } + + public TaskStatusContainerStatus withContainerID(String containerID) { + this.containerID = containerID; + return this; + } + + public TaskStatusContainerStatus withPid(Long pid) { + this.pid = pid; + return this; + } + + /** + * + * @deprecated use {@link #withPid(Long)} + */ + @Deprecated + public TaskStatusContainerStatus withPid(Integer pid) { + this.pid = pid != null ? pid.longValue() : null; + return this; + } + + public TaskStatusContainerStatus withExitCode(Long exitCode) { + this.exitCode = exitCode; + return this; + } + + /** + * + * @deprecated use {@link #withExitCode(Long)} + */ + @Deprecated + public TaskStatusContainerStatus withExitCode(Integer exitCode) { + this.exitCode = exitCode != null ? exitCode.longValue() : null; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/ThrottlingDataConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ThrottlingDataConfig.java new file mode 100644 index 000000000..e908ce8de --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/ThrottlingDataConfig.java @@ -0,0 +1,52 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * Used in {@link Statistics} + * + * @author Yuting Liu + */ +@EqualsAndHashCode +@ToString +public class ThrottlingDataConfig extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("periods") + private Long periods; + + @JsonProperty("throttled_periods") + private Long throttledPeriods; + + @JsonProperty("throttled_time") + private Long throttledTime; + + /** + * @see #periods + */ + @CheckForNull + public Long getPeriods() { + return periods; + } + + /** + * @see #throttledPeriods + */ + @CheckForNull + public Long getThrottledPeriods() { + return throttledPeriods; + } + + /** + * @see #throttledTime + */ + @CheckForNull + public Long getThrottledTime() { + return throttledTime; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/TmpfsOptions.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TmpfsOptions.java new file mode 100644 index 000000000..e64adaac2 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/TmpfsOptions.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_29} + */ +@EqualsAndHashCode +@ToString +public class TmpfsOptions extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + @JsonProperty("SizeBytes") + //The size for the tmpfs mount in bytes. + private Long sizeBytes; + + //The permission mode for the tmpfs mount in an integer. + @JsonProperty("Mode") + private Integer mode; + + public Long getSizeBytes() { + return sizeBytes; + } + + public TmpfsOptions withSizeBytes(Long sizeBytes) { + this.sizeBytes = sizeBytes; + return this; + } + + public Integer getMode() { + return mode; + } + + public TmpfsOptions withMode(Integer mode) { + this.mode = mode; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Ulimit.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Ulimit.java new file mode 100644 index 000000000..24dbd764e --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Ulimit.java @@ -0,0 +1,71 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonIgnore; +import com.fasterxml.jackson.annotation.JsonProperty; +import com.fasterxml.jackson.annotation.JsonPropertyOrder; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +import static java.util.Objects.requireNonNull; + +/** + * @author Vangie Du (duwan@live.com) + */ +@JsonPropertyOrder({"Name", "Soft", "Hard"}) +@EqualsAndHashCode +@ToString +public class Ulimit extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("Name") + private String name; + + private Long soft; + + private Long hard; + + public Ulimit() { + } + + @Deprecated + public Ulimit(String name, int soft, int hard) { + this(name, (long) soft, (long) hard); + } + + @JsonCreator + public Ulimit(@JsonProperty("Name") String name, @JsonProperty("Soft") long soft, @JsonProperty("Hard") long hard) { + requireNonNull(name, "Name is null"); + this.name = name; + this.soft = soft; + this.hard = hard; + } + + public String getName() { + return name; + } + + @Deprecated + @JsonIgnore + public Integer getSoft() { + return soft != null ? soft.intValue() : null; + } + + @Deprecated + @JsonIgnore + public Integer getHard() { + return hard != null ? hard.intValue() : null; + } + + @JsonProperty("Soft") + public Long getSoftLong() { + return soft; + } + + @JsonProperty("Hard") + public Long getHardLong() { + return hard; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateConfig.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateConfig.java new file mode 100644 index 000000000..a70e137a9 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateConfig.java @@ -0,0 +1,139 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class UpdateConfig extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("Parallelism") + private Long parallelism; + + /** + * @since 1.24 + */ + @JsonProperty("Delay") + private Long delay; + + /** + * @since 1.24 + */ + @JsonProperty("FailureAction") + private UpdateFailureAction failureAction; + + /** + * @since 1.25 + */ + @JsonProperty("MaxFailureRatio") + private Float maxFailureRatio; + + /** + * @since 1.25 + */ + @JsonProperty("Monitor") + private Long monitor; + + /** + * @since 1.36 + */ + @JsonProperty("Order") + private UpdateOrder order; + + + /** + * @see #parallelism + */ + @CheckForNull + public Long getParallelism() { + return parallelism; + } + + /** + * @see #parallelism + */ + public UpdateConfig withParallelism(long parallelism) { + this.parallelism = parallelism; + return this; + } + + /** + * @see #delay + */ + @CheckForNull + public Long getDelay() { + return delay; + } + + /** + * @see #delay + */ + public UpdateConfig setDelay(Long delay) { + this.delay = delay; + return this; + } + + /** + * @see #failureAction + */ + @CheckForNull + public UpdateFailureAction getFailureAction() { + return failureAction; + } + + /** + * @see #failureAction + */ + public UpdateConfig withFailureAction(UpdateFailureAction failureAction) { + this.failureAction = failureAction; + return this; + } + + public UpdateConfig withParallelism(Long parallelism) { + this.parallelism = parallelism; + return this; + } + + public UpdateConfig withDelay(Long delay) { + this.delay = delay; + return this; + } + + public Float getMaxFailureRatio() { + return maxFailureRatio; + } + + public UpdateConfig withMaxFailureRatio(Float maxFailureRatio) { + this.maxFailureRatio = maxFailureRatio; + return this; + } + + public Long getMonitor() { + return monitor; + } + + public UpdateConfig withMonitor(Long monitor) { + this.monitor = monitor; + return this; + } + + public UpdateOrder getOrder() { + return order; + } + + public UpdateConfig withOrder(UpdateOrder order) { + this.order = order; + return this; + } +} diff --git a/src/main/java/com/github/dockerjava/api/model/UpdateContainerResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateContainerResponse.java similarity index 82% rename from src/main/java/com/github/dockerjava/api/model/UpdateContainerResponse.java rename to docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateContainerResponse.java index b1b1188d0..5b7b849b5 100644 --- a/src/main/java/com/github/dockerjava/api/model/UpdateContainerResponse.java +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateContainerResponse.java @@ -1,8 +1,6 @@ package com.github.dockerjava.api.model; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.core.RemoteApiVersion; import javax.annotation.CheckForNull; import java.util.List; @@ -14,7 +12,6 @@ * https://docs.docker.com/engine/reference/api/docker_remote_api_v1.22/ * @since {@link RemoteApiVersion#VERSION_1_22} */ -@JsonIgnoreProperties(ignoreUnknown = true) public class UpdateContainerResponse extends ResponseItem { private static final long serialVersionUID = 1L; diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateFailureAction.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateFailureAction.java new file mode 100644 index 000000000..f4b1fb009 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateFailureAction.java @@ -0,0 +1,19 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public enum UpdateFailureAction { + + @JsonProperty("pause") + PAUSE, + + @JsonProperty("continue") + CONTINUE, + + @JsonProperty("rollback") + ROLLBACK + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateOrder.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateOrder.java new file mode 100644 index 000000000..3f74388de --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/UpdateOrder.java @@ -0,0 +1,14 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +/** + * @since {@link RemoteApiVersion#VERSION_1_36} + */ +public enum UpdateOrder { + @JsonProperty("stop-first") + STOP_FIRST, + + @JsonProperty("start-first") + START_FIRST +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Version.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Version.java new file mode 100644 index 000000000..1a05726fc --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Version.java @@ -0,0 +1,141 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.command.VersionCmd; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.List; + +/** + * Used for `/version` + * + * @author Konstantin Pelykh (kpelykh@gmail.com) + * @see VersionCmd + */ +@EqualsAndHashCode +@ToString +public class Version extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("ApiVersion") + private String apiVersion; + + @JsonProperty("Arch") + private String arch; + + @JsonProperty("GitCommit") + private String gitCommit; + + @JsonProperty("GoVersion") + private String goVersion; + + @JsonProperty("KernelVersion") + private String kernelVersion; + + @JsonProperty("Os") + private String operatingSystem; + + @JsonProperty("Version") + private String version; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} + */ + @JsonProperty("BuildTime") + private String buildTime; + + /** + * @since ~{@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_20} + */ + @JsonProperty("Experimental") + private Boolean experimental; + + /** + * @since ~{@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_25} + */ + @JsonProperty("MinAPIVersion") + private String minAPIVersion; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_35} + */ + @JsonProperty("Platform") + private VersionPlatform platform; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_35} + */ + @JsonProperty("Components") + private List components; + + public String getVersion() { + return version; + } + + public String getGitCommit() { + return gitCommit; + } + + public String getGoVersion() { + return goVersion; + } + + public String getKernelVersion() { + return kernelVersion; + } + + public String getArch() { + return arch; + } + + public String getOperatingSystem() { + return operatingSystem; + } + + public String getApiVersion() { + return apiVersion; + } + + /** + * @see #buildTime + */ + @CheckForNull + public String getBuildTime() { + return buildTime; + } + + /** + * @see #experimental + */ + @CheckForNull + public Boolean getExperimental() { + return experimental; + } + + /** + * @see #minAPIVersion + */ + @CheckForNull + public String getMinAPIVersion() { + return minAPIVersion; + } + + /** + * @see #platform + */ + @CheckForNull + public VersionPlatform getPlatform() { + return platform; + } + + /** + * @see #components + */ + @CheckForNull + public List getComponents() { + return components; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/VersionComponent.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VersionComponent.java new file mode 100644 index 000000000..9a9fb9071 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VersionComponent.java @@ -0,0 +1,78 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; +import java.util.Map; + +/** + * Part of {@link Version} + * + * @since {@link RemoteApiVersion#VERSION_1_35} + * @author Dmitry Tretyakov + */ +@EqualsAndHashCode +@ToString +public class VersionComponent extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + @JsonProperty("Details") + private Map details; + + @JsonProperty("Name") + private String name; + + @JsonProperty("Version") + private String version; + + /** + * @see #details + */ + @CheckForNull + public Map getDetails() { + return details; + } + + /** + * @see #details + */ + public VersionComponent withDetails(Map details) { + this.details = details; + return this; + } + + /** + * @see #name + */ + @CheckForNull + public String getName() { + return name; + } + + /** + * @see #name + */ + public VersionComponent withName(String name) { + this.name = name; + return this; + } + + /** + * @see #version + */ + @CheckForNull + public String getVersion() { + return version; + } + + /** + * @see #version + */ + public VersionComponent withVersion(String version) { + this.version = version; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/VersionPlatform.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VersionPlatform.java new file mode 100644 index 000000000..72d29a44b --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VersionPlatform.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.annotation.CheckForNull; +import java.io.Serializable; + +/** + * Part of {@link Version} + * + * @since {@link RemoteApiVersion#VERSION_1_35} + * @author Dmitry Tretyakov + */ +@EqualsAndHashCode +@ToString +public class VersionPlatform extends DockerObject implements Serializable { + public static final long serialVersionUID = 1L; + + @JsonProperty("Name") + private String name; + + /** + * @see #name + */ + @CheckForNull + public String getName() { + return name; + } + + /** + * @see #name + */ + public VersionPlatform withName(String name) { + this.name = name; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Volume.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Volume.java new file mode 100644 index 000000000..bc511476f --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Volume.java @@ -0,0 +1,71 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonProperty; +import com.fasterxml.jackson.annotation.JsonValue; +import lombok.EqualsAndHashCode; + +import javax.annotation.Nonnull; +import java.io.Serializable; +import java.util.Map; + +/** + * Represents a bind mounted volume in a Docker container. + *

+ * Due to an inconsistency in the Docker REST API implementation the response to a container + * command might include either {@code "Mounts" : [ { "Destination" : "/path/to/mount" } ]} or + * {@code "Mounts" : [ { "Destination" : { "path" : "/path/to/mount" } } ]} JSON snippets. However, + * both variants have to be mapped to this class. Therefore, both a single-argument constructor + * as well as a single-argument factory method is provided either of which handles the former or + * latter variant (with {@code path} key), respectively. + * + * @see Bind + */ +@EqualsAndHashCode +public class Volume implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * Handles the {@code { "Destination" : { "path" : "/path/to/mount" } }} variant. + * @param path the destination path of the bind mounted volume + * @return a volume instance referring to the given path. + * @deprecated use {@link #parse(Map)} + */ + @Nonnull + @Deprecated + public static Volume parse(@JsonProperty("path") String path) { + return new Volume(path); + } + + /** + * Handles the {@code { "Destination" : { "path" : "/path/to/mount" } }} variant. + * @param path the destination path of the bind mounted volume + * @return a volume instance referring to the given path. + */ + @Nonnull + @JsonCreator + public static Volume parse(Map primitive) { + return new Volume(primitive.get("path")); + } + + private String path; + + /** + * Creates a volume referring to the given path. + * Handles the {@code { "Destination" : "/path/to/mount" }} variant. + * @param path the destination path of the bind mounted volume + */ + public Volume(String path) { + this.path = path; + } + + public String getPath() { + return path; + } + + @Override + @JsonValue + public String toString() { + return getPath(); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeBind.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeBind.java new file mode 100644 index 000000000..93c106070 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeBind.java @@ -0,0 +1,32 @@ +package com.github.dockerjava.api.model; + +import lombok.EqualsAndHashCode; + +import java.io.Serializable; + +@EqualsAndHashCode +public class VolumeBind extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + private final String hostPath; + + private final String containerPath; + + public VolumeBind(String hostPath, String containerPath) { + this.hostPath = hostPath; + this.containerPath = containerPath; + } + + public String getContainerPath() { + return containerPath; + } + + public String getHostPath() { + return hostPath; + } + + @Override + public String toString() { + return hostPath + ":" + containerPath; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeBinds.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeBinds.java new file mode 100644 index 000000000..1fbc0ca12 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeBinds.java @@ -0,0 +1,43 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; +import java.util.Map; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; +import lombok.ToString; + +@ToString +public class VolumeBinds implements Serializable { + private static final long serialVersionUID = 1L; + + private final VolumeBind[] binds; + + public VolumeBinds(VolumeBind... binds) { + this.binds = binds; + } + + public VolumeBind[] getBinds() { + return binds; + } + + @JsonCreator + public static VolumeBinds fromPrimitive(Map primitive) { + return new VolumeBinds( + primitive.entrySet().stream() + .map(it -> new VolumeBind(it.getValue(), it.getKey())) + .toArray(VolumeBind[]::new) + ); + } + + @JsonValue + public Map toPrimitive() { + return Stream.of(binds).collect(Collectors.toMap( + VolumeBind::getContainerPath, + VolumeBind::getHostPath + )); + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeOptions.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeOptions.java new file mode 100644 index 000000000..740ffcd51 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeOptions.java @@ -0,0 +1,80 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; +import java.util.Map; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +@EqualsAndHashCode +@ToString +public class VolumeOptions extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * @since 1.24 + */ + @JsonProperty("NoCopy") + private Boolean noCopy; + + /** + * @since 1.24 + */ + @JsonProperty("Labels") + private Map labels; + + /** + * @since 1.24 + */ + @JsonProperty("DriverConfig") + private Driver driverConfig; + + /** + * @see #noCopy + */ + public Boolean getNoCopy() { + return noCopy; + } + + /** + * @see #noCopy + */ + public VolumeOptions withNoCopy(Boolean noCopy) { + this.noCopy = noCopy; + return this; + } + + /** + * @see #labels + */ + public Map getLabels() { + return labels; + } + + /** + * @see #labels + */ + public VolumeOptions withLabels(Map labels) { + this.labels = labels; + return this; + } + + /** + * @see #driverConfig + */ + public Driver getDriverConfig() { + return driverConfig; + } + + /** + * @see #driverConfig + */ + public VolumeOptions withDriverConfig(Driver driverConfig) { + this.driverConfig = driverConfig; + return this; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeRW.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeRW.java new file mode 100644 index 000000000..66ba57ced --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumeRW.java @@ -0,0 +1,66 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; +import java.util.Collections; +import java.util.Map; +import java.util.Map.Entry; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; +import lombok.EqualsAndHashCode; + +/** + * Represents a bind mounted volume in a Docker container. + * + * @see Bind + * @deprecated since {@link RemoteApiVersion#VERSION_1_20} + */ +@Deprecated +@EqualsAndHashCode +public class VolumeRW implements Serializable { + private static final long serialVersionUID = 1L; + + private Volume volume; + + private AccessMode accessMode = AccessMode.rw; + + public VolumeRW(Volume volume) { + this.volume = volume; + } + + public VolumeRW(Volume volume, AccessMode accessMode) { + this.volume = volume; + this.accessMode = accessMode; + } + + public Volume getVolume() { + return volume; + } + + public AccessMode getAccessMode() { + return accessMode; + } + + /** + * Returns a string representation of this {@link VolumeRW} suitable for inclusion in a JSON message. The returned String is simply the + * container path, {@link #getPath()}. + * + * @return a string representation of this {@link VolumeRW} + */ + @Override + public String toString() { + return getVolume() + ":" + getAccessMode(); + } + + @JsonCreator + public static VolumeRW fromPrimitive(Map map) { + Entry entry = map.entrySet().iterator().next(); + return new VolumeRW(new Volume(entry.getKey()), AccessMode.fromBoolean(entry.getValue())); + } + + @JsonValue + public Map toPrimitive() { + return Collections.singletonMap(volume.getPath(), accessMode.toBoolean()); + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/Volumes.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Volumes.java new file mode 100644 index 000000000..825b8481a --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/Volumes.java @@ -0,0 +1,46 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; +import java.util.List; +import java.util.Map; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; +import lombok.ToString; + +@ToString +public class Volumes implements Serializable { + private static final long serialVersionUID = 1L; + + private Volume[] volumes; + + public Volumes(Volume... volumes) { + this.volumes = volumes; + } + + public Volumes(List volumes) { + this.volumes = volumes.toArray(new Volume[volumes.size()]); + } + + public Volume[] getVolumes() { + return volumes; + } + + @JsonCreator + public static Volumes fromPrimitive(Map map) { + return new Volumes( + map.keySet().stream().map(Volume::new).toArray(Volume[]::new) + ); + } + + @JsonValue + public Map toPrimitive() { + return Stream.of(volumes).collect(Collectors.toMap( + Volume::getPath, + __ -> new Object() + )); + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumesFrom.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumesFrom.java new file mode 100644 index 000000000..98165afa9 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumesFrom.java @@ -0,0 +1,76 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; +import lombok.EqualsAndHashCode; + +@EqualsAndHashCode +public class VolumesFrom implements Serializable { + private static final long serialVersionUID = 1L; + + private String container; + + private AccessMode accessMode; + + public VolumesFrom(String container) { + this(container, AccessMode.DEFAULT); + } + + public VolumesFrom(String container, AccessMode accessMode) { + this.container = container; + this.accessMode = accessMode; + } + + public String getContainer() { + return container; + } + + public AccessMode getAccessMode() { + return accessMode; + } + + /** + * Parses a volume from specification to a {@link VolumesFrom}. + * + * @param serialized + * the specification, e.g. container:ro + * @return a {@link VolumesFrom} matching the specification + * @throws IllegalArgumentException + * if the specification cannot be parsed + */ + @JsonCreator + public static VolumesFrom parse(String serialized) { + try { + String[] parts = serialized.split(":"); + switch (parts.length) { + case 1: { + return new VolumesFrom(parts[0]); + } + case 2: { + return new VolumesFrom(parts[0], AccessMode.valueOf(parts[1])); + } + + default: { + throw new IllegalArgumentException(); + } + } + } catch (Exception e) { + throw new IllegalArgumentException("Error parsing Bind '" + serialized + "'"); + } + } + + /** + * Returns a string representation of this {@link VolumesFrom} suitable for inclusion in a JSON message. The format is + * <container>:<access mode>, like the argument in {@link #parse(String)}. + * + * @return a string representation of this {@link VolumesFrom} + */ + @Override + @JsonValue + public String toString() { + return container + ":" + accessMode.toString(); + } + +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumesRW.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumesRW.java new file mode 100644 index 000000000..93e95b68c --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/VolumesRW.java @@ -0,0 +1,43 @@ +package com.github.dockerjava.api.model; + +import java.io.Serializable; +import java.util.Map; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonValue; +import lombok.ToString; + +// This is not going to be serialized +@ToString +public class VolumesRW implements Serializable { + private static final long serialVersionUID = 1L; + + private final VolumeRW[] volumesRW; + + public VolumesRW(VolumeRW... binds) { + this.volumesRW = binds; + } + + public VolumeRW[] getVolumesRW() { + return volumesRW; + } + + @JsonCreator + public static VolumesRW fromPrimitive(Map map) { + return new VolumesRW( + map.entrySet().stream() + .map(entry -> new VolumeRW(new Volume(entry.getKey()), AccessMode.fromBoolean(entry.getValue()))) + .toArray(VolumeRW[]::new) + ); + } + + @JsonValue + public Map toPrimitive() { + return Stream.of(volumesRW).collect(Collectors.toMap( + it -> it.getVolume().getPath(), + it -> it.getAccessMode().toBoolean() + )); + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/WaitContainerCondition.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/WaitContainerCondition.java new file mode 100644 index 000000000..8af0efa35 --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/WaitContainerCondition.java @@ -0,0 +1,26 @@ +package com.github.dockerjava.api.model; + +import javax.annotation.Nonnull; + +/** + * Docker Engine API wait conditions (added in v1.30). + * + * @since {@link RemoteApiVersion#VERSION_1_30} + */ +public enum WaitContainerCondition { + NOT_RUNNING("not-running"), + NEXT_EXIT("next-exit"), + REMOVED("removed"); + + @Nonnull + private final String value; + + WaitContainerCondition(@Nonnull String value) { + this.value = value; + } + + @Nonnull + public String getValue() { + return value; + } +} diff --git a/docker-java-api/src/main/java/com/github/dockerjava/api/model/WaitResponse.java b/docker-java-api/src/main/java/com/github/dockerjava/api/model/WaitResponse.java new file mode 100644 index 000000000..eed22870f --- /dev/null +++ b/docker-java-api/src/main/java/com/github/dockerjava/api/model/WaitResponse.java @@ -0,0 +1,23 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.io.Serializable; + +/** + * Represents a wait container command response + */ +@EqualsAndHashCode +@ToString +public class WaitResponse extends DockerObject implements Serializable { + private static final long serialVersionUID = 1L; + + @JsonProperty("StatusCode") + private Integer statusCode; + + public Integer getStatusCode() { + return statusCode; + } +} diff --git a/docker-java-api/src/test/java/com/github/dockerjava/api/model/DockerObjectArchTest.java b/docker-java-api/src/test/java/com/github/dockerjava/api/model/DockerObjectArchTest.java new file mode 100644 index 000000000..2df7051e9 --- /dev/null +++ b/docker-java-api/src/test/java/com/github/dockerjava/api/model/DockerObjectArchTest.java @@ -0,0 +1,46 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.CreateConfigResponse; +import com.github.dockerjava.api.command.DockerCmdExecFactory; +import com.tngtech.archunit.base.DescribedPredicate; +import com.tngtech.archunit.core.domain.JavaClass; +import com.tngtech.archunit.core.domain.JavaClasses; +import com.tngtech.archunit.core.importer.ClassFileImporter; +import com.tngtech.archunit.core.importer.ImportOption; +import org.junit.jupiter.api.Test; + +import static com.tngtech.archunit.lang.syntax.ArchRuleDefinition.classes; + +class DockerObjectArchTest { + + static JavaClasses CLASSES = new ClassFileImporter() + .withImportOption(ImportOption.Predefined.DO_NOT_INCLUDE_TESTS) + .withImportOption(ImportOption.Predefined.DO_NOT_INCLUDE_JARS) + .importPackagesOf( + Container.class, + CreateConfigResponse.class + ); + + @Test + void modelClassMustExtendDockerObject() { + classes() + .that().areNotEnums() + .and().areNotInterfaces() + .and().areNotAnnotatedWith(Deprecated.class) + .and().doNotImplement(ResultCallback.class) + .and().doNotImplement(DockerCmdExecFactory.class) + .and().doNotBelongToAnyOf(DockerObjectAccessor.class) + .and(new DescribedPredicate("not @JsonCreator-based object") { + @Override + public boolean apply(JavaClass input) { + return input.getAllMethods().stream().noneMatch(method -> { + return method.isAnnotatedWith(JsonCreator.class); + }); + } + }) + .should().beAssignableTo(DockerObject.class) + .check(CLASSES); + } +} diff --git a/docker-java-bom/pom.xml b/docker-java-bom/pom.xml new file mode 100644 index 000000000..7066b3a67 --- /dev/null +++ b/docker-java-bom/pom.xml @@ -0,0 +1,67 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java-bom + pom + + docker-java + https://github.com/docker-java/docker-java + Java API Client for Docker + + + + + ${project.groupId} + docker-java + ${project.version} + + + ${project.groupId} + docker-java-api + ${project.version} + + + ${project.groupId} + docker-java-core + ${project.version} + + + ${project.groupId} + docker-java-transport + ${project.version} + + + ${project.groupId} + docker-java-transport-httpclient5 + ${project.version} + + + ${project.groupId} + docker-java-transport-jersey + ${project.version} + + + ${project.groupId} + docker-java-transport-netty + ${project.version} + + + ${project.groupId} + docker-java-transport-okhttp + ${project.version} + + + ${project.groupId} + docker-java-transport-zerodep + ${project.version} + + + + diff --git a/docker-java-core/pom.xml b/docker-java-core/pom.xml new file mode 100644 index 000000000..89c72c024 --- /dev/null +++ b/docker-java-core/pom.xml @@ -0,0 +1,107 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java-core + jar + + docker-java-core + https://github.com/docker-java/docker-java + Java API Client for Docker + + + com.github.dockerjava.core + + + + + ${project.groupId} + docker-java-api + ${project.version} + + + ${project.groupId} + docker-java-transport + ${project.version} + + + + org.slf4j + slf4j-api + ${slf4j-api.version} + + + + commons-io + commons-io + ${commons-io.version} + + + + org.apache.commons + commons-compress + ${commons-compress.version} + + + + org.apache.commons + commons-lang3 + ${commons-lang3.version} + + + + com.fasterxml.jackson.core + jackson-databind + 2.20.1 + + + + com.google.guava + guava + ${guava.version} + + + + org.bouncycastle + bcpkix-jdk18on + ${bouncycastle.version} + + + + com.google.code.findbugs + annotations + 3.0.1u2 + provided + + + + + + + org.apache.felix + maven-bundle-plugin + true + + + com.github.dockerjava.core.* + + + + + org.apache.maven.plugins + maven-compiler-plugin + + 8 + 8 + true + + + + + diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/AbstractDockerCmdExecFactory.java b/docker-java-core/src/main/java/com/github/dockerjava/core/AbstractDockerCmdExecFactory.java new file mode 100644 index 000000000..e04ab8e3e --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/AbstractDockerCmdExecFactory.java @@ -0,0 +1,599 @@ +package com.github.dockerjava.core; + +import java.util.Objects; + +import com.github.dockerjava.api.command.AttachContainerCmd; +import com.github.dockerjava.api.command.AuthCmd; +import com.github.dockerjava.api.command.BuildImageCmd; +import com.github.dockerjava.api.command.CommitCmd; +import com.github.dockerjava.api.command.ConnectToNetworkCmd; +import com.github.dockerjava.api.command.ContainerDiffCmd; +import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; +import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; +import com.github.dockerjava.api.command.CopyFileFromContainerCmd; +import com.github.dockerjava.api.command.CreateConfigCmd; +import com.github.dockerjava.api.command.CreateContainerCmd; +import com.github.dockerjava.api.command.CreateImageCmd; +import com.github.dockerjava.api.command.CreateNetworkCmd; +import com.github.dockerjava.api.command.CreateSecretCmd; +import com.github.dockerjava.api.command.CreateServiceCmd; +import com.github.dockerjava.api.command.CreateVolumeCmd; +import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; +import com.github.dockerjava.api.command.DockerCmdExecFactory; +import com.github.dockerjava.api.command.EventsCmd; +import com.github.dockerjava.api.command.ExecCreateCmd; +import com.github.dockerjava.api.command.ExecStartCmd; +import com.github.dockerjava.api.command.InfoCmd; +import com.github.dockerjava.api.command.InitializeSwarmCmd; +import com.github.dockerjava.api.command.InspectConfigCmd; +import com.github.dockerjava.api.command.InspectContainerCmd; +import com.github.dockerjava.api.command.InspectExecCmd; +import com.github.dockerjava.api.command.InspectImageCmd; +import com.github.dockerjava.api.command.InspectNetworkCmd; +import com.github.dockerjava.api.command.InspectServiceCmd; +import com.github.dockerjava.api.command.InspectSwarmCmd; +import com.github.dockerjava.api.command.InspectSwarmNodeCmd; +import com.github.dockerjava.api.command.InspectVolumeCmd; +import com.github.dockerjava.api.command.JoinSwarmCmd; +import com.github.dockerjava.api.command.KillContainerCmd; +import com.github.dockerjava.api.command.LeaveSwarmCmd; +import com.github.dockerjava.api.command.ListConfigsCmd; +import com.github.dockerjava.api.command.ListContainersCmd; +import com.github.dockerjava.api.command.ListImagesCmd; +import com.github.dockerjava.api.command.ListNetworksCmd; +import com.github.dockerjava.api.command.ListSecretsCmd; +import com.github.dockerjava.api.command.ListServicesCmd; +import com.github.dockerjava.api.command.ListSwarmNodesCmd; +import com.github.dockerjava.api.command.ListTasksCmd; +import com.github.dockerjava.api.command.ListVolumesCmd; +import com.github.dockerjava.api.command.LoadImageAsyncCmd; +import com.github.dockerjava.api.command.LoadImageCmd; +import com.github.dockerjava.api.command.LogContainerCmd; +import com.github.dockerjava.api.command.LogSwarmObjectCmd; +import com.github.dockerjava.api.command.PauseContainerCmd; +import com.github.dockerjava.api.command.PingCmd; +import com.github.dockerjava.api.command.PruneCmd; +import com.github.dockerjava.api.command.PullImageCmd; +import com.github.dockerjava.api.command.PushImageCmd; +import com.github.dockerjava.api.command.RemoveConfigCmd; +import com.github.dockerjava.api.command.RemoveContainerCmd; +import com.github.dockerjava.api.command.RemoveImageCmd; +import com.github.dockerjava.api.command.RemoveNetworkCmd; +import com.github.dockerjava.api.command.RemoveSecretCmd; +import com.github.dockerjava.api.command.RemoveServiceCmd; +import com.github.dockerjava.api.command.RemoveSwarmNodeCmd; +import com.github.dockerjava.api.command.RemoveVolumeCmd; +import com.github.dockerjava.api.command.RenameContainerCmd; +import com.github.dockerjava.api.command.ResizeContainerCmd; +import com.github.dockerjava.api.command.ResizeExecCmd; +import com.github.dockerjava.api.command.RestartContainerCmd; +import com.github.dockerjava.api.command.SaveImageCmd; +import com.github.dockerjava.api.command.SaveImagesCmd; +import com.github.dockerjava.api.command.SearchImagesCmd; +import com.github.dockerjava.api.command.StartContainerCmd; +import com.github.dockerjava.api.command.StatsCmd; +import com.github.dockerjava.api.command.StopContainerCmd; +import com.github.dockerjava.api.command.TagImageCmd; +import com.github.dockerjava.api.command.TopContainerCmd; +import com.github.dockerjava.api.command.UnpauseContainerCmd; +import com.github.dockerjava.api.command.UpdateContainerCmd; +import com.github.dockerjava.api.command.UpdateServiceCmd; +import com.github.dockerjava.api.command.UpdateSwarmCmd; +import com.github.dockerjava.api.command.UpdateSwarmNodeCmd; +import com.github.dockerjava.api.command.VersionCmd; +import com.github.dockerjava.api.command.WaitContainerCmd; +import com.github.dockerjava.core.exec.AttachContainerCmdExec; +import com.github.dockerjava.core.exec.AuthCmdExec; +import com.github.dockerjava.core.exec.BuildImageCmdExec; +import com.github.dockerjava.core.exec.CommitCmdExec; +import com.github.dockerjava.core.exec.ConnectToNetworkCmdExec; +import com.github.dockerjava.core.exec.ContainerDiffCmdExec; +import com.github.dockerjava.core.exec.CopyArchiveFromContainerCmdExec; +import com.github.dockerjava.core.exec.CopyArchiveToContainerCmdExec; +import com.github.dockerjava.core.exec.CopyFileFromContainerCmdExec; +import com.github.dockerjava.core.exec.CreateConfigCmdExec; +import com.github.dockerjava.core.exec.CreateContainerCmdExec; +import com.github.dockerjava.core.exec.CreateImageCmdExec; +import com.github.dockerjava.core.exec.CreateNetworkCmdExec; +import com.github.dockerjava.core.exec.CreateSecretCmdExec; +import com.github.dockerjava.core.exec.CreateServiceCmdExec; +import com.github.dockerjava.core.exec.CreateVolumeCmdExec; +import com.github.dockerjava.core.exec.DisconnectFromNetworkCmdExec; +import com.github.dockerjava.core.exec.EventsCmdExec; +import com.github.dockerjava.core.exec.ExecCreateCmdExec; +import com.github.dockerjava.core.exec.ExecStartCmdExec; +import com.github.dockerjava.core.exec.InspectConfigCmdExec; +import com.github.dockerjava.core.exec.ListConfigsCmdExec; +import com.github.dockerjava.core.exec.LoadImageAsyncCmdExec; +import com.github.dockerjava.core.exec.RemoveConfigCmdExec; +import com.github.dockerjava.core.exec.ResizeContainerCmdExec; +import com.github.dockerjava.core.exec.ResizeExecCmdExec; +import com.github.dockerjava.core.exec.InfoCmdExec; +import com.github.dockerjava.core.exec.InitializeSwarmCmdExec; +import com.github.dockerjava.core.exec.InspectContainerCmdExec; +import com.github.dockerjava.core.exec.InspectExecCmdExec; +import com.github.dockerjava.core.exec.InspectImageCmdExec; +import com.github.dockerjava.core.exec.InspectNetworkCmdExec; +import com.github.dockerjava.core.exec.InspectServiceCmdExec; +import com.github.dockerjava.core.exec.InspectSwarmCmdExec; +import com.github.dockerjava.core.exec.InspectSwarmNodeCmdExec; +import com.github.dockerjava.core.exec.InspectVolumeCmdExec; +import com.github.dockerjava.core.exec.JoinSwarmCmdExec; +import com.github.dockerjava.core.exec.KillContainerCmdExec; +import com.github.dockerjava.core.exec.LeaveSwarmCmdExec; +import com.github.dockerjava.core.exec.ListContainersCmdExec; +import com.github.dockerjava.core.exec.ListImagesCmdExec; +import com.github.dockerjava.core.exec.ListNetworksCmdExec; +import com.github.dockerjava.core.exec.ListSecretsCmdExec; +import com.github.dockerjava.core.exec.ListServicesCmdExec; +import com.github.dockerjava.core.exec.ListSwarmNodesCmdExec; +import com.github.dockerjava.core.exec.ListTasksCmdExec; +import com.github.dockerjava.core.exec.ListVolumesCmdExec; +import com.github.dockerjava.core.exec.LoadImageCmdExec; +import com.github.dockerjava.core.exec.LogContainerCmdExec; +import com.github.dockerjava.core.exec.LogSwarmObjectExec; +import com.github.dockerjava.core.exec.PauseContainerCmdExec; +import com.github.dockerjava.core.exec.PingCmdExec; +import com.github.dockerjava.core.exec.PruneCmdExec; +import com.github.dockerjava.core.exec.PullImageCmdExec; +import com.github.dockerjava.core.exec.PushImageCmdExec; +import com.github.dockerjava.core.exec.RemoveContainerCmdExec; +import com.github.dockerjava.core.exec.RemoveImageCmdExec; +import com.github.dockerjava.core.exec.RemoveNetworkCmdExec; +import com.github.dockerjava.core.exec.RemoveSecretCmdExec; +import com.github.dockerjava.core.exec.RemoveServiceCmdExec; +import com.github.dockerjava.core.exec.RemoveSwarmNodeCmdExec; +import com.github.dockerjava.core.exec.RemoveVolumeCmdExec; +import com.github.dockerjava.core.exec.RenameContainerCmdExec; +import com.github.dockerjava.core.exec.RestartContainerCmdExec; +import com.github.dockerjava.core.exec.SaveImageCmdExec; +import com.github.dockerjava.core.exec.SaveImagesCmdExec; +import com.github.dockerjava.core.exec.SearchImagesCmdExec; +import com.github.dockerjava.core.exec.StartContainerCmdExec; +import com.github.dockerjava.core.exec.StatsCmdExec; +import com.github.dockerjava.core.exec.StopContainerCmdExec; +import com.github.dockerjava.core.exec.TagImageCmdExec; +import com.github.dockerjava.core.exec.TopContainerCmdExec; +import com.github.dockerjava.core.exec.UnpauseContainerCmdExec; +import com.github.dockerjava.core.exec.UpdateContainerCmdExec; +import com.github.dockerjava.core.exec.UpdateServiceCmdExec; +import com.github.dockerjava.core.exec.UpdateSwarmCmdExec; +import com.github.dockerjava.core.exec.UpdateSwarmNodeCmdExec; +import com.github.dockerjava.core.exec.VersionCmdExec; +import com.github.dockerjava.core.exec.WaitContainerCmdExec; + +public abstract class AbstractDockerCmdExecFactory implements DockerCmdExecFactory, DockerClientConfigAware { + + private DockerClientConfig dockerClientConfig; + + protected Integer connectTimeout; + protected Integer readTimeout; + + protected DockerClientConfig getDockerClientConfig() { + Objects.requireNonNull(dockerClientConfig, + "Factor not initialized, dockerClientConfig not set. You probably forgot to call init()!"); + return dockerClientConfig; + } + + @Override + public void init(DockerClientConfig dockerClientConfig) { + this.dockerClientConfig = Objects.requireNonNull(dockerClientConfig, "config was not specified"); + } + + @Override + public CopyArchiveFromContainerCmd.Exec createCopyArchiveFromContainerCmdExec() { + return new CopyArchiveFromContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public CopyArchiveToContainerCmd.Exec createCopyArchiveToContainerCmdExec() { + return new CopyArchiveToContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + /** + * Configure connection timeout in milliseconds + */ + public AbstractDockerCmdExecFactory withConnectTimeout(Integer connectTimeout) { + this.connectTimeout = connectTimeout; + return this; + } + + /** + * Configure read timeout in milliseconds + */ + public AbstractDockerCmdExecFactory withReadTimeout(Integer readTimeout) { + this.readTimeout = readTimeout; + return this; + } + + @Override + public AuthCmd.Exec createAuthCmdExec() { + return new AuthCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public InfoCmd.Exec createInfoCmdExec() { + return new InfoCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public PingCmd.Exec createPingCmdExec() { + return new PingCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public VersionCmd.Exec createVersionCmdExec() { + return new VersionCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public PullImageCmd.Exec createPullImageCmdExec() { + return new PullImageCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public PushImageCmd.Exec createPushImageCmdExec() { + return new PushImageCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public SaveImageCmd.Exec createSaveImageCmdExec() { + return new SaveImageCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public SaveImagesCmd.Exec createSaveImagesCmdExec() { + return new SaveImagesCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public CreateImageCmd.Exec createCreateImageCmdExec() { + return new CreateImageCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public LoadImageCmd.Exec createLoadImageCmdExec() { + return new LoadImageCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public LoadImageAsyncCmd.Exec createLoadImageAsyncCmdExec() { + return new LoadImageAsyncCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public SearchImagesCmd.Exec createSearchImagesCmdExec() { + return new SearchImagesCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public RemoveImageCmd.Exec createRemoveImageCmdExec() { + return new RemoveImageCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ListImagesCmd.Exec createListImagesCmdExec() { + return new ListImagesCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public InspectImageCmd.Exec createInspectImageCmdExec() { + return new InspectImageCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ListContainersCmd.Exec createListContainersCmdExec() { + return new ListContainersCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public CreateContainerCmd.Exec createCreateContainerCmdExec() { + return new CreateContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public StartContainerCmd.Exec createStartContainerCmdExec() { + return new StartContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public InspectContainerCmd.Exec createInspectContainerCmdExec() { + return new InspectContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ExecCreateCmd.Exec createExecCmdExec() { + return new ExecCreateCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public RemoveContainerCmd.Exec createRemoveContainerCmdExec() { + return new RemoveContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public WaitContainerCmd.Exec createWaitContainerCmdExec() { + return new WaitContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public AttachContainerCmd.Exec createAttachContainerCmdExec() { + return new AttachContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ResizeContainerCmd.Exec createResizeContainerCmdExec() { + return new ResizeContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ExecStartCmd.Exec createExecStartCmdExec() { + return new ExecStartCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ResizeExecCmd.Exec createResizeExecCmdExec() { + return new ResizeExecCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public InspectExecCmd.Exec createInspectExecCmdExec() { + return new InspectExecCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public LogContainerCmd.Exec createLogContainerCmdExec() { + return new LogContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public CopyFileFromContainerCmd.Exec createCopyFileFromContainerCmdExec() { + return new CopyFileFromContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public StopContainerCmd.Exec createStopContainerCmdExec() { + return new StopContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ContainerDiffCmd.Exec createContainerDiffCmdExec() { + return new ContainerDiffCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public KillContainerCmd.Exec createKillContainerCmdExec() { + return new KillContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public UpdateContainerCmd.Exec createUpdateContainerCmdExec() { + return new UpdateContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public RenameContainerCmd.Exec createRenameContainerCmdExec() { + return new RenameContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public RestartContainerCmd.Exec createRestartContainerCmdExec() { + return new RestartContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public CommitCmd.Exec createCommitCmdExec() { + return new CommitCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public BuildImageCmd.Exec createBuildImageCmdExec() { + return new BuildImageCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public TopContainerCmd.Exec createTopContainerCmdExec() { + return new TopContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public TagImageCmd.Exec createTagImageCmdExec() { + return new TagImageCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public PauseContainerCmd.Exec createPauseContainerCmdExec() { + return new PauseContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public UnpauseContainerCmd.Exec createUnpauseContainerCmdExec() { + return new UnpauseContainerCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public EventsCmd.Exec createEventsCmdExec() { + return new EventsCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public StatsCmd.Exec createStatsCmdExec() { + return new StatsCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public CreateVolumeCmd.Exec createCreateVolumeCmdExec() { + return new CreateVolumeCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public InspectVolumeCmd.Exec createInspectVolumeCmdExec() { + return new InspectVolumeCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public RemoveVolumeCmd.Exec createRemoveVolumeCmdExec() { + return new RemoveVolumeCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ListVolumesCmd.Exec createListVolumesCmdExec() { + return new ListVolumesCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ListNetworksCmd.Exec createListNetworksCmdExec() { + return new ListNetworksCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public InspectNetworkCmd.Exec createInspectNetworkCmdExec() { + return new InspectNetworkCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public CreateNetworkCmd.Exec createCreateNetworkCmdExec() { + return new CreateNetworkCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public RemoveNetworkCmd.Exec createRemoveNetworkCmdExec() { + return new RemoveNetworkCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ConnectToNetworkCmd.Exec createConnectToNetworkCmdExec() { + return new ConnectToNetworkCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public DisconnectFromNetworkCmd.Exec createDisconnectFromNetworkCmdExec() { + return new DisconnectFromNetworkCmdExec(getBaseResource(), getDockerClientConfig()); + } + + // swarm + @Override + public InitializeSwarmCmd.Exec createInitializeSwarmCmdExec() { + return new InitializeSwarmCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public InspectSwarmCmd.Exec createInspectSwarmCmdExec() { + return new InspectSwarmCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public JoinSwarmCmd.Exec createJoinSwarmCmdExec() { + return new JoinSwarmCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public LeaveSwarmCmd.Exec createLeaveSwarmCmdExec() { + return new LeaveSwarmCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public UpdateSwarmCmd.Exec createUpdateSwarmCmdExec() { + return new UpdateSwarmCmdExec(getBaseResource(), getDockerClientConfig()); + } + + // services + @Override + public ListServicesCmd.Exec createListServicesCmdExec() { + return new ListServicesCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public CreateServiceCmd.Exec createCreateServiceCmdExec() { + return new CreateServiceCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public InspectServiceCmd.Exec createInspectServiceCmdExec() { + return new InspectServiceCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public UpdateServiceCmd.Exec createUpdateServiceCmdExec() { + return new UpdateServiceCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public RemoveServiceCmd.Exec createRemoveServiceCmdExec() { + return new RemoveServiceCmdExec(getBaseResource(), getDockerClientConfig()); + } + + // nodes + @Override + public ListSwarmNodesCmd.Exec listSwarmNodeCmdExec() { + return new ListSwarmNodesCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public InspectSwarmNodeCmd.Exec inspectSwarmNodeCmdExec() { + return new InspectSwarmNodeCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public RemoveSwarmNodeCmd.Exec removeSwarmNodeCmdExec() { + return new RemoveSwarmNodeCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public UpdateSwarmNodeCmd.Exec updateSwarmNodeCmdExec() { + return new UpdateSwarmNodeCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ListTasksCmd.Exec listTasksCmdExec() { + return new ListTasksCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public LogSwarmObjectCmd.Exec logSwarmObjectExec(String endpoint) { + return new LogSwarmObjectExec(getBaseResource(), getDockerClientConfig(), endpoint); + } + + @Override + public PruneCmd.Exec pruneCmdExec() { + return new PruneCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ListSecretsCmd.Exec createListSecretsCmdExec() { + return new ListSecretsCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public CreateSecretCmd.Exec createCreateSecretCmdExec() { + return new CreateSecretCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public RemoveSecretCmd.Exec createRemoveSecretCmdExec() { + return new RemoveSecretCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public ListConfigsCmd.Exec createListConfigsCmdExec() { + return new ListConfigsCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public CreateConfigCmd.Exec createCreateConfigCmdExec() { + return new CreateConfigCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public InspectConfigCmd.Exec createInspectConfigCmdExec() { + return new InspectConfigCmdExec(getBaseResource(), getDockerClientConfig()); + } + + @Override + public RemoveConfigCmd.Exec createRemoveConfigCmdExec() { + return new RemoveConfigCmdExec(getBaseResource(), getDockerClientConfig()); + } + + + protected abstract WebTarget getBaseResource(); +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/DefaultDockerClientConfig.java b/docker-java-core/src/main/java/com/github/dockerjava/core/DefaultDockerClientConfig.java new file mode 100644 index 000000000..dad75b360 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/DefaultDockerClientConfig.java @@ -0,0 +1,523 @@ +package com.github.dockerjava.core; + +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.AuthConfigurations; +import com.github.dockerjava.core.NameParser.HostnameReposName; +import com.github.dockerjava.core.NameParser.ReposTag; + +import java.util.Map.Entry; +import java.util.Optional; +import org.apache.commons.lang3.StringUtils; +import org.apache.commons.lang3.SystemUtils; +import org.apache.commons.lang3.builder.EqualsBuilder; +import org.apache.commons.lang3.builder.HashCodeBuilder; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import java.io.File; +import java.io.FileInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.Serializable; +import java.net.URI; +import java.util.HashSet; +import java.util.Map; +import java.util.Objects; +import java.util.Properties; +import java.util.Set; + +import static org.apache.commons.lang3.BooleanUtils.isTrue; + +/** + * Respects some of the docker CLI options. See https://docs.docker.com/engine/reference/commandline/cli/#environment-variables + */ +public class DefaultDockerClientConfig implements Serializable, DockerClientConfig { + + private static final long serialVersionUID = 1L; + + public static final String DOCKER_HOST = "DOCKER_HOST"; + + public static final String DOCKER_CONTEXT = "DOCKER_CONTEXT"; + + public static final String DOCKER_TLS_VERIFY = "DOCKER_TLS_VERIFY"; + + public static final String DOCKER_CONFIG = "DOCKER_CONFIG"; + + public static final String DOCKER_CERT_PATH = "DOCKER_CERT_PATH"; + + public static final String API_VERSION = "api.version"; + + public static final String REGISTRY_USERNAME = "registry.username"; + + public static final String REGISTRY_PASSWORD = "registry.password"; + + public static final String REGISTRY_EMAIL = "registry.email"; + + public static final String REGISTRY_URL = "registry.url"; + + private static final String DOCKER_JAVA_PROPERTIES = "docker-java.properties"; + + private static final Set CONFIG_KEYS = new HashSet<>(); + + static final Properties DEFAULT_PROPERTIES = new Properties(); + + static final String DEFAULT_DOCKER_HOST = "unix:///var/run/docker.sock"; + + static final String WINDOWS_DEFAULT_DOCKER_HOST = "npipe:////./pipe/docker_engine"; + + static { + CONFIG_KEYS.add(DOCKER_HOST); + CONFIG_KEYS.add(DOCKER_TLS_VERIFY); + CONFIG_KEYS.add(DOCKER_CONFIG); + CONFIG_KEYS.add(DOCKER_CERT_PATH); + CONFIG_KEYS.add(API_VERSION); + CONFIG_KEYS.add(REGISTRY_USERNAME); + CONFIG_KEYS.add(REGISTRY_PASSWORD); + CONFIG_KEYS.add(REGISTRY_EMAIL); + CONFIG_KEYS.add(REGISTRY_URL); + + DEFAULT_PROPERTIES.put(DOCKER_CONFIG, "${user.home}/.docker"); + DEFAULT_PROPERTIES.put(REGISTRY_URL, "https://index.docker.io/v1/"); + DEFAULT_PROPERTIES.put(REGISTRY_USERNAME, "${user.name}"); + } + + private final URI dockerHost; + + private final String registryUsername, registryPassword, registryEmail, registryUrl, dockerConfigPath; + + private final SSLConfig sslConfig; + + private final RemoteApiVersion apiVersion; + + private final DockerConfigFile dockerConfig; + + DefaultDockerClientConfig(URI dockerHost, DockerConfigFile dockerConfigFile, String dockerConfigPath, String apiVersion, + String registryUrl, String registryUsername, String registryPassword, String registryEmail, + SSLConfig sslConfig) { + this.dockerHost = checkDockerHostScheme(dockerHost); + this.dockerConfig = dockerConfigFile; + this.dockerConfigPath = dockerConfigPath; + this.apiVersion = RemoteApiVersion.parseConfigWithDefault(apiVersion); + this.sslConfig = sslConfig; + this.registryUsername = registryUsername; + this.registryPassword = registryPassword; + this.registryEmail = registryEmail; + this.registryUrl = registryUrl; + } + + private URI checkDockerHostScheme(URI dockerHost) { + if (dockerHost == null) { + throw new DockerClientException("'dockerHost' is null"); + } + return dockerHost; + } + + private static Properties loadIncludedDockerProperties(Properties systemProperties) { + Properties p = new Properties(); + p.putAll(DEFAULT_PROPERTIES); + try (InputStream is = DefaultDockerClientConfig.class.getResourceAsStream("/" + DOCKER_JAVA_PROPERTIES)) { + if (is != null) { + p.load(is); + } + } catch (IOException e) { + throw new RuntimeException(e); + } + replaceProperties(p, systemProperties); + return p; + } + + private static void replaceProperties(Properties properties, Properties replacements) { + for (Entry entry : properties.entrySet()) { + final String key = entry.getKey().toString(); + // no entry.getValue here because it does not have the same semantics as getProperty (defaults handling) + final String value = properties.getProperty(key); + entry.setValue(replaceProperties(value, replacements)); + } + } + + private static String replaceProperties(String s, Properties replacements) { + for (Map.Entry entry : replacements.entrySet()) { + String key = "${" + entry.getKey() + "}"; + while (s.contains(key)) { + s = s.replace(key, String.valueOf(entry.getValue())); + } + } + return s; + } + + /** + * Creates a new Properties object containing values overridden from ${user.home}/.docker.io.properties + * + * @param p + * The original set of properties to override + * @return A copy of the original Properties with overridden values + */ + private static Properties overrideDockerPropertiesWithSettingsFromUserHome(Properties p, Properties systemProperties) { + Properties overriddenProperties = new Properties(); + overriddenProperties.putAll(p); + + final File usersDockerPropertiesFile = new File(systemProperties.getProperty("user.home"), + "." + DOCKER_JAVA_PROPERTIES); + if (usersDockerPropertiesFile.isFile()) { + try (FileInputStream in = new FileInputStream(usersDockerPropertiesFile)) { + overriddenProperties.load(in); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + return overriddenProperties; + } + + private static Properties overrideDockerPropertiesWithEnv(Properties properties, Map env) { + Properties overriddenProperties = new Properties(); + overriddenProperties.putAll(properties); + + // special case which is a sensible default + if (env.containsKey(DOCKER_HOST)) { + String value = env.get(DOCKER_HOST); + if (value != null && value.trim().length() != 0) { + overriddenProperties.setProperty(DOCKER_HOST, value); + } + } + + if (env.containsKey(DOCKER_CONTEXT)) { + String value = env.get(DOCKER_CONTEXT); + if (value != null && value.trim().length() != 0) { + overriddenProperties.setProperty(DOCKER_CONTEXT, value); + } + } + + for (Map.Entry envEntry : env.entrySet()) { + String envKey = envEntry.getKey(); + if (CONFIG_KEYS.contains(envKey)) { + String value = envEntry.getValue(); + if (value != null && value.trim().length() != 0) { + overriddenProperties.setProperty(envKey, value); + } + } + } + + return overriddenProperties; + } + + /** + * Creates a new Properties object containing values overridden from the System properties + * + * @param p + * The original set of properties to override + * @return A copy of the original Properties with overridden values + */ + private static Properties overrideDockerPropertiesWithSystemProperties(Properties p, Properties systemProperties) { + Properties overriddenProperties = new Properties(); + overriddenProperties.putAll(p); + + for (String key : CONFIG_KEYS) { + if (systemProperties.containsKey(key)) { + overriddenProperties.setProperty(key, systemProperties.getProperty(key)); + } + } + return overriddenProperties; + } + + public static Builder createDefaultConfigBuilder() { + return createDefaultConfigBuilder(System.getenv(), (Properties) System.getProperties().clone()); + } + + /** + * Allows you to build the config without system environment interfering for more robust testing + */ + static Builder createDefaultConfigBuilder(Map env, Properties systemProperties) { + Properties properties = loadIncludedDockerProperties(systemProperties); + properties = overrideDockerPropertiesWithSettingsFromUserHome(properties, systemProperties); + properties = overrideDockerPropertiesWithEnv(properties, env); + properties = overrideDockerPropertiesWithSystemProperties(properties, systemProperties); + return new Builder().withProperties(properties); + } + + @Override + public URI getDockerHost() { + return dockerHost; + } + + @Override + public RemoteApiVersion getApiVersion() { + return apiVersion; + } + + @Override + public String getRegistryUsername() { + return registryUsername; + } + + @Override + public String getRegistryPassword() { + return registryPassword; + } + + @Override + public String getRegistryEmail() { + return registryEmail; + } + + @Override + public String getRegistryUrl() { + return registryUrl; + } + + @CheckForNull + public String getDockerConfigPath() { + return dockerConfigPath; + } + + @Nonnull + public DockerConfigFile getDockerConfig() { + return dockerConfig; + } + + private AuthConfig getAuthConfig() { + AuthConfig authConfig = null; + if (getRegistryUsername() != null && getRegistryPassword() != null && getRegistryUrl() != null) { + authConfig = new AuthConfig() + .withUsername(getRegistryUsername()) + .withPassword(getRegistryPassword()) + .withEmail(getRegistryEmail()) + .withRegistryAddress(getRegistryUrl()); + } + return authConfig; + } + + @Override + public AuthConfig effectiveAuthConfig(String imageName) { + AuthConfig authConfig = getAuthConfig(); + + if (authConfig != null) { + return authConfig; + } + + DockerConfigFile dockerCfg = getDockerConfig(); + + ReposTag reposTag = NameParser.parseRepositoryTag(imageName); + HostnameReposName hostnameReposName = NameParser.resolveRepositoryName(reposTag.repos); + + return dockerCfg.resolveAuthConfig(hostnameReposName.hostname); + } + + @Override + public AuthConfigurations getAuthConfigurations() { + return getDockerConfig().getAuthConfigurations(); + } + + @Override + public SSLConfig getSSLConfig() { + return sslConfig; + } + + @Override + public boolean equals(Object o) { + return EqualsBuilder.reflectionEquals(this, o); + } + + @Override + public int hashCode() { + return HashCodeBuilder.reflectionHashCode(this); + } + + @Override + public String toString() { + return ToStringBuilder.reflectionToString(this, ToStringStyle.SHORT_PREFIX_STYLE); + } + + public static class Builder { + private URI dockerHost; + + private String apiVersion, registryUsername, registryPassword, registryEmail, registryUrl, dockerConfig, + dockerCertPath, dockerContext; + + private Boolean dockerTlsVerify; + + private SSLConfig customSslConfig = null; + + /** + * This will set all fields in the builder to those contained in the Properties object. The Properties object should contain the + * following docker-java configuration keys: DOCKER_HOST, DOCKER_TLS_VERIFY, api.version, registry.username, registry.password, + * registry.email, DOCKER_CERT_PATH, and DOCKER_CONFIG. + */ + public Builder withProperties(Properties p) { + + if (p.getProperty(DOCKER_HOST) != null) { + withDockerHost(p.getProperty(DOCKER_HOST)); + } + + return withDockerTlsVerify(p.getProperty(DOCKER_TLS_VERIFY)) + .withDockerContext(p.getProperty(DOCKER_CONTEXT)) + .withDockerConfig(p.getProperty(DOCKER_CONFIG)) + .withDockerCertPath(p.getProperty(DOCKER_CERT_PATH)) + .withApiVersion(p.getProperty(API_VERSION)) + .withRegistryUsername(p.getProperty(REGISTRY_USERNAME)) + .withRegistryPassword(p.getProperty(REGISTRY_PASSWORD)) + .withRegistryEmail(p.getProperty(REGISTRY_EMAIL)) + .withRegistryUrl(p.getProperty(REGISTRY_URL)); + } + + /** + * configure DOCKER_HOST + */ + public final Builder withDockerHost(String dockerHost) { + Objects.requireNonNull(dockerHost, "uri was not specified"); + this.dockerHost = URI.create(dockerHost); + return this; + } + + public final Builder withApiVersion(RemoteApiVersion apiVersion) { + this.apiVersion = apiVersion.getVersion(); + return this; + } + + public final Builder withApiVersion(String apiVersion) { + this.apiVersion = apiVersion; + return this; + } + + public final Builder withRegistryUsername(String registryUsername) { + this.registryUsername = registryUsername; + return this; + } + + public final Builder withRegistryPassword(String registryPassword) { + this.registryPassword = registryPassword; + return this; + } + + public final Builder withRegistryEmail(String registryEmail) { + this.registryEmail = registryEmail; + return this; + } + + public Builder withRegistryUrl(String registryUrl) { + this.registryUrl = registryUrl; + return this; + } + + public final Builder withDockerCertPath(String dockerCertPath) { + this.dockerCertPath = dockerCertPath; + return this; + } + + public final Builder withDockerConfig(String dockerConfig) { + this.dockerConfig = dockerConfig; + return this; + } + + public final Builder withDockerContext(String dockerContext) { + this.dockerContext = dockerContext; + return this; + } + + public final Builder withDockerTlsVerify(String dockerTlsVerify) { + if (dockerTlsVerify != null) { + String trimmed = dockerTlsVerify.trim(); + this.dockerTlsVerify = "true".equalsIgnoreCase(trimmed) || "1".equals(trimmed); + } else { + this.dockerTlsVerify = false; + } + return this; + } + + public final Builder withDockerTlsVerify(Boolean dockerTlsVerify) { + this.dockerTlsVerify = dockerTlsVerify; + return this; + } + + public final boolean isDockerHostSetExplicitly() { + return dockerHost != null; + } + + /** + * Overrides the default {@link SSLConfig} that is used when calling {@link Builder#withDockerTlsVerify(java.lang.Boolean)} and + * {@link Builder#withDockerCertPath(String)}. This way it is possible to pass a custom {@link SSLConfig} to the resulting + * {@link DockerClientConfig} that may be created by other means than the local file system. + */ + public final Builder withCustomSslConfig(SSLConfig customSslConfig) { + this.customSslConfig = customSslConfig; + return this; + } + + private void applyContextConfiguration(final String context) { + final Optional dockerContextMetaFile = + Optional.ofNullable(context) + .flatMap(ctx -> DockerContextMetaFile.resolveContextMetaFile(DockerClientConfig.getDefaultObjectMapper(), + new File(this.dockerConfig), ctx)); + final Optional dockerContextTLSFile = + Optional.ofNullable(context) + .flatMap(ctx -> DockerContextMetaFile.resolveContextTLSFile(new File(this.dockerConfig), ctx)); + + if (dockerContextMetaFile.isPresent()) { + final Optional dockerEndpoint = + dockerContextMetaFile.map(metaFile -> metaFile.endpoints).map(endpoint -> endpoint.docker); + if (this.dockerHost == null) { + this.dockerHost = dockerEndpoint.map(endpoint -> endpoint.host).map(URI::create).orElse(null); + } + } + if (dockerContextTLSFile.isPresent() && this.dockerCertPath == null) { + this.dockerCertPath = dockerContextTLSFile.get().getAbsolutePath(); + this.dockerTlsVerify = true; + } + } + + public DefaultDockerClientConfig build() { + final DockerConfigFile dockerConfigFile = readDockerConfig(); + final String context = (dockerContext != null) ? dockerContext : dockerConfigFile.getCurrentContext(); + applyContextConfiguration(context); + + SSLConfig sslConfig = null; + + if (customSslConfig == null) { + if (isTrue(dockerTlsVerify)) { + dockerCertPath = checkDockerCertPath(dockerCertPath); + sslConfig = new LocalDirectorySSLConfig(dockerCertPath); + } + } else { + sslConfig = customSslConfig; + } + + URI dockerHostUri = dockerHost != null + ? dockerHost + : URI.create(SystemUtils.IS_OS_WINDOWS ? WINDOWS_DEFAULT_DOCKER_HOST : DEFAULT_DOCKER_HOST); + + return new DefaultDockerClientConfig(dockerHostUri, dockerConfigFile, dockerConfig, apiVersion, registryUrl, registryUsername, + registryPassword, registryEmail, sslConfig); + } + + private DockerConfigFile readDockerConfig() { + try { + return DockerConfigFile.loadConfig(DockerClientConfig.getDefaultObjectMapper(), dockerConfig); + } catch (IOException e) { + throw new DockerClientException("Failed to parse docker configuration file", e); + } + } + + private String checkDockerCertPath(String dockerCertPath) { + if (StringUtils.isEmpty(dockerCertPath)) { + throw new DockerClientException( + "Enabled TLS verification (DOCKER_TLS_VERIFY=1) but certificate path (DOCKER_CERT_PATH) is not defined."); + } + + File certPath = new File(dockerCertPath); + + if (!certPath.exists()) { + throw new DockerClientException( + "Enabled TLS verification (DOCKER_TLS_VERIFY=1) but certificate path (DOCKER_CERT_PATH) '" + + dockerCertPath + "' doesn't exist."); + } else if (!certPath.isDirectory()) { + throw new DockerClientException( + "Enabled TLS verification (DOCKER_TLS_VERIFY=1) but certificate path (DOCKER_CERT_PATH) '" + + dockerCertPath + "' doesn't point to a directory."); + } + + return dockerCertPath; + } + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/DefaultDockerCmdExecFactory.java b/docker-java-core/src/main/java/com/github/dockerjava/core/DefaultDockerCmdExecFactory.java new file mode 100644 index 000000000..54722e6c7 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/DefaultDockerCmdExecFactory.java @@ -0,0 +1,160 @@ +package com.github.dockerjava.core; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.dockerjava.transport.DockerHttpClient; +import com.google.common.collect.HashMultimap; +import com.google.common.collect.ImmutableList; +import com.google.common.collect.MultimapBuilder; +import com.google.common.collect.SetMultimap; +import com.google.common.escape.Escaper; +import com.google.common.net.UrlEscapers; +import org.apache.commons.lang3.StringUtils; + +import java.io.IOException; +import java.util.Map; +import java.util.Objects; +import java.util.Set; +import java.util.stream.Collectors; + +public final class DefaultDockerCmdExecFactory extends AbstractDockerCmdExecFactory { + + private final DockerHttpClient dockerHttpClient; + + private final ObjectMapper objectMapper; + + public DefaultDockerCmdExecFactory( + DockerHttpClient dockerHttpClient, + ObjectMapper objectMapper + ) { + this.dockerHttpClient = dockerHttpClient; + this.objectMapper = objectMapper; + } + + public DockerHttpClient getDockerHttpClient() { + return dockerHttpClient; + } + + @Override + protected WebTarget getBaseResource() { + return new DefaultWebTarget(); + } + + @Override + public void close() throws IOException { + dockerHttpClient.close(); + } + + private class DefaultWebTarget implements WebTarget { + + final ImmutableList path; + + final SetMultimap queryParams; + + DefaultWebTarget() { + this( + ImmutableList.of(), + MultimapBuilder.hashKeys().hashSetValues().build() + ); + } + + DefaultWebTarget( + ImmutableList path, + SetMultimap queryParams + ) { + this.path = path; + this.queryParams = queryParams; + } + + @Override + public String toString() { + return String.format("DefaultWebTarget{path=%s, queryParams=%s}", path, queryParams); + } + + @Override + public InvocationBuilder request() { + String resource = StringUtils.join(path, "/"); + + if (!resource.startsWith("/")) { + resource = "/" + resource; + } + + RemoteApiVersion apiVersion = getDockerClientConfig().getApiVersion(); + if (apiVersion != RemoteApiVersion.UNKNOWN_VERSION) { + resource = "/" + apiVersion.asWebPathPart() + resource; + } + + if (!queryParams.isEmpty()) { + Escaper urlFormParameterEscaper = UrlEscapers.urlFormParameterEscaper(); + resource = queryParams.asMap().entrySet().stream() + .flatMap(entry -> { + return entry.getValue().stream().map(s -> { + return entry.getKey() + "=" + urlFormParameterEscaper.escape(s); + }); + }) + .collect(Collectors.joining("&", resource + "?", "")); + } + + return new DefaultInvocationBuilder( + dockerHttpClient, objectMapper, resource + ); + } + + @Override + public DefaultWebTarget path(String... components) { + ImmutableList newPath = ImmutableList.builder() + .addAll(path) + .add(components) + .build(); + return new DefaultWebTarget(newPath, queryParams); + } + + @Override + public DefaultWebTarget resolveTemplate(String name, Object value) { + ImmutableList.Builder newPath = ImmutableList.builder(); + for (String component : path) { + component = component.replaceAll( + "\\{" + name + "\\}", + UrlEscapers.urlPathSegmentEscaper().escape(value.toString()) + ); + newPath.add(component); + } + + return new DefaultWebTarget(newPath.build(), queryParams); + } + + @Override + public DefaultWebTarget queryParam(String name, Object value) { + if (value == null) { + return this; + } + + SetMultimap newQueryParams = HashMultimap.create(queryParams); + newQueryParams.put(name, value.toString()); + + return new DefaultWebTarget(path, newQueryParams); + } + + @Override + public DefaultWebTarget queryParamsSet(String name, Set values) { + SetMultimap newQueryParams = HashMultimap.create(queryParams); + newQueryParams.replaceValues(name, values.stream().filter(Objects::nonNull).map(Object::toString).collect(Collectors.toSet())); + + return new DefaultWebTarget(path, newQueryParams); + } + + @Override + public DefaultWebTarget queryParamsJsonMap(String name, Map values) { + if (values == null || values.isEmpty()) { + return this; + } + + // when param value is JSON string + try { + return queryParam(name, objectMapper.writeValueAsString(values)); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + } + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/DefaultInvocationBuilder.java b/docker-java-core/src/main/java/com/github/dockerjava/core/DefaultInvocationBuilder.java new file mode 100644 index 000000000..c7525279c --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/DefaultInvocationBuilder.java @@ -0,0 +1,322 @@ +package com.github.dockerjava.core; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.MappingIterator; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.exception.BadRequestException; +import com.github.dockerjava.api.exception.ConflictException; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.InternalServerErrorException; +import com.github.dockerjava.api.exception.NotAcceptableException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.exception.NotModifiedException; +import com.github.dockerjava.api.exception.UnauthorizedException; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.transport.DockerHttpClient; +import org.apache.commons.io.IOUtils; + +import java.io.FilterInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.nio.charset.StandardCharsets; +import java.util.Objects; +import java.util.function.Consumer; + +class DefaultInvocationBuilder implements InvocationBuilder { + + private final DockerHttpClient.Request.Builder requestBuilder; + private final DockerHttpClient dockerHttpClient; + private final ObjectMapper objectMapper; + + DefaultInvocationBuilder(DockerHttpClient dockerHttpClient, ObjectMapper objectMapper, String path) { + this.requestBuilder = DockerHttpClient.Request.builder().path(path); + this.dockerHttpClient = dockerHttpClient; + this.objectMapper = objectMapper; + } + + @Override + public DefaultInvocationBuilder accept(MediaType mediaType) { + return header("accept", mediaType.getMediaType()); + } + + @Override + public DefaultInvocationBuilder header(String name, String value) { + requestBuilder.putHeader(name, value); + return this; + } + + @Override + public void delete() { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.DELETE) + .build(); + + execute(request).close(); + } + + @Override + public void get(ResultCallback resultCallback) { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.GET) + .build(); + + executeAndStream( + request, + resultCallback, + new FramedInputStreamConsumer(resultCallback) + ); + } + + @Override + public T get(TypeReference typeReference) { + try (InputStream inputStream = get()) { + return objectMapper.readValue(inputStream, typeReference); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Override + public void get(TypeReference typeReference, ResultCallback resultCallback) { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.GET) + .build(); + + executeAndStream( + request, + resultCallback, + new JsonSink<>(typeReference, resultCallback) + ); + } + + @Override + public InputStream post(Object entity) { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.POST) + .putHeader("content-type", "application/json") + .bodyBytes(encode(entity)) + .build(); + + DockerHttpClient.Response response = execute(request); + return new FilterInputStream(response.getBody()) { + @Override + public void close() throws IOException { + try { + super.close(); + } finally { + response.close(); + } + } + }; + } + + @Override + public T post(Object entity, TypeReference typeReference) { + try { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.POST) + .putHeader("content-type", "application/json") + .bodyBytes(encode(entity)) + .build(); + + try (DockerHttpClient.Response response = execute(request)) { + return objectMapper.readValue(response.getBody(), typeReference); + } + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Override + public void post(Object entity, TypeReference typeReference, ResultCallback resultCallback) { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.POST) + .putHeader("content-type", "application/json") + .bodyBytes(encode(entity)) + .build(); + + executeAndStream( + request, + resultCallback, + new JsonSink<>(typeReference, resultCallback) + ); + } + + @Override + public T post(TypeReference typeReference, InputStream body) { + try (InputStream inputStream = post(body)) { + return objectMapper.readValue(inputStream, typeReference); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Override + public void post(Object entity, InputStream stdin, ResultCallback resultCallback) { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.POST) + .putHeader("content-type", "application/json") + .bodyBytes(encode(entity)) + .hijackedInput(stdin) + .build(); + + executeAndStream( + request, + resultCallback, + new FramedInputStreamConsumer(resultCallback) + ); + } + + @Override + public void post(TypeReference typeReference, ResultCallback resultCallback, InputStream body) { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.POST) + .body(body) + .build(); + + executeAndStream( + request, + resultCallback, + new JsonSink<>(typeReference, resultCallback) + ); + } + + @Override + public void postStream(InputStream body) { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.POST) + .body(body) + .build(); + + execute(request).close(); + } + + @Override + public InputStream get() { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.GET) + .build(); + + DockerHttpClient.Response response = execute(request); + return new FilterInputStream(response.getBody()) { + @Override + public void close() throws IOException { + try { + super.close(); + } finally { + response.close(); + } + } + }; + } + + @Override + public void put(InputStream body, MediaType mediaType) { + DockerHttpClient.Request request = requestBuilder + .method(DockerHttpClient.Request.Method.PUT) + .putHeader("content-type", mediaType.toString()) + .body(body) + .build(); + + execute(request).close(); + } + + protected DockerHttpClient.Response execute(DockerHttpClient.Request request) { + try { + DockerHttpClient.Response response = dockerHttpClient.execute(request); + int statusCode = response.getStatusCode(); + if (statusCode < 200 || statusCode > 299) { + try { + String body = IOUtils.toString(response.getBody(), StandardCharsets.UTF_8); + switch (statusCode) { + case 304: + throw new NotModifiedException(body); + case 400: + throw new BadRequestException(body); + case 401: + throw new UnauthorizedException(body); + case 404: + throw new NotFoundException(body); + case 406: + throw new NotAcceptableException(body); + case 409: + throw new ConflictException(body); + case 500: + throw new InternalServerErrorException(body); + default: + throw new DockerException(body, statusCode); + } + } finally { + response.close(); + } + } else { + return response; + } + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + protected void executeAndStream( + DockerHttpClient.Request request, + ResultCallback callback, + Consumer sourceConsumer + ) { + Thread thread = new Thread(() -> { + Thread streamingThread = Thread.currentThread(); + try (DockerHttpClient.Response response = execute(request)) { + callback.onStart(() -> { + streamingThread.interrupt(); + response.close(); + }); + + sourceConsumer.accept(response); + callback.onComplete(); + } catch (Exception e) { + callback.onError(e); + } + }, "docker-java-stream-" + Objects.hashCode(request)); + thread.setDaemon(true); + + thread.start(); + } + + private byte[] encode(Object entity) { + if (entity == null) { + return null; + } + + try { + return objectMapper.writeValueAsBytes(entity); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + } + + private class JsonSink implements Consumer { + + private final TypeReference typeReference; + + private final ResultCallback resultCallback; + + JsonSink(TypeReference typeReference, ResultCallback resultCallback) { + this.typeReference = typeReference; + this.resultCallback = resultCallback; + } + + @Override + public void accept(DockerHttpClient.Response response) { + try { + InputStream body = response.getBody(); + MappingIterator iterator = objectMapper.readerFor(typeReference).readValues(body); + while (iterator.hasNextValue()) { + resultCallback.onNext((T) iterator.nextValue()); + } + } catch (Exception e) { + resultCallback.onError(e); + } + } + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientConfig.java b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientConfig.java new file mode 100644 index 000000000..e3961661a --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientConfig.java @@ -0,0 +1,143 @@ +/* + * Created on 08.06.2016 + */ +package com.github.dockerjava.core; + +import com.fasterxml.jackson.core.JsonParser; +import com.fasterxml.jackson.databind.BeanDescription; +import com.fasterxml.jackson.databind.DeserializationConfig; +import com.fasterxml.jackson.databind.DeserializationContext; +import com.fasterxml.jackson.databind.DeserializationFeature; +import com.fasterxml.jackson.databind.JsonDeserializer; +import com.fasterxml.jackson.databind.JsonNode; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.SerializationFeature; +import com.fasterxml.jackson.databind.deser.BeanDeserializerModifier; +import com.fasterxml.jackson.databind.deser.std.DelegatingDeserializer; +import com.fasterxml.jackson.databind.module.SimpleModule; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.AuthConfigurations; +import com.github.dockerjava.api.model.DockerObject; +import com.github.dockerjava.api.model.DockerObjectAccessor; + +import java.io.IOException; +import java.net.URI; +import java.util.HashMap; + +/** + * Interface that describes the docker client configuration. + * + * @author Marcus Linke + * + */ +public interface DockerClientConfig { + + static ObjectMapper getDefaultObjectMapper() { + return DefaultObjectMapperHolder.INSTANCE.getObjectMapper().copy(); + } + + URI getDockerHost(); + + RemoteApiVersion getApiVersion(); + + String getRegistryUsername(); + + String getRegistryPassword(); + + String getRegistryEmail(); + + String getRegistryUrl(); + + AuthConfig effectiveAuthConfig(String imageName); + + AuthConfigurations getAuthConfigurations(); + + /** + * Returns an {@link SSLConfig} when secure connection is configured or null if not. + */ + SSLConfig getSSLConfig(); + + default ObjectMapper getObjectMapper() { + return getDefaultObjectMapper(); + } +} + +enum DefaultObjectMapperHolder { + INSTANCE; + + private final ObjectMapper objectMapper = new ObjectMapper() + // TODO .setDefaultPropertyInclusion(JsonInclude.Include.NON_NULL) + .disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES) + .disable(SerializationFeature.FAIL_ON_EMPTY_BEANS); + + { + ObjectMapper originalObjectMapper = objectMapper.copy(); + objectMapper.registerModule(new SimpleModule("docker-java") { + @Override + public void setupModule(SetupContext context) { + super.setupModule(context); + context.addBeanDeserializerModifier(new BeanDeserializerModifier() { + @Override + public JsonDeserializer modifyDeserializer( + DeserializationConfig config, + BeanDescription beanDescription, + JsonDeserializer originalDeserializer + ) { + if (!beanDescription.getType().isTypeOrSubTypeOf(DockerObject.class)) { + return originalDeserializer; + } + + return new DockerObjectDeserializer( + originalDeserializer, + beanDescription, + originalObjectMapper + ); + } + }); + } + }); + } + + public ObjectMapper getObjectMapper() { + return objectMapper; + } +} + +class DockerObjectDeserializer extends DelegatingDeserializer { + + private final BeanDescription beanDescription; + + private final ObjectMapper originalMapper; + + DockerObjectDeserializer( + JsonDeserializer delegate, + BeanDescription beanDescription, + ObjectMapper originalMapper + ) { + super(delegate); + this.beanDescription = beanDescription; + this.originalMapper = originalMapper; + } + + @Override + protected JsonDeserializer newDelegatingInstance(JsonDeserializer newDelegatee) { + return new DockerObjectDeserializer(newDelegatee, beanDescription, originalMapper); + } + + @Override + @SuppressWarnings({"deprecation", "unchecked"}) + public Object deserialize(JsonParser p, DeserializationContext ctxt) throws IOException { + JsonNode jsonNode = p.readValueAsTree(); + + Object deserializedObject = originalMapper.treeToValue(jsonNode, beanDescription.getBeanClass()); + + if (deserializedObject instanceof DockerObject) { + DockerObjectAccessor.overrideRawValues( + ((DockerObject) deserializedObject), + originalMapper.convertValue(jsonNode, HashMap.class) + ); + } + + return deserializedObject; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientConfigAware.java b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientConfigAware.java new file mode 100644 index 000000000..5cee4d6cb --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientConfigAware.java @@ -0,0 +1,6 @@ +package com.github.dockerjava.core; + +public interface DockerClientConfigAware { + + void init(DockerClientConfig dockerClientConfig); +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientConfigDelegate.java b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientConfigDelegate.java new file mode 100644 index 000000000..86c6ac5de --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientConfigDelegate.java @@ -0,0 +1,57 @@ +package com.github.dockerjava.core; + +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.AuthConfigurations; + +import java.net.URI; + +@SuppressWarnings("unused") +public class DockerClientConfigDelegate implements DockerClientConfig { + + final DockerClientConfig original; + + public DockerClientConfigDelegate(DockerClientConfig original) { + this.original = original; + } + + public URI getDockerHost() { + return original.getDockerHost(); + } + + public RemoteApiVersion getApiVersion() { + return original.getApiVersion(); + } + + public String getRegistryUsername() { + return original.getRegistryUsername(); + } + + public String getRegistryPassword() { + return original.getRegistryPassword(); + } + + public String getRegistryEmail() { + return original.getRegistryEmail(); + } + + public String getRegistryUrl() { + return original.getRegistryUrl(); + } + + public AuthConfig effectiveAuthConfig(String imageName) { + return original.effectiveAuthConfig(imageName); + } + + public AuthConfigurations getAuthConfigurations() { + return original.getAuthConfigurations(); + } + + public SSLConfig getSSLConfig() { + return original.getSSLConfig(); + } + + public ObjectMapper getObjectMapper() { + return original.getObjectMapper(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientImpl.java new file mode 100644 index 000000000..55f530057 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerClientImpl.java @@ -0,0 +1,723 @@ +package com.github.dockerjava.core; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.AttachContainerCmd; +import com.github.dockerjava.api.command.AuthCmd; +import com.github.dockerjava.api.command.BuildImageCmd; +import com.github.dockerjava.api.command.CommitCmd; +import com.github.dockerjava.api.command.ConnectToNetworkCmd; +import com.github.dockerjava.api.command.ContainerDiffCmd; +import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; +import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; +import com.github.dockerjava.api.command.CopyFileFromContainerCmd; +import com.github.dockerjava.api.command.CreateConfigCmd; +import com.github.dockerjava.api.command.CreateContainerCmd; +import com.github.dockerjava.api.command.CreateImageCmd; +import com.github.dockerjava.api.command.CreateNetworkCmd; +import com.github.dockerjava.api.command.CreateSecretCmd; +import com.github.dockerjava.api.command.CreateServiceCmd; +import com.github.dockerjava.api.command.CreateVolumeCmd; +import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; +import com.github.dockerjava.api.command.DockerCmdExecFactory; +import com.github.dockerjava.api.command.EventsCmd; +import com.github.dockerjava.api.command.ExecCreateCmd; +import com.github.dockerjava.api.command.ExecStartCmd; +import com.github.dockerjava.api.command.InfoCmd; +import com.github.dockerjava.api.command.InitializeSwarmCmd; +import com.github.dockerjava.api.command.InspectConfigCmd; +import com.github.dockerjava.api.command.InspectContainerCmd; +import com.github.dockerjava.api.command.InspectExecCmd; +import com.github.dockerjava.api.command.InspectImageCmd; +import com.github.dockerjava.api.command.InspectNetworkCmd; +import com.github.dockerjava.api.command.InspectServiceCmd; +import com.github.dockerjava.api.command.InspectSwarmCmd; +import com.github.dockerjava.api.command.InspectVolumeCmd; +import com.github.dockerjava.api.command.JoinSwarmCmd; +import com.github.dockerjava.api.command.KillContainerCmd; +import com.github.dockerjava.api.command.LeaveSwarmCmd; +import com.github.dockerjava.api.command.ListConfigsCmd; +import com.github.dockerjava.api.command.ListContainersCmd; +import com.github.dockerjava.api.command.ListImagesCmd; +import com.github.dockerjava.api.command.ListNetworksCmd; +import com.github.dockerjava.api.command.ListSecretsCmd; +import com.github.dockerjava.api.command.ListServicesCmd; +import com.github.dockerjava.api.command.ListSwarmNodesCmd; +import com.github.dockerjava.api.command.ListTasksCmd; +import com.github.dockerjava.api.command.ListVolumesCmd; +import com.github.dockerjava.api.command.LoadImageAsyncCmd; +import com.github.dockerjava.api.command.LoadImageCmd; +import com.github.dockerjava.api.command.LogContainerCmd; +import com.github.dockerjava.api.command.LogSwarmObjectCmd; +import com.github.dockerjava.api.command.PauseContainerCmd; +import com.github.dockerjava.api.command.PingCmd; +import com.github.dockerjava.api.command.PruneCmd; +import com.github.dockerjava.api.command.PullImageCmd; +import com.github.dockerjava.api.command.PushImageCmd; +import com.github.dockerjava.api.command.RemoveConfigCmd; +import com.github.dockerjava.api.command.RemoveContainerCmd; +import com.github.dockerjava.api.command.RemoveImageCmd; +import com.github.dockerjava.api.command.RemoveNetworkCmd; +import com.github.dockerjava.api.command.RemoveSecretCmd; +import com.github.dockerjava.api.command.RemoveServiceCmd; +import com.github.dockerjava.api.command.RemoveSwarmNodeCmd; +import com.github.dockerjava.api.command.RemoveVolumeCmd; +import com.github.dockerjava.api.command.RenameContainerCmd; +import com.github.dockerjava.api.command.ResizeContainerCmd; +import com.github.dockerjava.api.command.ResizeExecCmd; +import com.github.dockerjava.api.command.RestartContainerCmd; +import com.github.dockerjava.api.command.SaveImageCmd; +import com.github.dockerjava.api.command.SaveImagesCmd; +import com.github.dockerjava.api.command.SearchImagesCmd; +import com.github.dockerjava.api.command.StartContainerCmd; +import com.github.dockerjava.api.command.StatsCmd; +import com.github.dockerjava.api.command.StopContainerCmd; +import com.github.dockerjava.api.command.TagImageCmd; +import com.github.dockerjava.api.command.TopContainerCmd; +import com.github.dockerjava.api.command.UnpauseContainerCmd; +import com.github.dockerjava.api.command.UpdateContainerCmd; +import com.github.dockerjava.api.command.UpdateServiceCmd; +import com.github.dockerjava.api.command.UpdateSwarmCmd; +import com.github.dockerjava.api.command.UpdateSwarmNodeCmd; +import com.github.dockerjava.api.command.VersionCmd; +import com.github.dockerjava.api.command.WaitContainerCmd; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.Identifier; +import com.github.dockerjava.api.model.PruneType; +import com.github.dockerjava.api.model.SecretSpec; +import com.github.dockerjava.api.model.ServiceSpec; +import com.github.dockerjava.api.model.SwarmSpec; +import com.github.dockerjava.core.command.AttachContainerCmdImpl; +import com.github.dockerjava.core.command.AuthCmdImpl; +import com.github.dockerjava.core.command.BuildImageCmdImpl; +import com.github.dockerjava.core.command.CommitCmdImpl; +import com.github.dockerjava.core.command.ConnectToNetworkCmdImpl; +import com.github.dockerjava.core.command.ContainerDiffCmdImpl; +import com.github.dockerjava.core.command.CopyArchiveFromContainerCmdImpl; +import com.github.dockerjava.core.command.CopyArchiveToContainerCmdImpl; +import com.github.dockerjava.core.command.CopyFileFromContainerCmdImpl; +import com.github.dockerjava.core.command.CreateConfigCmdImpl; +import com.github.dockerjava.core.command.CreateContainerCmdImpl; +import com.github.dockerjava.core.command.CreateImageCmdImpl; +import com.github.dockerjava.core.command.CreateNetworkCmdImpl; +import com.github.dockerjava.core.command.CreateSecretCmdImpl; +import com.github.dockerjava.core.command.CreateServiceCmdImpl; +import com.github.dockerjava.core.command.CreateVolumeCmdImpl; +import com.github.dockerjava.core.command.DisconnectFromNetworkCmdImpl; +import com.github.dockerjava.core.command.EventsCmdImpl; +import com.github.dockerjava.core.command.ExecCreateCmdImpl; +import com.github.dockerjava.core.command.ExecStartCmdImpl; +import com.github.dockerjava.core.command.InfoCmdImpl; +import com.github.dockerjava.core.command.InitializeSwarmCmdImpl; +import com.github.dockerjava.core.command.InpectNetworkCmdImpl; +import com.github.dockerjava.core.command.InspectConfigCmdImpl; +import com.github.dockerjava.core.command.InspectContainerCmdImpl; +import com.github.dockerjava.core.command.InspectExecCmdImpl; +import com.github.dockerjava.core.command.InspectImageCmdImpl; +import com.github.dockerjava.core.command.InspectServiceCmdImpl; +import com.github.dockerjava.core.command.InspectSwarmCmdImpl; +import com.github.dockerjava.core.command.InspectVolumeCmdImpl; +import com.github.dockerjava.core.command.JoinSwarmCmdImpl; +import com.github.dockerjava.core.command.KillContainerCmdImpl; +import com.github.dockerjava.core.command.LeaveSwarmCmdImpl; +import com.github.dockerjava.core.command.ListConfigsCmdImpl; +import com.github.dockerjava.core.command.ListContainersCmdImpl; +import com.github.dockerjava.core.command.ListImagesCmdImpl; +import com.github.dockerjava.core.command.ListNetworksCmdImpl; +import com.github.dockerjava.core.command.ListSecretsCmdImpl; +import com.github.dockerjava.core.command.ListServicesCmdImpl; +import com.github.dockerjava.core.command.ListSwarmNodesCmdImpl; +import com.github.dockerjava.core.command.ListTasksCmdImpl; +import com.github.dockerjava.core.command.ListVolumesCmdImpl; +import com.github.dockerjava.core.command.LoadImageAsyncCmdImpl; +import com.github.dockerjava.core.command.LoadImageCmdImpl; +import com.github.dockerjava.core.command.LogContainerCmdImpl; +import com.github.dockerjava.core.command.LogSwarmObjectImpl; +import com.github.dockerjava.core.command.PauseContainerCmdImpl; +import com.github.dockerjava.core.command.PingCmdImpl; +import com.github.dockerjava.core.command.PruneCmdImpl; +import com.github.dockerjava.core.command.PullImageCmdImpl; +import com.github.dockerjava.core.command.PushImageCmdImpl; +import com.github.dockerjava.core.command.RemoveConfigCmdImpl; +import com.github.dockerjava.core.command.RemoveContainerCmdImpl; +import com.github.dockerjava.core.command.RemoveImageCmdImpl; +import com.github.dockerjava.core.command.RemoveNetworkCmdImpl; +import com.github.dockerjava.core.command.RemoveSecretCmdImpl; +import com.github.dockerjava.core.command.RemoveServiceCmdImpl; +import com.github.dockerjava.core.command.RemoveSwarmNodeCmdImpl; +import com.github.dockerjava.core.command.RemoveVolumeCmdImpl; +import com.github.dockerjava.core.command.RenameContainerCmdImpl; +import com.github.dockerjava.core.command.ResizeContainerCmdImpl; +import com.github.dockerjava.core.command.ResizeExecCmdImpl; +import com.github.dockerjava.core.command.RestartContainerCmdImpl; +import com.github.dockerjava.core.command.SaveImageCmdImpl; +import com.github.dockerjava.core.command.SaveImagesCmdImpl; +import com.github.dockerjava.core.command.SearchImagesCmdImpl; +import com.github.dockerjava.core.command.StartContainerCmdImpl; +import com.github.dockerjava.core.command.StatsCmdImpl; +import com.github.dockerjava.core.command.StopContainerCmdImpl; +import com.github.dockerjava.core.command.TagImageCmdImpl; +import com.github.dockerjava.core.command.TopContainerCmdImpl; +import com.github.dockerjava.core.command.UnpauseContainerCmdImpl; +import com.github.dockerjava.core.command.UpdateContainerCmdImpl; +import com.github.dockerjava.core.command.UpdateServiceCmdImpl; +import com.github.dockerjava.core.command.UpdateSwarmCmdImpl; +import com.github.dockerjava.core.command.UpdateSwarmNodeCmdImpl; +import com.github.dockerjava.core.command.VersionCmdImpl; +import com.github.dockerjava.core.command.WaitContainerCmdImpl; +import com.github.dockerjava.transport.DockerHttpClient; + +import javax.annotation.Nonnull; +import javax.annotation.Nullable; +import java.io.Closeable; +import java.io.File; +import java.io.IOException; +import java.io.InputStream; +import java.util.Objects; + +/** + * @author Konstantin Pelykh (kpelykh@gmail.com) + * @see "https://github.com/docker/docker/blob/master/api/client/commands.go" + */ +public class DockerClientImpl implements Closeable, DockerClient { + + private final DockerClientConfig dockerClientConfig; + + DockerCmdExecFactory dockerCmdExecFactory; + + DockerClientImpl(DockerClientConfig dockerClientConfig) { + this.dockerClientConfig = Objects.requireNonNull(dockerClientConfig, "config was not specified"); + } + + /** + * + * @deprecated use {@link #getInstance(DockerClientConfig, DockerHttpClient)} + */ + @Deprecated + public static DockerClientImpl getInstance() { + return new DockerClientImpl(DefaultDockerClientConfig.createDefaultConfigBuilder().build()); + } + + /** + * + * @deprecated use {@link #getInstance(DockerClientConfig, DockerHttpClient)} + */ + @Deprecated + public static DockerClientImpl getInstance(DockerClientConfig dockerClientConfig) { + return new DockerClientImpl(dockerClientConfig); + } + + public static DockerClient getInstance(DockerClientConfig dockerClientConfig, DockerHttpClient dockerHttpClient) { + return new DockerClientImpl(dockerClientConfig) + .withHttpClient(dockerHttpClient); + } + + /** + * + * @deprecated use {@link #getInstance(DockerClientConfig, DockerHttpClient)} + */ + @Deprecated + public static DockerClientImpl getInstance(String serverUrl) { + return new DockerClientImpl( + DefaultDockerClientConfig.createDefaultConfigBuilder() + .withDockerHost(serverUrl) + .build() + ); + } + + DockerClientImpl withHttpClient(DockerHttpClient httpClient) { + return withDockerCmdExecFactory(new DefaultDockerCmdExecFactory(httpClient, dockerClientConfig.getObjectMapper())); + } + + /** + * + * @return {@link DockerHttpClient} or null if not set + */ + @Nullable + public DockerHttpClient getHttpClient() { + if (dockerCmdExecFactory instanceof DefaultDockerCmdExecFactory) { + return ((DefaultDockerCmdExecFactory) dockerCmdExecFactory).getDockerHttpClient(); + } else { + return null; + } + } + + /** + * @deprecated use {@link #getInstance(DockerClientConfig, DockerHttpClient)} + */ + @Deprecated + public DockerClientImpl withDockerCmdExecFactory(DockerCmdExecFactory dockerCmdExecFactory) { + Objects.requireNonNull(dockerCmdExecFactory, "dockerCmdExecFactory was not specified"); + this.dockerCmdExecFactory = dockerCmdExecFactory; + if (dockerCmdExecFactory instanceof DockerClientConfigAware) { + ((DockerClientConfigAware) dockerCmdExecFactory).init(dockerClientConfig); + } + return this; + } + + @Deprecated + private DockerCmdExecFactory getDockerCmdExecFactory() { + Objects.requireNonNull(dockerCmdExecFactory, "dockerCmdExecFactory was not specified"); + return dockerCmdExecFactory; + } + + @Override + public AuthConfig authConfig() { + Objects.requireNonNull(dockerClientConfig.getRegistryUsername(), "Configured username is null."); + Objects.requireNonNull(dockerClientConfig.getRegistryUrl(), "Configured serverAddress is null."); + + return new AuthConfig() + .withUsername(dockerClientConfig.getRegistryUsername()) + .withPassword(dockerClientConfig.getRegistryPassword()) + .withEmail(dockerClientConfig.getRegistryEmail()) + .withRegistryAddress(dockerClientConfig.getRegistryUrl()); + } + + /** + * * MISC API * + */ + + /** + * Authenticate with the server, useful for checking authentication. + */ + @Override + public AuthCmd authCmd() { + return new AuthCmdImpl(getDockerCmdExecFactory().createAuthCmdExec(), authConfig()); + } + + @Override + public InfoCmd infoCmd() { + return new InfoCmdImpl(getDockerCmdExecFactory().createInfoCmdExec()); + } + + @Override + public PingCmd pingCmd() { + return new PingCmdImpl(getDockerCmdExecFactory().createPingCmdExec()); + } + + @Override + public VersionCmd versionCmd() { + return new VersionCmdImpl(getDockerCmdExecFactory().createVersionCmdExec()); + } + + /** + * * IMAGE API * + */ + @Override + public PullImageCmd pullImageCmd(String repository) { + return new PullImageCmdImpl(getDockerCmdExecFactory().createPullImageCmdExec(), + dockerClientConfig.effectiveAuthConfig(repository), repository); + } + + @Override + public PushImageCmd pushImageCmd(String name) { + PushImageCmd cmd = new PushImageCmdImpl(getDockerCmdExecFactory().createPushImageCmdExec(), + dockerClientConfig.effectiveAuthConfig(name), name); + return cmd; + } + + @Override + public PushImageCmd pushImageCmd(Identifier identifier) { + PushImageCmd cmd = pushImageCmd(identifier.repository.name); + if (identifier.tag.isPresent()) { + cmd.withTag(identifier.tag.get()); + } + + AuthConfig cfg = dockerClientConfig.effectiveAuthConfig(identifier.repository.name); + if (cfg != null) { + cmd.withAuthConfig(cfg); + } + + return cmd; + } + + @Override + public SaveImageCmd saveImageCmd(String name) { + return new SaveImageCmdImpl(getDockerCmdExecFactory().createSaveImageCmdExec(), name); + } + + @Override + public SaveImagesCmd saveImagesCmd() { + return new SaveImagesCmdImpl(getDockerCmdExecFactory().createSaveImagesCmdExec()); + } + + @Override + public CreateImageCmd createImageCmd(String repository, InputStream imageStream) { + return new CreateImageCmdImpl(getDockerCmdExecFactory().createCreateImageCmdExec(), repository, imageStream); + } + + @Override + public LoadImageCmd loadImageCmd(@Nonnull InputStream imageStream) { + return new LoadImageCmdImpl(getDockerCmdExecFactory().createLoadImageCmdExec(), imageStream); + } + + @Override + public LoadImageAsyncCmd loadImageAsyncCmd(@Nonnull InputStream imageStream) { + return new LoadImageAsyncCmdImpl(getDockerCmdExecFactory().createLoadImageAsyncCmdExec(), imageStream); + } + + @Override + public SearchImagesCmd searchImagesCmd(String term) { + return new SearchImagesCmdImpl(getDockerCmdExecFactory().createSearchImagesCmdExec(), term); + } + + @Override + public RemoveImageCmd removeImageCmd(String imageId) { + return new RemoveImageCmdImpl(getDockerCmdExecFactory().createRemoveImageCmdExec(), imageId); + } + + @Override + public ListImagesCmd listImagesCmd() { + return new ListImagesCmdImpl(getDockerCmdExecFactory().createListImagesCmdExec()); + } + + @Override + public InspectImageCmd inspectImageCmd(String imageId) { + return new InspectImageCmdImpl(getDockerCmdExecFactory().createInspectImageCmdExec(), imageId); + } + + /** + * * CONTAINER API * + */ + + @Override + public ListContainersCmd listContainersCmd() { + return new ListContainersCmdImpl(getDockerCmdExecFactory().createListContainersCmdExec()); + } + + @Override + public CreateContainerCmd createContainerCmd(String image) { + return new CreateContainerCmdImpl(getDockerCmdExecFactory() + .createCreateContainerCmdExec(), dockerClientConfig.effectiveAuthConfig(image), image); + } + + @Override + public StartContainerCmd startContainerCmd(String containerId) { + return new StartContainerCmdImpl(getDockerCmdExecFactory().createStartContainerCmdExec(), containerId); + } + + @Override + public InspectContainerCmd inspectContainerCmd(String containerId) { + return new InspectContainerCmdImpl(getDockerCmdExecFactory().createInspectContainerCmdExec(), containerId); + } + + @Override + public ExecCreateCmd execCreateCmd(String containerId) { + return new ExecCreateCmdImpl(getDockerCmdExecFactory().createExecCmdExec(), containerId); + } + + @Override + public ResizeExecCmd resizeExecCmd(@Nonnull String execId) { + return new ResizeExecCmdImpl(getDockerCmdExecFactory().createResizeExecCmdExec(), execId); + } + + @Override + public RemoveContainerCmd removeContainerCmd(String containerId) { + return new RemoveContainerCmdImpl(getDockerCmdExecFactory().createRemoveContainerCmdExec(), containerId); + } + + @Override + public WaitContainerCmd waitContainerCmd(String containerId) { + return new WaitContainerCmdImpl(getDockerCmdExecFactory().createWaitContainerCmdExec(), containerId); + } + + @Override + public AttachContainerCmd attachContainerCmd(String containerId) { + return new AttachContainerCmdImpl(getDockerCmdExecFactory().createAttachContainerCmdExec(), containerId); + } + + @Override + public ExecStartCmd execStartCmd(String execId) { + return new ExecStartCmdImpl(getDockerCmdExecFactory().createExecStartCmdExec(), execId); + } + + @Override + public InspectExecCmd inspectExecCmd(String execId) { + return new InspectExecCmdImpl(getDockerCmdExecFactory().createInspectExecCmdExec(), execId); + } + + @Override + public LogContainerCmd logContainerCmd(String containerId) { + return new LogContainerCmdImpl(getDockerCmdExecFactory().createLogContainerCmdExec(), containerId); + } + + @Override + public CopyFileFromContainerCmd copyFileFromContainerCmd(String containerId, String resource) { + return new CopyFileFromContainerCmdImpl(getDockerCmdExecFactory().createCopyFileFromContainerCmdExec(), + containerId, resource); + } + + @Override + public CopyArchiveFromContainerCmd copyArchiveFromContainerCmd(String containerId, String resource) { + return new CopyArchiveFromContainerCmdImpl(getDockerCmdExecFactory().createCopyArchiveFromContainerCmdExec(), + containerId, resource); + } + + @Override + public CopyArchiveToContainerCmd copyArchiveToContainerCmd(String containerId) { + return new CopyArchiveToContainerCmdImpl(getDockerCmdExecFactory().createCopyArchiveToContainerCmdExec(), + containerId); + } + + @Override + public ContainerDiffCmd containerDiffCmd(String containerId) { + return new ContainerDiffCmdImpl(getDockerCmdExecFactory().createContainerDiffCmdExec(), containerId); + } + + @Override + public StopContainerCmd stopContainerCmd(String containerId) { + return new StopContainerCmdImpl(getDockerCmdExecFactory().createStopContainerCmdExec(), containerId); + } + + @Override + public KillContainerCmd killContainerCmd(String containerId) { + return new KillContainerCmdImpl(getDockerCmdExecFactory().createKillContainerCmdExec(), containerId); + } + + @Override + public UpdateContainerCmd updateContainerCmd(@Nonnull String containerId) { + return new UpdateContainerCmdImpl(getDockerCmdExecFactory().createUpdateContainerCmdExec(), containerId); + } + + @Override + public RenameContainerCmd renameContainerCmd(@Nonnull String containerId) { + return new RenameContainerCmdImpl(getDockerCmdExecFactory().createRenameContainerCmdExec(), containerId); + } + + @Override + public RestartContainerCmd restartContainerCmd(String containerId) { + return new RestartContainerCmdImpl(getDockerCmdExecFactory().createRestartContainerCmdExec(), containerId); + } + + @Override + public ResizeContainerCmd resizeContainerCmd(@Nonnull String containerId) { + return new ResizeContainerCmdImpl(getDockerCmdExecFactory().createResizeContainerCmdExec(), containerId); + } + + @Override + public CommitCmd commitCmd(String containerId) { + return new CommitCmdImpl(getDockerCmdExecFactory().createCommitCmdExec(), containerId); + } + + @Override + public BuildImageCmd buildImageCmd() { + return new BuildImageCmdImpl(getDockerCmdExecFactory().createBuildImageCmdExec()); + } + + @Override + public BuildImageCmd buildImageCmd(File dockerFileOrFolder) { + return new BuildImageCmdImpl(getDockerCmdExecFactory().createBuildImageCmdExec(), dockerFileOrFolder); + } + + @Override + public BuildImageCmd buildImageCmd(InputStream tarInputStream) { + return new BuildImageCmdImpl(getDockerCmdExecFactory().createBuildImageCmdExec(), tarInputStream); + } + + @Override + public TopContainerCmd topContainerCmd(String containerId) { + return new TopContainerCmdImpl(getDockerCmdExecFactory().createTopContainerCmdExec(), containerId); + } + + @Override + public TagImageCmd tagImageCmd(String imageId, String imageNameWithRepository, String tag) { + return new TagImageCmdImpl(getDockerCmdExecFactory().createTagImageCmdExec(), imageId, imageNameWithRepository, tag); + } + + @Override + public PauseContainerCmd pauseContainerCmd(String containerId) { + return new PauseContainerCmdImpl(getDockerCmdExecFactory().createPauseContainerCmdExec(), containerId); + } + + @Override + public UnpauseContainerCmd unpauseContainerCmd(String containerId) { + return new UnpauseContainerCmdImpl(getDockerCmdExecFactory().createUnpauseContainerCmdExec(), containerId); + } + + @Override + public EventsCmd eventsCmd() { + return new EventsCmdImpl(getDockerCmdExecFactory().createEventsCmdExec()); + } + + @Override + public StatsCmd statsCmd(String containerId) { + return new StatsCmdImpl(getDockerCmdExecFactory().createStatsCmdExec(), containerId); + } + + @Override + public CreateVolumeCmd createVolumeCmd() { + return new CreateVolumeCmdImpl(getDockerCmdExecFactory().createCreateVolumeCmdExec()); + } + + @Override + public InspectVolumeCmd inspectVolumeCmd(String name) { + return new InspectVolumeCmdImpl(getDockerCmdExecFactory().createInspectVolumeCmdExec(), name); + } + + @Override + public RemoveVolumeCmd removeVolumeCmd(String name) { + return new RemoveVolumeCmdImpl(getDockerCmdExecFactory().createRemoveVolumeCmdExec(), name); + } + + @Override + public ListVolumesCmd listVolumesCmd() { + return new ListVolumesCmdImpl(getDockerCmdExecFactory().createListVolumesCmdExec()); + } + + @Override + public ListNetworksCmd listNetworksCmd() { + return new ListNetworksCmdImpl(getDockerCmdExecFactory().createListNetworksCmdExec()); + } + + @Override + public InspectNetworkCmd inspectNetworkCmd() { + return new InpectNetworkCmdImpl(getDockerCmdExecFactory().createInspectNetworkCmdExec()); + } + + @Override + public CreateNetworkCmd createNetworkCmd() { + return new CreateNetworkCmdImpl(getDockerCmdExecFactory().createCreateNetworkCmdExec()); + } + + @Override + public RemoveNetworkCmd removeNetworkCmd(String networkId) { + return new RemoveNetworkCmdImpl(getDockerCmdExecFactory().createRemoveNetworkCmdExec(), networkId); + } + + @Override + public ConnectToNetworkCmd connectToNetworkCmd() { + return new ConnectToNetworkCmdImpl(getDockerCmdExecFactory().createConnectToNetworkCmdExec()); + } + + @Override + public DisconnectFromNetworkCmd disconnectFromNetworkCmd() { + return new DisconnectFromNetworkCmdImpl(getDockerCmdExecFactory().createDisconnectFromNetworkCmdExec()); + } + + @Override + public InitializeSwarmCmd initializeSwarmCmd(SwarmSpec swarmSpec) { + return new InitializeSwarmCmdImpl(getDockerCmdExecFactory().createInitializeSwarmCmdExec(), swarmSpec); + } + + @Override + public InspectSwarmCmd inspectSwarmCmd() { + return new InspectSwarmCmdImpl(getDockerCmdExecFactory().createInspectSwarmCmdExec()); + } + + @Override + public JoinSwarmCmd joinSwarmCmd() { + return new JoinSwarmCmdImpl(getDockerCmdExecFactory().createJoinSwarmCmdExec()); + } + + @Override + public LeaveSwarmCmd leaveSwarmCmd() { + return new LeaveSwarmCmdImpl(getDockerCmdExecFactory().createLeaveSwarmCmdExec()); + } + + @Override + public UpdateSwarmCmd updateSwarmCmd(SwarmSpec swarmSpec) { + return new UpdateSwarmCmdImpl(getDockerCmdExecFactory().createUpdateSwarmCmdExec(), swarmSpec); + } + + @Override + public UpdateSwarmNodeCmd updateSwarmNodeCmd() { + return new UpdateSwarmNodeCmdImpl(getDockerCmdExecFactory().updateSwarmNodeCmdExec()); + } + + @Override + public RemoveSwarmNodeCmd removeSwarmNodeCmd(String swarmNodeId) { + return new RemoveSwarmNodeCmdImpl(getDockerCmdExecFactory().removeSwarmNodeCmdExec(), swarmNodeId); + } + + @Override + public ListSwarmNodesCmd listSwarmNodesCmd() { + return new ListSwarmNodesCmdImpl(getDockerCmdExecFactory().listSwarmNodeCmdExec()); + } + + @Override + public ListServicesCmd listServicesCmd() { + return new ListServicesCmdImpl(getDockerCmdExecFactory().createListServicesCmdExec()); + } + + @Override + public CreateServiceCmd createServiceCmd(ServiceSpec serviceSpec) { + return new CreateServiceCmdImpl(getDockerCmdExecFactory().createCreateServiceCmdExec(), serviceSpec); + } + + @Override + public InspectServiceCmd inspectServiceCmd(String serviceId) { + return new InspectServiceCmdImpl(getDockerCmdExecFactory().createInspectServiceCmdExec(), serviceId); + } + + @Override + public UpdateServiceCmd updateServiceCmd(String serviceId, ServiceSpec serviceSpec) { + return new UpdateServiceCmdImpl(getDockerCmdExecFactory().createUpdateServiceCmdExec(), serviceId, serviceSpec); + } + + @Override + public RemoveServiceCmd removeServiceCmd(String serviceId) { + return new RemoveServiceCmdImpl(getDockerCmdExecFactory().createRemoveServiceCmdExec(), serviceId); + } + + @Override + public LogSwarmObjectCmd logServiceCmd(String serviceId) { + return new LogSwarmObjectImpl(getDockerCmdExecFactory().logSwarmObjectExec("services"), serviceId); + } + + @Override + public LogSwarmObjectCmd logTaskCmd(String taskId) { + return new LogSwarmObjectImpl(getDockerCmdExecFactory().logSwarmObjectExec("tasks"), taskId); + } + + @Override + public PruneCmd pruneCmd(PruneType pruneType) { + return new PruneCmdImpl(getDockerCmdExecFactory().pruneCmdExec(), pruneType); + } + + @Override + public ListSecretsCmd listSecretsCmd() { + return new ListSecretsCmdImpl(getDockerCmdExecFactory().createListSecretsCmdExec()); + } + + @Override + public CreateSecretCmd createSecretCmd(SecretSpec secretSpec) { + return new CreateSecretCmdImpl(getDockerCmdExecFactory().createCreateSecretCmdExec(), secretSpec); + } + + @Override + public RemoveSecretCmd removeSecretCmd(String secretId) { + return new RemoveSecretCmdImpl(getDockerCmdExecFactory().createRemoveSecretCmdExec(), secretId); + } + + @Override + public ListConfigsCmd listConfigsCmd() { + return new ListConfigsCmdImpl(getDockerCmdExecFactory().createListConfigsCmdExec()); + } + + @Override + public CreateConfigCmd createConfigCmd() { + return new CreateConfigCmdImpl(getDockerCmdExecFactory().createCreateConfigCmdExec()); + } + + @Override + public InspectConfigCmd inspectConfigCmd(String configId) { + return new InspectConfigCmdImpl(getDockerCmdExecFactory().createInspectConfigCmdExec(), configId); + } + + + @Override + public RemoveConfigCmd removeConfigCmd(String configId) { + return new RemoveConfigCmdImpl(getDockerCmdExecFactory().createRemoveConfigCmdExec(), configId); + } + + + @Override + public ListTasksCmd listTasksCmd() { + return new ListTasksCmdImpl(getDockerCmdExecFactory().listTasksCmdExec()); + } + + @Override + public void close() throws IOException { + getDockerCmdExecFactory().close(); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/DockerConfigFile.java b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerConfigFile.java new file mode 100644 index 000000000..39ef15271 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerConfigFile.java @@ -0,0 +1,250 @@ +package com.github.dockerjava.core; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.fasterxml.jackson.annotation.JsonSetter; +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.AuthConfigurations; +import java.util.Objects; +import org.apache.commons.io.FileUtils; +import org.apache.commons.lang3.StringUtils; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import java.io.File; +import java.io.IOException; +import java.nio.charset.StandardCharsets; +import java.util.Base64; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +public class DockerConfigFile { + private static final String DOCKER_LEGACY_CFG = ".dockercfg"; + private static final String DOCKER_CFG = "config.json"; + + private static final TypeReference> CONFIG_MAP_TYPE = new TypeReference>() { + }; + + @JsonProperty + private Map auths; + + @JsonProperty + private String currentContext; + + public DockerConfigFile() { + this(new HashMap<>()); + } + + private DockerConfigFile(Map authConfigMap) { + auths = authConfigMap; + } + + @Nonnull + public Map getAuths() { + return auths; + } + + @JsonSetter + public void setAuths(Map authConfigMap) { + auths = (authConfigMap == null || authConfigMap.size() == 0) ? new HashMap<>() : authConfigMap; + } + + void addAuthConfig(AuthConfig config) { + auths.put(config.getRegistryAddress(), config); + } + + void setCurrentContext(String currentContext) { + this.currentContext = currentContext; + } + + public String getCurrentContext() { + return currentContext; + } + + @CheckForNull + public AuthConfig resolveAuthConfig(@CheckForNull String hostname) { + if (StringUtils.isEmpty(hostname) || AuthConfig.DEFAULT_SERVER_ADDRESS.equals(hostname)) { + return auths.get(AuthConfig.DEFAULT_SERVER_ADDRESS); + } + + AuthConfig c = auths.get(hostname); + if (c != null) { + return c; + } + + // Maybe they have a legacy config file, we will iterate the keys converting + // them to the new format and testing + String normalizedHostname = convertToHostname(hostname); + for (Map.Entry entry : auths.entrySet()) { + String registry = entry.getKey(); + AuthConfig config = entry.getValue(); + if (convertToHostname(registry).equals(normalizedHostname)) { + return config; + } + } + + return null; + } + + @Nonnull + public AuthConfigurations getAuthConfigurations() { + final AuthConfigurations authConfigurations = new AuthConfigurations(); + for (Map.Entry authConfigEntry : auths.entrySet()) { + authConfigurations.addConfig(authConfigEntry.getValue()); + } + + return authConfigurations; + } + + // CHECKSTYLE:OFF + @Override + public int hashCode() { + final int prime = 31; + int result = 1; + result = prime * result + ((auths == null) ? 0 : auths.hashCode()); + return result; + } + + @Override + public boolean equals(Object obj) { + if (this == obj) + return true; + if (obj == null) + return false; + if (getClass() != obj.getClass()) + return false; + DockerConfigFile other = (DockerConfigFile) obj; + if (auths == null) { + if (other.auths != null) + return false; + } else if (!auths.equals(other.auths)) + return false; + if (!Objects.equals(currentContext, other.currentContext)) { + return false; + } + return true; + } + + // CHECKSTYLE:ON + + @Override + public String toString() { + return "DockerConfigFile [auths=" + auths + ", currentContext='" + currentContext + "']"; + } + + @Nonnull + @Deprecated + public static DockerConfigFile loadConfig(@CheckForNull String dockerConfigPath) throws IOException { + return loadConfig(DefaultObjectMapperHolder.INSTANCE.getObjectMapper(), dockerConfigPath); + } + + @Nonnull + public static DockerConfigFile loadConfig(ObjectMapper objectMapper, @CheckForNull String dockerConfigPath) throws IOException { + // no any configs, but for empty auths return non null object + if (dockerConfigPath == null) { + return new DockerConfigFile(); + } + + //parse new docker config file format + DockerConfigFile dockerConfig = loadCurrentConfig(objectMapper, dockerConfigPath); + + //parse old auth config file format + if (dockerConfig == null) { + dockerConfig = loadLegacyConfig(objectMapper, dockerConfigPath); + } + + //otherwise create default config + if (dockerConfig == null) { + dockerConfig = new DockerConfigFile(); + } + + for (Map.Entry entry : dockerConfig.getAuths().entrySet()) { + AuthConfig authConfig = entry.getValue(); + decodeAuth(authConfig); + authConfig.withAuth(null); + authConfig.withRegistryAddress(entry.getKey()); + } + + return dockerConfig; + } + + @CheckForNull + private static DockerConfigFile loadCurrentConfig(ObjectMapper objectMapper, @CheckForNull String dockerConfigPath) throws IOException { + File dockerCfgFile = new File(dockerConfigPath, DOCKER_CFG); + + if (!dockerCfgFile.exists() || !dockerCfgFile.isFile()) { + return null; + } + + try { + return objectMapper.readValue(dockerCfgFile, DockerConfigFile.class); + } catch (IOException e) { + throw new IOException("Failed to parse docker " + DOCKER_CFG, e); + } + } + + @CheckForNull + private static DockerConfigFile loadLegacyConfig(ObjectMapper objectMapper, String dockerConfigPath) throws IOException { + File dockerLegacyCfgFile = new File(dockerConfigPath, DOCKER_LEGACY_CFG); + + if (!dockerLegacyCfgFile.exists() || !dockerLegacyCfgFile.isFile()) { + return null; + } + + //parse legacy auth config file format + try { + return new DockerConfigFile(objectMapper.>readValue(dockerLegacyCfgFile, CONFIG_MAP_TYPE)); + } catch (IOException e) { + // pass + } + + List authFileContent = FileUtils.readLines(dockerLegacyCfgFile, StandardCharsets.UTF_8); + if (authFileContent.size() < 2) { + throw new IOException("The Auth Config file is empty"); + } + + AuthConfig config = new AuthConfig(); + String[] origAuth = authFileContent.get(0).split(" = "); + if (origAuth.length != 2) { + throw new IOException("Invalid Auth config file"); + } + + config.withAuth(origAuth[1]); + + String[] origEmail = authFileContent.get(1).split(" = "); + if (origEmail.length != 2) { + throw new IOException("Invalid Auth config file"); + } + config.withEmail(origEmail[1]); + + return new DockerConfigFile(new HashMap<>(Collections.singletonMap(config.getRegistryAddress(), config))); + } + + private static void decodeAuth(AuthConfig config) throws IOException { + if (config.getAuth() == null) { + return; + } + + String str = new String(Base64.getDecoder().decode(config.getAuth()), StandardCharsets.UTF_8); + String[] parts = str.split(":", 2); + if (parts.length != 2) { + throw new IOException("Invalid auth configuration file"); + } + config.withUsername(parts[0]); + config.withPassword(parts[1]); + } + + static String convertToHostname(String server) { + String stripped = server; + if (server.startsWith("http://")) { + stripped = server.substring(7); + } else if (server.startsWith("https://")) { + stripped = server.substring(8); + } + String[] numParts = stripped.split("/", 2); + return numParts[0]; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/DockerContextMetaFile.java b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerContextMetaFile.java new file mode 100644 index 000000000..e10db4498 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/DockerContextMetaFile.java @@ -0,0 +1,71 @@ +package com.github.dockerjava.core; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.google.common.hash.HashFunction; +import com.google.common.hash.Hashing; +import java.io.File; +import java.io.IOException; +import java.nio.charset.StandardCharsets; +import java.util.Optional; + +public class DockerContextMetaFile { + private static HashFunction metaHashFunction = Hashing.sha256(); + + @JsonProperty("Name") + String name; + + @JsonProperty("Endpoints") + Endpoints endpoints; + + + public static class Endpoints { + @JsonProperty("docker") + Docker docker; + + public static class Docker { + @JsonProperty("Host") + String host; + + @JsonProperty("SkipTLSVerify") + boolean skipTLSVerify; + } + } + + + public static Optional resolveContextMetaFile(ObjectMapper objectMapper, File dockerConfigPath, String context) { + final File path = dockerConfigPath.toPath() + .resolve("contexts") + .resolve("meta") + .resolve(metaHashFunction.hashString(context, StandardCharsets.UTF_8).toString()) + .resolve("meta.json") + .toFile(); + return Optional.ofNullable(loadContextMetaFile(objectMapper, path)); + } + + public static Optional resolveContextTLSFile(File dockerConfigPath, String context) { + final File path = dockerConfigPath.toPath() + .resolve("contexts") + .resolve("tls") + .resolve(metaHashFunction.hashString(context, StandardCharsets.UTF_8).toString()) + .resolve("docker") + .toFile(); + return Optional.ofNullable(path).filter(File::exists); + } + + public static DockerContextMetaFile loadContextMetaFile(ObjectMapper objectMapper, File dockerContextMetaFile) { + try { + return parseContextMetaFile(objectMapper, dockerContextMetaFile); + } catch (Exception exception) { + return null; + } + } + + public static DockerContextMetaFile parseContextMetaFile(ObjectMapper objectMapper, File dockerContextMetaFile) throws IOException { + try { + return objectMapper.readValue(dockerContextMetaFile, DockerContextMetaFile.class); + } catch (IOException e) { + throw new IOException("Failed to parse docker context meta file " + dockerContextMetaFile, e); + } + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/FramedInputStreamConsumer.java b/docker-java-core/src/main/java/com/github/dockerjava/core/FramedInputStreamConsumer.java new file mode 100644 index 000000000..c81521076 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/FramedInputStreamConsumer.java @@ -0,0 +1,100 @@ +package com.github.dockerjava.core; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.api.model.StreamType; +import com.github.dockerjava.transport.DockerHttpClient; + +import java.io.InputStream; +import java.util.Arrays; +import java.util.function.Consumer; + +class FramedInputStreamConsumer implements Consumer { + + private final ResultCallback resultCallback; + + FramedInputStreamConsumer(ResultCallback resultCallback) { + this.resultCallback = resultCallback; + } + + @Override + public void accept(DockerHttpClient.Response response) { + try { + InputStream body = response.getBody(); + + byte[] buffer = new byte[1024]; + while (true) { + // See https://docs.docker.com/engine/api/v1.37/#operation/ContainerAttach + // [8]byte{STREAM_TYPE, 0, 0, 0, SIZE1, SIZE2, SIZE3, SIZE4}[]byte{OUTPUT} + + int streamTypeByte = body.read(); + if (streamTypeByte < 0) { + return; + } + + StreamType streamType = streamType(streamTypeByte); + + if (streamType == StreamType.RAW) { + resultCallback.onNext(new Frame(StreamType.RAW, new byte[]{(byte) streamTypeByte})); + + int readBytes; + while ((readBytes = body.read(buffer)) >= 0) { + if (readBytes == buffer.length) { + resultCallback.onNext(new Frame(StreamType.RAW, buffer)); + } else { + resultCallback.onNext(new Frame(StreamType.RAW, Arrays.copyOf(buffer, readBytes))); + } + } + return; + } + + // Skip 3 bytes + for (int i = 0; i < 3; i++) { + if (body.read() < 0) { + return; + } + } + + // uint32 encoded as big endian. + int bytesToRead = 0; + for (int i = 0; i < 4; i++) { + int readByte = body.read(); + if (readByte < 0) { + return; + } + bytesToRead |= (readByte & 0xff) << (8 * (3 - i)); + } + + do { + int readBytes = body.read(buffer, 0, Math.min(buffer.length, bytesToRead)); + if (readBytes < 0) { + // TODO log? + return; + } + + if (readBytes == buffer.length) { + resultCallback.onNext(new Frame(streamType, buffer)); + } else { + resultCallback.onNext(new Frame(streamType, Arrays.copyOf(buffer, readBytes))); + } + bytesToRead -= readBytes; + } while (bytesToRead > 0); + } + } catch (Exception e) { + resultCallback.onError(e); + } + } + + private static StreamType streamType(int streamType) { + switch (streamType) { + case 0: + return StreamType.STDIN; + case 1: + return StreamType.STDOUT; + case 2: + return StreamType.STDERR; + default: + return StreamType.RAW; + } + } +} diff --git a/src/main/java/com/github/dockerjava/core/GoLangFileMatch.java b/docker-java-core/src/main/java/com/github/dockerjava/core/GoLangFileMatch.java similarity index 91% rename from src/main/java/com/github/dockerjava/core/GoLangFileMatch.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/GoLangFileMatch.java index b26b7db44..811b60ce5 100644 --- a/src/main/java/com/github/dockerjava/core/GoLangFileMatch.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/GoLangFileMatch.java @@ -6,9 +6,15 @@ import java.io.File; import java.util.ArrayList; import java.util.List; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.TimeUnit; import java.util.regex.Pattern; -import org.apache.commons.lang.StringUtils; +import com.google.common.cache.CacheBuilder; +import com.google.common.cache.CacheLoader; +import com.google.common.cache.LoadingCache; +import com.google.common.util.concurrent.UncheckedExecutionException; +import org.apache.commons.lang3.StringUtils; import com.github.dockerjava.core.exception.GoLangFileMatchException; @@ -52,6 +58,11 @@ private GoLangFileMatch() { private static final String PATTERN_CHARS_TO_ESCAPE = "\\.[]{}()*+-?^$|"; + private static final LoadingCache PATTERN_CACHE = CacheBuilder.newBuilder() + .expireAfterAccess(1, TimeUnit.HOURS) + .maximumSize(10_000) + .build(CacheLoader.from(GoLangFileMatch::buildPattern)); + public static boolean match(List patterns, File file) { return !match(patterns, file.getPath()).isEmpty(); } @@ -64,7 +75,7 @@ public static boolean match(String pattern, File file) { * Returns the matching patterns for the given string */ public static List match(List patterns, String name) { - List matches = new ArrayList(); + List matches = new ArrayList<>(); for (String pattern : patterns) { if (match(pattern, name)) { matches.add(pattern); @@ -74,7 +85,11 @@ public static List match(List patterns, String name) { } public static boolean match(String pattern, String name) { - return buildPattern(pattern).matcher(name).matches(); + try { + return PATTERN_CACHE.get(pattern).matcher(name).matches(); + } catch (ExecutionException | UncheckedExecutionException e) { + throw new GoLangFileMatchException(e.getCause().getMessage()); + } } private static Pattern buildPattern(String pattern) { diff --git a/src/main/java/com/github/dockerjava/core/GoLangMatchFileFilter.java b/docker-java-core/src/main/java/com/github/dockerjava/core/GoLangMatchFileFilter.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/GoLangMatchFileFilter.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/GoLangMatchFileFilter.java diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/InvocationBuilder.java b/docker-java-core/src/main/java/com/github/dockerjava/core/InvocationBuilder.java new file mode 100644 index 000000000..88b8707cf --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/InvocationBuilder.java @@ -0,0 +1,94 @@ +package com.github.dockerjava.core; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.core.async.ResultCallbackTemplate; + +import java.io.IOException; +import java.io.InputStream; +import java.util.concurrent.CountDownLatch; + +public interface InvocationBuilder { + + InvocationBuilder accept(MediaType mediaType); + + InvocationBuilder header(String name, String value); + + void delete(); + + void get(ResultCallback resultCallback); + + T get(TypeReference typeReference); + + void get(TypeReference typeReference, ResultCallback resultCallback); + + InputStream post(Object entity); + + void post(Object entity, InputStream stdin, ResultCallback resultCallback); + + T post(Object entity, TypeReference typeReference); + + void post(Object entity, TypeReference typeReference, ResultCallback resultCallback); + + T post(TypeReference typeReference, InputStream body); + + void post(TypeReference typeReference, ResultCallback resultCallback, InputStream body); + + void postStream(InputStream body); + + InputStream get(); + + void put(InputStream body, MediaType mediaType); + + /** + * Implementation of {@link ResultCallback} with the single result event expected. + */ + class AsyncResultCallback + extends ResultCallbackTemplate, A_RES_T> { + + private A_RES_T result = null; + + private final CountDownLatch resultReady = new CountDownLatch(1); + + @Override + public void onNext(A_RES_T object) { + onResult(object); + } + + private void onResult(A_RES_T object) { + if (resultReady.getCount() == 0) { + throw new IllegalStateException("Result has already been set"); + } + + try { + result = object; + } finally { + resultReady.countDown(); + } + } + + @Override + public void close() throws IOException { + try { + super.close(); + } finally { + resultReady.countDown(); + } + } + + /** + * Blocks until {@link ResultCallback#onNext(Object)} was called for the first time + */ + @SuppressWarnings("unchecked") + public A_RES_T awaitResult() { + try { + resultReady.await(); + } catch (InterruptedException e) { + throw new RuntimeException(e); + } + throwFirstError(); + return result; + } + } +} diff --git a/src/main/java/com/github/dockerjava/core/KeystoreSSLConfig.java b/docker-java-core/src/main/java/com/github/dockerjava/core/KeystoreSSLConfig.java similarity index 95% rename from src/main/java/com/github/dockerjava/core/KeystoreSSLConfig.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/KeystoreSSLConfig.java index 43946a53b..73491bb43 100644 --- a/src/main/java/com/github/dockerjava/core/KeystoreSSLConfig.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/KeystoreSSLConfig.java @@ -1,7 +1,5 @@ package com.github.dockerjava.core; -import static com.google.common.base.Preconditions.checkNotNull; - import java.io.File; import java.io.FileInputStream; import java.io.IOException; @@ -14,6 +12,7 @@ import java.security.UnrecoverableKeyException; import java.security.cert.CertificateException; import java.security.cert.X509Certificate; +import java.util.Objects; import javax.net.ssl.KeyManagerFactory; import javax.net.ssl.SSLContext; @@ -37,8 +36,7 @@ public class KeystoreSSLConfig implements SSLConfig, Serializable { */ public KeystoreSSLConfig(KeyStore keystore, String keystorePassword) { this.keystorePassword = keystorePassword; - checkNotNull(keystore); - this.keystore = keystore; + this.keystore = Objects.requireNonNull(keystore); } /** @@ -54,8 +52,8 @@ public KeystoreSSLConfig(KeyStore keystore, String keystorePassword) { */ public KeystoreSSLConfig(File pfxFile, String password) throws KeyStoreException, IOException, CertificateException, NoSuchAlgorithmException { - checkNotNull(pfxFile); - checkNotNull(password); + Objects.requireNonNull(pfxFile); + Objects.requireNonNull(password); keystore = KeyStore.getInstance("pkcs12"); try (FileInputStream fs = new FileInputStream(pfxFile)) { keystore.load(fs, password.toCharArray()); diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/LocalDirectorySSLConfig.java b/docker-java-core/src/main/java/com/github/dockerjava/core/LocalDirectorySSLConfig.java new file mode 100644 index 000000000..0f50f561d --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/LocalDirectorySSLConfig.java @@ -0,0 +1,115 @@ +package com.github.dockerjava.core; + +import java.io.File; +import java.io.Serializable; +import java.nio.file.Files; +import java.nio.file.Paths; +import java.security.AccessController; +import java.security.PrivilegedAction; +import java.security.Security; +import java.util.Objects; + +import javax.net.ssl.KeyManagerFactory; +import javax.net.ssl.SSLContext; +import javax.net.ssl.TrustManagerFactory; + +import org.bouncycastle.jce.provider.BouncyCastleProvider; + +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.core.util.CertificateUtils; + +/** + * SSL Config from local files. + */ +public class LocalDirectorySSLConfig implements SSLConfig, Serializable { + + private static final long serialVersionUID = -4736328026418377358L; + + private final String dockerCertPath; + + public LocalDirectorySSLConfig(String dockerCertPath) { + this.dockerCertPath = Objects.requireNonNull(dockerCertPath); + } + + public String getDockerCertPath() { + return dockerCertPath; + } + + @Override + public SSLContext getSSLContext() { + + boolean certificatesExist = CertificateUtils.verifyCertificatesExist(dockerCertPath); + + if (certificatesExist) { + + try { + + Security.addProvider(new BouncyCastleProvider()); + + String caPemPath = dockerCertPath + File.separator + "ca.pem"; + String keyPemPath = dockerCertPath + File.separator + "key.pem"; + String certPemPath = dockerCertPath + File.separator + "cert.pem"; + + String keypem = new String(Files.readAllBytes(Paths.get(keyPemPath))); + String certpem = new String(Files.readAllBytes(Paths.get(certPemPath))); + String capem = new String(Files.readAllBytes(Paths.get(caPemPath))); + + String kmfAlgorithm = AccessController.doPrivileged(getSystemProperty("ssl.keyManagerFactory.algorithm", + KeyManagerFactory.getDefaultAlgorithm())); + KeyManagerFactory keyManagerFactory = KeyManagerFactory.getInstance(kmfAlgorithm); + keyManagerFactory.init(CertificateUtils.createKeyStore(keypem, certpem), "docker".toCharArray()); + + String tmfAlgorithm = AccessController.doPrivileged(getSystemProperty("ssl.trustManagerFactory.algorithm", + TrustManagerFactory.getDefaultAlgorithm())); + TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(tmfAlgorithm); + trustManagerFactory.init(CertificateUtils.createTrustStore(capem)); + + SSLContext sslContext = SSLContext.getInstance(AccessController.doPrivileged(getSystemProperty("ssl.protocol", + "TLSv1.2"))); + sslContext.init(keyManagerFactory.getKeyManagers(), trustManagerFactory.getTrustManagers(), null); + + return sslContext; + + } catch (Exception e) { + throw new DockerClientException(e.getMessage(), e); + } + + } + + return null; + + } + + private PrivilegedAction getSystemProperty(final String name, final String def) { + return () -> System.getProperty(name, def); + } + + @Override + public boolean equals(Object o) { + if (this == o) { + return true; + } + if (o == null || getClass() != o.getClass()) { + return false; + } + + LocalDirectorySSLConfig that = (LocalDirectorySSLConfig) o; + + if (!dockerCertPath.equals(that.dockerCertPath)) { + return false; + } + + return true; + } + + @Override + public int hashCode() { + return dockerCertPath.hashCode(); + } + + @Override + public String toString() { + return new StringBuilder().append(this.getClass().getSimpleName()).append("{").append("dockerCertPath=") + .append(dockerCertPath).append("}").toString(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/MediaType.java b/docker-java-core/src/main/java/com/github/dockerjava/core/MediaType.java new file mode 100644 index 000000000..dbcc77149 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/MediaType.java @@ -0,0 +1,22 @@ +package com.github.dockerjava.core; + +/** + * + * @author Marcus Linke + */ +public enum MediaType { + + APPLICATION_JSON("application/json"), + APPLICATION_OCTET_STREAM("application/octet-stream"), + APPLICATION_X_TAR("application/x-tar"); + + private String mediaType; + + MediaType(String mediaType) { + this.mediaType = mediaType; + } + + public String getMediaType() { + return mediaType; + } +} diff --git a/src/main/java/com/github/dockerjava/core/NameParser.java b/docker-java-core/src/main/java/com/github/dockerjava/core/NameParser.java similarity index 78% rename from src/main/java/com/github/dockerjava/core/NameParser.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/NameParser.java index 1db59e3a0..f06adb6d8 100644 --- a/src/main/java/com/github/dockerjava/core/NameParser.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/NameParser.java @@ -5,9 +5,10 @@ import java.util.regex.Pattern; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; -import org.apache.commons.lang.builder.ToStringStyle; +import org.apache.commons.lang3.StringUtils; +import org.apache.commons.lang3.builder.EqualsBuilder; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; import com.github.dockerjava.api.model.AuthConfig; import com.github.dockerjava.core.exception.InvalidRepositoryNameException; @@ -20,6 +21,9 @@ private NameParser() { // CHECKSTYLE:OFF private static final int RepositoryNameTotalLengthMax = 255; + private static final String SHA256_SEPARATOR = "@sha256:"; + private static final String COLON_SEPARATOR = ":"; + private static final Pattern RepositoryNameComponentRegexp = Pattern.compile("[a-z0-9]+(?:[._-][a-z0-9]+)*"); private static final Pattern RepositoryNameComponentAnchoredRegexp = Pattern.compile("^" @@ -37,6 +41,9 @@ public static ReposTag parseRepositoryTag(String name) { return new ReposTag(name, ""); } String tag = name.substring(n + 1); + if (StringUtils.containsIgnoreCase(name, SHA256_SEPARATOR)) { + return new ReposTag(name, ""); + } if (!tag.contains("/")) { return new ReposTag(name.substring(0, n), tag); } @@ -94,13 +101,19 @@ public static void validateRepoName(String name) { public static HostnameReposName resolveRepositoryName(String reposName) { if (reposName.contains("://")) { - // It cannot contain a scheme! - throw new InvalidRepositoryNameException(); + throw new InvalidRepositoryNameException("RepositoryName shouldn't contain a scheme"); } String[] nameParts = reposName.split("/", 2); if (nameParts.length == 1 || (!nameParts[0].contains(".") && !nameParts[0].contains(":") && !nameParts[0].equals("localhost"))) { + if (StringUtils.containsIgnoreCase(reposName, SHA256_SEPARATOR)) { + reposName = StringUtils.substringBeforeLast(reposName, SHA256_SEPARATOR); + } + + if (StringUtils.contains(reposName, COLON_SEPARATOR)) { + reposName = StringUtils.substringBeforeLast(reposName, COLON_SEPARATOR); + } return new HostnameReposName(AuthConfig.DEFAULT_SERVER_ADDRESS, reposName); } @@ -110,6 +123,13 @@ public static HostnameReposName resolveRepositoryName(String reposName) { throw new InvalidRepositoryNameException(String.format("Invalid repository name, try \"%s\" instead", reposName)); } + if (StringUtils.containsIgnoreCase(reposName, SHA256_SEPARATOR)) { + reposName = StringUtils.substringBeforeLast(reposName, SHA256_SEPARATOR); + } + + if (StringUtils.contains(reposName, COLON_SEPARATOR)) { + reposName = StringUtils.substringBeforeLast(reposName, COLON_SEPARATOR); + } validateRepoName(reposName); return new HostnameReposName(hostname, reposName); diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/RemoteApiVersion.java b/docker-java-core/src/main/java/com/github/dockerjava/core/RemoteApiVersion.java new file mode 100644 index 000000000..373a67332 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/RemoteApiVersion.java @@ -0,0 +1,212 @@ +package com.github.dockerjava.core; + +import com.google.common.base.MoreObjects; +import com.google.common.base.Objects; +import com.google.common.base.Preconditions; +import com.google.common.base.Strings; + +import java.io.Serializable; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +/** + * Bean to encapsulate the version of the Docker Remote (REST) + * API + *

+ * Contains the minor and major version of the API as well as operations to compare API versions. + * + * @author Marcus Thiesen + */ +public class RemoteApiVersion implements Serializable { + private static final long serialVersionUID = -5382212999262115459L; + + private static final Pattern VERSION_REGEX = Pattern.compile("v?(\\d+)\\.(\\d+)"); + + /** + * Online documentation is not available anymore. + */ + public static final RemoteApiVersion VERSION_1_7 = RemoteApiVersion.create(1, 7); + + /** + * @see Docker API 1.16 + */ + public static final RemoteApiVersion VERSION_1_16 = RemoteApiVersion.create(1, 16); + + /** + * @see Docker API 1.17 + */ + public static final RemoteApiVersion VERSION_1_17 = RemoteApiVersion.create(1, 17); + + /** + * @see Docker API 1.18 + */ + public static final RemoteApiVersion VERSION_1_18 = RemoteApiVersion.create(1, 18); + + /** + * @see Docker API 1.19 + */ + public static final RemoteApiVersion VERSION_1_19 = RemoteApiVersion.create(1, 19); + + /** + * @see Docker API 1.20 + */ + public static final RemoteApiVersion VERSION_1_20 = RemoteApiVersion.create(1, 20); + + /** + * @see Docker API 1.21 + */ + public static final RemoteApiVersion VERSION_1_21 = RemoteApiVersion.create(1, 21); + + /** + * @see Docker API 1.22 + */ + public static final RemoteApiVersion VERSION_1_22 = RemoteApiVersion.create(1, 22); + + /** + * @see Docker API 1.23 + */ + public static final RemoteApiVersion VERSION_1_23 = RemoteApiVersion.create(1, 23); + + /** + * @see Docker API 1.24 + */ + public static final RemoteApiVersion VERSION_1_24 = RemoteApiVersion.create(1, 24); + + /* + * @see Docker API 1.25 + */ + public static final RemoteApiVersion VERSION_1_25 = RemoteApiVersion.create(1, 25); + + public static final RemoteApiVersion VERSION_1_26 = RemoteApiVersion.create(1, 26); + public static final RemoteApiVersion VERSION_1_27 = RemoteApiVersion.create(1, 27); + public static final RemoteApiVersion VERSION_1_28 = RemoteApiVersion.create(1, 28); + public static final RemoteApiVersion VERSION_1_29 = RemoteApiVersion.create(1, 29); + public static final RemoteApiVersion VERSION_1_30 = RemoteApiVersion.create(1, 30); + public static final RemoteApiVersion VERSION_1_31 = RemoteApiVersion.create(1, 31); + public static final RemoteApiVersion VERSION_1_32 = RemoteApiVersion.create(1, 32); + public static final RemoteApiVersion VERSION_1_33 = RemoteApiVersion.create(1, 33); + public static final RemoteApiVersion VERSION_1_34 = RemoteApiVersion.create(1, 34); + public static final RemoteApiVersion VERSION_1_35 = RemoteApiVersion.create(1, 35); + public static final RemoteApiVersion VERSION_1_36 = RemoteApiVersion.create(1, 36); + public static final RemoteApiVersion VERSION_1_37 = RemoteApiVersion.create(1, 37); + public static final RemoteApiVersion VERSION_1_38 = RemoteApiVersion.create(1, 38); + public static final RemoteApiVersion VERSION_1_40 = RemoteApiVersion.create(1, 40); + public static final RemoteApiVersion VERSION_1_41 = RemoteApiVersion.create(1, 41); + public static final RemoteApiVersion VERSION_1_42 = RemoteApiVersion.create(1, 42); + public static final RemoteApiVersion VERSION_1_43 = RemoteApiVersion.create(1, 43); + public static final RemoteApiVersion VERSION_1_44 = RemoteApiVersion.create(1, 44); + + + /** + * Unknown, docker doesn't reflect reality. I.e. we implemented method, but for javadoc it not clear when it was added. + */ + public static final RemoteApiVersion UNKNOWN_VERSION = new RemoteApiVersion(0, 0) { + + @Override + public boolean isGreaterOrEqual(final RemoteApiVersion other) { + return false; + } + + @Override + public boolean isGreater(RemoteApiVersion other) { + return false; + } + + @Override + public String toString() { + return MoreObjects.toStringHelper(this).addValue("UNKNOWN_VERSION").toString(); + } + + @Override + public String asWebPathPart() { + return ""; + } + }; + + private final int major; + + private final int minor; + + private RemoteApiVersion(final int major, final int minor) { + this.major = major; + this.minor = minor; + } + + public static RemoteApiVersion create(final int major, final int minor) { + Preconditions.checkArgument(major > 0, "Major version must be bigger than 0 but is " + major); + Preconditions.checkArgument(minor > 0, "Minor version must be bigger than 0 but is " + minor); + return new RemoteApiVersion(major, minor); + } + + public static RemoteApiVersion unknown() { + return UNKNOWN_VERSION; + } + + public static RemoteApiVersion parseConfig(final String version) { + Preconditions.checkArgument(version != null, "Version must not be null"); + final Matcher matcher = VERSION_REGEX.matcher(version); + if (matcher.matches()) { + return create(Integer.parseInt(matcher.group(1)), Integer.parseInt(matcher.group(2))); + } + throw new IllegalArgumentException(version + " can not be parsed"); + } + + public static RemoteApiVersion parseConfigWithDefault(final String version) { + if (Strings.isNullOrEmpty(version)) { + return UNKNOWN_VERSION; + } + + try { + return parseConfig(version); + } catch (IllegalArgumentException e) { + return UNKNOWN_VERSION; + } + } + + public boolean isGreaterOrEqual(final RemoteApiVersion other) { + if (major >= other.major && minor >= other.minor) { + return true; + } + return false; + } + + public boolean isGreater(final RemoteApiVersion other) { + return major > other.major || (major == other.major && minor > other.minor); + + } + + /** + * @return String representation of version. i.e. "1.22" + */ + public String getVersion() { + return major + "." + minor; + } + + // CHECKSTYLE:OFF + @Override + public boolean equals(final Object o) { + if (this == o) + return true; + if (o == null || getClass() != o.getClass()) + return false; + final RemoteApiVersion that = (RemoteApiVersion) o; + return Objects.equal(major, that.major) && Objects.equal(minor, that.minor); + } + + // CHECKSTYLE:ON + + @Override + public int hashCode() { + return Objects.hashCode(major, minor); + } + + @Override + public String toString() { + return MoreObjects.toStringHelper(this).add("major", major).add("minor", minor).toString(); + } + + public String asWebPathPart() { + return "v" + major + "." + minor; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/SSLConfig.java b/docker-java-core/src/main/java/com/github/dockerjava/core/SSLConfig.java new file mode 100644 index 000000000..c3a05efb2 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/SSLConfig.java @@ -0,0 +1,10 @@ +package com.github.dockerjava.core; + +/** + * + * @deprecated use {@link com.github.dockerjava.transport.SSLConfig} + */ +@Deprecated +public interface SSLConfig extends com.github.dockerjava.transport.SSLConfig { + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/WebTarget.java b/docker-java-core/src/main/java/com/github/dockerjava/core/WebTarget.java new file mode 100644 index 000000000..d5a14d0a9 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/WebTarget.java @@ -0,0 +1,19 @@ +package com.github.dockerjava.core; + +import java.util.Map; +import java.util.Set; + +public interface WebTarget { + + WebTarget path(String... components); + + InvocationBuilder request(); + + WebTarget resolveTemplate(String name, Object value); + + WebTarget queryParam(String name, Object value); + + WebTarget queryParamsSet(String name, Set values); + + WebTarget queryParamsJsonMap(String name, Map values); +} diff --git a/src/main/java/com/github/dockerjava/core/async/FrameStreamProcessor.java b/docker-java-core/src/main/java/com/github/dockerjava/core/async/FrameStreamProcessor.java similarity index 97% rename from src/main/java/com/github/dockerjava/core/async/FrameStreamProcessor.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/async/FrameStreamProcessor.java index 3070930d6..4c3136417 100644 --- a/src/main/java/com/github/dockerjava/core/async/FrameStreamProcessor.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/async/FrameStreamProcessor.java @@ -16,6 +16,8 @@ * @author Marcus Linke * */ +@SuppressWarnings("unused") +@Deprecated public class FrameStreamProcessor implements ResponseStreamProcessor { @Override diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/async/JsonStreamProcessor.java b/docker-java-core/src/main/java/com/github/dockerjava/core/async/JsonStreamProcessor.java new file mode 100644 index 000000000..6bc5ad385 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/async/JsonStreamProcessor.java @@ -0,0 +1,84 @@ +/* + * Created on 18.06.2015 + */ +package com.github.dockerjava.core.async; + +import java.io.IOException; +import java.io.InputStream; + +import com.fasterxml.jackson.core.JsonFactory; +import com.fasterxml.jackson.core.JsonParser; +import com.fasterxml.jackson.core.JsonToken; +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.node.ObjectNode; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.core.DockerClientConfig; + +/** + * + * @author Marcus Linke + * + */ +@SuppressWarnings("unused") +@Deprecated +public class JsonStreamProcessor implements ResponseStreamProcessor { + + private static final JsonFactory JSON_FACTORY = new JsonFactory(); + + private final TypeReference typeReference; + + private final ObjectMapper objectMapper; + + @Deprecated + public JsonStreamProcessor(Class clazz) { + this( + DockerClientConfig.getDefaultObjectMapper(), + new TypeReference() { + } + ); + } + + public JsonStreamProcessor(ObjectMapper objectMapper, TypeReference typeReference) { + this.typeReference = typeReference; + this.objectMapper = objectMapper.copy().enable(JsonParser.Feature.AUTO_CLOSE_SOURCE); + } + + @Override + public void processResponseStream(InputStream response, ResultCallback resultCallback) { + + resultCallback.onStart(response); + + try { + JsonParser jp = JSON_FACTORY.createParser(response); + Boolean closed = jp.isClosed(); + JsonToken nextToken = jp.nextToken(); + while (!closed && nextToken != null && nextToken != JsonToken.END_OBJECT) { + try { + ObjectNode objectNode = objectMapper.readTree(jp); + // exclude empty item serialization into class #461 + if (!objectNode.isEmpty(null)) { + T next = objectMapper.convertValue(objectNode, typeReference); + resultCallback.onNext(next); + } + } catch (Exception e) { + resultCallback.onError(e); + } + + closed = jp.isClosed(); + nextToken = jp.nextToken(); + } + } catch (Throwable t) { + resultCallback.onError(t); + } finally { + try { + response.close(); + } catch (IOException e) { + resultCallback.onError(e); + } finally { + resultCallback.onComplete(); + } + } + + } +} diff --git a/src/main/java/com/github/dockerjava/core/async/ResponseStreamProcessor.java b/docker-java-core/src/main/java/com/github/dockerjava/core/async/ResponseStreamProcessor.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/async/ResponseStreamProcessor.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/async/ResponseStreamProcessor.java diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/async/ResultCallbackTemplate.java b/docker-java-core/src/main/java/com/github/dockerjava/core/async/ResultCallbackTemplate.java new file mode 100644 index 000000000..32c5de018 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/async/ResultCallbackTemplate.java @@ -0,0 +1,19 @@ +/* + * Created on 16.06.2015 + */ +package com.github.dockerjava.core.async; + +import com.github.dockerjava.api.async.ResultCallback; + +/** + * Abstract template implementation of {@link ResultCallback} + * + * @author Marcus Linke + * @deprecated use {@link com.github.dockerjava.api.async.ResultCallbackTemplate} + * + */ +@Deprecated +public abstract class ResultCallbackTemplate, A_RES_T> + extends com.github.dockerjava.api.async.ResultCallbackTemplate { + +} diff --git a/src/main/java/com/github/dockerjava/core/command/AbstrAsyncDockerCmd.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/AbstrAsyncDockerCmd.java similarity index 75% rename from src/main/java/com/github/dockerjava/core/command/AbstrAsyncDockerCmd.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/AbstrAsyncDockerCmd.java index 5849fc3fb..9f83c0b4f 100644 --- a/src/main/java/com/github/dockerjava/core/command/AbstrAsyncDockerCmd.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/AbstrAsyncDockerCmd.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.async.ResultCallback; import com.github.dockerjava.api.command.AsyncDockerCmd; @@ -9,11 +9,10 @@ public abstract class AbstrAsyncDockerCmd, A_RES_T> implements AsyncDockerCmd { - protected DockerCmdAsyncExec execution; + protected transient DockerCmdAsyncExec execution; public AbstrAsyncDockerCmd(DockerCmdAsyncExec execution) { - checkNotNull(execution, "execution was not specified"); - this.execution = execution; + this.execution = Objects.requireNonNull(execution, "execution was not specified"); } @Override diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/AbstrDockerCmd.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/AbstrDockerCmd.java new file mode 100644 index 000000000..17ff9a88e --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/AbstrDockerCmd.java @@ -0,0 +1,53 @@ +package com.github.dockerjava.core.command; + +import java.io.IOException; +import java.util.Base64; +import java.util.Objects; + +import org.apache.commons.lang3.builder.ReflectionToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.dockerjava.api.command.DockerCmd; +import com.github.dockerjava.api.command.DockerCmdSyncExec; +import com.github.dockerjava.api.command.SyncDockerCmd; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.AuthConfig; + +public abstract class AbstrDockerCmd, RES_T> implements SyncDockerCmd { + + private static final Logger LOGGER = LoggerFactory.getLogger(AbstrDockerCmd.class); + + protected transient DockerCmdSyncExec execution; + + public AbstrDockerCmd(DockerCmdSyncExec execution) { + this.execution = Objects.requireNonNull(execution, "execution was not specified"); + } + + @Override + @SuppressWarnings("unchecked") + public RES_T exec() throws DockerException { + LOGGER.debug("Cmd: {}", this); + return execution.exec((CMD_T) this); + } + + @Override + public void close() { + } + + @Override + public String toString() { + return ReflectionToStringBuilder.toString(this, ToStringStyle.SIMPLE_STYLE); + } + + @Deprecated + protected String registryAuth(AuthConfig authConfig) { + try { + return Base64.getEncoder().encodeToString(new ObjectMapper().writeValueAsBytes(authConfig)); + } catch (IOException e) { + throw new RuntimeException(e); + } + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/AttachContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/AttachContainerCmdImpl.java new file mode 100644 index 000000000..30a411dba --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/AttachContainerCmdImpl.java @@ -0,0 +1,103 @@ +package com.github.dockerjava.core.command; + +import java.io.InputStream; +import java.util.Objects; + +import com.github.dockerjava.api.command.AttachContainerCmd; +import com.github.dockerjava.api.model.Frame; + +/** + * Attach to container. + */ +public class AttachContainerCmdImpl extends AbstrAsyncDockerCmd + implements AttachContainerCmd { + + private String containerId; + + private Boolean logs, followStream, timestamps, stdout, stderr; + + private InputStream stdin; + + public AttachContainerCmdImpl(AttachContainerCmd.Exec exec, String containerId) { + super(exec); + withContainerId(containerId); + } + + @Override + public String getContainerId() { + return containerId; + } + + @Override + public Boolean hasLogsEnabled() { + return logs; + } + + @Override + public Boolean hasFollowStreamEnabled() { + return followStream; + } + + @Override + public Boolean hasTimestampsEnabled() { + return timestamps; + } + + @Override + public Boolean hasStdoutEnabled() { + return stdout; + } + + @Override + public Boolean hasStderrEnabled() { + return stderr; + } + + @Override + public InputStream getStdin() { + return stdin; + } + + @Override + public AttachContainerCmd withContainerId(String containerId) { + Objects.requireNonNull(containerId, "containerId was not specified"); + this.containerId = containerId; + return this; + } + + @Override + public AttachContainerCmd withFollowStream(Boolean followStream) { + this.followStream = followStream; + return this; + } + + @Override + public AttachContainerCmd withTimestamps(Boolean timestamps) { + this.timestamps = timestamps; + return this; + } + + @Override + public AttachContainerCmd withStdOut(Boolean stdout) { + this.stdout = stdout; + return this; + } + + @Override + public AttachContainerCmd withStdErr(Boolean stderr) { + this.stderr = stderr; + return this; + } + + @Override + public AttachContainerCmd withStdIn(InputStream stdin) { + this.stdin = Objects.requireNonNull(stdin, "stdin was not specified"); + return this; + } + + @Override + public AttachContainerCmd withLogs(Boolean logs) { + this.logs = logs; + return this; + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/AttachContainerResultCallback.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/AttachContainerResultCallback.java similarity index 86% rename from src/main/java/com/github/dockerjava/core/command/AttachContainerResultCallback.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/AttachContainerResultCallback.java index bddddfd4f..1e4d22cc8 100644 --- a/src/main/java/com/github/dockerjava/core/command/AttachContainerResultCallback.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/AttachContainerResultCallback.java @@ -13,7 +13,9 @@ * * @author Marcus Linke * + * @deprecated use {@link com.github.dockerjava.api.async.ResultCallback.Adapter} */ +@Deprecated public class AttachContainerResultCallback extends ResultCallbackTemplate { private static final Logger LOGGER = LoggerFactory.getLogger(AttachContainerResultCallback.class); diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/AuthCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/AuthCmdImpl.java new file mode 100644 index 000000000..e22ad67f2 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/AuthCmdImpl.java @@ -0,0 +1,35 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.AuthCmd; +import com.github.dockerjava.api.exception.UnauthorizedException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.AuthResponse; + +/** + * + * Authenticate with the server, useful for checking authentication. + * + */ +public class AuthCmdImpl extends AbstrDockerCmd implements AuthCmd { + + private AuthConfig authConfig; + + public AuthCmdImpl(AuthCmd.Exec exec, AuthConfig authConfig) { + super(exec); + withAuthConfig(authConfig); + } + + @Override + public AuthResponse exec() throws UnauthorizedException { + return super.exec(); + } + + public AuthConfig getAuthConfig() { + return authConfig; + } + + public AuthCmd withAuthConfig(AuthConfig authConfig) { + this.authConfig = authConfig; + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/BuildImageCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/BuildImageCmdImpl.java new file mode 100644 index 000000000..5d9e62909 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/BuildImageCmdImpl.java @@ -0,0 +1,423 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.BuildImageCmd; +import com.github.dockerjava.api.model.AuthConfigurations; +import com.github.dockerjava.api.model.BuildResponseItem; +import com.github.dockerjava.core.dockerfile.Dockerfile; +import com.github.dockerjava.core.util.FilePathUtil; + +import javax.annotation.CheckForNull; +import java.io.File; +import java.io.IOException; +import java.io.InputStream; +import java.net.URI; +import java.util.HashMap; +import java.util.Map; +import java.util.Objects; +import java.util.Set; + +/** + * Build an image from Dockerfile. + */ +public class BuildImageCmdImpl extends AbstrAsyncDockerCmd implements BuildImageCmd { + + private InputStream tarInputStream; + + @Deprecated + private String tag; + + private Set tags; + + private Set cacheFrom; + + private Boolean noCache; + + private Boolean remove = true; + + private Boolean quiet; + + private Boolean pull; + + private AuthConfigurations buildAuthConfigs; + + private File dockerFile; + + private String dockerFilePath; + + private File baseDirectory; + + private String cpusetcpus; + + private Long memory; + + private String cpushares; + + private Boolean forcerm; + + private Long memswap; + + private Long shmsize; + + private URI remote; + + private Map buildArgs; + + private Map labels; + + private String networkMode; + + private String platform; + + private String target; + + private Set extraHosts; + + public BuildImageCmdImpl(BuildImageCmd.Exec exec) { + super(exec); + } + + public BuildImageCmdImpl(BuildImageCmd.Exec exec, File dockerFileOrFolder) { + super(exec); + Objects.requireNonNull(dockerFileOrFolder, "dockerFolder is null"); + + if (dockerFileOrFolder.isDirectory()) { + withBaseDirectory(dockerFileOrFolder); + withDockerfile(new File(dockerFileOrFolder, "Dockerfile")); + } else { + withDockerfile(dockerFileOrFolder); + } + } + + public BuildImageCmdImpl(BuildImageCmd.Exec exec, InputStream tarInputStream) { + super(exec); + Objects.requireNonNull(tarInputStream, "tarInputStream is null"); + withTarInputStream(tarInputStream); + } + + // getters API + + @Deprecated + @Override + public String getTag() { + return tag; + } + + @CheckForNull + public Set getTags() { + return tags; + } + + @CheckForNull + public Set getCacheFrom() { + return cacheFrom; + } + + @Override + public URI getRemote() { + return remote; + } + + @Override + public Boolean hasNoCacheEnabled() { + return noCache; + } + + @Override + public Boolean hasRemoveEnabled() { + return remove; + } + + @Override + public Boolean isForcerm() { + return forcerm; + } + + @Override + public Boolean isQuiet() { + return quiet; + } + + @Override + public Boolean hasPullEnabled() { + return pull; + } + + @Override + public String getPathToDockerfile() { + if (dockerFilePath != null) { + return dockerFilePath; + } else if (baseDirectory != null && dockerFile != null) { + return FilePathUtil.relativize(baseDirectory, dockerFile); + } else { + return null; + } + } + + @Override + public Long getMemory() { + return memory; + } + + @Override + public Long getMemswap() { + return memswap; + } + + @Override + public String getCpushares() { + return cpushares; + } + + @Override + public String getCpusetcpus() { + return cpusetcpus; + } + + @Override + public Map getBuildArgs() { + return buildArgs; + } + + @Override + public Map getLabels() { + return labels; + } + + @Override + public String getNetworkMode() { + return networkMode; + } + + @Override + public String getPlatform() { + return platform; + } + + @Override + public String getTarget() { + return target; + } + + // getter lib specific + + @Override + public AuthConfigurations getBuildAuthConfigs() { + return buildAuthConfigs; + } + + @Override + public InputStream getTarInputStream() { + return tarInputStream; + } + + /** + * @see #shmsize + */ + @Override + public Long getShmsize() { + return shmsize; + } + + @Override + public Set getExtraHosts() { + return extraHosts; + } + + // setters + + /** + * @deprecated use #withTags() + */ + @Deprecated + @Override + public BuildImageCmdImpl withTag(String tag) { + this.tag = Objects.requireNonNull(tag, "Tag is null"); + return this; + } + + @Override + public BuildImageCmd withTags(Set tags) { + this.tags = tags; + return this; + } + + @Override + public BuildImageCmd withCacheFrom(Set cacheFrom) { + this.cacheFrom = cacheFrom; + return this; + } + + @Override + public BuildImageCmd withRemote(URI remote) { + this.remote = remote; + return this; + } + + @Override + public BuildImageCmdImpl withNoCache(Boolean noCache) { + this.noCache = noCache; + return this; + } + + @Override + public BuildImageCmdImpl withRemove(Boolean rm) { + this.remove = rm; + return this; + } + + @Override + public BuildImageCmd withForcerm(Boolean forcerm) { + this.forcerm = forcerm; + return this; + } + + @Override + public BuildImageCmdImpl withQuiet(Boolean quiet) { + this.quiet = quiet; + return this; + } + + @Override + public BuildImageCmdImpl withPull(Boolean pull) { + this.pull = pull; + return this; + } + + @Override + public BuildImageCmd withMemory(Long memory) { + this.memory = memory; + return this; + } + + @Override + public BuildImageCmd withMemswap(Long memswap) { + this.memswap = memswap; + return this; + } + + @Override + public BuildImageCmd withCpushares(String cpushares) { + this.cpushares = cpushares; + return this; + } + + @Override + public BuildImageCmd withCpusetcpus(String cpusetcpus) { + this.cpusetcpus = cpusetcpus; + return this; + } + + @Override + public BuildImageCmd withBuildArg(String key, String value) { + if (this.buildArgs == null) { + this.buildArgs = new HashMap<>(); + } + this.buildArgs.put(key, value); + return this; + } + + // lib specific + + @Override + public BuildImageCmd withBaseDirectory(File baseDirectory) { + this.baseDirectory = baseDirectory; + return this; + } + + @Override + public BuildImageCmdImpl withDockerfile(File dockerfile) { + Objects.requireNonNull(dockerfile); + if (!dockerfile.exists()) { + throw new IllegalArgumentException("Dockerfile does not exist"); + } + if (!dockerfile.isFile()) { + throw new IllegalArgumentException("Dockerfile is not a file"); + } + + if (baseDirectory == null) { + withBaseDirectory(dockerfile.getParentFile()); + } + + this.dockerFile = dockerfile; + + try { + withTarInputStream(new Dockerfile(dockerfile, baseDirectory).parse().buildDockerFolderTar()); + } catch (IOException e) { + // we just created the file this should never happen. + throw new RuntimeException(e); + } + return this; + } + + @Override + public BuildImageCmd withDockerfilePath(String dockerfilePath) { + this.dockerFilePath = Objects.requireNonNull(dockerfilePath, "dockerfilePath is null"); + return this; + } + + @Override + public BuildImageCmdImpl withTarInputStream(InputStream tarInputStream) { + this.tarInputStream = Objects.requireNonNull(tarInputStream, "tarInputStream is null"); + return this; + } + + @Override + public BuildImageCmd withBuildAuthConfigs(AuthConfigurations authConfigs) { + this.buildAuthConfigs = Objects.requireNonNull(authConfigs, "authConfig is null"); + return this; + } + + /** + * @see #shmsize + */ + @Override + public BuildImageCmd withShmsize(Long shmsize) { + this.shmsize = shmsize; + return this; + } + + /** + * @see #labels + */ + @Override + public BuildImageCmd withLabels(Map labels) { + this.labels = labels; + return this; + } + + @Override + public BuildImageCmd withNetworkMode(String networkMode) { + this.networkMode = networkMode; + return this; + } + + @Override + public BuildImageCmd withPlatform(String platform) { + this.platform = platform; + return this; + } + + @Override + public BuildImageCmd withTarget(String target) { + this.target = target; + return this; + } + + @Override + public BuildImageCmd withExtraHosts(Set extraHosts) { + this.extraHosts = extraHosts; + return this; + } + + @Override + public void close() { + super.close(); + + try { + tarInputStream.close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/BuildImageResultCallback.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/BuildImageResultCallback.java new file mode 100644 index 000000000..f94a751d4 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/BuildImageResultCallback.java @@ -0,0 +1,82 @@ +/* + * Created on 21.07.2015 + */ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.model.BuildResponseItem; +import com.github.dockerjava.core.async.ResultCallbackTemplate; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.concurrent.TimeUnit; + +/** + * + * @author Marcus Linke + * + * @deprecated use {@link com.github.dockerjava.api.command.BuildImageResultCallback} + */ +@Deprecated +public class BuildImageResultCallback extends ResultCallbackTemplate { + + private static final Logger LOGGER = LoggerFactory.getLogger(BuildImageResultCallback.class); + + private String imageId; + + private String error; + + @Override + public void onNext(BuildResponseItem item) { + if (item.isBuildSuccessIndicated()) { + this.imageId = item.getImageId(); + } else if (item.isErrorIndicated()) { + this.error = item.getError(); + } + LOGGER.debug(item.toString()); + } + + /** + * Awaits the image id from the response stream. + * + * @throws DockerClientException + * if the build fails. + */ + public String awaitImageId() { + try { + awaitCompletion(); + } catch (InterruptedException e) { + throw new DockerClientException("", e); + } + + return getImageId(); + } + + /** + * Awaits the image id from the response stream. + * + * @throws DockerClientException + * if the build fails or the timeout occurs. + */ + public String awaitImageId(long timeout, TimeUnit timeUnit) { + try { + awaitCompletion(timeout, timeUnit); + } catch (InterruptedException e) { + throw new DockerClientException("Awaiting image id interrupted: ", e); + } + + return getImageId(); + } + + private String getImageId() { + if (imageId != null) { + return imageId; + } + + if (error == null) { + throw new DockerClientException("Could not build image"); + } + + throw new DockerClientException("Could not build image: " + error); + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/CommitCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CommitCmdImpl.java similarity index 77% rename from src/main/java/com/github/dockerjava/core/command/CommitCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/CommitCmdImpl.java index 534cce52c..8458c7f7e 100644 --- a/src/main/java/com/github/dockerjava/core/command/CommitCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CommitCmdImpl.java @@ -1,6 +1,7 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Map; +import java.util.Objects; import com.fasterxml.jackson.annotation.JsonProperty; import com.github.dockerjava.api.command.CommitCmd; @@ -43,6 +44,12 @@ public class CommitCmdImpl extends AbstrDockerCmd implements @JsonProperty("Hostname") private String hostname; + /** + * @since 1.19 + */ + @JsonProperty("Labels") + private Map labels; + @JsonProperty("Memory") private Integer memory; @@ -82,8 +89,7 @@ public String getContainerId() { @Override public CommitCmdImpl withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } @@ -132,8 +138,7 @@ public CommitCmdImpl withAttachStdout(Boolean attachStdout) { @Override public CommitCmdImpl withCmd(String... cmd) { - checkNotNull(cmd, "cmd was not specified"); - this.cmd = cmd; + this.cmd = Objects.requireNonNull(cmd, "cmd was not specified"); return this; } @@ -145,29 +150,25 @@ public CommitCmdImpl withDisableNetwork(Boolean disableNetwork) { @Override public CommitCmdImpl withAuthor(String author) { - checkNotNull(author, "author was not specified"); - this.author = author; + this.author = Objects.requireNonNull(author, "author was not specified"); return this; } @Override public CommitCmdImpl withMessage(String message) { - checkNotNull(message, "message was not specified"); - this.message = message; + this.message = Objects.requireNonNull(message, "message was not specified"); return this; } @Override public CommitCmdImpl withTag(String tag) { - checkNotNull(tag, "tag was not specified"); - this.tag = tag; + this.tag = Objects.requireNonNull(tag, "tag was not specified"); return this; } @Override public CommitCmdImpl withRepository(String repository) { - checkNotNull(repository, "repository was not specified"); - this.repository = repository; + this.repository = Objects.requireNonNull(repository, "repository was not specified"); return this; } @@ -184,8 +185,18 @@ public String[] getEnv() { @Override public CommitCmdImpl withEnv(String... env) { - checkNotNull(env, "env was not specified"); - this.env = env; + this.env = Objects.requireNonNull(env, "env was not specified"); + return this; + } + + @Override + public Map getLabels() { + return labels; + } + + @Override + public CommitCmdImpl withLabels(Map labels) { + this.labels = labels; return this; } @@ -196,8 +207,7 @@ public ExposedPorts getExposedPorts() { @Override public CommitCmdImpl withExposedPorts(ExposedPorts exposedPorts) { - checkNotNull(exposedPorts, "exposedPorts was not specified"); - this.exposedPorts = exposedPorts; + this.exposedPorts = Objects.requireNonNull(exposedPorts, "exposedPorts was not specified"); return this; } @@ -208,8 +218,7 @@ public String getHostname() { @Override public CommitCmdImpl withHostname(String hostname) { - checkNotNull(hostname, "hostname was not specified"); - this.hostname = hostname; + this.hostname = Objects.requireNonNull(hostname, "hostname was not specified"); return this; } @@ -220,8 +229,7 @@ public Integer getMemory() { @Override public CommitCmdImpl withMemory(Integer memory) { - checkNotNull(memory, "memory was not specified"); - this.memory = memory; + this.memory = Objects.requireNonNull(memory, "memory was not specified"); return this; } @@ -232,8 +240,7 @@ public Integer getMemorySwap() { @Override public CommitCmdImpl withMemorySwap(Integer memorySwap) { - checkNotNull(memorySwap, "memorySwap was not specified"); - this.memorySwap = memorySwap; + this.memorySwap = Objects.requireNonNull(memorySwap, "memorySwap was not specified"); return this; } @@ -244,8 +251,7 @@ public Boolean isOpenStdin() { @Override public CommitCmdImpl withOpenStdin(Boolean openStdin) { - checkNotNull(openStdin, "openStdin was not specified"); - this.openStdin = openStdin; + this.openStdin = Objects.requireNonNull(openStdin, "openStdin was not specified"); return this; } @@ -256,8 +262,7 @@ public String[] getPortSpecs() { @Override public CommitCmdImpl withPortSpecs(String... portSpecs) { - checkNotNull(portSpecs, "portSpecs was not specified"); - this.portSpecs = portSpecs; + this.portSpecs = Objects.requireNonNull(portSpecs, "portSpecs was not specified"); return this; } @@ -290,8 +295,7 @@ public String getUser() { @Override public CommitCmdImpl withUser(String user) { - checkNotNull(user, "user was not specified"); - this.user = user; + this.user = Objects.requireNonNull(user, "user was not specified"); return this; } @@ -302,8 +306,7 @@ public Volumes getVolumes() { @Override public CommitCmdImpl withVolumes(Volumes volumes) { - checkNotNull(volumes, "volumes was not specified"); - this.volumes = volumes; + this.volumes = Objects.requireNonNull(volumes, "volumes was not specified"); return this; } @@ -314,8 +317,7 @@ public String getWorkingDir() { @Override public CommitCmdImpl withWorkingDir(String workingDir) { - checkNotNull(workingDir, "workingDir was not specified"); - this.workingDir = workingDir; + this.workingDir = Objects.requireNonNull(workingDir, "workingDir was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/ConnectToNetworkCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ConnectToNetworkCmdImpl.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/command/ConnectToNetworkCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/ConnectToNetworkCmdImpl.java diff --git a/src/main/java/com/github/dockerjava/core/command/ContainerDiffCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ContainerDiffCmdImpl.java similarity index 88% rename from src/main/java/com/github/dockerjava/core/command/ContainerDiffCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/ContainerDiffCmdImpl.java index eb6baa5ab..14fc683ce 100644 --- a/src/main/java/com/github/dockerjava/core/command/ContainerDiffCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ContainerDiffCmdImpl.java @@ -1,8 +1,7 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; - import java.util.List; +import java.util.Objects; import com.github.dockerjava.api.command.ContainerDiffCmd; import com.github.dockerjava.api.exception.DockerException; @@ -33,8 +32,7 @@ public String getContainerId() { @Override public ContainerDiffCmdImpl withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/CopyArchiveFromContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CopyArchiveFromContainerCmdImpl.java similarity index 80% rename from src/main/java/com/github/dockerjava/core/command/CopyArchiveFromContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/CopyArchiveFromContainerCmdImpl.java index 8ca305616..e827b37c8 100644 --- a/src/main/java/com/github/dockerjava/core/command/CopyArchiveFromContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CopyArchiveFromContainerCmdImpl.java @@ -1,8 +1,7 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; - import java.io.InputStream; +import java.util.Objects; import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; import com.github.dockerjava.api.exception.NotFoundException; @@ -37,15 +36,13 @@ public String getResource() { @Override public CopyArchiveFromContainerCmdImpl withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } @Override public CopyArchiveFromContainerCmdImpl withResource(String resource) { - checkNotNull(resource, "resource was not specified"); - this.resource = resource; + this.resource = Objects.requireNonNull(resource, "resource was not specified"); return this; } @@ -56,8 +53,7 @@ public String getHostPath() { @Override public CopyArchiveFromContainerCmdImpl withHostPath(String hostPath) { - checkNotNull(hostPath, "hostPath was not specified"); - this.hostPath = hostPath; + this.hostPath = Objects.requireNonNull(hostPath, "hostPath was not specified"); return this; } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/CopyArchiveToContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CopyArchiveToContainerCmdImpl.java new file mode 100644 index 000000000..a9b42e921 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CopyArchiveToContainerCmdImpl.java @@ -0,0 +1,164 @@ +package com.github.dockerjava.core.command; + +import java.io.IOException; +import java.io.InputStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.Objects; + +import org.apache.commons.lang3.StringUtils; +import org.apache.commons.lang3.builder.ToStringBuilder; + +import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.core.util.CompressArchiveUtil; + +public class CopyArchiveToContainerCmdImpl extends AbstrDockerCmd implements + CopyArchiveToContainerCmd { + + private String containerId; + + private String remotePath = "."; + + private InputStream tarInputStream; + + private String hostResource; + + private boolean noOverwriteDirNonDir = false; + + private boolean dirChildrenOnly = false; + + private boolean copyUIDGID = false; + + public CopyArchiveToContainerCmdImpl(CopyArchiveToContainerCmd.Exec exec, String containerId) { + super(exec); + withContainerId(containerId); + } + + @Override + public CopyArchiveToContainerCmd withContainerId(String containerId) { + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); + return this; + } + + @Override + public CopyArchiveToContainerCmd withHostResource(String hostResource) { + this.hostResource = Objects.requireNonNull(hostResource, "hostResource was not specified"); + return this; + } + + @Override + public CopyArchiveToContainerCmd withNoOverwriteDirNonDir(boolean noOverwriteDirNonDir) { + this.noOverwriteDirNonDir = noOverwriteDirNonDir; + return this; + } + + @Override + public CopyArchiveToContainerCmd withCopyUIDGID(boolean copyUIDGID) { + this.copyUIDGID = copyUIDGID; + return this; + } + + @Override + public CopyArchiveToContainerCmd withRemotePath(String remotePath) { + this.remotePath = Objects.requireNonNull(remotePath, "remotePath was not specified"); + return this; + } + + @Override + public CopyArchiveToContainerCmd withTarInputStream(InputStream tarInputStream) { + this.tarInputStream = Objects.requireNonNull(tarInputStream, "tarInputStream was not specified"); + return this; + } + + @Override + public CopyArchiveToContainerCmd withDirChildrenOnly(boolean dirChildrenOnly) { + this.dirChildrenOnly = dirChildrenOnly; + return this; + } + + @Override + public InputStream getTarInputStream() { + return tarInputStream; + } + + @Override + public String getContainerId() { + return this.containerId; + } + + @Override + public String getHostResource() { + return this.hostResource; + } + + @Override + public boolean isNoOverwriteDirNonDir() { + return this.noOverwriteDirNonDir; + } + + @Override + public String getRemotePath() { + return this.remotePath; + } + + @Override + public boolean isDirChildrenOnly() { + return this.dirChildrenOnly; + } + + @Override + public boolean isCopyUIDGID() { + return this.copyUIDGID; + } + + @Override + public String toString() { + return new ToStringBuilder(this).append("cp ").append(String.format("-a=%b ", isCopyUIDGID())) + .append(hostResource).append(" ").append(containerId).append(":").append(remotePath).toString(); + } + + /** + * @throws com.github.dockerjava.api.exception.NotFoundException + * No such container + */ + @Override + public Void exec() throws NotFoundException { + if (StringUtils.isNotEmpty(this.hostResource)) { + // User set host resource and not directly a stream + if (this.tarInputStream != null) { + throw new DockerClientException( + "Only one of host resource or tar input stream should be defined to perform the copy, not both"); + } + // create TAR package for the given path so docker can consume it + Path toUpload = null; + try { + toUpload = Files.createTempFile("docker-java", ".tar.gz"); + CompressArchiveUtil.tar(Paths.get(hostResource), toUpload, true, dirChildrenOnly); + } catch (IOException createFileIOException) { + if (toUpload != null) { + // remove tmp docker-javaxxx.tar.gz + toUpload.toFile().delete(); + } + throw new DockerClientException("Unable to perform tar on host resource " + this.hostResource, createFileIOException); + } + // send the tar stream, call exec so that the stream is consumed and then closed by try-with-resources + try (InputStream uploadStream = Files.newInputStream(toUpload)) { + this.tarInputStream = uploadStream; + return super.exec(); + } catch (IOException e) { + throw new DockerClientException("Unable to read temp file " + toUpload.toFile().getAbsolutePath(), e); + } finally { + this.tarInputStream = null; + // remove tmp docker-javaxxx.tar.gz + toUpload.toFile().delete(); + } + } else if (this.tarInputStream == null) { + throw new DockerClientException("One of host resource or tar input stream must be defined to perform the copy"); + } + // User set a stream, so we will just consume it and let the user close it by him self + return super.exec(); + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/CopyFileFromContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CopyFileFromContainerCmdImpl.java similarity index 81% rename from src/main/java/com/github/dockerjava/core/command/CopyFileFromContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/CopyFileFromContainerCmdImpl.java index 3976834a4..4c8b3447d 100644 --- a/src/main/java/com/github/dockerjava/core/command/CopyFileFromContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CopyFileFromContainerCmdImpl.java @@ -1,8 +1,7 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; - import java.io.InputStream; +import java.util.Objects; import com.fasterxml.jackson.annotation.JsonProperty; import com.github.dockerjava.api.command.CopyFileFromContainerCmd; @@ -42,15 +41,13 @@ public String getResource() { @Override public CopyFileFromContainerCmdImpl withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } @Override public CopyFileFromContainerCmdImpl withResource(String resource) { - checkNotNull(resource, "resource was not specified"); - this.resource = resource; + this.resource = Objects.requireNonNull(resource, "resource was not specified"); return this; } @@ -61,8 +58,7 @@ public String getHostPath() { @Override public CopyFileFromContainerCmdImpl withHostPath(String hostPath) { - checkNotNull(hostPath, "hostPath was not specified"); - this.hostPath = hostPath; + this.hostPath = Objects.requireNonNull(hostPath, "hostPath was not specified"); return this; } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateConfigCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateConfigCmdImpl.java new file mode 100644 index 000000000..3a4f1cf01 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateConfigCmdImpl.java @@ -0,0 +1,63 @@ +package com.github.dockerjava.core.command; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.command.CreateConfigCmd; +import com.github.dockerjava.api.command.CreateConfigResponse; + +import java.util.Base64; +import java.util.Map; +import java.util.Objects; + +/** + * Creates a new config + */ +public class CreateConfigCmdImpl extends AbstrDockerCmd implements CreateConfigCmd { + + @JsonProperty("Name") + private String name; + + @JsonProperty("Data") + private String data; + + @JsonProperty("Labels") + private Map labels; + + @Override + public String getName() { + return name; + } + + @Override + public String getData() { + return data; + } + + @Override + public Map getLabels() { + return labels; + } + + public CreateConfigCmdImpl(CreateConfigCmd.Exec exec) { + super(exec); + } + + @Override + public CreateConfigCmd withName(String name) { + this.name = Objects.requireNonNull(name, "name was not specified"); + return this; + } + + @Override + public CreateConfigCmd withData(byte[] data) { + Objects.requireNonNull(data, "data was not specified"); + this.data = Base64.getEncoder().encodeToString(data); + return this; + } + + @Override + public CreateConfigCmd withLabels(Map labels) { + this.labels = Objects.requireNonNull(labels, "labels was not specified"); + return this; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateContainerCmdImpl.java new file mode 100644 index 000000000..9b7f8a8fe --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateContainerCmdImpl.java @@ -0,0 +1,640 @@ +package com.github.dockerjava.core.command; + +import com.fasterxml.jackson.annotation.JsonAutoDetect; +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.command.CreateContainerCmd; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.ConflictException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.ContainerNetwork; +import com.github.dockerjava.api.model.ExposedPort; +import com.github.dockerjava.api.model.ExposedPorts; +import com.github.dockerjava.api.model.HealthCheck; +import com.github.dockerjava.api.model.HostConfig; +import com.github.dockerjava.api.model.Volume; +import com.github.dockerjava.api.model.Volumes; +import org.apache.commons.lang3.builder.EqualsBuilder; +import org.apache.commons.lang3.builder.HashCodeBuilder; +import org.apache.commons.lang3.builder.ToStringBuilder; + +import javax.annotation.CheckForNull; +import java.util.Arrays; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +import static java.util.Collections.singletonMap; + +/** + * Creates a new container. + * `/containers/create` + */ +@JsonAutoDetect( + fieldVisibility = JsonAutoDetect.Visibility.NONE, + setterVisibility = JsonAutoDetect.Visibility.NONE, + getterVisibility = JsonAutoDetect.Visibility.NONE, + isGetterVisibility = JsonAutoDetect.Visibility.NONE, + creatorVisibility = JsonAutoDetect.Visibility.NONE +) +public class CreateContainerCmdImpl extends AbstrDockerCmd implements + CreateContainerCmd { + + private String name; + + @JsonProperty("Hostname") + private String hostName; + + @JsonProperty("Domainname") + private String domainName; + + @JsonProperty("User") + private String user; + + @JsonProperty("AttachStdin") + private Boolean attachStdin; + + @JsonProperty("AttachStdout") + private Boolean attachStdout; + + @JsonProperty("AttachStderr") + private Boolean attachStderr; + + @JsonProperty("PortSpecs") + private String[] portSpecs; + + @JsonProperty("Tty") + private Boolean tty; + + @JsonProperty("OpenStdin") + private Boolean stdinOpen; + + @JsonProperty("StdinOnce") + private Boolean stdInOnce; + + @JsonProperty("Env") + private String[] env; + + @JsonProperty("Cmd") + private String[] cmd; + + @JsonProperty("Healthcheck") + private HealthCheck healthcheck; + + @JsonProperty("ArgsEscaped") + private Boolean argsEscaped; + + @JsonProperty("Entrypoint") + private String[] entrypoint; + + @JsonProperty("Image") + private String image; + + @JsonProperty("Volumes") + private Volumes volumes = new Volumes(); + + @JsonProperty("WorkingDir") + private String workingDir; + + @JsonProperty("MacAddress") + private String macAddress; + + @JsonProperty("OnBuild") + private List onBuild; + + @JsonProperty("NetworkDisabled") + private Boolean networkDisabled; + + @JsonProperty("ExposedPorts") + private ExposedPorts exposedPorts = new ExposedPorts(); + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_21} + */ + @JsonProperty("StopSignal") + private String stopSignal; + + @JsonProperty("StopTimeout") + private Integer stopTimeout; + + @JsonProperty("HostConfig") + private HostConfig hostConfig = new HostConfig(); + + @JsonProperty("Labels") + private Map labels; + + @JsonProperty("Shell") + private List shell; + + @JsonProperty("NetworkingConfig") + private NetworkingConfig networkingConfig; + + private String ipv4Address = null; + + private String ipv6Address = null; + + private List aliases = null; + + private AuthConfig authConfig; + + private String platform; + + public CreateContainerCmdImpl(CreateContainerCmd.Exec exec, AuthConfig authConfig, String image) { + super(exec); + withAuthConfig(authConfig); + withImage(image); + } + + public AuthConfig getAuthConfig() { + return authConfig; + } + + public CreateContainerCmd withAuthConfig(AuthConfig authConfig) { + this.authConfig = authConfig; + return this; + } + + @Override + public List getAliases() { + return aliases; + } + + @Override + public CreateContainerCmd withAliases(String... aliases) { + this.aliases = Arrays.asList(aliases); + return this; + } + + @Override + public CreateContainerCmd withAliases(List aliases) { + Objects.requireNonNull(aliases, "aliases was not specified"); + this.aliases = aliases; + return this; + } + + + @Override + public String[] getCmd() { + return cmd; + } + + @Override + public CreateContainerCmd withCmd(String... cmd) { + Objects.requireNonNull(cmd, "cmd was not specified"); + this.cmd = cmd; + return this; + } + + @Override + public CreateContainerCmd withCmd(List cmd) { + Objects.requireNonNull(cmd, "cmd was not specified"); + return withCmd(cmd.toArray(new String[0])); + } + + @CheckForNull + public HealthCheck getHealthcheck() { + return healthcheck; + } + + public CreateContainerCmdImpl withHealthcheck(HealthCheck healthcheck) { + this.healthcheck = healthcheck; + return this; + } + + public Boolean getArgsEscaped() { + return argsEscaped; + } + + public CreateContainerCmdImpl withArgsEscaped(Boolean argsEscaped) { + this.argsEscaped = argsEscaped; + return this; + } + + @Override + public String getDomainName() { + return domainName; + } + + @Override + public CreateContainerCmd withDomainName(String domainName) { + Objects.requireNonNull(domainName, "no domainName was specified"); + this.domainName = domainName; + return this; + } + + @Override + public String[] getEntrypoint() { + return entrypoint; + } + + @Override + public CreateContainerCmd withEntrypoint(String... entrypoint) { + Objects.requireNonNull(entrypoint, "entrypoint was not specified"); + this.entrypoint = entrypoint; + return this; + } + + @Override + public CreateContainerCmd withEntrypoint(List entrypoint) { + Objects.requireNonNull(entrypoint, "entrypoint was not specified"); + return withEntrypoint(entrypoint.toArray(new String[0])); + } + + @Override + public String[] getEnv() { + return env; + } + + @Override + public CreateContainerCmd withEnv(String... env) { + Objects.requireNonNull(env, "env was not specified"); + this.env = env; + return this; + } + + @Override + public CreateContainerCmd withEnv(List env) { + Objects.requireNonNull(env, "env was not specified"); + return withEnv(env.toArray(new String[0])); + } + + @Override + public ExposedPort[] getExposedPorts() { + return exposedPorts.getExposedPorts(); + } + + @Override + public CreateContainerCmd withExposedPorts(ExposedPort... exposedPorts) { + Objects.requireNonNull(exposedPorts, "exposedPorts was not specified"); + this.exposedPorts = new ExposedPorts(exposedPorts); + return this; + } + + @Override + public CreateContainerCmd withExposedPorts(List exposedPorts) { + Objects.requireNonNull(exposedPorts, "exposedPorts was not specified"); + return withExposedPorts(exposedPorts.toArray(new ExposedPort[0])); + } + + /** + * @see #stopSignal + */ + @Override + public String getStopSignal() { + return stopSignal; + } + + @Override + public CreateContainerCmd withStopSignal(String stopSignal) { + Objects.requireNonNull(stopSignal, "stopSignal wasn't specified."); + this.stopSignal = stopSignal; + return this; + } + + @Override + public Integer getStopTimeout() { + return stopTimeout; + } + + @Override + public CreateContainerCmd withStopTimeout(Integer stopTimeout) { + this.stopTimeout = stopTimeout; + return this; + } + + @Override + public String getHostName() { + return hostName; + } + + @Override + public CreateContainerCmd withHostName(String hostName) { + Objects.requireNonNull(hostName, "no hostName was specified"); + this.hostName = hostName; + return this; + } + + @Override + public String getImage() { + return image; + } + + @Override + public CreateContainerCmd withImage(String image) { + Objects.requireNonNull(image, "no image was specified"); + this.image = image; + return this; + } + + @Override + public Map getLabels() { + return labels; + } + + @Override + public CreateContainerCmd withLabels(Map labels) { + Objects.requireNonNull(labels, "labels was not specified"); + this.labels = labels; + return this; + } + + @Override + public String getMacAddress() { + return macAddress; + } + + @Override + public CreateContainerCmd withMacAddress(String macAddress) { + Objects.requireNonNull(macAddress, "macAddress was not specified"); + this.macAddress = macAddress; + return this; + } + + + @Override + public String getName() { + return name; + } + + @Override + public CreateContainerCmd withName(String name) { + Objects.requireNonNull(name, "name was not specified"); + this.name = name; + return this; + } + + @Override + public String[] getPortSpecs() { + return portSpecs; + } + + @Override + public CreateContainerCmd withPortSpecs(String... portSpecs) { + Objects.requireNonNull(portSpecs, "portSpecs was not specified"); + this.portSpecs = portSpecs; + return this; + } + + @Override + public CreateContainerCmd withPortSpecs(List portSpecs) { + Objects.requireNonNull(portSpecs, "portSpecs was not specified"); + return withPortSpecs(portSpecs.toArray(new String[0])); + } + + @Override + public String getUser() { + return user; + } + + @Override + public CreateContainerCmd withUser(String user) { + Objects.requireNonNull(user, "user was not specified"); + this.user = user; + return this; + } + + @Override + public Boolean isAttachStderr() { + return attachStderr; + } + + @Override + public CreateContainerCmd withAttachStderr(Boolean attachStderr) { + Objects.requireNonNull(attachStderr, "attachStderr was not specified"); + this.attachStderr = attachStderr; + return this; + } + + @Override + public Boolean isAttachStdin() { + return attachStdin; + } + + @Override + public CreateContainerCmd withAttachStdin(Boolean attachStdin) { + Objects.requireNonNull(attachStdin, "attachStdin was not specified"); + this.attachStdin = attachStdin; + return this; + } + + @Override + public Boolean isAttachStdout() { + return attachStdout; + } + + @Override + public CreateContainerCmd withAttachStdout(Boolean attachStdout) { + Objects.requireNonNull(attachStdout, "attachStdout was not specified"); + this.attachStdout = attachStdout; + return this; + } + + @Override + public Volume[] getVolumes() { + return volumes.getVolumes(); + } + + @Override + public CreateContainerCmd withVolumes(Volume... volumes) { + Objects.requireNonNull(volumes, "volumes was not specified"); + this.volumes = new Volumes(volumes); + return this; + } + + @Override + public CreateContainerCmd withVolumes(List volumes) { + Objects.requireNonNull(volumes, "volumes was not specified"); + return withVolumes(volumes.toArray(new Volume[0])); + } + + @Override + public String getWorkingDir() { + return workingDir; + } + + @Override + public CreateContainerCmd withWorkingDir(String workingDir) { + Objects.requireNonNull(workingDir, "workingDir was not specified"); + this.workingDir = workingDir; + return this; + } + + @Override + public Boolean isNetworkDisabled() { + return networkDisabled; + } + + @Override + public CreateContainerCmd withNetworkDisabled(Boolean disableNetwork) { + Objects.requireNonNull(disableNetwork, "disableNetwork was not specified"); + this.networkDisabled = disableNetwork; + return this; + } + + + @Override + public Boolean isStdInOnce() { + return stdInOnce; + } + + @Override + public CreateContainerCmd withStdInOnce(Boolean stdInOnce) { + Objects.requireNonNull(stdInOnce, "no stdInOnce was specified"); + this.stdInOnce = stdInOnce; + return this; + } + + @Override + public Boolean isStdinOpen() { + return stdinOpen; + } + + @Override + public CreateContainerCmd withStdinOpen(Boolean stdinOpen) { + Objects.requireNonNull(stdinOpen, "no stdinOpen was specified"); + this.stdinOpen = stdinOpen; + return this; + } + + + @Override + public Boolean isTty() { + return tty; + } + + @Override + public CreateContainerCmd withTty(Boolean tty) { + Objects.requireNonNull(tty, "no tty was specified"); + this.tty = tty; + return this; + } + + @Override + public HostConfig getHostConfig() { + return hostConfig; + } + + @Override + public CreateContainerCmd withHostConfig(HostConfig hostConfig) { + this.hostConfig = hostConfig; + return this; + } + + @Override + public String getIpv4Address() { + return ipv4Address; + } + + @Override + public CreateContainerCmd withIpv4Address(String ipv4Address) { + Objects.requireNonNull(ipv4Address, "no ipv4Address was specified"); + this.ipv4Address = ipv4Address; + return this; + } + + @Override + public String getIpv6Address() { + return ipv6Address; + } + + @Override + public CreateContainerCmd withIpv6Address(String ipv6Address) { + Objects.requireNonNull(ipv6Address, "no ipv6Address was specified"); + this.ipv6Address = ipv6Address; + return this; + } + + @CheckForNull + public List getOnBuild() { + return onBuild; + } + + public CreateContainerCmdImpl withOnBuild(List onBuild) { + this.onBuild = onBuild; + return this; + } + + @CheckForNull + @Override + public String getPlatform() { + return platform; + } + + @Override + public CreateContainerCmd withPlatform(String platform) { + this.platform = platform; + return this; + } + + /** + * @throws NotFoundException No such container + * @throws ConflictException Named container already exists + */ + @Override + public CreateContainerResponse exec() throws NotFoundException, ConflictException { + //code flow taken from https://github.com/docker/docker/blob/master/runconfig/opts/parse.go + ContainerNetwork containerNetwork = null; + + if (ipv4Address != null || ipv6Address != null) { + containerNetwork = new ContainerNetwork() + .withIpamConfig(new ContainerNetwork.Ipam() + .withIpv4Address(ipv4Address) + .withIpv6Address(ipv6Address) + ); + + } + + if (hostConfig.isUserDefinedNetwork() && hostConfig.getLinks().length > 0) { + if (containerNetwork == null) { + containerNetwork = new ContainerNetwork(); + } + + containerNetwork.withLinks(hostConfig.getLinks()); + } + + if (aliases != null) { + if (containerNetwork == null) { + containerNetwork = new ContainerNetwork(); + } + + containerNetwork.withAliases(aliases); + } + + if (containerNetwork != null && hostConfig.getNetworkMode() != null) { + networkingConfig = new NetworkingConfig() + .withEndpointsConfig(singletonMap(hostConfig.getNetworkMode(), containerNetwork)); + } + + return super.exec(); + } + + + @Override + public String toString() { + return ToStringBuilder.reflectionToString(this); + } + + @Override + public boolean equals(Object o) { + return EqualsBuilder.reflectionEquals(this, o); + } + + @Override + public int hashCode() { + return HashCodeBuilder.reflectionHashCode(this); + } + + public static class NetworkingConfig { + @JsonProperty("EndpointsConfig") + public Map endpointsConfig; + + public Map getEndpointsConfig() { + return endpointsConfig; + } + + public NetworkingConfig withEndpointsConfig(Map endpointsConfig) { + this.endpointsConfig = endpointsConfig; + return this; + } + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/CreateImageCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateImageCmdImpl.java similarity index 76% rename from src/main/java/com/github/dockerjava/core/command/CreateImageCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateImageCmdImpl.java index 57ddc2908..cab9607d0 100644 --- a/src/main/java/com/github/dockerjava/core/command/CreateImageCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateImageCmdImpl.java @@ -1,8 +1,7 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; - import java.io.InputStream; +import java.util.Objects; import com.github.dockerjava.api.command.CreateImageCmd; import com.github.dockerjava.api.command.CreateImageResponse; @@ -12,7 +11,7 @@ */ public class CreateImageCmdImpl extends AbstrDockerCmd implements CreateImageCmd { - private String repository, tag; + private String repository, tag, platform; private InputStream imageStream; @@ -38,6 +37,11 @@ public String getTag() { return tag; } + @Override + public String getPlatform() { + return platform; + } + @Override public InputStream getImageStream() { return imageStream; @@ -49,7 +53,7 @@ public InputStream getImageStream() { */ @Override public CreateImageCmdImpl withRepository(String repository) { - checkNotNull(repository, "repository was not specified"); + Objects.requireNonNull(repository, "repository was not specified"); this.repository = repository; return this; } @@ -60,7 +64,7 @@ public CreateImageCmdImpl withRepository(String repository) { */ @Override public CreateImageCmdImpl withImageStream(InputStream imageStream) { - checkNotNull(imageStream, "imageStream was not specified"); + Objects.requireNonNull(imageStream, "imageStream was not specified"); this.imageStream = imageStream; return this; } @@ -71,8 +75,17 @@ public CreateImageCmdImpl withImageStream(InputStream imageStream) { */ @Override public CreateImageCmdImpl withTag(String tag) { - checkNotNull(tag, "tag was not specified"); + Objects.requireNonNull(tag, "tag was not specified"); this.tag = tag; return this; } + + /** + * {@inheritDoc} + */ + @Override + public CreateImageCmd withPlatform(String platform) { + this.platform = platform; + return this; + } } diff --git a/src/main/java/com/github/dockerjava/core/command/CreateNetworkCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateNetworkCmdImpl.java similarity index 77% rename from src/main/java/com/github/dockerjava/core/command/CreateNetworkCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateNetworkCmdImpl.java index 834aa6905..db2766796 100644 --- a/src/main/java/com/github/dockerjava/core/command/CreateNetworkCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateNetworkCmdImpl.java @@ -2,6 +2,7 @@ import java.util.HashMap; import java.util.Map; +import java.util.Objects; import com.fasterxml.jackson.annotation.JsonProperty; import com.github.dockerjava.api.command.CreateNetworkCmd; @@ -34,6 +35,12 @@ public class CreateNetworkCmdImpl extends AbstrDockerCmd labels; + public CreateNetworkCmdImpl(DockerCmdSyncExec execution) { super(execution); } @@ -114,4 +121,33 @@ public CreateNetworkCmd withEnableIpv6(boolean enableIpv6) { this.enableIpv6 = enableIpv6; return this; } + + @Override + public Boolean getAttachable() { + return this.attachable; + } + + /** + * {@inheritDoc} + */ + @Override + public CreateNetworkCmd withAttachable(Boolean attachable) { + this.attachable = attachable; + return this; + } + + @Override + public Map getLabels() { + return labels; + } + + /** + * {@inheritDoc} + */ + @Override + public CreateNetworkCmd withLabels(Map labels) { + Objects.requireNonNull(labels, "labels was not specified"); + this.labels = labels; + return this; + } } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateSecretCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateSecretCmdImpl.java new file mode 100644 index 000000000..19891325a --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateSecretCmdImpl.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.CreateSecretCmd; +import com.github.dockerjava.api.command.CreateSecretResponse; +import com.github.dockerjava.api.model.SecretSpec; + +import java.util.Objects; + +/** + * Creates a new secret + */ +public class CreateSecretCmdImpl extends AbstrDockerCmd implements + CreateSecretCmd { + + private SecretSpec secretSpec; + + public CreateSecretCmdImpl(Exec exec, SecretSpec secretSpec) { + super(exec); + Objects.requireNonNull(secretSpec, "secretSpec was not specified"); + withSecretSpec(secretSpec); + } + + @Override + public SecretSpec getSecretSpec() { + return secretSpec; + } + + @Override + public CreateSecretCmd withSecretSpec(SecretSpec secretSpec) { + Objects.requireNonNull(secretSpec, "secretSpec was not specified"); + this.secretSpec = secretSpec; + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateServiceCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateServiceCmdImpl.java new file mode 100644 index 000000000..1da533b88 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateServiceCmdImpl.java @@ -0,0 +1,49 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.CreateServiceCmd; +import com.github.dockerjava.api.command.CreateServiceResponse; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.ServiceSpec; + +import java.util.Objects; + +/** + * Creates a new service + */ +public class CreateServiceCmdImpl extends AbstrDockerCmd implements + CreateServiceCmd { + + private ServiceSpec serviceSpec; + + private AuthConfig authConfig; + + public CreateServiceCmdImpl(CreateServiceCmd.Exec exec, ServiceSpec serviceSpec) { + super(exec); + Objects.requireNonNull(serviceSpec, "serviceSpec was not specified"); + withServiceSpec(serviceSpec); + } + + @Override + public ServiceSpec getServiceSpec() { + return serviceSpec; + } + + @Override + public AuthConfig getAuthConfig() { + return authConfig; + } + + @Override + public CreateServiceCmd withServiceSpec(ServiceSpec serviceSpec) { + Objects.requireNonNull(serviceSpec, "serviceSpec was not specified"); + this.serviceSpec = serviceSpec; + return this; + } + + @Override + public CreateServiceCmd withAuthConfig(AuthConfig authConfig) { + Objects.requireNonNull(authConfig, "authConfig was not specified"); + this.authConfig = authConfig; + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateVolumeCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateVolumeCmdImpl.java new file mode 100644 index 000000000..6d3a8bb7b --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/CreateVolumeCmdImpl.java @@ -0,0 +1,81 @@ +package com.github.dockerjava.core.command; + +import java.util.Map; +import java.util.Objects; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.command.CreateVolumeCmd; +import com.github.dockerjava.api.command.CreateVolumeResponse; + +/** + * Create a volume. + * + * @author Marcus Linke + */ +public class CreateVolumeCmdImpl extends AbstrDockerCmd implements + CreateVolumeCmd { + + @JsonProperty("Name") + private String name; + + @JsonProperty("Labels") + private Map labels; + + @JsonProperty("Driver") + private String driver; + + @JsonProperty("DriverOpts") + private Map driverOpts; + + public CreateVolumeCmdImpl(CreateVolumeCmd.Exec exec) { + super(exec); + } + + @Override + public String getName() { + return name; + } + + @Override + public Map getLabels() { + return labels; + } + + @Override + public String getDriver() { + return driver; + } + + @Override + public Map getDriverOpts() { + return driverOpts; + } + + @Override + public CreateVolumeCmdImpl withName(String name) { + Objects.requireNonNull(name, "name was not specified"); + this.name = name; + return this; + } + + @Override + public CreateVolumeCmdImpl withLabels(Map labels) { + Objects.requireNonNull(labels, "labels was not specified"); + this.labels = labels; + return this; + } + + @Override + public CreateVolumeCmdImpl withDriver(String driver) { + Objects.requireNonNull(driver, "driver was not specified"); + this.driver = driver; + return this; + } + + @Override + public CreateVolumeCmd withDriverOpts(Map driverOpts) { + Objects.requireNonNull(driverOpts, "driverOpts was not specified"); + this.driverOpts = driverOpts; + return this; + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/DisconnectFromNetworkCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/DisconnectFromNetworkCmdImpl.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/command/DisconnectFromNetworkCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/DisconnectFromNetworkCmdImpl.java diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/EventsCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/EventsCmdImpl.java new file mode 100644 index 000000000..f578b3a19 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/EventsCmdImpl.java @@ -0,0 +1,95 @@ +package com.github.dockerjava.core.command; + +import java.util.List; +import java.util.Map; +import java.util.Objects; + +import com.github.dockerjava.api.command.EventsCmd; +import com.github.dockerjava.api.model.Event; +import com.github.dockerjava.core.util.FiltersBuilder; + +/** + * Stream docker events + */ +public class EventsCmdImpl extends AbstrAsyncDockerCmd implements EventsCmd { + + private String since; + + private String until; + + private FiltersBuilder filters = new FiltersBuilder(); + + public EventsCmdImpl(EventsCmd.Exec exec) { + super(exec); + } + + @Override + public EventsCmd withSince(String since) { + this.since = since; + return this; + } + + @Override + public EventsCmd withUntil(String until) { + this.until = until; + return this; + } + + @Override + public EventsCmd withContainerFilter(String... container) { + Objects.requireNonNull(container, "container have not been specified"); + this.filters.withContainers(container); + return this; + } + + @Override + public EventsCmd withImageFilter(String... image) { + Objects.requireNonNull(image, "image have not been specified"); + this.filters.withImages(image); + return this; + } + + @Override + public EventsCmd withEventFilter(String... event) { + Objects.requireNonNull(event, "event have not been specified"); + this.filters.withFilter("event", event); + return this; + } + + @Override + public EventsCmd withEventTypeFilter(String... eventTypes) { + Objects.requireNonNull(eventTypes, "event types have not been specified"); + this.filters.withEventTypes(eventTypes); + return this; + } + + @Override + public EventsCmd withLabelFilter(String... label) { + Objects.requireNonNull(label, "label have not been specified"); + this.filters.withLabels(label); + return this; + } + + @Override + public EventsCmd withLabelFilter(Map labels) { + Objects.requireNonNull(labels, "labels have not been specified"); + this.filters.withLabels(labels); + return this; + } + + @Override + public String getSince() { + return since; + } + + @Override + public String getUntil() { + return until; + } + + @Override + public Map> getFilters() { + return filters.build(); + } + +} diff --git a/src/main/java/com/github/dockerjava/core/command/EventsResultCallback.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/EventsResultCallback.java similarity index 85% rename from src/main/java/com/github/dockerjava/core/command/EventsResultCallback.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/EventsResultCallback.java index cd6205f6d..e6df7fbc1 100644 --- a/src/main/java/com/github/dockerjava/core/command/EventsResultCallback.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/EventsResultCallback.java @@ -13,7 +13,9 @@ * * @author Marcus Linke * + * @deprecated use {@link com.github.dockerjava.api.async.ResultCallback.Adapter} */ +@Deprecated public class EventsResultCallback extends ResultCallbackTemplate { private static final Logger LOGGER = LoggerFactory.getLogger(EventsResultCallback.class); diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ExecCreateCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ExecCreateCmdImpl.java new file mode 100644 index 000000000..8ea6e275f --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ExecCreateCmdImpl.java @@ -0,0 +1,173 @@ +package com.github.dockerjava.core.command; + +import java.util.List; +import java.util.Objects; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.command.ExecCreateCmd; +import com.github.dockerjava.api.command.ExecCreateCmdResponse; +import com.github.dockerjava.api.exception.NotFoundException; + +public class ExecCreateCmdImpl extends AbstrDockerCmd implements ExecCreateCmd { + + private String containerId; + + @JsonProperty("AttachStdin") + private Boolean attachStdin; + + @JsonProperty("AttachStdout") + private Boolean attachStdout; + + @JsonProperty("AttachStderr") + private Boolean attachStderr; + + @JsonProperty("Tty") + private Boolean tty; + + /** + * @since 1.21 + **/ + @JsonProperty("Privileged") + private Boolean privileged; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_19} + */ + @JsonProperty("User") + private String user; + + @JsonProperty("Cmd") + private String[] cmd; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_25} + */ + @JsonProperty("Env") + private List env; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_35} + */ + @JsonProperty("WorkingDir") + private String workingDir; + + public ExecCreateCmdImpl(ExecCreateCmd.Exec exec, String containerId) { + super(exec); + withContainerId(containerId); + } + + @Override + public ExecCreateCmd withContainerId(String containerId) { + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); + return this; + } + + @Override + public ExecCreateCmd withAttachStdin(Boolean attachStdin) { + this.attachStdin = attachStdin; + return this; + } + + @Override + public ExecCreateCmd withAttachStdout(Boolean attachStdout) { + this.attachStdout = attachStdout; + return this; + } + + @Override + public ExecCreateCmd withAttachStderr(Boolean attachStderr) { + this.attachStderr = attachStderr; + return this; + } + + @Override + public ExecCreateCmd withTty(Boolean tty) { + this.tty = tty; + return this; + } + + @Override + public ExecCreateCmd withUser(String user) { + this.user = user; + return this; + } + + @Override + public ExecCreateCmd withCmd(String... cmd) { + this.cmd = cmd; + return this; + } + + @Override + public ExecCreateCmd withEnv(List env) { + this.env = env; + return this; + } + + @Override + public ExecCreateCmd withPrivileged(Boolean privileged) { + this.privileged = privileged; + return this; + } + + @Override + public ExecCreateCmd withWorkingDir(String workingDir) { + this.workingDir = workingDir; + return this; + } + + @Override + public String getContainerId() { + return containerId; + } + + @Override + public Boolean hasAttachStdinEnabled() { + return attachStdin; + } + + @Override + public Boolean hasAttachStdoutEnabled() { + return attachStdout; + } + + @Override + public Boolean hasAttachStderrEnabled() { + return attachStderr; + } + + @Override + public Boolean hasTtyEnabled() { + return tty; + } + + @Override + public List getEnv() { + return env; + } + + @Override + public Boolean getPrivileged() { + return privileged; + } + + @Override + public String getUser() { + return user; + } + + @Override + public String getWorkingDir() { + return workingDir; + } + + /** + * @throws NotFoundException + * No such container + */ + @Override + public ExecCreateCmdResponse exec() throws NotFoundException { + return super.exec(); + } + +} diff --git a/src/main/java/com/github/dockerjava/core/command/ExecStartCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ExecStartCmdImpl.java similarity index 86% rename from src/main/java/com/github/dockerjava/core/command/ExecStartCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/ExecStartCmdImpl.java index 01532090a..d5990ef80 100644 --- a/src/main/java/com/github/dockerjava/core/command/ExecStartCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ExecStartCmdImpl.java @@ -1,19 +1,15 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; - import java.io.InputStream; +import java.util.Objects; import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; import com.fasterxml.jackson.annotation.JsonProperty; import com.github.dockerjava.api.async.ResultCallback; import com.github.dockerjava.api.command.ExecStartCmd; import com.github.dockerjava.api.exception.NotFoundException; import com.github.dockerjava.api.model.Frame; -@JsonInclude(Include.NON_NULL) public class ExecStartCmdImpl extends AbstrAsyncDockerCmd implements ExecStartCmd { @JsonIgnore @@ -40,8 +36,7 @@ public String getExecId() { @Override public ExecStartCmd withExecId(String execId) { - checkNotNull(execId, "execId was not specified"); - this.execId = execId; + this.execId = Objects.requireNonNull(execId, "execId was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/ExecStartResultCallback.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ExecStartResultCallback.java similarity index 94% rename from src/main/java/com/github/dockerjava/core/command/ExecStartResultCallback.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/ExecStartResultCallback.java index 847d62057..39c0e1e8c 100644 --- a/src/main/java/com/github/dockerjava/core/command/ExecStartResultCallback.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ExecStartResultCallback.java @@ -1,19 +1,20 @@ package com.github.dockerjava.core.command; -import java.io.IOException; -import java.io.OutputStream; - +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.core.async.ResultCallbackTemplate; import org.slf4j.Logger; import org.slf4j.LoggerFactory; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.core.async.ResultCallbackTemplate; +import java.io.IOException; +import java.io.OutputStream; /** * * @author Marcus Linke * + * @deprecated use {@link com.github.dockerjava.api.async.ResultCallback.Adapter} */ +@Deprecated public class ExecStartResultCallback extends ResultCallbackTemplate { private static final Logger LOGGER = LoggerFactory.getLogger(ExecStartResultCallback.class); diff --git a/src/main/java/com/github/dockerjava/core/command/FrameReader.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/FrameReader.java similarity index 99% rename from src/main/java/com/github/dockerjava/core/command/FrameReader.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/FrameReader.java index 9c9c31e74..1dc5d3503 100644 --- a/src/main/java/com/github/dockerjava/core/command/FrameReader.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/FrameReader.java @@ -14,6 +14,7 @@ *

* See: {@link }http://docs.docker.com/v1.6/reference/api/docker_remote_api_v1.13/#attach-to-a-container} */ +@Deprecated public class FrameReader implements AutoCloseable { private static final int HEADER_SIZE = 8; diff --git a/src/main/java/com/github/dockerjava/core/command/InfoCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InfoCmdImpl.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/command/InfoCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/InfoCmdImpl.java diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/InitializeSwarmCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InitializeSwarmCmdImpl.java new file mode 100644 index 000000000..f13f307ad --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InitializeSwarmCmdImpl.java @@ -0,0 +1,95 @@ +package com.github.dockerjava.core.command; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.command.InitializeSwarmCmd; +import com.github.dockerjava.api.model.SwarmSpec; +import org.apache.commons.lang3.builder.EqualsBuilder; +import org.apache.commons.lang3.builder.HashCodeBuilder; +import org.apache.commons.lang3.builder.ToStringBuilder; + +import javax.annotation.CheckForNull; + +public class InitializeSwarmCmdImpl extends AbstrDockerCmd implements + InitializeSwarmCmd { + + @JsonProperty("ListenAddr") + private String listenAddr = "0.0.0.0"; + + @JsonProperty("AdvertiseAddr") + private String advertiseAddr; + + @JsonProperty("ForceNewCluster") + private boolean forceNewCluster; + + @JsonProperty("Spec") + private SwarmSpec spec; + + public InitializeSwarmCmdImpl(InitializeSwarmCmd.Exec exec, SwarmSpec swarmSpec) { + super(exec); + this.spec = swarmSpec; + } + + @Override + @CheckForNull + public String getListenAddr() { + return listenAddr; + } + + @Override + public InitializeSwarmCmd withListenAddr(String listenAddr) { + this.listenAddr = listenAddr; + return this; + } + + @Override + @CheckForNull + public String getAdvertiseAddr() { + return advertiseAddr; + } + + @Override + public InitializeSwarmCmd withAdvertiseAddr(String advertiseAddr) { + this.advertiseAddr = advertiseAddr; + return this; + } + + @Override + @CheckForNull + public Boolean isForceNewCluster() { + return forceNewCluster; + } + + @Override + public InitializeSwarmCmd withForceNewCluster(Boolean forceNewCluster) { + this.forceNewCluster = forceNewCluster; + return this; + } + + @Override + @CheckForNull + public SwarmSpec getSwarmSpec() { + return spec; + } + + @Override + public InitializeSwarmCmd withSwarmSpec(SwarmSpec swarmSpec) { + this.spec = swarmSpec; + return this; + } + + @Override + public String toString() { + return ToStringBuilder.reflectionToString(this); + } + + @Override + public boolean equals(Object o) { + return EqualsBuilder.reflectionEquals(this, o); + } + + @Override + public int hashCode() { + return HashCodeBuilder.reflectionHashCode(this); + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/InpectNetworkCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InpectNetworkCmdImpl.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/command/InpectNetworkCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/InpectNetworkCmdImpl.java diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectConfigCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectConfigCmdImpl.java new file mode 100644 index 000000000..eff4170cb --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectConfigCmdImpl.java @@ -0,0 +1,40 @@ +package com.github.dockerjava.core.command; + +import java.util.Objects; + +import com.github.dockerjava.api.command.InspectConfigCmd; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Config; + +/** + * Inspect the details of a config. + */ +public class InspectConfigCmdImpl extends AbstrDockerCmd implements InspectConfigCmd { + + private String configId; + + public InspectConfigCmdImpl(Exec exec, String configId) { + super(exec); + withConfigId(configId); + } + + @Override + public String getConfigId() { + return configId; + } + + @Override + public InspectConfigCmd withConfigId(String configId) { + this.configId = Objects.requireNonNull(configId, "configId was not specified"); + return this; + } + + /** + * @throws NotFoundException + * No such config + */ + @Override + public Config exec() throws NotFoundException { + return super.exec(); + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/InspectContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectContainerCmdImpl.java similarity index 87% rename from src/main/java/com/github/dockerjava/core/command/InspectContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectContainerCmdImpl.java index ed3c01b00..ab8c2989a 100644 --- a/src/main/java/com/github/dockerjava/core/command/InspectContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectContainerCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.InspectContainerCmd; import com.github.dockerjava.api.command.InspectContainerResponse; @@ -27,8 +27,7 @@ public String getContainerId() { @Override public InspectContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/InspectExecCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectExecCmdImpl.java similarity index 85% rename from src/main/java/com/github/dockerjava/core/command/InspectExecCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectExecCmdImpl.java index 22d6d70f4..36ad73e28 100644 --- a/src/main/java/com/github/dockerjava/core/command/InspectExecCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectExecCmdImpl.java @@ -1,9 +1,10 @@ package com.github.dockerjava.core.command; +import java.util.Objects; + import com.github.dockerjava.api.command.InspectExecCmd; import com.github.dockerjava.api.command.InspectExecResponse; import com.github.dockerjava.api.exception.NotFoundException; -import com.google.common.base.Preconditions; public class InspectExecCmdImpl extends AbstrDockerCmd implements InspectExecCmd { private String execId; @@ -20,8 +21,7 @@ public String getExecId() { @Override public InspectExecCmd withExecId(String execId) { - Preconditions.checkNotNull(execId, "execId was not specified"); - this.execId = execId; + this.execId = Objects.requireNonNull(execId, "execId was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/InspectImageCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectImageCmdImpl.java similarity index 85% rename from src/main/java/com/github/dockerjava/core/command/InspectImageCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectImageCmdImpl.java index 48dd30757..5ffa7c5d7 100644 --- a/src/main/java/com/github/dockerjava/core/command/InspectImageCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectImageCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.InspectImageCmd; import com.github.dockerjava.api.command.InspectImageResponse; @@ -26,8 +26,7 @@ public String getImageId() { @Override public InspectImageCmd withImageId(String imageId) { - checkNotNull(imageId, "imageId was not specified"); - this.imageId = imageId; + this.imageId = Objects.requireNonNull(imageId, "imageId was not specified"); return this; } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectServiceCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectServiceCmdImpl.java new file mode 100644 index 000000000..804710ce1 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectServiceCmdImpl.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.core.command; + +import java.util.Objects; + +import com.github.dockerjava.api.command.InspectServiceCmd; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Service; + +/** + * Inspect the details of a container. + */ +public class InspectServiceCmdImpl extends AbstrDockerCmd implements + InspectServiceCmd { + + private String serviceId; + + public InspectServiceCmdImpl(Exec exec, String serviceId) { + super(exec); + withServiceId(serviceId); + } + + @Override + public String getServiceId() { + return serviceId; + } + + @Override + public InspectServiceCmd withServiceId(String serviceId) { + this.serviceId = Objects.requireNonNull(serviceId, "serviceId was not specified"); + return this; + } + + /** + * @throws NotFoundException + * No such service + */ + @Override + public Service exec() throws NotFoundException { + return super.exec(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectSwarmCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectSwarmCmdImpl.java new file mode 100644 index 000000000..c084c8af9 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectSwarmCmdImpl.java @@ -0,0 +1,21 @@ +package com.github.dockerjava.core.command; + + +import com.github.dockerjava.api.command.InspectSwarmCmd; +import com.github.dockerjava.api.model.Swarm; + +/** + * Inspect a swarm. + */ +public class InspectSwarmCmdImpl extends AbstrDockerCmd implements + InspectSwarmCmd { + + public InspectSwarmCmdImpl(InspectSwarmCmd.Exec exec) { + super(exec); + } + + @Override + public Swarm exec() { + return super.exec(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectSwarmNodeCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectSwarmNodeCmdImpl.java new file mode 100644 index 000000000..9289878d1 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectSwarmNodeCmdImpl.java @@ -0,0 +1,44 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.InspectSwarmNodeCmd; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.SwarmNode; + +import javax.annotation.CheckForNull; + +import java.util.Objects; + +/** + * Inspect the details of a swarmNode. + */ +public class InspectSwarmNodeCmdImpl extends AbstrDockerCmd implements + InspectSwarmNodeCmd { + + private String swarmNodeId; + + public InspectSwarmNodeCmdImpl(Exec exec, String swarmNodeId) { + super(exec); + withSwarmNodeId(swarmNodeId); + } + + @Override + @CheckForNull + public String getSwarmNodeId() { + return swarmNodeId; + } + + @Override + public InspectSwarmNodeCmd withSwarmNodeId(String swarmNodeId) { + this.swarmNodeId = Objects.requireNonNull(swarmNodeId, "swarmNodeId was not specified"); + return this; + } + + /** + * @throws NotFoundException + * No such swarmNode + */ + @Override + public SwarmNode exec() throws NotFoundException { + return super.exec(); + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/InspectVolumeCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectVolumeCmdImpl.java similarity index 86% rename from src/main/java/com/github/dockerjava/core/command/InspectVolumeCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectVolumeCmdImpl.java index c789d91c9..5b76310db 100644 --- a/src/main/java/com/github/dockerjava/core/command/InspectVolumeCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/InspectVolumeCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.InspectVolumeCmd; import com.github.dockerjava.api.command.InspectVolumeResponse; @@ -26,8 +26,7 @@ public String getName() { @Override public InspectVolumeCmd withName(String name) { - checkNotNull(name, "name was not specified"); - this.name = name; + this.name = Objects.requireNonNull(name, "name was not specified"); return this; } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/JoinSwarmCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/JoinSwarmCmdImpl.java new file mode 100644 index 000000000..d19ed8ea2 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/JoinSwarmCmdImpl.java @@ -0,0 +1,94 @@ +package com.github.dockerjava.core.command; + + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.command.JoinSwarmCmd; +import org.apache.commons.lang3.builder.EqualsBuilder; +import org.apache.commons.lang3.builder.HashCodeBuilder; +import org.apache.commons.lang3.builder.ToStringBuilder; + +import javax.annotation.CheckForNull; +import java.util.List; + +public class JoinSwarmCmdImpl extends AbstrDockerCmd implements + JoinSwarmCmd { + + @JsonProperty("ListenAddr") + private String listenAddr = "0.0.0.0"; + + @JsonProperty("AdvertiseAddr") + private String advertiseAddr; + + @JsonProperty("RemoteAddrs") + private List remoteAddrs; + + @JsonProperty("JoinToken") + private String joinToken; + + public JoinSwarmCmdImpl(JoinSwarmCmd.Exec exec) { + super(exec); + } + + @Override + @CheckForNull + public String getListenAddr() { + return listenAddr; + } + + @Override + public JoinSwarmCmd withListenAddr(String listenAddr) { + this.listenAddr = listenAddr; + return this; + } + + @Override + @CheckForNull + public String getAdvertiseAddr() { + return advertiseAddr; + } + + @Override + public JoinSwarmCmd withAdvertiseAddr(String advertiseAddr) { + this.advertiseAddr = advertiseAddr; + return this; + } + + @Override + @CheckForNull + public List getRemoteAddrs() { + return remoteAddrs; + } + + @Override + public JoinSwarmCmd withRemoteAddrs(List remoteAddrs) { + this.remoteAddrs = remoteAddrs; + return this; + } + + @Override + @CheckForNull + public String getJoinToken() { + return joinToken; + } + + @Override + public JoinSwarmCmd withJoinToken(String joinToken) { + this.joinToken = joinToken; + return this; + } + + @Override + public String toString() { + return ToStringBuilder.reflectionToString(this); + } + + @Override + public boolean equals(Object o) { + return EqualsBuilder.reflectionEquals(this, o); + } + + @Override + public int hashCode() { + return HashCodeBuilder.reflectionHashCode(this); + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/KillContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/KillContainerCmdImpl.java similarity index 80% rename from src/main/java/com/github/dockerjava/core/command/KillContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/KillContainerCmdImpl.java index a23d0caa5..bc7207829 100644 --- a/src/main/java/com/github/dockerjava/core/command/KillContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/KillContainerCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.KillContainerCmd; import com.github.dockerjava.api.exception.NotFoundException; @@ -29,15 +29,13 @@ public String getSignal() { @Override public KillContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } @Override public KillContainerCmd withSignal(String signal) { - checkNotNull(signal, "signal was not specified"); - this.signal = signal; + this.signal = Objects.requireNonNull(signal, "signal was not specified"); return this; } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/LeaveSwarmCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/LeaveSwarmCmdImpl.java new file mode 100644 index 000000000..a5731bc1e --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/LeaveSwarmCmdImpl.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.command; + + +import com.github.dockerjava.api.command.LeaveSwarmCmd; + +import javax.annotation.CheckForNull; + +public class LeaveSwarmCmdImpl extends AbstrDockerCmd implements LeaveSwarmCmd { + + private Boolean force; + + public LeaveSwarmCmdImpl(LeaveSwarmCmd.Exec exec) { + super(exec); + } + + @Override + @CheckForNull + public Boolean hasForceEnabled() { + return force; + } + + @Override + public LeaveSwarmCmd withForceEnabled(Boolean force) { + this.force = force; + return this; + } + + @Override + public Void exec() { + return super.exec(); + } + + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListConfigsCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListConfigsCmdImpl.java new file mode 100644 index 000000000..f67dd30b8 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListConfigsCmdImpl.java @@ -0,0 +1,31 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.ListConfigsCmd; +import com.github.dockerjava.api.model.Config; + +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +/** + * List configs. + */ +public class ListConfigsCmdImpl extends AbstrDockerCmd> implements ListConfigsCmd { + + private Map> filters = Collections.emptyMap(); + + public ListConfigsCmdImpl(Exec exec) { + super(exec); + } + + @Override + public Map> getFilters() { + return filters; + } + + public ListConfigsCmd withFilters(Map> filters) { + this.filters = Objects.requireNonNull(filters, "filters was not specified"); + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListContainersCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListContainersCmdImpl.java new file mode 100644 index 000000000..94de5daff --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListContainersCmdImpl.java @@ -0,0 +1,151 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.ListContainersCmd; +import com.github.dockerjava.api.model.Container; +import com.github.dockerjava.core.util.FiltersBuilder; + +import java.util.Collection; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +import static com.google.common.base.Preconditions.checkArgument; + +/** + * List containers. + */ +public class ListContainersCmdImpl extends AbstrDockerCmd> implements + ListContainersCmd { + + private Integer limit = -1; + + private Boolean showSize, showAll = false; + + private String sinceId, beforeId; + + private FiltersBuilder filters = new FiltersBuilder(); + + public ListContainersCmdImpl(ListContainersCmd.Exec exec) { + super(exec); + } + + @Override + public Integer getLimit() { + return limit; + } + + @Override + public Boolean hasShowSizeEnabled() { + return showSize; + } + + @Override + public Boolean hasShowAllEnabled() { + return showAll; + } + + @Override + public String getSinceId() { + return sinceId; + } + + @Override + public String getBeforeId() { + return beforeId; + } + + @Override + public Map> getFilters() { + return filters.build(); + } + + @Override + public ListContainersCmd withShowAll(Boolean showAll) { + this.showAll = showAll; + return this; + } + + @Override + public ListContainersCmd withShowSize(Boolean showSize) { + this.showSize = showSize; + return this; + } + + @Override + public ListContainersCmd withLimit(Integer limit) { + Objects.requireNonNull(limit, "limit was not specified"); + checkArgument(limit > 0, "limit must be greater 0"); + this.limit = limit; + return this; + } + + @Override + public ListContainersCmd withSince(String since) { + this.sinceId = Objects.requireNonNull(since, "since was not specified"); + return this; + } + + @Override + public ListContainersCmd withBefore(String before) { + this.beforeId = Objects.requireNonNull(before, "before was not specified"); + return this; + } + + @Override + public ListContainersCmd withNameFilter(Collection name) { + return withFilter("name", name); + } + + @Override + public ListContainersCmd withIdFilter(Collection id) { + return withFilter("id", id); + } + + @Override + public ListContainersCmd withAncestorFilter(Collection ancestor) { + return withFilter("ancestor", ancestor); + } + + @Override + public ListContainersCmd withVolumeFilter(Collection volume) { + return withFilter("volume", volume); + } + + @Override + public ListContainersCmd withNetworkFilter(Collection network) { + return withFilter("network", network); + } + + @Override + public ListContainersCmd withLabelFilter(Collection labels) { + return withFilter("label", labels); + } + + @Override + public ListContainersCmd withLabelFilter(Map labels) { + Objects.requireNonNull(labels, "labels was not specified"); + this.filters.withLabels(labels); + return this; + } + + @Override + public ListContainersCmd withExitedFilter(Integer exited) { + Objects.requireNonNull(exited, "exited was not specified"); + this.filters.withFilter("exited", exited.toString()); + return this; + } + + @Override + public ListContainersCmd withFilter(String filterName, Collection filterValues) { + Objects.requireNonNull(filterValues, filterName + " was not specified"); + this.filters.withFilter(filterName, filterValues); + return this; + } + + @Override + public ListContainersCmd withStatusFilter(Collection status) { + Objects.requireNonNull(status, "status was not specified"); + this.filters.withFilter("status", status); + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListImagesCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListImagesCmdImpl.java new file mode 100644 index 000000000..40d378c8c --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListImagesCmdImpl.java @@ -0,0 +1,99 @@ +package com.github.dockerjava.core.command; + +import java.util.Collection; +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +import org.apache.commons.lang3.builder.ReflectionToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; + +import com.github.dockerjava.api.command.ListImagesCmd; +import com.github.dockerjava.api.model.Image; +import com.github.dockerjava.core.util.FiltersBuilder; + +/** + * List images + */ +public class ListImagesCmdImpl extends AbstrDockerCmd> implements ListImagesCmd { + + private String imageNameFilter; + + private Boolean showAll = false; + + private FiltersBuilder filters = new FiltersBuilder(); + + public ListImagesCmdImpl(ListImagesCmd.Exec exec) { + super(exec); + } + + @Override + public Map> getFilters() { + return filters.build(); + } + + @Override + public Boolean hasShowAllEnabled() { + return showAll; + } + + @Override + public ListImagesCmd withShowAll(Boolean showAll) { + this.showAll = showAll; + return this; + } + + @Override + public ListImagesCmd withDanglingFilter(Boolean dangling) { + Objects.requireNonNull(dangling, "dangling have not been specified"); + withFilter("dangling", Collections.singletonList(dangling.toString())); + return this; + } + + @Override + public ListImagesCmd withLabelFilter(String... labels) { + Objects.requireNonNull(labels, "labels have not been specified"); + filters.withLabels(labels); + return this; + } + + @Override + public ListImagesCmd withLabelFilter(Map labels) { + Objects.requireNonNull(labels, "labels have not been specified"); + filters.withLabels(labels); + return this; + } + + @Override + public ListImagesCmd withImageNameFilter(String imageNameFilter) { + Objects.requireNonNull(imageNameFilter, "image name filter not specified"); + this.imageNameFilter = imageNameFilter; + return this; + } + + @Override + public ListImagesCmd withReferenceFilter(String reference) { + Objects.requireNonNull(reference, "reference filter not specified"); + withFilter("reference", Collections.singletonList(reference)); + return this; + } + + @Override + public ListImagesCmd withFilter(String key, Collection values) { + Objects.requireNonNull(key, "key not specified"); + Objects.requireNonNull(values, "values not specified"); + filters.withFilter(key, values); + return this; + } + + @Override + public String getImageNameFilter() { + return this.imageNameFilter; + } + + @Override + public String toString() { + return ReflectionToStringBuilder.toString(this, ToStringStyle.SHORT_PREFIX_STYLE); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListNetworksCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListNetworksCmdImpl.java new file mode 100644 index 000000000..b3be94b8e --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListNetworksCmdImpl.java @@ -0,0 +1,43 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.ListNetworksCmd; +import com.github.dockerjava.api.model.Network; +import com.github.dockerjava.core.util.FiltersBuilder; + +import java.util.Collection; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +public class ListNetworksCmdImpl extends AbstrDockerCmd> implements ListNetworksCmd { + + private FiltersBuilder filtersBuilder = new FiltersBuilder(); + + public ListNetworksCmdImpl(ListNetworksCmd.Exec exec) { + super(exec); + } + + @Override + public Map> getFilters() { + return filtersBuilder.build(); + } + + @Override + public ListNetworksCmd withIdFilter(String... networkId) { + this.filtersBuilder.withFilter("id", networkId); + return this; + } + + @Override + public ListNetworksCmd withNameFilter(String... networkName) { + this.filtersBuilder.withFilter("name", networkName); + return this; + } + + @Override + public ListNetworksCmd withFilter(String filterName, Collection filterValues) { + Objects.requireNonNull(filterValues, filterName + " was not specified"); + this.filtersBuilder.withFilter(filterName, filterValues); + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListSecretsCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListSecretsCmdImpl.java new file mode 100644 index 000000000..bed3f9e51 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListSecretsCmdImpl.java @@ -0,0 +1,48 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.ListSecretsCmd; +import com.github.dockerjava.api.model.Secret; +import com.github.dockerjava.core.util.FiltersBuilder; + +import java.util.List; +import java.util.Map; +import java.util.Objects; + +/** + * List services. + */ +public class ListSecretsCmdImpl extends AbstrDockerCmd> implements + ListSecretsCmd { + + private FiltersBuilder filters = new FiltersBuilder(); + + public ListSecretsCmdImpl(Exec exec) { + super(exec); + } + + @Override + public Map> getFilters() { + return filters.build(); + } + + @Override + public ListSecretsCmd withIdFilter(List ids) { + Objects.requireNonNull(ids, "ids was not specified"); + this.filters.withFilter("id", ids); + return this; + } + + @Override + public ListSecretsCmd withNameFilter(List names) { + Objects.requireNonNull(names, "names was not specified"); + this.filters.withFilter("name", names); + return this; + } + + @Override + public ListSecretsCmd withLabelFilter(Map labels) { + Objects.requireNonNull(labels, "labels was not specified"); + this.filters.withLabels(labels); + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListServicesCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListServicesCmdImpl.java new file mode 100644 index 000000000..1245d14b3 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListServicesCmdImpl.java @@ -0,0 +1,48 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.ListServicesCmd; +import com.github.dockerjava.api.model.Service; +import com.github.dockerjava.core.util.FiltersBuilder; + +import java.util.List; +import java.util.Map; +import java.util.Objects; + +/** + * List services. + */ +public class ListServicesCmdImpl extends AbstrDockerCmd> implements + ListServicesCmd { + + private FiltersBuilder filters = new FiltersBuilder(); + + public ListServicesCmdImpl(Exec exec) { + super(exec); + } + + @Override + public Map> getFilters() { + return filters.build(); + } + + @Override + public ListServicesCmd withIdFilter(List ids) { + Objects.requireNonNull(ids, "ids was not specified"); + this.filters.withFilter("id", ids); + return this; + } + + @Override + public ListServicesCmd withNameFilter(List names) { + Objects.requireNonNull(names, "names was not specified"); + this.filters.withFilter("name", names); + return this; + } + + @Override + public ListServicesCmd withLabelFilter(Map labels) { + Objects.requireNonNull(labels, "labels was not specified"); + this.filters.withLabels(labels); + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListSwarmNodesCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListSwarmNodesCmdImpl.java new file mode 100644 index 000000000..a35751627 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListSwarmNodesCmdImpl.java @@ -0,0 +1,56 @@ +package com.github.dockerjava.core.command; + + +import com.github.dockerjava.api.command.ListSwarmNodesCmd; +import com.github.dockerjava.api.model.SwarmNode; +import com.github.dockerjava.core.util.SwarmNodesFiltersBuilder; + +import java.util.List; +import java.util.Map; +import java.util.Objects; + +/** + * List SwarmNodes + */ +public class ListSwarmNodesCmdImpl extends AbstrDockerCmd> implements + ListSwarmNodesCmd { + + private SwarmNodesFiltersBuilder filters = new SwarmNodesFiltersBuilder(); + + public ListSwarmNodesCmdImpl(Exec exec) { + super(exec); + } + + @Override + public Map> getFilters() { + return filters.build(); + } + + @Override + public ListSwarmNodesCmd withIdFilter(List ids) { + Objects.requireNonNull(ids, "ids was not specified"); + this.filters.withIds(ids); + return this; + } + + @Override + public ListSwarmNodesCmd withNameFilter(List names) { + Objects.requireNonNull(names, "names was not specified"); + this.filters.withNames(names); + return this; + } + + @Override + public ListSwarmNodesCmd withMembershipFilter(List memberships) { + Objects.requireNonNull(memberships, "memberships was not specified"); + this.filters.withMemberships(memberships); + return this; + } + + @Override + public ListSwarmNodesCmd withRoleFilter(List roles) { + Objects.requireNonNull(roles, "roles was not specified"); + this.filters.withRoles(roles); + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListTasksCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListTasksCmdImpl.java new file mode 100644 index 000000000..789922ad5 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListTasksCmdImpl.java @@ -0,0 +1,70 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.ListTasksCmd; +import com.github.dockerjava.api.model.Task; +import com.github.dockerjava.api.model.TaskState; +import com.github.dockerjava.core.util.FiltersBuilder; + +import javax.annotation.CheckForNull; +import java.util.List; +import java.util.Map; + +public class ListTasksCmdImpl extends AbstrDockerCmd> implements ListTasksCmd { + private FiltersBuilder filters = new FiltersBuilder(); + + public ListTasksCmdImpl(Exec exec) { + super(exec); + } + + @CheckForNull + @Override + public Map> getFilters() { + return filters.build(); + } + + @Override + public ListTasksCmd withLabelFilter(Map labels) { + filters.withLabels(labels); + return this; + } + + @Override + public ListTasksCmd withLabelFilter(String... labels) { + filters.withLabels(labels); + return this; + } + + @Override + public ListTasksCmd withIdFilter(String... ids) { + filters.withFilter("id", ids); + return this; + } + + @Override + public ListTasksCmd withNameFilter(String... names) { + filters.withFilter("name", names); + return this; + } + + @Override + public ListTasksCmd withNodeFilter(String... nodeNames) { + filters.withFilter("node", nodeNames); + return this; + } + + @Override + public ListTasksCmd withServiceFilter(String... serviceNames) { + filters.withFilter("service", serviceNames); + return this; + } + + @Override + public ListTasksCmd withStateFilter(TaskState... desiredStates) { + String[] states = new String[desiredStates.length]; + for (int i = 0; i < desiredStates.length; i++) { + states[i] = desiredStates[i].getValue(); + } + filters.withFilter("desired-state", states); + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListVolumesCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListVolumesCmdImpl.java new file mode 100644 index 000000000..78d39c2c7 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ListVolumesCmdImpl.java @@ -0,0 +1,43 @@ +package com.github.dockerjava.core.command; + +import java.util.Collection; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +import com.github.dockerjava.api.command.ListVolumesCmd; +import com.github.dockerjava.api.command.ListVolumesResponse; +import com.github.dockerjava.core.util.FiltersBuilder; + +/** + * + * @author Marcus Linke + * + */ +public class ListVolumesCmdImpl extends AbstrDockerCmd implements ListVolumesCmd { + + private FiltersBuilder filters = new FiltersBuilder(); + + public ListVolumesCmdImpl(ListVolumesCmd.Exec exec) { + super(exec); + } + + @Override + public Map> getFilters() { + return filters.build(); + } + + @Override + public ListVolumesCmd withDanglingFilter(Boolean dangling) { + Objects.requireNonNull(dangling, "dangling have not been specified"); + this.filters.withFilter("dangling", dangling.toString()); + return this; + } + + @Override + public ListVolumesCmd withFilter(String filterName, Collection filterValues) { + Objects.requireNonNull(filterValues, filterName + " was not specified"); + this.filters.withFilter(filterName, filterValues); + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/LoadImageAsyncCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/LoadImageAsyncCmdImpl.java new file mode 100644 index 000000000..3de1dfa4d --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/LoadImageAsyncCmdImpl.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.LoadImageAsyncCmd; +import com.github.dockerjava.api.model.LoadResponseItem; + +import java.io.IOException; +import java.io.InputStream; +import java.util.Objects; + +public class LoadImageAsyncCmdImpl extends AbstrAsyncDockerCmd implements LoadImageAsyncCmd { + + private InputStream inputStream; + + public LoadImageAsyncCmdImpl(LoadImageAsyncCmd.Exec exec, InputStream inputStream) { + super(exec); + this.inputStream = inputStream; + } + + @Override + public InputStream getImageStream() { + return this.inputStream; + } + + @Override + public LoadImageAsyncCmd withImageStream(InputStream imageStream) { + Objects.requireNonNull(imageStream, "imageStream was not specified"); + this.inputStream = imageStream; + return this; + } + + @Override + public void close() { + super.close(); + + try { + this.inputStream.close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/LoadImageCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/LoadImageCmdImpl.java similarity index 82% rename from src/main/java/com/github/dockerjava/core/command/LoadImageCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/LoadImageCmdImpl.java index 496ea3176..0b8cbea94 100644 --- a/src/main/java/com/github/dockerjava/core/command/LoadImageCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/LoadImageCmdImpl.java @@ -1,8 +1,7 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; - import java.io.InputStream; +import java.util.Objects; import com.github.dockerjava.api.command.LoadImageCmd; @@ -32,8 +31,7 @@ public InputStream getImageStream() { */ @Override public LoadImageCmdImpl withImageStream(@Nonnull InputStream imageStream) { - checkNotNull(imageStream, "imageStream was not specified"); - this.imageStream = imageStream; + this.imageStream = Objects.requireNonNull(imageStream, "imageStream was not specified"); return this; } } diff --git a/src/main/java/com/github/dockerjava/core/command/LogContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/LogContainerCmdImpl.java similarity index 83% rename from src/main/java/com/github/dockerjava/core/command/LogContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/LogContainerCmdImpl.java index aa267fc8f..65321a318 100644 --- a/src/main/java/com/github/dockerjava/core/command/LogContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/LogContainerCmdImpl.java @@ -1,9 +1,9 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; -import org.apache.commons.lang.builder.ReflectionToStringBuilder; -import org.apache.commons.lang.builder.ToStringStyle; +import org.apache.commons.lang3.builder.ReflectionToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; import com.github.dockerjava.api.command.LogContainerCmd; import com.github.dockerjava.api.model.Frame; @@ -24,6 +24,8 @@ * @param since * - UNIX timestamp (integer) to filter logs. Specifying a timestamp will only output log-entries since that timestamp. Default: * 0 (unfiltered) + * @param until + * - Only return logs before this time, as a UNIX timestamp. Default: 0 */ public class LogContainerCmdImpl extends AbstrAsyncDockerCmd implements LogContainerCmd { @@ -31,7 +33,7 @@ public class LogContainerCmdImpl extends AbstrAsyncDockerCmd { private static final Logger LOGGER = LoggerFactory.getLogger(LogContainerResultCallback.class); diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/LogSwarmObjectImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/LogSwarmObjectImpl.java new file mode 100644 index 000000000..d775582fc --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/LogSwarmObjectImpl.java @@ -0,0 +1,121 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.DockerCmdAsyncExec; +import com.github.dockerjava.api.command.LogSwarmObjectCmd; +import com.github.dockerjava.api.model.Frame; + +import javax.annotation.Nonnull; + +public class LogSwarmObjectImpl extends AbstrAsyncDockerCmd implements LogSwarmObjectCmd { + private String id; + private Boolean followStream; + private Boolean timestamps; + private Boolean stdout; + private Boolean stderr; + private Boolean tailAll; + private Boolean follow; + private Integer tail; + private Integer since; + private Boolean details; + + public LogSwarmObjectImpl(DockerCmdAsyncExec execution, String id) { + super(execution); + this.id = id; + } + + @Override + public LogSwarmObjectImpl withId(@Nonnull String id) { + this.id = id; + return this; + } + + public String getId() { + return id; + } + + public Boolean getFollowStream() { + return followStream; + } + + public LogSwarmObjectImpl withFollowStream(Boolean followStream) { + this.followStream = followStream; + return this; + } + + public Boolean getTimestamps() { + return timestamps; + } + + @Override + public LogSwarmObjectImpl withTimestamps(Boolean timestamps) { + this.timestamps = timestamps; + return this; + } + + public Boolean getStdout() { + return stdout; + } + + public LogSwarmObjectImpl withStdout(Boolean stdout) { + this.stdout = stdout; + return this; + } + + public Boolean getStderr() { + return stderr; + } + + public LogSwarmObjectImpl withStderr(Boolean stderr) { + this.stderr = stderr; + return this; + } + + public Boolean getTailAll() { + return tailAll; + } + + public LogSwarmObjectImpl withTailAll(Boolean tailAll) { + this.tailAll = tailAll; + return this; + } + + public Integer getTail() { + return tail; + } + + @Override + public LogSwarmObjectImpl withTail(Integer tail) { + this.tail = tail; + return this; + } + + public Integer getSince() { + return since; + } + + @Override + public LogSwarmObjectImpl withSince(Integer since) { + this.since = since; + return this; + } + + public Boolean getFollow() { + return follow; + } + + @Override + public LogSwarmObjectImpl withFollow(Boolean follow) { + this.follow = follow; + return this; + } + + @Override + public LogSwarmObjectCmd withDetails(Boolean details) { + this.details = details; + return this; + } + + public Boolean getDetails() { + return details; + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/PauseContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PauseContainerCmdImpl.java similarity index 84% rename from src/main/java/com/github/dockerjava/core/command/PauseContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/PauseContainerCmdImpl.java index a8caf3091..f646e8ced 100644 --- a/src/main/java/com/github/dockerjava/core/command/PauseContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PauseContainerCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.PauseContainerCmd; import com.github.dockerjava.api.exception.NotFoundException; @@ -28,8 +28,7 @@ public String getContainerId() { @Override public PauseContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/PingCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PingCmdImpl.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/command/PingCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/PingCmdImpl.java diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/PruneCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PruneCmdImpl.java new file mode 100644 index 000000000..e08f64b02 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PruneCmdImpl.java @@ -0,0 +1,101 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.PruneCmd; +import com.github.dockerjava.api.model.PruneResponse; +import com.github.dockerjava.api.model.PruneType; +import com.github.dockerjava.core.util.FiltersBuilder; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +/** + * Delete unused content (containers, images, volumes, networks, build relicts) + */ +public class PruneCmdImpl extends AbstrDockerCmd implements PruneCmd { + + private static final String BUILD_API_PATH = "/build/prune"; + private static final String CONTAINERS_API_PATH = "/containers/prune"; + private static final String IMAGES_API_PATH = "/images/prune"; + private static final String VOLUMES_API_PATH = "/volumes/prune"; + private static final String NETWORKS_API_PATH = "/networks/prune"; + + private FiltersBuilder filters = new FiltersBuilder(); + private PruneType pruneType; + + public PruneCmdImpl(Exec exec, PruneType pruneType) { + super(exec); + this.pruneType = pruneType; + } + + @Nonnull + @Override + public PruneType getPruneType() { + return pruneType; + } + + @Nonnull + @Override + public String getApiPath() { + String apiPath; + switch (getPruneType()) { + case BUILD: + apiPath = BUILD_API_PATH; + break; + case IMAGES: + apiPath = IMAGES_API_PATH; + break; + case NETWORKS: + apiPath = NETWORKS_API_PATH; + break; + case VOLUMES: + apiPath = VOLUMES_API_PATH; + break; + default: + apiPath = CONTAINERS_API_PATH; + break; + } + return apiPath; + } + + @CheckForNull + @Override + public Map> getFilters() { + return filters.build(); + } + + @Override + public PruneCmd withPruneType(final PruneType pruneType) { + Objects.requireNonNull(pruneType, "pruneType has not been specified"); + this.pruneType = pruneType; + return this; + } + + @Override + public PruneCmd withDangling(Boolean dangling) { + Objects.requireNonNull(dangling, "dangling has not been specified"); + filters.withFilter("dangling", dangling ? "1" : "0"); + return this; + } + + @Override + public PruneCmd withUntilFilter(final String until) { + Objects.requireNonNull(until, "until has not been specified"); + filters.withUntil(until); + return this; + } + + @Override + public PruneCmd withLabelFilter(final String... labels) { + Objects.requireNonNull(labels, "labels have not been specified"); + filters.withLabels(labels); + return this; + } + + @Override + public PruneResponse exec() { + return super.exec(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/PullImageCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PullImageCmdImpl.java new file mode 100644 index 000000000..7f70ac3b7 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PullImageCmdImpl.java @@ -0,0 +1,79 @@ +package com.github.dockerjava.core.command; + +import java.util.Objects; + +import com.github.dockerjava.api.command.PullImageCmd; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.PullResponseItem; + +/** + * + * Pull image from repository. + * + */ +public class PullImageCmdImpl extends AbstrAsyncDockerCmd implements PullImageCmd { + + private String repository, tag, platform, registry; + + private AuthConfig authConfig; + + public PullImageCmdImpl(PullImageCmd.Exec exec, AuthConfig authConfig, String repository) { + super(exec); + withAuthConfig(authConfig); + withRepository(repository); + } + + public AuthConfig getAuthConfig() { + return authConfig; + } + + public PullImageCmd withAuthConfig(AuthConfig authConfig) { + this.authConfig = authConfig; + return this; + } + + @Override + public String getRepository() { + return repository; + } + + @Override + public String getTag() { + return tag; + } + + @Override + public String getPlatform() { + return platform; + } + + @Override + public String getRegistry() { + return registry; + } + + @Override + public PullImageCmd withRepository(String repository) { + this.repository = Objects.requireNonNull(repository, "repository was not specified"); + return this; + } + + @Override + public PullImageCmd withTag(String tag) { + this.tag = Objects.requireNonNull(tag, "tag was not specified"); + return this; + } + + @Override + public PullImageCmd withPlatform(String platform) { + this.platform = platform; + return this; + } + + @Override + public PullImageCmd withRegistry(String registry) { + this.registry = Objects.requireNonNull(registry, "registry was not specified"); + return this; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/PullImageResultCallback.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PullImageResultCallback.java new file mode 100644 index 000000000..67b6b5e48 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PullImageResultCallback.java @@ -0,0 +1,131 @@ +/* + * Created on 21.07.2015 + */ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.model.PullResponseItem; +import com.github.dockerjava.core.async.ResultCallbackTemplate; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.annotation.CheckForNull; +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.TimeUnit; + +/** + * + * @author Marcus Linke + * + * @deprecated use {@link com.github.dockerjava.api.command.PullImageResultCallback} + */ +@Deprecated +public class PullImageResultCallback extends ResultCallbackTemplate { + + private static final Logger LOGGER = LoggerFactory.getLogger(PullImageResultCallback.class); + + private boolean isSwarm = false; + private Map results = null; + + @CheckForNull + private PullResponseItem latestItem = null; + + @Override + public void onNext(PullResponseItem item) { + // only do it once + if (results == null && latestItem == null) { + checkForDockerSwarmResponse(item); + } + + if (isSwarm) { + handleDockerSwarmResponse(item); + } else { + handleDockerClientResponse(item); + } + + LOGGER.debug(item.toString()); + } + + private void checkForDockerSwarmResponse(PullResponseItem item) { + if (item.getStatus().matches("Pulling\\s.+\\.{3}$")) { + isSwarm = true; + LOGGER.debug("Communicating with Docker Swarm."); + } + } + + private void handleDockerSwarmResponse(final PullResponseItem item) { + if (results == null) { + results = new HashMap<>(); + } + + // Swarm terminates a pull sometimes with an empty line. + // Therefore keep first success message + PullResponseItem currentItem = results.get(item.getId()); + if (currentItem == null || !currentItem.isPullSuccessIndicated()) { + results.put(item.getId(), item); + } + } + + private void handleDockerClientResponse(PullResponseItem item) { + latestItem = item; + } + + private void checkDockerSwarmPullSuccessful() { + if (results.isEmpty()) { + throw new DockerClientException("Could not pull image through Docker Swarm"); + } else { + boolean pullFailed = false; + StringBuilder sb = new StringBuilder(); + + for (PullResponseItem pullResponseItem : results.values()) { + if (!pullResponseItem.isPullSuccessIndicated()) { + pullFailed = true; + sb.append("[" + pullResponseItem.getId() + ":" + messageFromPullResult(pullResponseItem) + "]"); + } + } + + if (pullFailed) { + throw new DockerClientException("Could not pull image: " + sb.toString()); + } + } + } + + private void checkDockerClientPullSuccessful() { + if (latestItem == null) { + throw new DockerClientException("Could not pull image"); + } else if (!latestItem.isPullSuccessIndicated()) { + throw new DockerClientException("Could not pull image: " + messageFromPullResult(latestItem)); + } + } + + private String messageFromPullResult(PullResponseItem pullResponseItem) { + return (pullResponseItem.getError() != null) ? pullResponseItem.getError() : pullResponseItem.getStatus(); + } + + @Override + protected void throwFirstError() { + super.throwFirstError(); + + if (isSwarm) { + checkDockerSwarmPullSuccessful(); + } else { + checkDockerClientPullSuccessful(); + } + } + + /** + * Awaits the image to be pulled successful. + * + * @deprecated use {@link #awaitCompletion()} or {@link #awaitCompletion(long, TimeUnit)} instead + * @throws DockerClientException + * if the pull fails. + */ + public void awaitSuccess() { + try { + awaitCompletion(); + } catch (InterruptedException e) { + throw new DockerClientException("", e); + } + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/PushImageCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PushImageCmdImpl.java new file mode 100644 index 000000000..8e1fa5cec --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PushImageCmdImpl.java @@ -0,0 +1,67 @@ +package com.github.dockerjava.core.command; + +import java.util.Objects; + +import com.github.dockerjava.api.command.PushImageCmd; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.PushResponseItem; + +/** + * Push the latest image to the repository. + * + * @param name + * The name, e.g. "alexec/busybox" or just "busybox" if you want to default. Not null. + */ +public class PushImageCmdImpl extends AbstrAsyncDockerCmd implements PushImageCmd { + + private String name; + + private String tag; + + private AuthConfig authConfig; + + public PushImageCmdImpl(PushImageCmd.Exec exec, AuthConfig authConfig, String name) { + super(exec); + withName(name); + withAuthConfig(authConfig); + } + + @Override + public String getName() { + return name; + } + + @Override + public String getTag() { + return tag; + } + + /** + * @param name + * The name, e.g. "alexec/busybox" or just "busybox" if you want to default. Not null. + */ + @Override + public PushImageCmd withName(String name) { + this.name = Objects.requireNonNull(name, "name was not specified"); + return this; + } + + /** + * @param tag + * The image's tag. Can be null or empty. + */ + @Override + public PushImageCmd withTag(String tag) { + this.tag = Objects.requireNonNull(tag, "tag was not specified"); + return this; + } + + public AuthConfig getAuthConfig() { + return authConfig; + } + + public PushImageCmd withAuthConfig(AuthConfig authConfig) { + this.authConfig = authConfig; + return this; + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/PushImageResultCallback.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PushImageResultCallback.java similarity index 81% rename from src/main/java/com/github/dockerjava/core/command/PushImageResultCallback.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/PushImageResultCallback.java index 55dfb5650..06f1fbc8d 100644 --- a/src/main/java/com/github/dockerjava/core/command/PushImageResultCallback.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/PushImageResultCallback.java @@ -3,20 +3,22 @@ */ package com.github.dockerjava.core.command; -import javax.annotation.CheckForNull; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - import com.github.dockerjava.api.exception.DockerClientException; import com.github.dockerjava.api.model.PushResponseItem; import com.github.dockerjava.core.async.ResultCallbackTemplate; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.annotation.CheckForNull; +import java.util.concurrent.TimeUnit; /** * * @author Marcus Linke * + * @deprecated use {@link com.github.dockerjava.api.async.ResultCallback.Adapter} */ +@Deprecated public class PushImageResultCallback extends ResultCallbackTemplate { private static final Logger LOGGER = LoggerFactory.getLogger(PushImageResultCallback.class); @@ -30,9 +32,21 @@ public void onNext(PushResponseItem item) { LOGGER.debug(item.toString()); } + @Override + protected void throwFirstError() { + super.throwFirstError(); + + if (latestItem == null) { + throw new DockerClientException("Could not push image"); + } else if (latestItem.isErrorIndicated()) { + throw new DockerClientException("Could not push image: " + latestItem.getError()); + } + } + /** * Awaits the image to be pulled successful. * + * @deprecated use {@link #awaitCompletion()} or {@link #awaitCompletion(long, TimeUnit)} instead * @throws DockerClientException * if the push fails. */ @@ -42,11 +56,5 @@ public void awaitSuccess() { } catch (InterruptedException e) { throw new DockerClientException("", e); } - - if (latestItem == null) { - throw new DockerClientException("Could not push image"); - } else if (latestItem.isErrorIndicated()) { - throw new DockerClientException("Could not push image: " + latestItem.getError()); - } } } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveConfigCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveConfigCmdImpl.java new file mode 100644 index 000000000..e2e7d06fd --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveConfigCmdImpl.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.core.command; + +import java.util.Objects; + +import com.github.dockerjava.api.command.RemoveConfigCmd; +import com.github.dockerjava.api.exception.NotFoundException; + +/** + * Remove a config. + */ +public class RemoveConfigCmdImpl extends AbstrDockerCmd implements RemoveConfigCmd { + + private String configId; + + public RemoveConfigCmdImpl(Exec exec, String configId) { + super(exec); + withConfigId(configId); + } + + @Override + public String getConfigId() { + return configId; + } + + @Override + public RemoveConfigCmd withConfigId(String configId) { + this.configId = Objects.requireNonNull(configId, "configId was not specified"); + return this; + } + + /** + * @throws NotFoundException + * No such secret + */ + @Override + public Void exec() throws NotFoundException { + return super.exec(); + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/RemoveContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveContainerCmdImpl.java similarity index 90% rename from src/main/java/com/github/dockerjava/core/command/RemoveContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveContainerCmdImpl.java index 0fc2ab626..cd8f4a9f3 100644 --- a/src/main/java/com/github/dockerjava/core/command/RemoveContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveContainerCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.RemoveContainerCmd; import com.github.dockerjava.api.exception.NotFoundException; @@ -41,8 +41,7 @@ public Boolean hasForceEnabled() { @Override public RemoveContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/RemoveImageCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveImageCmdImpl.java similarity index 89% rename from src/main/java/com/github/dockerjava/core/command/RemoveImageCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveImageCmdImpl.java index 038a27f2d..a77357b59 100644 --- a/src/main/java/com/github/dockerjava/core/command/RemoveImageCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveImageCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.RemoveImageCmd; import com.github.dockerjava.api.exception.NotFoundException; @@ -38,8 +38,7 @@ public Boolean hasNoPruneEnabled() { @Override public RemoveImageCmd withImageId(String imageId) { - checkNotNull(imageId, "imageId was not specified"); - this.imageId = imageId; + this.imageId = Objects.requireNonNull(imageId, "imageId was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/RemoveNetworkCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveNetworkCmdImpl.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/command/RemoveNetworkCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveNetworkCmdImpl.java diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveSecretCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveSecretCmdImpl.java new file mode 100644 index 000000000..5c8d0e075 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveSecretCmdImpl.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.RemoveSecretCmd; +import com.github.dockerjava.api.exception.NotFoundException; + +import java.util.Objects; + +/** + * Remove a secret. + */ +public class RemoveSecretCmdImpl extends AbstrDockerCmd implements RemoveSecretCmd { + + private String secretId; + + public RemoveSecretCmdImpl(Exec exec, String secretId) { + super(exec); + withSecretId(secretId); + } + + @Override + public String getSecretId() { + return secretId; + } + + @Override + public RemoveSecretCmd withSecretId(String secretId) { + this.secretId = Objects.requireNonNull(secretId, "secretId was not specified"); + return this; + } + + /** + * @throws NotFoundException + * No such secret + */ + @Override + public Void exec() throws NotFoundException { + return super.exec(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveServiceCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveServiceCmdImpl.java new file mode 100644 index 000000000..6fed721c7 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveServiceCmdImpl.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.RemoveServiceCmd; +import com.github.dockerjava.api.exception.NotFoundException; + +import java.util.Objects; + +/** + * Remove a service. + */ +public class RemoveServiceCmdImpl extends AbstrDockerCmd implements RemoveServiceCmd { + + private String serviceId; + + public RemoveServiceCmdImpl(Exec exec, String serviceId) { + super(exec); + withServiceId(serviceId); + } + + @Override + public String getServiceId() { + return serviceId; + } + + @Override + public RemoveServiceCmd withServiceId(String serviceId) { + this.serviceId = Objects.requireNonNull(serviceId, "serviceId was not specified"); + return this; + } + + /** + * @throws NotFoundException + * No such service + */ + @Override + public Void exec() throws NotFoundException { + return super.exec(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveSwarmNodeCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveSwarmNodeCmdImpl.java new file mode 100644 index 000000000..ef8a86943 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveSwarmNodeCmdImpl.java @@ -0,0 +1,56 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.RemoveSwarmNodeCmd; +import com.github.dockerjava.api.exception.NotFoundException; + +import java.util.Objects; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * Remove a container. + */ +public class RemoveSwarmNodeCmdImpl extends AbstrDockerCmd implements RemoveSwarmNodeCmd { + + private String swarmNodeId; + + private Boolean force; + + public RemoveSwarmNodeCmdImpl(RemoveSwarmNodeCmd.Exec exec, String swarmNodeId) { + super(exec); + withSwarmNodeId(swarmNodeId); + } + + @Override + @CheckForNull + public String getSwarmNodeId() { + return swarmNodeId; + } + + @Override + @CheckForNull + public Boolean hasForceEnabled() { + return force; + } + + @Override + public RemoveSwarmNodeCmd withSwarmNodeId(@Nonnull String swarmNodeId) { + this.swarmNodeId = Objects.requireNonNull(swarmNodeId, "swarmNodeId was not specified"); + return this; + } + + @Override + public RemoveSwarmNodeCmd withForce(Boolean force) { + this.force = force; + return this; + } + + /** + * @throws NotFoundException No such swarmNode + */ + @Override + public Void exec() throws NotFoundException { + return super.exec(); + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/RemoveVolumeCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveVolumeCmdImpl.java similarity index 83% rename from src/main/java/com/github/dockerjava/core/command/RemoveVolumeCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveVolumeCmdImpl.java index ee294a130..c62316842 100644 --- a/src/main/java/com/github/dockerjava/core/command/RemoveVolumeCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RemoveVolumeCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.RemoveVolumeCmd; import com.github.dockerjava.api.exception.NotFoundException; @@ -26,8 +26,7 @@ public String getName() { @Override public RemoveVolumeCmd withName(String name) { - checkNotNull(name, "name was not specified"); - this.name = name; + this.name = Objects.requireNonNull(name, "name was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/RenameContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RenameContainerCmdImpl.java similarity index 81% rename from src/main/java/com/github/dockerjava/core/command/RenameContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/RenameContainerCmdImpl.java index ebf3ebabf..b146382f3 100644 --- a/src/main/java/com/github/dockerjava/core/command/RenameContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RenameContainerCmdImpl.java @@ -5,7 +5,7 @@ import javax.annotation.Nonnull; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; public class RenameContainerCmdImpl extends AbstrDockerCmd implements RenameContainerCmd { @@ -30,15 +30,13 @@ public String getName() { @Override public RenameContainerCmd withContainerId(@Nonnull String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } @Override public RenameContainerCmd withName(@Nonnull String name) { - checkNotNull(name, "name was not specified"); - this.name = name; + this.name = Objects.requireNonNull(name, "name was not specified"); return this; } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ResizeContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ResizeContainerCmdImpl.java new file mode 100644 index 000000000..188802a41 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ResizeContainerCmdImpl.java @@ -0,0 +1,56 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.ResizeContainerCmd; +import com.github.dockerjava.api.exception.NotFoundException; + +import java.util.Objects; + +public class ResizeContainerCmdImpl extends AbstrDockerCmd implements ResizeContainerCmd { + + private String containerId; + + private Integer height; + + private Integer width; + + public ResizeContainerCmdImpl(ResizeContainerCmd.Exec exec, String execId) { + super(exec); + withContainerId(execId); + } + + @Override + public String getContainerId() { + return containerId; + } + + @Override + public Integer getHeight() { + return height; + } + + @Override + public Integer getWidth() { + return width; + } + + @Override + public ResizeContainerCmd withContainerId(String containerId) { + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); + return this; + } + + @Override + public ResizeContainerCmd withSize(int height, int width) { + this.height = height; + this.width = width; + return this; + } + + /** + * @throws NotFoundException no such exec instance + */ + @Override + public Void exec() throws NotFoundException { + return super.exec(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/ResizeExecCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ResizeExecCmdImpl.java new file mode 100644 index 000000000..3aa02c7e9 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/ResizeExecCmdImpl.java @@ -0,0 +1,57 @@ +package com.github.dockerjava.core.command; + +import java.util.Objects; + +import com.github.dockerjava.api.command.ResizeExecCmd; +import com.github.dockerjava.api.exception.NotFoundException; + +public class ResizeExecCmdImpl extends AbstrDockerCmd implements ResizeExecCmd { + + private String execId; + + private Integer height; + + private Integer width; + + public ResizeExecCmdImpl(ResizeExecCmd.Exec exec, String execId) { + super(exec); + withExecId(execId); + } + + @Override + public String getExecId() { + return execId; + } + + @Override + public Integer getHeight() { + return height; + } + + @Override + public Integer getWidth() { + return width; + } + + @Override + public ResizeExecCmd withExecId(String execId) { + this.execId = Objects.requireNonNull(execId, "execId was not specified"); + return this; + } + + @Override + public ResizeExecCmd withSize(int height, int width) { + this.height = height; + this.width = width; + return this; + } + + /** + * @throws NotFoundException no such exec instance + */ + @Override + public Void exec() throws NotFoundException { + return super.exec(); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/RestartContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RestartContainerCmdImpl.java new file mode 100644 index 000000000..3b1df465b --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/RestartContainerCmdImpl.java @@ -0,0 +1,79 @@ +package com.github.dockerjava.core.command; + +import static com.google.common.base.Preconditions.checkArgument; + +import java.util.Objects; + +import com.github.dockerjava.api.command.RestartContainerCmd; +import com.github.dockerjava.api.exception.NotFoundException; + +import javax.annotation.CheckForNull; + +/** + * Restart a running container. + * + * @param signal - Signal to send to the container as an integer or string (e.g. SIGINT). + * @param timeout - Timeout in seconds before killing the container. Defaults to 10 seconds. + */ +public class RestartContainerCmdImpl extends AbstrDockerCmd implements RestartContainerCmd { + + private String containerId; + + private Integer timeout = 10; + + private String signal; + + public RestartContainerCmdImpl(RestartContainerCmd.Exec exec, String containerId) { + super(exec); + withContainerId(containerId); + } + + @Override + public String getContainerId() { + return containerId; + } + + @Override + public Integer getTimeout() { + return timeout; + } + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_42} + */ + @CheckForNull + @Override + public String getSignal() { + return signal; + } + + @Override + public RestartContainerCmd withContainerId(String containerId) { + Objects.requireNonNull(containerId, "containerId was not specified"); + this.containerId = containerId; + return this; + } + + @Override + public RestartContainerCmd withTimeout(Integer timeout) { + Objects.requireNonNull(timeout, "timeout was not specified"); + checkArgument(timeout >= 0, "timeout must be greater or equal 0"); + this.timeout = timeout; + return this; + } + + @Override + public RestartContainerCmd withSignal(String signal) { + Objects.requireNonNull(signal, "signal was not specified"); + this.signal = signal; + return this; + } + + /** + * @throws NotFoundException No such container + */ + @Override + public Void exec() throws NotFoundException { + return super.exec(); + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/SaveImageCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/SaveImageCmdImpl.java similarity index 84% rename from src/main/java/com/github/dockerjava/core/command/SaveImageCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/SaveImageCmdImpl.java index 333e32df7..0ec72bcc5 100644 --- a/src/main/java/com/github/dockerjava/core/command/SaveImageCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/SaveImageCmdImpl.java @@ -1,8 +1,7 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; - import java.io.InputStream; +import java.util.Objects; import com.github.dockerjava.api.command.SaveImageCmd; import com.github.dockerjava.api.exception.NotFoundException; @@ -33,8 +32,7 @@ public String getTag() { */ @Override public SaveImageCmd withName(String name) { - checkNotNull(name, "name was not specified"); - this.name = name; + this.name = Objects.requireNonNull(name, "name was not specified"); return this; } @@ -44,8 +42,7 @@ public SaveImageCmd withName(String name) { */ @Override public SaveImageCmd withTag(String tag) { - checkNotNull(tag, "tag was not specified"); - this.tag = tag; + this.tag = Objects.requireNonNull(tag, "tag was not specified"); return this; } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/SaveImagesCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/SaveImagesCmdImpl.java new file mode 100644 index 000000000..43e11f609 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/SaveImagesCmdImpl.java @@ -0,0 +1,60 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.SaveImagesCmd; +import com.github.dockerjava.api.exception.NotFoundException; +import com.google.common.collect.ImmutableList; + +import javax.annotation.Nonnull; +import java.io.InputStream; +import java.util.List; +import java.util.Objects; + +public class SaveImagesCmdImpl extends AbstrDockerCmd implements SaveImagesCmd { + + private static class TaggedImageImpl implements TaggedImage { + private final String name; + private final String tag; + + private TaggedImageImpl(String name, String tag) { + this.name = Objects.requireNonNull(name, "image name was not specified"); + this.tag = Objects.requireNonNull(tag, "image tag was not specified"); + } + + @Override + public String asString() { + return name + ":" + tag; + } + + @Override + public String toString() { + return asString(); + } + } + + private final ImmutableList.Builder taggedImages = ImmutableList.builder(); + + public SaveImagesCmdImpl(final SaveImagesCmd.Exec exec) { + super(exec); + } + + @Override + public SaveImagesCmd withImage(@Nonnull final String name, @Nonnull final String tag) { + taggedImages.add(new TaggedImageImpl(name, tag)); + return this; + } + + + + @Override + public List getImages() { + return taggedImages.build(); + } + + /** + * @throws NotFoundException No such images + */ + @Override + public InputStream exec() throws NotFoundException { + return super.exec(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/SearchImagesCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/SearchImagesCmdImpl.java new file mode 100644 index 000000000..41b8cc844 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/SearchImagesCmdImpl.java @@ -0,0 +1,57 @@ +package com.github.dockerjava.core.command; + +import static com.google.common.base.Preconditions.checkArgument; + +import java.util.List; +import java.util.Objects; + +import com.github.dockerjava.api.command.SearchImagesCmd; +import com.github.dockerjava.api.model.SearchItem; + +import javax.annotation.Nonnull; + +/** + * Search images + * + * @param term + * - search term + * + */ +public class SearchImagesCmdImpl extends AbstrDockerCmd> implements SearchImagesCmd { + + private static final int MIN_LIMIT = 1; + private static final int MAX_LIMIT = 100; + + private String term; + private Integer limit; + + public SearchImagesCmdImpl(SearchImagesCmd.Exec exec, String term) { + super(exec); + withTerm(term); + } + + @Override + public String getTerm() { + return term; + } + + @Override + public SearchImagesCmd withTerm(String term) { + this.term = Objects.requireNonNull(term, "term was not specified"); + return this; + } + + @Override + public Integer getLimit() { + return limit; + } + + @Override + public SearchImagesCmd withLimit(@Nonnull Integer limit) { + String errorMessage = String.format("Limit %s is outside the range of [%s, %s]", limit, MIN_LIMIT, MAX_LIMIT); + checkArgument(limit <= MAX_LIMIT, errorMessage); + checkArgument(limit >= MIN_LIMIT, errorMessage); + this.limit = limit; + return this; + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/StartContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/StartContainerCmdImpl.java similarity index 77% rename from src/main/java/com/github/dockerjava/core/command/StartContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/StartContainerCmdImpl.java index 6f127da23..2c0e2c2b8 100644 --- a/src/main/java/com/github/dockerjava/core/command/StartContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/StartContainerCmdImpl.java @@ -1,10 +1,8 @@ package com.github.dockerjava.core.command; -import static com.fasterxml.jackson.annotation.JsonInclude.Include.NON_EMPTY; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonInclude; import com.github.dockerjava.api.command.StartContainerCmd; import com.github.dockerjava.api.exception.NotFoundException; import com.github.dockerjava.api.exception.NotModifiedException; @@ -12,7 +10,6 @@ /** * Start a container */ -@JsonInclude(NON_EMPTY) public class StartContainerCmdImpl extends AbstrDockerCmd implements StartContainerCmd { @JsonIgnore @@ -25,8 +22,7 @@ public StartContainerCmdImpl(StartContainerCmd.Exec exec, String containerId) { @Override public StartContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/StatsCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/StatsCmdImpl.java new file mode 100644 index 000000000..3e24bd5af --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/StatsCmdImpl.java @@ -0,0 +1,43 @@ +package com.github.dockerjava.core.command; + +import java.util.Objects; + +import com.github.dockerjava.api.command.StatsCmd; +import com.github.dockerjava.api.model.Statistics; + +/** + * Container stats + */ +public class StatsCmdImpl extends AbstrAsyncDockerCmd implements StatsCmd { + + private String containerId; + + private Boolean noStream; + + public StatsCmdImpl(StatsCmd.Exec exec, String containerId) { + super(exec); + withContainerId(containerId); + } + + @Override + public StatsCmd withContainerId(String containerId) { + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); + return this; + } + + @Override + public String getContainerId() { + return containerId; + } + + @Override + public Boolean hasNoStream() { + return noStream; + } + + @Override + public StatsCmd withNoStream(boolean noStream) { + this.noStream = noStream; + return this; + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/StopContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/StopContainerCmdImpl.java similarity index 87% rename from src/main/java/com/github/dockerjava/core/command/StopContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/StopContainerCmdImpl.java index 7b1e165f5..2cf5e37ae 100644 --- a/src/main/java/com/github/dockerjava/core/command/StopContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/StopContainerCmdImpl.java @@ -1,7 +1,8 @@ package com.github.dockerjava.core.command; import static com.google.common.base.Preconditions.checkArgument; -import static com.google.common.base.Preconditions.checkNotNull; + +import java.util.Objects; import com.github.dockerjava.api.command.StopContainerCmd; import com.github.dockerjava.api.exception.NotFoundException; @@ -39,14 +40,13 @@ public Integer getTimeout() { @Override public StopContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } @Override public StopContainerCmd withTimeout(Integer timeout) { - checkNotNull(timeout, "timeout was not specified"); + Objects.requireNonNull(timeout, "timeout was not specified"); checkArgument(timeout >= 0, "timeout must be greater or equal 0"); this.timeout = timeout; return this; diff --git a/src/main/java/com/github/dockerjava/core/command/TagImageCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/TagImageCmdImpl.java similarity index 81% rename from src/main/java/com/github/dockerjava/core/command/TagImageCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/TagImageCmdImpl.java index c1337112d..f7eebb8a4 100644 --- a/src/main/java/com/github/dockerjava/core/command/TagImageCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/TagImageCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.TagImageCmd; @@ -50,22 +50,19 @@ public Boolean hasForceEnabled() { @Override public TagImageCmd withImageId(String imageId) { - checkNotNull(imageId, "imageId was not specified"); - this.imageId = imageId; + this.imageId = Objects.requireNonNull(imageId, "imageId was not specified"); return this; } @Override public TagImageCmd withRepository(String repository) { - checkNotNull(repository, "repository was not specified"); - this.repository = repository; + this.repository = Objects.requireNonNull(repository, "repository was not specified"); return this; } @Override public TagImageCmd withTag(String tag) { - checkNotNull(tag, "tag was not specified"); - this.tag = tag; + this.tag = Objects.requireNonNull(tag, "tag was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/TopContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/TopContainerCmdImpl.java similarity index 82% rename from src/main/java/com/github/dockerjava/core/command/TopContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/TopContainerCmdImpl.java index 19faed3f2..5f5eee8d0 100644 --- a/src/main/java/com/github/dockerjava/core/command/TopContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/TopContainerCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.TopContainerCmd; import com.github.dockerjava.api.command.TopContainerResponse; @@ -33,15 +33,13 @@ public String getPsArgs() { @Override public TopContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } @Override public TopContainerCmd withPsArgs(String psArgs) { - checkNotNull(psArgs, "psArgs was not specified"); - this.psArgs = psArgs; + this.psArgs = Objects.requireNonNull(psArgs, "psArgs was not specified"); return this; } diff --git a/src/main/java/com/github/dockerjava/core/command/UnpauseContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/UnpauseContainerCmdImpl.java similarity index 84% rename from src/main/java/com/github/dockerjava/core/command/UnpauseContainerCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/UnpauseContainerCmdImpl.java index 26e9992b0..ef94c1979 100644 --- a/src/main/java/com/github/dockerjava/core/command/UnpauseContainerCmdImpl.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/UnpauseContainerCmdImpl.java @@ -1,6 +1,6 @@ package com.github.dockerjava.core.command; -import static com.google.common.base.Preconditions.checkNotNull; +import java.util.Objects; import com.github.dockerjava.api.command.UnpauseContainerCmd; import com.github.dockerjava.api.exception.NotFoundException; @@ -28,8 +28,7 @@ public String getContainerId() { @Override public UnpauseContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); return this; } diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateContainerCmdImpl.java new file mode 100644 index 000000000..47ab710eb --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateContainerCmdImpl.java @@ -0,0 +1,529 @@ +package com.github.dockerjava.core.command; + +import com.fasterxml.jackson.annotation.JsonIgnore; +import com.fasterxml.jackson.annotation.JsonProperty; +import com.github.dockerjava.api.command.UpdateContainerCmd; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.BlkioRateDevice; +import com.github.dockerjava.api.model.BlkioWeightDevice; +import com.github.dockerjava.api.model.Device; +import com.github.dockerjava.api.model.DeviceRequest; +import com.github.dockerjava.api.model.RestartPolicy; +import com.github.dockerjava.api.model.Ulimit; +import com.github.dockerjava.api.model.UpdateContainerResponse; +import com.github.dockerjava.core.RemoteApiVersion; +import org.apache.commons.lang3.builder.EqualsBuilder; +import org.apache.commons.lang3.builder.HashCodeBuilder; +import org.apache.commons.lang3.builder.ToStringBuilder; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import java.util.List; + +/** + * @author Kanstantsin Shautsou + * @see + * https://docs.docker.com/engine/reference/api/docker_remote_api_v1.22/ + * @since {@link RemoteApiVersion#VERSION_1_22} + */ +public class UpdateContainerCmdImpl extends AbstrDockerCmd + implements UpdateContainerCmd { + + @JsonIgnore + private String containerId; + + @JsonProperty("BlkioWeight") + private Integer blkioWeight; + + @JsonProperty("BlkioWeightDevice") + private List blkioWeightDevice; + + @JsonProperty("BlkioDeviceReadBps") + private List blkioDeviceReadBps; + + @JsonProperty("BlkioDeviceWriteBps") + private List blkioDeviceWriteBps; + + @JsonProperty("BlkioDeviceReadIOps") + private List blkioDeviceReadIOps; + + @JsonProperty("BlkioDeviceWriteIOps") + private List blkioDeviceWriteIOps; + + @JsonProperty("CpuShares") + private Integer cpuShares; + + @JsonProperty("CpuPeriod") + private Long cpuPeriod; + + @JsonProperty("CpuQuota") + private Long cpuQuota; + + @JsonProperty("CpuRealtimePeriod") + private Long cpuRealtimePeriod; + + @JsonProperty("CpuRealtimeRuntime") + private Long cpuRealtimeRuntime; + + @JsonProperty("NanoCpus") + private Long nanoCPUs; + + @JsonProperty("CpusetCpus") + private String cpusetCpus; + + @JsonProperty("CpusetMems") + private String cpusetMems; + + @JsonProperty("Devices") + private List devices; + + @JsonProperty("DeviceCgroupRules") + private List deviceCgroupRules; + + /** + * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_40} + */ + @JsonProperty("DeviceRequests") + private List deviceRequests; + + @JsonProperty("Memory") + private Long memory; + + @JsonProperty("MemorySwap") + private Long memorySwap; + + @JsonProperty("MemoryReservation") + private Long memoryReservation; + + @JsonProperty("KernelMemory") + private Long kernelMemory; + + @JsonProperty("OomKillDisable") + private Boolean oomKillDisable; + + @JsonProperty("Init") + private Boolean init; + + @JsonProperty("PidsLimit") + private Long pidsLimit; + + @JsonProperty("Ulimits") + private List ulimits; + + @JsonProperty("RestartPolicy") + private RestartPolicy restartPolicy; + + public UpdateContainerCmdImpl(UpdateContainerCmd.Exec exec, String containerId) { + super(exec); + withContainerId(containerId); + } + + /** + * @see #containerId + */ + @CheckForNull + @JsonIgnore + public String getContainerId() { + return containerId; + } + + /** + * @see #containerId + */ + public UpdateContainerCmd withContainerId(@Nonnull String containerId) { + this.containerId = containerId; + return this; + } + + /** + * @see #blkioWeight + */ + @CheckForNull + public Integer getBlkioWeight() { + return blkioWeight; + } + + /** + * @see #blkioWeight + */ + public UpdateContainerCmd withBlkioWeight(Integer blkioWeight) { + this.blkioWeight = blkioWeight; + return this; + } + + /** + * @see #blkioWeightDevice + */ + @CheckForNull + public List getBlkioWeightDevice() { + return blkioWeightDevice; + } + + public UpdateContainerCmd withBlkioWeightDevice(List blkioWeightDevice) { + this.blkioWeightDevice = blkioWeightDevice; + return this; + } + + /** + * @see #blkioDeviceReadBps + */ + @CheckForNull + public List getBlkioDeviceReadBps() { + return blkioDeviceReadBps; + } + + public UpdateContainerCmd withBlkioDeviceReadBps(List blkioDeviceReadBps) { + this.blkioDeviceReadBps = blkioDeviceReadBps; + return this; + } + + /** + * @see #blkioDeviceWriteBps + */ + @CheckForNull + public List getBlkioDeviceWriteBps() { + return blkioDeviceWriteBps; + } + + public UpdateContainerCmd withBlkioDeviceWriteBps(List blkioDeviceWriteBps) { + this.blkioDeviceWriteBps = blkioDeviceWriteBps; + return this; + } + + /** + * @see #blkioDeviceReadIOps + */ + @CheckForNull + public List getBlkioDeviceReadIOps() { + return blkioDeviceReadIOps; + } + + public UpdateContainerCmd withBlkioDeviceReadIOps(List blkioDeviceReadIOps) { + this.blkioDeviceReadIOps = blkioDeviceReadIOps; + return this; + } + + /** + * @see #blkioDeviceWriteIOps + */ + @CheckForNull + public List getBlkioDeviceWriteIOps() { + return blkioDeviceWriteIOps; + } + + @Override + public UpdateContainerCmd withBlkioDeviceWriteIOps(List blkioDeviceWriteIOps) { + this.blkioDeviceWriteIOps = blkioDeviceWriteIOps; + return this; + } + + /** + * @see #cpuPeriod + */ + @CheckForNull + public Long getCpuPeriod() { + return cpuPeriod; + } + + /** + * @see #cpuPeriod + */ + public UpdateContainerCmd withCpuPeriod(Long cpuPeriod) { + this.cpuPeriod = cpuPeriod; + return this; + } + + /** + * @see #cpuQuota + */ + @CheckForNull + public Long getCpuQuota() { + return cpuQuota; + } + + /** + * @see #cpuQuota + */ + public UpdateContainerCmd withCpuQuota(Long cpuQuota) { + this.cpuQuota = cpuQuota; + return this; + } + + /** + * @see #cpusetCpus + */ + @CheckForNull + public String getCpusetCpus() { + return cpusetCpus; + } + + /** + * @see #cpusetCpus + */ + public UpdateContainerCmd withCpusetCpus(String cpusetCpus) { + this.cpusetCpus = cpusetCpus; + return this; + } + + /** + * @see #cpusetMems + */ + @CheckForNull + public String getCpusetMems() { + return cpusetMems; + } + + /** + * @see #cpusetMems + */ + public UpdateContainerCmd withCpusetMems(String cpusetMems) { + this.cpusetMems = cpusetMems; + return this; + } + + /** + * @see #cpuShares + */ + @CheckForNull + public Integer getCpuShares() { + return cpuShares; + } + + /** + * @see #cpuShares + */ + public UpdateContainerCmd withCpuShares(Integer cpuShares) { + this.cpuShares = cpuShares; + return this; + } + + /** + * @see #cpuRealtimePeriod + */ + @CheckForNull + public Long getCpuRealtimePeriod() { + return cpuRealtimePeriod; + } + + public UpdateContainerCmd withCpuRealtimePeriod(Long cpuRealtimePeriod) { + this.cpuRealtimePeriod = cpuRealtimePeriod; + return this; + } + + /** + * @see #cpuRealtimeRuntime + */ + @CheckForNull + public Long getCpuRealtimeRuntime() { + return cpuRealtimeRuntime; + } + + public UpdateContainerCmd withCpuRealtimeRuntime(Long cpuRealtimeRuntime) { + this.cpuRealtimeRuntime = cpuRealtimeRuntime; + return this; + } + + /** + * @see #devices + */ + @CheckForNull + public List getDevices() { + return devices; + } + + public UpdateContainerCmd withDevices(List devices) { + this.devices = devices; + return this; + } + + /** + * @see #deviceCgroupRules + */ + @CheckForNull + public List getDeviceCgroupRules() { + return deviceCgroupRules; + } + + public UpdateContainerCmd withDeviceCgroupRules(List deviceCgroupRules) { + this.deviceCgroupRules = deviceCgroupRules; + return this; + } + + /** + * @see #deviceRequests + */ + @CheckForNull + public List getDeviceRequests() { + return deviceRequests; + } + + public UpdateContainerCmd withDeviceRequests(List deviceRequests) { + this.deviceRequests = deviceRequests; + return this; + } + + /** + * @see #kernelMemory + */ + @CheckForNull + public Long getKernelMemory() { + return kernelMemory; + } + + /** + * @see #kernelMemory + */ + public UpdateContainerCmd withKernelMemory(Long kernelMemory) { + this.kernelMemory = kernelMemory; + return this; + } + + /** + * @see #memory + */ + @CheckForNull + public Long getMemory() { + return memory; + } + + /** + * @see #memory + */ + public UpdateContainerCmd withMemory(Long memory) { + this.memory = memory; + return this; + } + + /** + * @see #memoryReservation + */ + @CheckForNull + public Long getMemoryReservation() { + return memoryReservation; + } + + /** + * @see #memoryReservation + */ + public UpdateContainerCmd withMemoryReservation(Long memoryReservation) { + this.memoryReservation = memoryReservation; + return this; + } + + /** + * @see #memorySwap + */ + @CheckForNull + public Long getMemorySwap() { + return memorySwap; + } + + /** + * @see #memorySwap + */ + public UpdateContainerCmd withMemorySwap(Long memorySwap) { + this.memorySwap = memorySwap; + return this; + } + + /** + * @see #nanoCPUs + */ + @CheckForNull + public Long getNanoCPUs() { + return nanoCPUs; + } + + public UpdateContainerCmd withNanoCPUs(Long nanoCPUs) { + this.nanoCPUs = nanoCPUs; + return this; + } + + /** + * @see #oomKillDisable + */ + @CheckForNull + public Boolean getOomKillDisable() { + return oomKillDisable; + } + + public UpdateContainerCmd withOomKillDisable(Boolean oomKillDisable) { + this.oomKillDisable = oomKillDisable; + return this; + } + + /** + * @see #init + */ + @CheckForNull + public Boolean getInit() { + return init; + } + + public UpdateContainerCmd withInit(Boolean init) { + this.init = init; + return this; + } + + /** + * @see #pidsLimit + */ + @CheckForNull + public Long getPidsLimit() { + return pidsLimit; + } + + public UpdateContainerCmd withPidsLimit(Long pidsLimit) { + this.pidsLimit = pidsLimit; + return this; + } + + /** + * @see #ulimits + */ + @CheckForNull + public List getUlimits() { + return ulimits; + } + + public UpdateContainerCmd withUlimits(List ulimits) { + this.ulimits = ulimits; + return this; + } + + /** + * @see #restartPolicy + */ + @CheckForNull + public RestartPolicy getRestartPolicy() { + return restartPolicy; + } + + public UpdateContainerCmd withRestartPolicy(RestartPolicy restartPolicy) { + this.restartPolicy = restartPolicy; + return this; + } + + /** + * @throws NotFoundException No such container + */ + @Override + public UpdateContainerResponse exec() throws NotFoundException { + return super.exec(); + } + + @Override + public String toString() { + return ToStringBuilder.reflectionToString(this); + } + + @Override + public boolean equals(Object o) { + return EqualsBuilder.reflectionEquals(this, o); + } + + @Override + public int hashCode() { + return HashCodeBuilder.reflectionHashCode(this); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateServiceCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateServiceCmdImpl.java new file mode 100644 index 000000000..7ff9412a9 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateServiceCmdImpl.java @@ -0,0 +1,112 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.UpdateServiceCmd; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.ServiceSpec; +import com.github.dockerjava.core.RemoteApiVersion; +import org.apache.commons.lang3.builder.EqualsBuilder; +import org.apache.commons.lang3.builder.HashCodeBuilder; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public class UpdateServiceCmdImpl extends AbstrDockerCmd + implements UpdateServiceCmd { + + /** + * @since 1.24 + */ + private String serviceId; + + /** + * @since 1.24 + */ + private ServiceSpec serviceSpec; + + /** + * @since 1.24 + */ + private Long version; + + public UpdateServiceCmdImpl(Exec exec, String serviceId, ServiceSpec serviceSpec) { + super(exec); + withServiceId(serviceId); + withServiceSpec(serviceSpec); + } + + /** + * @see #serviceId + */ + @CheckForNull + public String getServiceId() { + return serviceId; + } + + /** + * @see #serviceId + */ + public UpdateServiceCmd withServiceId(@Nonnull String serviceId) { + this.serviceId = serviceId; + return this; + } + + /** + * @see #serviceSpec + */ + @CheckForNull + public ServiceSpec getServiceSpec() { + return serviceSpec; + } + + /** + * @see #serviceSpec + */ + public UpdateServiceCmd withServiceSpec(ServiceSpec serviceSpec) { + this.serviceSpec = serviceSpec; + return this; + } + + /** + * @see #version + */ + @CheckForNull + public Long getVersion() { + return version; + } + + /** + * @see #version + */ + public UpdateServiceCmdImpl withVersion(Long version) { + this.version = version; + return this; + } + + /** + * @throws NotFoundException No such service + */ + @Override + public Void exec() throws NotFoundException { + return super.exec(); + } + + @Override + public String toString() { + return ToStringBuilder.reflectionToString(this, ToStringStyle.SHORT_PREFIX_STYLE); + } + + @Override + public boolean equals(Object o) { + return EqualsBuilder.reflectionEquals(this, o); + } + + @Override + public int hashCode() { + return HashCodeBuilder.reflectionHashCode(this); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateSwarmCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateSwarmCmdImpl.java new file mode 100644 index 000000000..372cd34ce --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateSwarmCmdImpl.java @@ -0,0 +1,68 @@ +package com.github.dockerjava.core.command; + + +import java.util.Objects; + +import com.github.dockerjava.api.command.UpdateSwarmCmd; +import com.github.dockerjava.api.model.SwarmSpec; + +/** + * Update a swarm. + */ +public class UpdateSwarmCmdImpl extends AbstrDockerCmd implements + UpdateSwarmCmd { + + private Long version; + private Boolean rotateWorkerToken; + private Boolean rotateManagerToken; + private SwarmSpec swarmSpec; + + public UpdateSwarmCmdImpl(Exec exec, SwarmSpec swarmSpec) { + super(exec); + this.swarmSpec = swarmSpec; + } + + @Override + public Long getVersion() { + return version; + } + + @Override + public UpdateSwarmCmd withVersion(Long version) { + this.version = version; + return this; + } + + @Override + public Boolean getRotateWorkerToken() { + return rotateWorkerToken; + } + + @Override + public UpdateSwarmCmd withRotateWorkerToken(Boolean rotateWorkerToken) { + this.rotateWorkerToken = rotateWorkerToken; + return this; + } + + @Override + public Boolean getRotateManagerToken() { + return rotateManagerToken; + } + + @Override + public UpdateSwarmCmd withRotateManagerToken(Boolean rotateManagerToken) { + this.rotateManagerToken = rotateManagerToken; + return this; + } + + @Override + public SwarmSpec getSwarmSpec() { + return swarmSpec; + } + + @Override + public UpdateSwarmCmd withSwarmSpec(SwarmSpec swarmSpec) { + this.swarmSpec = Objects.requireNonNull(swarmSpec, "swarmSpec was not specified"); + return this; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateSwarmNodeCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateSwarmNodeCmdImpl.java new file mode 100644 index 000000000..a2f22fa14 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/UpdateSwarmNodeCmdImpl.java @@ -0,0 +1,105 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.command.UpdateSwarmNodeCmd; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.SwarmNodeSpec; +import com.github.dockerjava.core.RemoteApiVersion; +import org.apache.commons.lang3.builder.EqualsBuilder; +import org.apache.commons.lang3.builder.HashCodeBuilder; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; + +/** + * Update swarmNode spec + * + * @since {@link RemoteApiVersion#VERSION_1_24} + */ +public class UpdateSwarmNodeCmdImpl extends AbstrDockerCmd + implements UpdateSwarmNodeCmd { + + private String swarmNodeId; + + private SwarmNodeSpec swarmNodeSpec; + + private Long version; + + public UpdateSwarmNodeCmdImpl(Exec exec) { + super(exec); + } + + public UpdateSwarmNodeCmdImpl(Exec exec, String swarmNodeId, SwarmNodeSpec swarmNodeSpec) { + super(exec); + withSwarmNodeId(swarmNodeId); + withSwarmNodeSpec(swarmNodeSpec); + } + + /** + * @see #swarmNodeId + */ + @CheckForNull + public String getSwarmNodeId() { + return swarmNodeId; + } + + /** + * @see #swarmNodeId + */ + public UpdateSwarmNodeCmd withSwarmNodeId(@Nonnull String swarmNodeId) { + this.swarmNodeId = swarmNodeId; + return this; + } + + /** + * @see #swarmNodeSpec + */ + @CheckForNull + public SwarmNodeSpec getSwarmNodeSpec() { + return swarmNodeSpec; + } + + /** + * @see #swarmNodeSpec + */ + public UpdateSwarmNodeCmd withSwarmNodeSpec(SwarmNodeSpec swarmNodeSpec) { + this.swarmNodeSpec = swarmNodeSpec; + return this; + } + + @Override + public UpdateSwarmNodeCmd withVersion(@Nonnull Long versionId) { + this.version = versionId; + return this; + } + + @CheckForNull + @Override + public Long getVersion() { + return version; + } + + /** + * @throws NotFoundException No such swarmNode + */ + @Override + public Void exec() throws NotFoundException { + return super.exec(); + } + + @Override + public String toString() { + return ToStringBuilder.reflectionToString(this, ToStringStyle.SHORT_PREFIX_STYLE); + } + + @Override + public boolean equals(Object o) { + return EqualsBuilder.reflectionEquals(this, o); + } + + @Override + public int hashCode() { + return HashCodeBuilder.reflectionHashCode(this); + } +} diff --git a/src/main/java/com/github/dockerjava/core/command/VersionCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/VersionCmdImpl.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/command/VersionCmdImpl.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/command/VersionCmdImpl.java diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/WaitContainerCmdImpl.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/WaitContainerCmdImpl.java new file mode 100644 index 000000000..b627e2ccd --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/WaitContainerCmdImpl.java @@ -0,0 +1,51 @@ +package com.github.dockerjava.core.command; + +import java.util.Objects; + +import javax.annotation.Nullable; + +import com.github.dockerjava.api.command.WaitContainerCmd; +import com.github.dockerjava.api.model.WaitContainerCondition; +import com.github.dockerjava.api.model.WaitResponse; + +/** + * Wait a container + * + * Block until container stops, then returns its exit code + */ +public class WaitContainerCmdImpl extends AbstrAsyncDockerCmd implements + WaitContainerCmd { + + private String containerId; + + private WaitContainerCondition condition; + + public WaitContainerCmdImpl(WaitContainerCmd.Exec exec, String containerId) { + super(exec); + withContainerId(containerId); + } + + @Override + public String getContainerId() { + return containerId; + } + + @Override + public WaitContainerCmd withContainerId(String containerId) { + this.containerId = Objects.requireNonNull(containerId, "containerId was not specified"); + return this; + } + + @Nullable + @Override + public WaitContainerCondition getCondition() { + return condition; + } + + @Override + public WaitContainerCmd withCondition(@Nullable WaitContainerCondition condition) { + this.condition = condition; + return this; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/command/WaitContainerResultCallback.java b/docker-java-core/src/main/java/com/github/dockerjava/core/command/WaitContainerResultCallback.java new file mode 100644 index 000000000..b9cd245e7 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/command/WaitContainerResultCallback.java @@ -0,0 +1,76 @@ +/* + * Created on 21.07.2015 + */ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.model.WaitResponse; +import com.github.dockerjava.core.async.ResultCallbackTemplate; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.annotation.CheckForNull; +import java.util.concurrent.TimeUnit; + +/** + * + * @author Marcus Linke + * + * @deprecated use {@link com.github.dockerjava.api.command.WaitContainerResultCallback} + */ +@Deprecated +public class WaitContainerResultCallback extends ResultCallbackTemplate { + + private static final Logger LOGGER = LoggerFactory.getLogger(WaitContainerResultCallback.class); + + @CheckForNull + private WaitResponse waitResponse = null; + + @Override + public void onNext(WaitResponse waitResponse) { + this.waitResponse = waitResponse; + LOGGER.debug(waitResponse.toString()); + } + + /** + * Awaits the status code from the container. + * + * @throws DockerClientException + * if the wait operation fails. + */ + public Integer awaitStatusCode() { + try { + awaitCompletion(); + } catch (InterruptedException e) { + throw new DockerClientException("", e); + } + + return getStatusCode(); + } + + /** + * Awaits the status code from the container. + * + * @throws DockerClientException + * if the wait operation fails. + */ + public Integer awaitStatusCode(long timeout, TimeUnit timeUnit) { + try { + if (!awaitCompletion(timeout, timeUnit)) { + throw new DockerClientException("Awaiting status code timeout."); + } + } catch (InterruptedException e) { + throw new DockerClientException("Awaiting status code interrupted: ", e); + } + + return getStatusCode(); + } + + private Integer getStatusCode() { + if (waitResponse == null) { + throw new DockerClientException("Error while wait container"); + } else { + return waitResponse.getStatusCode(); + } + } +} diff --git a/src/main/java/com/github/dockerjava/core/dockerfile/Dockerfile.java b/docker-java-core/src/main/java/com/github/dockerjava/core/dockerfile/Dockerfile.java similarity index 81% rename from src/main/java/com/github/dockerjava/core/dockerfile/Dockerfile.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/dockerfile/Dockerfile.java index 9dcd20199..752935e19 100644 --- a/src/main/java/com/github/dockerjava/core/dockerfile/Dockerfile.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/dockerfile/Dockerfile.java @@ -12,7 +12,6 @@ import org.apache.commons.io.FileUtils; import org.apache.commons.io.FilenameUtils; import org.apache.commons.io.IOUtils; -import org.apache.commons.io.filefilter.TrueFileFilter; import com.github.dockerjava.api.exception.DockerClientException; import com.github.dockerjava.core.GoLangFileMatch; @@ -20,7 +19,7 @@ import com.github.dockerjava.core.util.CompressArchiveUtil; import com.github.dockerjava.core.util.FilePathUtil; import com.google.common.base.Function; -import com.google.common.base.Objects; +import com.google.common.base.MoreObjects; import com.google.common.base.Optional; import com.google.common.collect.Collections2; @@ -85,7 +84,7 @@ public Iterable getStatements() throws IOException { } public List getIgnores() throws IOException { - List ignores = new ArrayList(); + List ignores = new ArrayList<>(); File dockerIgnoreFile = new File(baseDirectory, ".dockerignore"); if (dockerIgnoreFile.exists()) { int lineNumber = 0; @@ -119,7 +118,7 @@ public class ScannedResult { final List ignores; - final List filesToAdd = new ArrayList(); + final List filesToAdd = new ArrayList<>(); public InputStream buildDockerFolderTar() { return buildDockerFolderTar(baseDirectory); @@ -135,8 +134,6 @@ public InputStream buildDockerFolderTar(File directory) { dockerFolderTar = CompressArchiveUtil.archiveTARFiles(directory, filesToAdd, archiveNameWithOutExtension); - long length = dockerFolderTar.length(); - final FileInputStream tarInputStream = FileUtils.openInputStream(dockerFolderTar); final File tarFile = dockerFolderTar; @@ -172,7 +169,7 @@ public void close() throws IOException { @Override public String toString() { - return Objects.toStringHelper(this).add("ignores", ignores).add("filesToAdd", filesToAdd).toString(); + return MoreObjects.toStringHelper(this).add("ignores", ignores).add("filesToAdd", filesToAdd).toString(); } public ScannedResult() throws IOException { @@ -186,21 +183,48 @@ public ScannedResult() throws IOException { "Dockerfile is excluded by pattern '%s' in .dockerignore file", matchingIgnorePattern)); } - Collection filesInBuildContext = FileUtils.listFiles(baseDirectory, TrueFileFilter.INSTANCE, - TrueFileFilter.INSTANCE); + addFilesInDirectory(baseDirectory); + } + + /** + * Adds all files found in directory and subdirectories to + * filesToAdd collection. It also adds any empty directories + * if found. + * + * @param directory directory + * @throws DockerClientException when IO error occurs + */ + private void addFilesInDirectory(File directory) { + File[] files = directory.listFiles(); + + if (files == null) { + throw new DockerClientException("Failed to read build context directory: " + baseDirectory.getAbsolutePath()); + } - for (File f : filesInBuildContext) { - if (effectiveMatchingIgnorePattern(f) == null) { - filesToAdd.add(f); + if (files.length != 0) { + for (File f : files) { + if (f.isDirectory()) { + addFilesInDirectory(f); + } else if (effectiveMatchingIgnorePattern(f) == null) { + filesToAdd.add(f); + } } + // base directory should at least contains Dockerfile, but better check + } else if (!isBaseDirectory(directory)) { + // add empty directory + filesToAdd.add(directory); } } + private boolean isBaseDirectory(File directory) { + return directory.compareTo(baseDirectory) == 0; + } + /** * Returns all matching ignore patterns for the given file name. */ private List matchingIgnorePatterns(String fileName) { - List matches = new ArrayList(); + List matches = new ArrayList<>(); int lineNumber = 0; for (String pattern : ignores) { @@ -224,7 +248,8 @@ private List matchingIgnorePatterns(String fileName) { * will be respected. */ private String effectiveMatchingIgnorePattern(File file) { - String relativeFilename = FilePathUtil.relativize(baseDirectory, file); + // normalize path to replace '/' to '\' on Windows + String relativeFilename = FilenameUtils.normalize(FilePathUtil.relativize(baseDirectory, file)); List matchingPattern = matchingIgnorePatterns(relativeFilename); diff --git a/src/main/java/com/github/dockerjava/core/dockerfile/DockerfileStatement.java b/docker-java-core/src/main/java/com/github/dockerjava/core/dockerfile/DockerfileStatement.java similarity index 84% rename from src/main/java/com/github/dockerjava/core/dockerfile/DockerfileStatement.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/dockerfile/DockerfileStatement.java index 4f0341c7a..fb2447c2d 100644 --- a/src/main/java/com/github/dockerjava/core/dockerfile/DockerfileStatement.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/dockerfile/DockerfileStatement.java @@ -8,13 +8,11 @@ import java.util.regex.Matcher; import java.util.regex.Pattern; -import org.apache.commons.lang.StringUtils; +import org.apache.commons.lang3.StringUtils; import com.github.dockerjava.api.exception.DockerClientException; -import com.google.common.base.Function; -import com.google.common.base.Objects; +import com.google.common.base.MoreObjects; import com.google.common.base.Optional; -import com.google.common.base.Predicate; import com.google.common.collect.Collections2; /** @@ -92,28 +90,19 @@ private Add(Collection sources, String destination) { @Override public Add transform(final Map env) { - Collection resources = Collections2.transform(sources, new Function() { - @Override - public String apply(String source) { - return filterForEnvironmentVars(env, source).trim(); - } - }); + Collection resources = Collections2.transform(sources, source -> filterForEnvironmentVars(env, source).trim()); return new Add(resources, destination); } public Iterable getFileResources() { - return Collections2.filter(sources, new Predicate() { - - @Override - public boolean apply(String source) { - URI uri; - try { - uri = new URI(source); - } catch (URISyntaxException e) { - return false; - } - return uri.getScheme() == null || "file".equals(uri.getScheme()); + return Collections2.filter(sources, source -> { + URI uri; + try { + uri = new URI(source); + } catch (URISyntaxException e) { + return false; } + return uri.getScheme() == null || "file".equals(uri.getScheme()); }); } @@ -155,7 +144,7 @@ public static Optional create(String statement) { @Override public String toString() { - return Objects.toStringHelper(this).add("sources", sources).add("destination", destination).toString(); + return MoreObjects.toStringHelper(this).add("sources", sources).add("destination", destination).toString(); } } @@ -192,7 +181,7 @@ public static Optional create(String statement) { @Override public String toString() { - return Objects.toStringHelper(this).add("variable", variable).add("value", value).toString(); + return MoreObjects.toStringHelper(this).add("variable", variable).add("value", value).toString(); } } diff --git a/src/main/java/com/github/dockerjava/core/exception/GoLangFileMatchException.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exception/GoLangFileMatchException.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/exception/GoLangFileMatchException.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/exception/GoLangFileMatchException.java diff --git a/src/main/java/com/github/dockerjava/core/exception/InvalidRepositoryNameException.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exception/InvalidRepositoryNameException.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/exception/InvalidRepositoryNameException.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/exception/InvalidRepositoryNameException.java diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AbstrAsyncDockerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AbstrAsyncDockerCmdExec.java new file mode 100644 index 000000000..5655d74bd --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AbstrAsyncDockerCmdExec.java @@ -0,0 +1,63 @@ +package com.github.dockerjava.core.exec; + +import java.io.Closeable; +import java.io.IOException; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.AsyncDockerCmd; +import com.github.dockerjava.api.command.DockerCmdAsyncExec; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; + +public abstract class AbstrAsyncDockerCmdExec, A_RES_T> extends + AbstrDockerCmdExec implements DockerCmdAsyncExec { + + public AbstrAsyncDockerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + public Void exec(CMD_T command, ResultCallback resultCallback) { + return execute(command, resultCallback); + } + + protected final Void execute(final CMD_T command, final ResultCallback resultCallback) { + + ResultCallback delegatingResultCallback = new ResultCallback() { + + @Override + public void close() throws IOException { + resultCallback.close(); + command.close(); + } + + @Override + public void onStart(Closeable closeable) { + resultCallback.onStart(closeable); + } + + @Override + public void onNext(A_RES_T object) { + resultCallback.onNext(object); + } + + @Override + public void onError(Throwable throwable) { + resultCallback.onError(throwable); + } + + @Override + public void onComplete() { + resultCallback.onComplete(); + command.close(); + } + }; + + execute0(command, delegatingResultCallback); + + return null; + } + + protected abstract Void execute0(final CMD_T command, final ResultCallback resultCallback); + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AbstrDockerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AbstrDockerCmdExec.java new file mode 100644 index 000000000..ad7f285f4 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AbstrDockerCmdExec.java @@ -0,0 +1,99 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.node.ObjectNode; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.AuthConfigurations; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.InvocationBuilder; +import com.github.dockerjava.core.RemoteApiVersion; +import com.github.dockerjava.core.WebTarget; +import com.google.common.io.BaseEncoding; + +import javax.annotation.CheckForNull; +import javax.annotation.Nonnull; +import java.io.IOException; +import java.util.Objects; + +import static com.github.dockerjava.core.RemoteApiVersion.UNKNOWN_VERSION; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_19; + +public abstract class AbstrDockerCmdExec { + + private final transient DockerClientConfig dockerClientConfig; + + private final transient WebTarget baseResource; + + public AbstrDockerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + this.baseResource = Objects.requireNonNull(baseResource, "baseResource was not specified"); + this.dockerClientConfig = Objects.requireNonNull(dockerClientConfig, "dockerClientConfig was not specified"); + } + + protected WebTarget getBaseResource() { + return baseResource; + } + + @CheckForNull + protected AuthConfigurations getBuildAuthConfigs() { + return dockerClientConfig.getAuthConfigurations(); + } + + protected String registryAuth(@Nonnull AuthConfig authConfig) { + try { + return BaseEncoding.base64Url().encode(dockerClientConfig.getObjectMapper().writeValueAsString(authConfig).getBytes()); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Nonnull + protected String registryConfigs(@Nonnull AuthConfigurations authConfigs) { + try { + final String json; + final RemoteApiVersion apiVersion = dockerClientConfig.getApiVersion(); + ObjectMapper objectMapper = dockerClientConfig.getObjectMapper(); + + if (apiVersion.equals(UNKNOWN_VERSION)) { + ObjectNode rootNode = objectMapper.valueToTree(authConfigs.getConfigs()); // all registries + final ObjectNode authNodes = objectMapper.valueToTree(authConfigs); // wrapped in "configs":{} + rootNode.setAll(authNodes); // merge 2 variants + json = rootNode.toString(); + } else if (apiVersion.isGreaterOrEqual(VERSION_1_19)) { + json = objectMapper.writeValueAsString(authConfigs.getConfigs()); + } else { + json = objectMapper.writeValueAsString(authConfigs); + } + return BaseEncoding.base64Url().encode(json.getBytes()); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Nonnull + protected InvocationBuilder resourceWithAuthConfig(@Nonnull AuthConfig authConfig, + @Nonnull InvocationBuilder request) { + return request.header("X-Registry-Auth", registryAuth(authConfig)); + } + + @Nonnull + protected InvocationBuilder resourceWithOptionalAuthConfig(@CheckForNull AuthConfig authConfig, + @Nonnull InvocationBuilder request) { + if (authConfig != null) { + request = resourceWithAuthConfig(authConfig, request); + } + return request; + } + + protected boolean bool(Boolean bool) { + return bool != null && bool; + } + + protected WebTarget booleanQueryParam(WebTarget webTarget, String name, Boolean value) { + if (bool(value)) { + webTarget = webTarget.queryParam(name, bool(value) + ""); + } + + return webTarget; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AbstrSyncDockerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AbstrSyncDockerCmdExec.java new file mode 100644 index 000000000..de3e3b029 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AbstrSyncDockerCmdExec.java @@ -0,0 +1,33 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.DockerCmd; +import com.github.dockerjava.api.command.DockerCmdSyncExec; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; + +public abstract class AbstrSyncDockerCmdExec, RES_T> extends AbstrDockerCmdExec + implements DockerCmdSyncExec { + + public AbstrSyncDockerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + public RES_T exec(CMD_T command) { + // this hack works because of ResponseStatusExceptionFilter + try (CMD_T cmd = command) { + try { + return execute(cmd); + } catch (RuntimeException e) { + if (e.getCause() instanceof DockerException) { + throw (DockerException) e.getCause(); + } else { + throw e; + } + } + } + } + + protected abstract RES_T execute(CMD_T command); +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AttachContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AttachContainerCmdExec.java new file mode 100644 index 000000000..618d4e6b1 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AttachContainerCmdExec.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.AttachContainerCmd; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; + +public class AttachContainerCmdExec extends AbstrAsyncDockerCmdExec implements + AttachContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(AttachContainerCmdExec.class); + + public AttachContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute0(AttachContainerCmd command, ResultCallback resultCallback) { + + WebTarget webTarget = getBaseResource().path("/containers/{id}/attach").resolveTemplate("id", + command.getContainerId()); + + webTarget = booleanQueryParam(webTarget, "logs", command.hasLogsEnabled()); + webTarget = booleanQueryParam(webTarget, "stdout", command.hasStdoutEnabled()); + webTarget = booleanQueryParam(webTarget, "stderr", command.hasStderrEnabled()); + webTarget = booleanQueryParam(webTarget, "stdin", command.getStdin() != null); + webTarget = booleanQueryParam(webTarget, "stream", command.hasFollowStreamEnabled()); + + LOGGER.trace("POST: {}", webTarget); + + webTarget.request().post(null, command.getStdin(), resultCallback); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AuthCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AuthCmdExec.java new file mode 100644 index 000000000..ef9fe5792 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/AuthCmdExec.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.AuthCmd; +import com.github.dockerjava.api.model.AuthResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class AuthCmdExec extends AbstrSyncDockerCmdExec implements AuthCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(AuthCmdExec.class); + + public AuthCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected AuthResponse execute(AuthCmd command) { + WebTarget webResource = getBaseResource().path("/auth"); + LOGGER.trace("POST: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON) + .post(command.getAuthConfig(), new TypeReference() { + }); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/BuildImageCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/BuildImageCmdExec.java new file mode 100644 index 000000000..aa65fff40 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/BuildImageCmdExec.java @@ -0,0 +1,136 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.BuildImageCmd; +import com.github.dockerjava.api.model.AuthConfigurations; +import com.github.dockerjava.api.model.BuildResponseItem; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.InvocationBuilder; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.annotation.CheckForNull; + +import static com.github.dockerjava.core.util.CacheFromEncoder.jsonEncode; +import static org.apache.commons.lang3.StringUtils.isNotBlank; + +public class BuildImageCmdExec extends AbstrAsyncDockerCmdExec implements + BuildImageCmd.Exec { + private static final Logger LOGGER = LoggerFactory.getLogger(BuildImageCmdExec.class); + + public BuildImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + private InvocationBuilder resourceWithOptionalAuthConfig(BuildImageCmd command, InvocationBuilder request) { + final AuthConfigurations authConfigs = firstNonNull(command.getBuildAuthConfigs(), getBuildAuthConfigs()); + if (authConfigs != null && !authConfigs.getConfigs().isEmpty()) { + request = request.header("X-Registry-Config", registryConfigs(authConfigs)); + } + return request; + } + + @CheckForNull + private static AuthConfigurations firstNonNull(@CheckForNull final AuthConfigurations fromCommand, + @CheckForNull final AuthConfigurations fromConfig) { + if (fromCommand != null) { + return fromCommand; + } + if (fromConfig != null) { + return fromConfig; + } + return null; + } + + @Override + protected Void execute0(BuildImageCmd command, ResultCallback resultCallback) { + + WebTarget webTarget = getBaseResource().path("/build"); + String dockerFilePath = command.getPathToDockerfile(); + + if (dockerFilePath != null && command.getRemote() == null && !"Dockerfile".equals(dockerFilePath)) { + webTarget = webTarget.queryParam("dockerfile", dockerFilePath); + } + + if (command.getTags() != null && !command.getTags().isEmpty()) { + webTarget = webTarget.queryParamsSet("t", command.getTags()); + } else if (isNotBlank(command.getTag())) { + webTarget = webTarget.queryParam("t", command.getTag()); + } + + if (command.getCacheFrom() != null && !command.getCacheFrom().isEmpty()) { + webTarget = webTarget.queryParam("cachefrom", jsonEncode(command.getCacheFrom())); + } + + if (command.getRemote() != null) { + webTarget = webTarget.queryParam("remote", command.getRemote().toString()); + } + + webTarget = booleanQueryParam(webTarget, "q", command.isQuiet()); + webTarget = booleanQueryParam(webTarget, "nocache", command.hasNoCacheEnabled()); + webTarget = booleanQueryParam(webTarget, "pull", command.hasPullEnabled()); + webTarget = booleanQueryParam(webTarget, "rm", command.hasRemoveEnabled()); + webTarget = booleanQueryParam(webTarget, "forcerm", command.isForcerm()); + + // this has to be handled differently as it should switch to 'false' + if (command.hasRemoveEnabled() == null || !command.hasRemoveEnabled()) { + webTarget = webTarget.queryParam("rm", "false"); + } + + if (command.getMemory() != null) { + webTarget = webTarget.queryParam("memory", command.getMemory()); + } + if (command.getMemswap() != null) { + webTarget = webTarget.queryParam("memswap", command.getMemswap()); + } + if (command.getCpushares() != null) { + webTarget = webTarget.queryParam("cpushares", command.getCpushares()); + } + if (command.getCpusetcpus() != null) { + webTarget = webTarget.queryParam("cpusetcpus", command.getCpusetcpus()); + } + + if (command.getBuildArgs() != null) { + webTarget = webTarget.queryParamsJsonMap("buildargs", command.getBuildArgs()); + } + + if (command.getShmsize() != null) { + webTarget = webTarget.queryParam("shmsize", command.getShmsize()); + } + + if (command.getLabels() != null) { + webTarget = webTarget.queryParamsJsonMap("labels", command.getLabels()); + } + + if (command.getNetworkMode() != null) { + webTarget = webTarget.queryParam("networkmode", command.getNetworkMode()); + } + + if (command.getPlatform() != null) { + webTarget = webTarget.queryParam("platform", command.getPlatform()); + } + + if (command.getTarget() != null) { + webTarget = webTarget.queryParam("target", command.getTarget()); + } + + if (command.getExtraHosts() != null) { + webTarget = webTarget.queryParamsSet("extrahosts", command.getExtraHosts()); + } + + LOGGER.trace("POST: {}", webTarget); + + InvocationBuilder builder = resourceWithOptionalAuthConfig(command, webTarget.request()) + .accept(MediaType.APPLICATION_JSON) + .header("Content-Type", "application/tar") + .header("encoding", "gzip"); + + builder.post(new TypeReference() { + }, resultCallback, command.getTarInputStream()); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CommitCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CommitCmdExec.java new file mode 100644 index 000000000..e9b78890c --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CommitCmdExec.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.node.ObjectNode; +import com.github.dockerjava.api.command.CommitCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class CommitCmdExec extends AbstrSyncDockerCmdExec implements CommitCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CommitCmdExec.class); + + public CommitCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected String execute(CommitCmd command) { + WebTarget webTarget = getBaseResource().path("/commit").queryParam("container", command.getContainerId()) + .queryParam("repo", command.getRepository()).queryParam("tag", command.getTag()) + .queryParam("m", command.getMessage()).queryParam("author", command.getAuthor()); + + webTarget = booleanQueryParam(webTarget, "pause", command.hasPauseEnabled()); + + LOGGER.trace("POST: {}", webTarget); + ObjectNode objectNode = webTarget.request().accept(MediaType.APPLICATION_JSON) + .post(command, new TypeReference() { + }); + + return objectNode.get("Id").asText(); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ConnectToNetworkCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ConnectToNetworkCmdExec.java new file mode 100644 index 000000000..b67eb296a --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ConnectToNetworkCmdExec.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.ConnectToNetworkCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +public class ConnectToNetworkCmdExec extends AbstrSyncDockerCmdExec + implements ConnectToNetworkCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ConnectToNetworkCmdExec.class); + + public ConnectToNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(ConnectToNetworkCmd command) { + + WebTarget webTarget = getBaseResource().path("/networks/" + command.getNetworkId() + "/connect"); + + LOGGER.trace("POST: {}", webTarget); + try { + webTarget.request().post(command).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ContainerDiffCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ContainerDiffCmdExec.java new file mode 100644 index 000000000..bed6501cc --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ContainerDiffCmdExec.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.exec; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ContainerDiffCmd; +import com.github.dockerjava.api.model.ChangeLog; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class ContainerDiffCmdExec extends AbstrSyncDockerCmdExec> implements + ContainerDiffCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ContainerDiffCmdExec.class); + + public ContainerDiffCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected List execute(ContainerDiffCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/changes").resolveTemplate("id", + command.getContainerId()); + + LOGGER.trace("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference>() { + }); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CopyArchiveFromContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CopyArchiveFromContainerCmdExec.java new file mode 100644 index 000000000..d9d98c923 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CopyArchiveFromContainerCmdExec.java @@ -0,0 +1,32 @@ +package com.github.dockerjava.core.exec; + +import java.io.InputStream; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class CopyArchiveFromContainerCmdExec extends AbstrSyncDockerCmdExec + implements CopyArchiveFromContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CopyArchiveFromContainerCmdExec.class); + + public CopyArchiveFromContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected InputStream execute(CopyArchiveFromContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/archive").resolveTemplate("id", + command.getContainerId()); + + LOGGER.trace("Get: " + webResource.toString()); + + return webResource.queryParam("path", command.getResource()).request().accept(MediaType.APPLICATION_X_TAR) + .get(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CopyArchiveToContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CopyArchiveToContainerCmdExec.java new file mode 100644 index 000000000..ea4a527b1 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CopyArchiveToContainerCmdExec.java @@ -0,0 +1,38 @@ +package com.github.dockerjava.core.exec; + +import java.io.InputStream; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class CopyArchiveToContainerCmdExec extends AbstrSyncDockerCmdExec implements + CopyArchiveToContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CopyArchiveFromContainerCmdExec.class); + + public CopyArchiveToContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(CopyArchiveToContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/archive").resolveTemplate("id", + command.getContainerId()); + + LOGGER.trace("PUT: " + webResource.toString()); + InputStream streamToUpload = command.getTarInputStream(); + + webResource.queryParam("path", command.getRemotePath()) + .queryParam("noOverwriteDirNonDir", command.isNoOverwriteDirNonDir()) + .queryParam("copyUIDGID", command.isCopyUIDGID()) + .request() + .put(streamToUpload, MediaType.APPLICATION_X_TAR); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CopyFileFromContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CopyFileFromContainerCmdExec.java new file mode 100644 index 000000000..2d54887a2 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CopyFileFromContainerCmdExec.java @@ -0,0 +1,33 @@ +package com.github.dockerjava.core.exec; + +import java.io.InputStream; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.CopyFileFromContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class CopyFileFromContainerCmdExec extends AbstrSyncDockerCmdExec + implements CopyFileFromContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CopyFileFromContainerCmdExec.class); + + public CopyFileFromContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected InputStream execute(CopyFileFromContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/copy").resolveTemplate("id", + command.getContainerId()); + + LOGGER.trace("POST: " + webResource.toString()); + + return webResource.request().accept(MediaType.APPLICATION_OCTET_STREAM) + .post(command); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateConfigCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateConfigCmdExec.java new file mode 100644 index 000000000..4ead9cb48 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateConfigCmdExec.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.CreateConfigCmd; +import com.github.dockerjava.api.command.CreateConfigResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class CreateConfigCmdExec extends AbstrSyncDockerCmdExec + implements CreateConfigCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CreateConfigCmdExec.class); + + public CreateConfigCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected CreateConfigResponse execute(CreateConfigCmd command) { + WebTarget webResource = getBaseResource().path("/configs/create"); + + LOGGER.trace("POST: {} ", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON) + .post(command, new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateContainerCmdExec.java new file mode 100644 index 000000000..87d2cca81 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateContainerCmdExec.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.CreateContainerCmd; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class CreateContainerCmdExec extends AbstrSyncDockerCmdExec + implements CreateContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CreateContainerCmdExec.class); + + public CreateContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected CreateContainerResponse execute(CreateContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/create"); + + if (command.getName() != null) { + webResource = webResource.queryParam("name", command.getName()); + } + + if (command.getPlatform() != null) { + webResource = webResource.queryParam("platform", command.getPlatform()); + } + + LOGGER.trace("POST: {} ", webResource); + return resourceWithOptionalAuthConfig(command.getAuthConfig(), webResource.request()) + .accept(MediaType.APPLICATION_JSON) + .post(command, new TypeReference() { }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateImageCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateImageCmdExec.java new file mode 100644 index 000000000..b1f4f23c0 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateImageCmdExec.java @@ -0,0 +1,36 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.CreateImageCmd; +import com.github.dockerjava.api.command.CreateImageResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class CreateImageCmdExec extends AbstrSyncDockerCmdExec implements + CreateImageCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CreateImageCmdExec.class); + + public CreateImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected CreateImageResponse execute(CreateImageCmd command) { + WebTarget webResource = getBaseResource().path("/images/create").queryParam("repo", command.getRepository()) + .queryParam("tag", command.getTag()).queryParam("fromSrc", "-"); + + if (command.getPlatform() != null) { + webResource = webResource.queryParam("platform", command.getPlatform()); + } + + LOGGER.trace("POST: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_OCTET_STREAM) + .post(new TypeReference() { + }, command.getImageStream()); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateNetworkCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateNetworkCmdExec.java new file mode 100644 index 000000000..c79767324 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateNetworkCmdExec.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.CreateNetworkCmd; +import com.github.dockerjava.api.command.CreateNetworkResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class CreateNetworkCmdExec extends AbstrSyncDockerCmdExec implements + CreateNetworkCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CreateNetworkCmdExec.class); + + public CreateNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected CreateNetworkResponse execute(CreateNetworkCmd command) { + WebTarget webResource = getBaseResource().path("/networks/create"); + + LOGGER.trace("POST: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON) + .post(command, new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateSecretCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateSecretCmdExec.java new file mode 100644 index 000000000..e2d695527 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateSecretCmdExec.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.CreateSecretCmd; +import com.github.dockerjava.api.command.CreateSecretResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class CreateSecretCmdExec extends AbstrSyncDockerCmdExec + implements CreateSecretCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CreateSecretCmdExec.class); + + public CreateSecretCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected CreateSecretResponse execute(CreateSecretCmd command) { + WebTarget webResource = getBaseResource().path("/secrets/create"); + + LOGGER.trace("POST: {} ", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON) + .post(command.getSecretSpec(), new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateServiceCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateServiceCmdExec.java new file mode 100644 index 000000000..6537aa930 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateServiceCmdExec.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.CreateServiceCmd; +import com.github.dockerjava.api.command.CreateServiceResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.InvocationBuilder; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class CreateServiceCmdExec extends AbstrSyncDockerCmdExec + implements CreateServiceCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CreateServiceCmdExec.class); + + public CreateServiceCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected CreateServiceResponse execute(CreateServiceCmd command) { + WebTarget webResource = getBaseResource().path("/services/create"); + + LOGGER.trace("POST: {} ", webResource); + + InvocationBuilder builder = resourceWithOptionalAuthConfig(command.getAuthConfig(), webResource.request()) + .accept(MediaType.APPLICATION_JSON); + + return builder.post(command.getServiceSpec(), new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateVolumeCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateVolumeCmdExec.java new file mode 100644 index 000000000..8d8c55cc6 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/CreateVolumeCmdExec.java @@ -0,0 +1,31 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.CreateVolumeCmd; +import com.github.dockerjava.api.command.CreateVolumeResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class CreateVolumeCmdExec extends AbstrSyncDockerCmdExec implements + CreateVolumeCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(CreateVolumeCmdExec.class); + + public CreateVolumeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected CreateVolumeResponse execute(CreateVolumeCmd command) { + WebTarget webResource = getBaseResource().path("/volumes/create"); + + LOGGER.trace("POST: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON) + .post(command, new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/DisconnectFromNetworkCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/DisconnectFromNetworkCmdExec.java new file mode 100644 index 000000000..abfb5df1a --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/DisconnectFromNetworkCmdExec.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +public class DisconnectFromNetworkCmdExec extends AbstrSyncDockerCmdExec + implements DisconnectFromNetworkCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(DisconnectFromNetworkCmdExec.class); + + public DisconnectFromNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(DisconnectFromNetworkCmd command) { + + WebTarget webTarget = getBaseResource().path("/networks/" + command.getNetworkId() + "/disconnect"); + + LOGGER.trace("POST: {}", webTarget); + try { + webTarget.request().post(command).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/EventsCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/EventsCmdExec.java new file mode 100644 index 000000000..2ee95c699 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/EventsCmdExec.java @@ -0,0 +1,40 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.EventsCmd; +import com.github.dockerjava.api.model.Event; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.util.FiltersEncoder; +import com.github.dockerjava.core.WebTarget; + +public class EventsCmdExec extends AbstrAsyncDockerCmdExec implements EventsCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(EventsCmdExec.class); + + public EventsCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute0(EventsCmd command, ResultCallback resultCallback) { + + WebTarget webTarget = getBaseResource().path("/events").queryParam("since", command.getSince()) + .queryParam("until", command.getUntil()); + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget + .queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + LOGGER.trace("GET: {}", webTarget); + + webTarget.request().get(new TypeReference() { + }, resultCallback); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ExecCreateCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ExecCreateCmdExec.java new file mode 100644 index 000000000..2f0ca1c6f --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ExecCreateCmdExec.java @@ -0,0 +1,33 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ExecCreateCmd; +import com.github.dockerjava.api.command.ExecCreateCmdResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class ExecCreateCmdExec extends AbstrSyncDockerCmdExec implements + ExecCreateCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ExecCreateCmdExec.class); + + public ExecCreateCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected ExecCreateCmdResponse execute(ExecCreateCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/exec").resolveTemplate("id", + command.getContainerId()); + + LOGGER.trace("POST: {}", webResource); + + return webResource.request().accept(MediaType.APPLICATION_JSON) + .post(command, new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ExecStartCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ExecStartCmdExec.java new file mode 100644 index 000000000..b33383562 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ExecStartCmdExec.java @@ -0,0 +1,24 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.ExecStartCmd; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class ExecStartCmdExec extends AbstrAsyncDockerCmdExec implements ExecStartCmd.Exec { + + public ExecStartCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute0(ExecStartCmd command, ResultCallback resultCallback) { + WebTarget webTarget = getBaseResource().path("/exec/{id}/start").resolveTemplate("id", command.getExecId()); + + webTarget.request().accept(MediaType.APPLICATION_JSON).post(command, command.getStdin(), resultCallback); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InfoCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InfoCmdExec.java new file mode 100644 index 000000000..e6ecb81e9 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InfoCmdExec.java @@ -0,0 +1,28 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.InfoCmd; +import com.github.dockerjava.api.model.Info; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; + +/** + * + * @author Marcus Linke + * + */ +public class InfoCmdExec implements InfoCmd.Exec { + + private WebTarget webResource; + + public InfoCmdExec(WebTarget webResource, DockerClientConfig dockerClientConfig) { + this.webResource = webResource; + } + + @Override + public Info exec(InfoCmd command) { + return webResource.path("info").request().get(new TypeReference() { + }); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InitializeSwarmCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InitializeSwarmCmdExec.java new file mode 100644 index 000000000..ad7df8fd2 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InitializeSwarmCmdExec.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.exec; + + +import com.github.dockerjava.api.command.InitializeSwarmCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +public class InitializeSwarmCmdExec extends AbstrSyncDockerCmdExec + implements InitializeSwarmCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(InitializeSwarmCmdExec.class); + + public InitializeSwarmCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(InitializeSwarmCmd command) { + WebTarget webResource = getBaseResource().path("/swarm/init"); + + LOGGER.trace("POST: {} ", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(command).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectConfigCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectConfigCmdExec.java new file mode 100644 index 000000000..b751c4655 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectConfigCmdExec.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.InspectConfigCmd; +import com.github.dockerjava.api.model.Config; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class InspectConfigCmdExec extends AbstrSyncDockerCmdExec + implements InspectConfigCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(InspectConfigCmdExec.class); + + public InspectConfigCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Config execute(InspectConfigCmd command) { + WebTarget webResource = getBaseResource().path("/configs/{id}") + .resolveTemplate("id", command.getConfigId()); + + LOGGER.debug("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { }); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectContainerCmdExec.java new file mode 100644 index 000000000..f76624be9 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectContainerCmdExec.java @@ -0,0 +1,35 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.InspectContainerCmd; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class InspectContainerCmdExec extends AbstrSyncDockerCmdExec + implements InspectContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(InspectContainerCmdExec.class); + + public InspectContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected InspectContainerResponse execute(InspectContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/json").resolveTemplate("id", + command.getContainerId()); + + webResource = booleanQueryParam(webResource, "size", command.getSize()); + + LOGGER.debug("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON) + .get(new TypeReference() { + }); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectExecCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectExecCmdExec.java new file mode 100644 index 000000000..5235a1624 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectExecCmdExec.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.InspectExecCmd; +import com.github.dockerjava.api.command.InspectExecResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class InspectExecCmdExec extends AbstrSyncDockerCmdExec implements + InspectExecCmd.Exec { + private static final Logger LOGGER = LoggerFactory.getLogger(InspectExecCmdExec.class); + + public InspectExecCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected InspectExecResponse execute(InspectExecCmd command) { + WebTarget webResource = getBaseResource().path("/exec/{id}/json").resolveTemplate("id", command.getExecId()); + + LOGGER.debug("GET: {}", webResource); + + return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectImageCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectImageCmdExec.java new file mode 100644 index 000000000..5cd7cf7df --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectImageCmdExec.java @@ -0,0 +1,32 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.InspectImageCmd; +import com.github.dockerjava.api.command.InspectImageResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class InspectImageCmdExec extends AbstrSyncDockerCmdExec implements + InspectImageCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(InspectImageCmdExec.class); + + public InspectImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected InspectImageResponse execute(InspectImageCmd command) { + WebTarget webResource = getBaseResource().path("/images/{id}/json").resolveTemplate("id", command.getImageId()); + + LOGGER.trace("GET: {}", webResource); + + return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { + }); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectNetworkCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectNetworkCmdExec.java new file mode 100644 index 000000000..196651c28 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectNetworkCmdExec.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.InspectNetworkCmd; +import com.github.dockerjava.api.model.Network; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class InspectNetworkCmdExec extends AbstrSyncDockerCmdExec implements + InspectNetworkCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(InspectNetworkCmdExec.class); + + public InspectNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Network execute(InspectNetworkCmd command) { + WebTarget webResource = getBaseResource().path("/networks/{id}").resolveTemplate("id", command.getNetworkId()); + + LOGGER.trace("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectServiceCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectServiceCmdExec.java new file mode 100644 index 000000000..c7cf9d011 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectServiceCmdExec.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.InspectServiceCmd; +import com.github.dockerjava.api.model.Service; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class InspectServiceCmdExec extends AbstrSyncDockerCmdExec + implements InspectServiceCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(InspectServiceCmdExec.class); + + public InspectServiceCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Service execute(InspectServiceCmd command) { + WebTarget webResource = getBaseResource().path("/services/{id}") + .resolveTemplate("id", command.getServiceId()); + + LOGGER.debug("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { }); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectSwarmCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectSwarmCmdExec.java new file mode 100644 index 000000000..41f62c7ff --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectSwarmCmdExec.java @@ -0,0 +1,31 @@ +package com.github.dockerjava.core.exec; + + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.InspectSwarmCmd; +import com.github.dockerjava.api.model.Swarm; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class InspectSwarmCmdExec extends AbstrSyncDockerCmdExec + implements InspectSwarmCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(InspectSwarmCmdExec.class); + + public InspectSwarmCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Swarm execute(InspectSwarmCmd command) { + WebTarget webResource = getBaseResource().path("/swarm"); + + LOGGER.debug("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON) + .get(new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectSwarmNodeCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectSwarmNodeCmdExec.java new file mode 100644 index 000000000..508e43b05 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectSwarmNodeCmdExec.java @@ -0,0 +1,32 @@ +package com.github.dockerjava.core.exec; + + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.InspectSwarmNodeCmd; +import com.github.dockerjava.api.model.SwarmNode; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class InspectSwarmNodeCmdExec extends AbstrSyncDockerCmdExec + implements InspectSwarmNodeCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(InspectSwarmNodeCmdExec.class); + + public InspectSwarmNodeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected SwarmNode execute(InspectSwarmNodeCmd command) { + WebTarget webResource = getBaseResource().path("/nodes/{id}").resolveTemplate("id", + command.getSwarmNodeId()); + + LOGGER.debug("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON) + .get(new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectVolumeCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectVolumeCmdExec.java new file mode 100644 index 000000000..5d1e84882 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/InspectVolumeCmdExec.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.InspectVolumeCmd; +import com.github.dockerjava.api.command.InspectVolumeResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class InspectVolumeCmdExec extends AbstrSyncDockerCmdExec implements + InspectVolumeCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(InspectVolumeCmdExec.class); + + public InspectVolumeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected InspectVolumeResponse execute(InspectVolumeCmd command) { + WebTarget webResource = getBaseResource().path("/volumes/{name}").resolveTemplate("name", command.getName()); + + LOGGER.trace("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/JoinSwarmCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/JoinSwarmCmdExec.java new file mode 100644 index 000000000..29070d1b6 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/JoinSwarmCmdExec.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.exec; + + +import com.github.dockerjava.api.command.JoinSwarmCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +public class JoinSwarmCmdExec extends AbstrSyncDockerCmdExec + implements JoinSwarmCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(InitializeSwarmCmdExec.class); + + public JoinSwarmCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(JoinSwarmCmd command) { + WebTarget webResource = getBaseResource().path("/swarm/join"); + + LOGGER.trace("POST: {} ", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(command).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/KillContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/KillContainerCmdExec.java new file mode 100644 index 000000000..c98b9f0f8 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/KillContainerCmdExec.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.KillContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +import java.io.IOException; + +public class KillContainerCmdExec extends AbstrSyncDockerCmdExec implements + KillContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(KillContainerCmdExec.class); + + public KillContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(KillContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/kill").resolveTemplate("id", + command.getContainerId()); + + if (command.getSignal() != null) { + webResource = webResource.queryParam("signal", command.getSignal()); + } + + LOGGER.trace("POST: {}", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LeaveSwarmCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LeaveSwarmCmdExec.java new file mode 100644 index 000000000..fee9cea7a --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LeaveSwarmCmdExec.java @@ -0,0 +1,35 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.LeaveSwarmCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +public class LeaveSwarmCmdExec extends AbstrSyncDockerCmdExec implements LeaveSwarmCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(LeaveSwarmCmdExec.class); + + public LeaveSwarmCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(LeaveSwarmCmd command) { + WebTarget webTarget = getBaseResource().path("/swarm/leave"); + + webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); + + LOGGER.trace("POST: {}", webTarget); + try { + webTarget.request().accept(MediaType.APPLICATION_JSON).post(null).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListConfigsCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListConfigsCmdExec.java new file mode 100644 index 000000000..89a1b83b1 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListConfigsCmdExec.java @@ -0,0 +1,43 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ListConfigsCmd; +import com.github.dockerjava.api.model.Config; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import com.github.dockerjava.core.util.FiltersEncoder; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +public class ListConfigsCmdExec extends AbstrSyncDockerCmdExec> implements ListConfigsCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ListConfigsCmdExec.class); + + public ListConfigsCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected List execute(ListConfigsCmd command) { + WebTarget webTarget = getBaseResource().path("/configs"); + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget + .queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + LOGGER.trace("GET: {}", webTarget); + + List configs = webTarget.request().accept(MediaType.APPLICATION_JSON) + .get(new TypeReference>() { + }); + + LOGGER.trace("Response: {}", configs); + + return configs; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListContainersCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListContainersCmdExec.java new file mode 100644 index 000000000..d4fb3cee0 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListContainersCmdExec.java @@ -0,0 +1,52 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ListContainersCmd; +import com.github.dockerjava.api.model.Container; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.util.FiltersEncoder; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +public class ListContainersCmdExec extends AbstrSyncDockerCmdExec> implements + ListContainersCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ListContainersCmdExec.class); + + public ListContainersCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected List execute(ListContainersCmd command) { + WebTarget webTarget = getBaseResource().path("/containers/json").queryParam("since", command.getSinceId()) + .queryParam("before", command.getBeforeId()); + + webTarget = booleanQueryParam(webTarget, "all", command.hasShowAllEnabled()); + webTarget = booleanQueryParam(webTarget, "size", command.hasShowSizeEnabled()); + + if (command.getLimit() != null && command.getLimit() >= 0) { + webTarget = webTarget.queryParam("limit", String.valueOf(command.getLimit())); + } + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget + .queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + LOGGER.trace("GET: {}", webTarget); + + List containers = webTarget.request().accept(MediaType.APPLICATION_JSON) + .get(new TypeReference>() { + }); + + LOGGER.trace("Response: {}", containers); + + return containers; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListImagesCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListImagesCmdExec.java new file mode 100644 index 000000000..9f53e76bd --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListImagesCmdExec.java @@ -0,0 +1,48 @@ +package com.github.dockerjava.core.exec; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ListImagesCmd; +import com.github.dockerjava.api.model.Image; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.util.FiltersEncoder; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class ListImagesCmdExec extends AbstrSyncDockerCmdExec> implements ListImagesCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ListImagesCmdExec.class); + + public ListImagesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected List execute(ListImagesCmd command) { + WebTarget webTarget = getBaseResource().path("/images/json"); + + webTarget = booleanQueryParam(webTarget, "all", command.hasShowAllEnabled()); + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget.queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + if (command.getImageNameFilter() != null) { + webTarget = webTarget.queryParam("filter", command.getImageNameFilter()); + } + + LOGGER.trace("GET: {}", webTarget); + + List images = webTarget.request().accept(MediaType.APPLICATION_JSON) + .get(new TypeReference>() { + }); + + LOGGER.trace("Response: {}", images); + + return images; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListNetworksCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListNetworksCmdExec.java new file mode 100644 index 000000000..ab47a5b78 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListNetworksCmdExec.java @@ -0,0 +1,38 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ListNetworksCmd; +import com.github.dockerjava.api.model.Network; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.util.FiltersEncoder; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +public class ListNetworksCmdExec extends AbstrSyncDockerCmdExec> implements + ListNetworksCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ListNetworksCmdExec.class); + + public ListNetworksCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected List execute(ListNetworksCmd command) { + WebTarget webTarget = getBaseResource().path("/networks"); + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget.queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + LOGGER.trace("GET: {}", webTarget); + + return webTarget.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference>() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListSecretsCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListSecretsCmdExec.java new file mode 100644 index 000000000..61a931057 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListSecretsCmdExec.java @@ -0,0 +1,44 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ListSecretsCmd; +import com.github.dockerjava.api.model.Secret; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import com.github.dockerjava.core.util.FiltersEncoder; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +public class ListSecretsCmdExec extends AbstrSyncDockerCmdExec> implements + ListSecretsCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ListSecretsCmdExec.class); + + public ListSecretsCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected List execute(ListSecretsCmd command) { + WebTarget webTarget = getBaseResource().path("/secrets"); + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget + .queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + LOGGER.trace("GET: {}", webTarget); + + List secrets = webTarget.request().accept(MediaType.APPLICATION_JSON) + .get(new TypeReference>() { + }); + + LOGGER.trace("Response: {}", secrets); + + return secrets; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListServicesCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListServicesCmdExec.java new file mode 100644 index 000000000..389d0cbad --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListServicesCmdExec.java @@ -0,0 +1,44 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ListServicesCmd; +import com.github.dockerjava.api.model.Service; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.util.FiltersEncoder; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +public class ListServicesCmdExec extends AbstrSyncDockerCmdExec> implements + ListServicesCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ListServicesCmdExec.class); + + public ListServicesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected List execute(ListServicesCmd command) { + WebTarget webTarget = getBaseResource().path("/services"); + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget + .queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + LOGGER.trace("GET: {}", webTarget); + + List services = webTarget.request().accept(MediaType.APPLICATION_JSON) + .get(new TypeReference>() { + }); + + LOGGER.trace("Response: {}", services); + + return services; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListSwarmNodesCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListSwarmNodesCmdExec.java new file mode 100644 index 000000000..1d6fe861f --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListSwarmNodesCmdExec.java @@ -0,0 +1,44 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ListSwarmNodesCmd; +import com.github.dockerjava.api.model.SwarmNode; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.util.FiltersEncoder; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +public class ListSwarmNodesCmdExec extends AbstrSyncDockerCmdExec> implements + ListSwarmNodesCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ListSwarmNodesCmdExec.class); + + public ListSwarmNodesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + + @Override + protected List execute(ListSwarmNodesCmd command) { + WebTarget webTarget = getBaseResource().path("/nodes"); + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget + .queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + LOGGER.trace("GET: {}", webTarget); + + List nodes = webTarget.request().accept(MediaType.APPLICATION_JSON) + .get(new TypeReference>() { + }); + + LOGGER.trace("Response: {}", nodes); + + return nodes; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListTasksCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListTasksCmdExec.java new file mode 100644 index 000000000..659449078 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListTasksCmdExec.java @@ -0,0 +1,44 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ListTasksCmd; +import com.github.dockerjava.api.model.Task; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.util.FiltersEncoder; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +public class ListTasksCmdExec extends AbstrSyncDockerCmdExec> implements + ListTasksCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ListTasksCmdExec.class); + + public ListTasksCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected List execute(ListTasksCmd command) { + WebTarget webTarget = getBaseResource().path("/tasks"); + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget + .queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + LOGGER.trace("GET: {}", webTarget); + + List tasks = webTarget.request().accept(MediaType.APPLICATION_JSON) + .get(new TypeReference>() { + }); + + LOGGER.trace("Response: {}", tasks); + + return tasks; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListVolumesCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListVolumesCmdExec.java new file mode 100644 index 000000000..fb287ca0a --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ListVolumesCmdExec.java @@ -0,0 +1,36 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.ListVolumesCmd; +import com.github.dockerjava.api.command.ListVolumesResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.util.FiltersEncoder; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class ListVolumesCmdExec extends AbstrSyncDockerCmdExec implements + ListVolumesCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ListVolumesCmdExec.class); + + public ListVolumesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected ListVolumesResponse execute(ListVolumesCmd command) { + WebTarget webTarget = getBaseResource().path("/volumes"); + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget.queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + LOGGER.trace("GET: {}", webTarget); + + return webTarget.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LoadImageAsyncCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LoadImageAsyncCmdExec.java new file mode 100644 index 000000000..47f1d52fc --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LoadImageAsyncCmdExec.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.LoadImageAsyncCmd; +import com.github.dockerjava.api.model.LoadResponseItem; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class LoadImageAsyncCmdExec extends AbstrAsyncDockerCmdExec implements LoadImageAsyncCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(LoadImageAsyncCmdExec.class); + + public LoadImageAsyncCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute0(LoadImageAsyncCmd command, ResultCallback resultCallback) { + WebTarget webTarget = getBaseResource().path("/images/load"); + + LOGGER.trace("POST: {}", webTarget); + + webTarget.request().post(new TypeReference() { }, resultCallback, command.getImageStream()); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LoadImageCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LoadImageCmdExec.java new file mode 100644 index 000000000..dd906d953 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LoadImageCmdExec.java @@ -0,0 +1,28 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.LoadImageCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; + +public class LoadImageCmdExec extends AbstrSyncDockerCmdExec implements + LoadImageCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(LoadImageCmdExec.class); + + public LoadImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(LoadImageCmd command) { + WebTarget webResource = getBaseResource().path("/images/load"); + + LOGGER.trace("POST: {}", webResource); + webResource.request().postStream(command.getImageStream()); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LogContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LogContainerCmdExec.java new file mode 100644 index 000000000..357af6d0e --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LogContainerCmdExec.java @@ -0,0 +1,50 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.LogContainerCmd; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; + +public class LogContainerCmdExec extends AbstrAsyncDockerCmdExec implements + LogContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(LogContainerCmdExec.class); + + public LogContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute0(LogContainerCmd command, ResultCallback resultCallback) { + + WebTarget webTarget = getBaseResource().path("/containers/{id}/logs").resolveTemplate("id", + command.getContainerId()); + + if (command.getTail() != null) { + webTarget = webTarget.queryParam("tail", command.getTail()); + } + + if (command.getSince() != null) { + webTarget = webTarget.queryParam("since", command.getSince()); + } + + if (command.getUntil() != null) { + webTarget = webTarget.queryParam("until", command.getUntil()); + } + + webTarget = booleanQueryParam(webTarget, "timestamps", command.hasTimestampsEnabled()); + webTarget = booleanQueryParam(webTarget, "stdout", command.hasStdoutEnabled()); + webTarget = booleanQueryParam(webTarget, "stderr", command.hasStderrEnabled()); + webTarget = booleanQueryParam(webTarget, "follow", command.hasFollowStreamEnabled()); + + LOGGER.trace("GET: {}", webTarget); + + webTarget.request().get(resultCallback); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LogSwarmObjectExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LogSwarmObjectExec.java new file mode 100644 index 000000000..e545ddbc1 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/LogSwarmObjectExec.java @@ -0,0 +1,48 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.LogSwarmObjectCmd; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.core.WebTarget; + +import com.github.dockerjava.core.DockerClientConfig; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class LogSwarmObjectExec extends AbstrAsyncDockerCmdExec implements + LogSwarmObjectCmd.Exec { + private String endpoint = ""; + private static final Logger LOGGER = LoggerFactory.getLogger(LogSwarmObjectExec.class); + + public LogSwarmObjectExec(com.github.dockerjava.core.WebTarget baseResource, DockerClientConfig dockerClientConfig, String endpoint) { + super(baseResource, dockerClientConfig); + this.endpoint = endpoint; + } + + @Override + protected Void execute0(LogSwarmObjectCmd command, ResultCallback resultCallback) { + + WebTarget webTarget = getBaseResource().path("/" + endpoint + "/{id}/logs").resolveTemplate("id", command.getId()); + + if (command.getTail() != null) { + webTarget = webTarget.queryParam("tail", command.getTail()); + } else { + webTarget = webTarget.queryParam("tail", "all"); + } + + if (command.getSince() != null) { + webTarget = webTarget.queryParam("since", command.getSince()); + } + + webTarget = booleanQueryParam(webTarget, "timestamps", command.getTimestamps()); + webTarget = booleanQueryParam(webTarget, "stdout", command.getStdout()); + webTarget = booleanQueryParam(webTarget, "stderr", command.getStderr()); + webTarget = booleanQueryParam(webTarget, "follow", command.getFollow()); + + LOGGER.trace("GET: {}", webTarget); + + webTarget.request().get(resultCallback); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PauseContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PauseContainerCmdExec.java new file mode 100644 index 000000000..a24b45a9b --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PauseContainerCmdExec.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.PauseContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +import java.io.IOException; + +public class PauseContainerCmdExec extends AbstrSyncDockerCmdExec implements + PauseContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(PauseContainerCmdExec.class); + + public PauseContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(PauseContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/pause").resolveTemplate("id", + command.getContainerId()); + + LOGGER.trace("POST: {}", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PingCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PingCmdExec.java new file mode 100644 index 000000000..bbb078cf7 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PingCmdExec.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.PingCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; + +import java.io.IOException; + +public class PingCmdExec extends AbstrSyncDockerCmdExec implements PingCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(PingCmdExec.class); + + public PingCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(PingCmd command) { + WebTarget webResource = getBaseResource().path("/_ping"); + + LOGGER.trace("GET: {}", webResource); + try { + webResource.request().get().close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PruneCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PruneCmdExec.java new file mode 100644 index 000000000..0b8832bcb --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PruneCmdExec.java @@ -0,0 +1,40 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.PruneCmd; +import com.github.dockerjava.api.model.PruneResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import com.github.dockerjava.core.util.FiltersEncoder; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class PruneCmdExec extends AbstrSyncDockerCmdExec implements PruneCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(PruneCmdExec.class); + + + public PruneCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected PruneResponse execute(PruneCmd command) { + WebTarget webTarget = getBaseResource().path(command.getApiPath()); + + if (command.getFilters() != null && !command.getFilters().isEmpty()) { + webTarget = webTarget.queryParam("filters", FiltersEncoder.jsonEncode(command.getFilters())); + } + + LOGGER.trace("POST: {}", webTarget); + + PruneResponse response = webTarget.request().accept(MediaType.APPLICATION_JSON) + .post(null, new TypeReference() { }); + + LOGGER.trace("Response: {}", response); + + return response; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PullImageCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PullImageCmdExec.java new file mode 100644 index 000000000..1ba0fd8c2 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PullImageCmdExec.java @@ -0,0 +1,40 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.PullImageCmd; +import com.github.dockerjava.api.model.PullResponseItem; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class PullImageCmdExec extends AbstrAsyncDockerCmdExec implements + PullImageCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(PullImageCmdExec.class); + + public PullImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute0(PullImageCmd command, ResultCallback resultCallback) { + + WebTarget webResource = getBaseResource().path("/images/create").queryParam("tag", command.getTag()) + .queryParam("fromImage", command.getRepository()).queryParam("registry", command.getRegistry()); + + if (command.getPlatform() != null) { + webResource = webResource.queryParam("platform", command.getPlatform()); + } + + LOGGER.trace("POST: {}", webResource); + resourceWithOptionalAuthConfig(command.getAuthConfig(), webResource.request()) + .accept(MediaType.APPLICATION_OCTET_STREAM) + .post(null, new TypeReference() { + }, resultCallback); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PushImageCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PushImageCmdExec.java new file mode 100644 index 000000000..4f4540891 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/PushImageCmdExec.java @@ -0,0 +1,40 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.PushImageCmd; +import com.github.dockerjava.api.model.PushResponseItem; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.InvocationBuilder; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class PushImageCmdExec extends AbstrAsyncDockerCmdExec implements + PushImageCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(PushImageCmdExec.class); + + public PushImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute0(PushImageCmd command, ResultCallback resultCallback) { + WebTarget webResource = getBaseResource().path("/images/{imageName}/push") + .resolveTemplate("imageName", command.getName()) + .queryParam("tag", command.getTag()); + + LOGGER.trace("POST: {}", webResource); + + InvocationBuilder builder = resourceWithAuthConfig(command.getAuthConfig(), webResource.request()) + .accept(MediaType.APPLICATION_JSON); + + builder.post(null, new TypeReference() { + }, resultCallback); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveConfigCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveConfigCmdExec.java new file mode 100644 index 000000000..1b81ef644 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveConfigCmdExec.java @@ -0,0 +1,28 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.RemoveConfigCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class RemoveConfigCmdExec extends AbstrSyncDockerCmdExec implements RemoveConfigCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(RemoveConfigCmdExec.class); + + public RemoveConfigCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(RemoveConfigCmd command) { + WebTarget webTarget = getBaseResource().path("/configs/" + command.getConfigId()); + + LOGGER.trace("DELETE: {}", webTarget); + webTarget.request().accept(MediaType.APPLICATION_JSON).delete(); + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveContainerCmdExec.java new file mode 100644 index 000000000..5408071d1 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveContainerCmdExec.java @@ -0,0 +1,33 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.RemoveContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class RemoveContainerCmdExec extends AbstrSyncDockerCmdExec implements + RemoveContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(RemoveContainerCmdExec.class); + + public RemoveContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(RemoveContainerCmd command) { + WebTarget webTarget = getBaseResource().path("/containers/" + command.getContainerId()); + + webTarget = booleanQueryParam(webTarget, "v", command.hasRemoveVolumesEnabled()); + webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); + + LOGGER.trace("DELETE: {}", webTarget); + webTarget.request().accept(MediaType.APPLICATION_JSON).delete(); + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveImageCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveImageCmdExec.java new file mode 100644 index 000000000..205c8670d --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveImageCmdExec.java @@ -0,0 +1,31 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.RemoveImageCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; + +public class RemoveImageCmdExec extends AbstrSyncDockerCmdExec implements RemoveImageCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(RemoveImageCmdExec.class); + + public RemoveImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(RemoveImageCmd command) { + WebTarget webTarget = getBaseResource().path("/images/" + command.getImageId()); + + webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); + webTarget = booleanQueryParam(webTarget, "noprune", command.hasNoPruneEnabled()); + + LOGGER.trace("DELETE: {}", webTarget); + webTarget.request().delete(); + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveNetworkCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveNetworkCmdExec.java new file mode 100644 index 000000000..40da76774 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveNetworkCmdExec.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.RemoveNetworkCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class RemoveNetworkCmdExec extends AbstrSyncDockerCmdExec implements + RemoveNetworkCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(RemoveNetworkCmdExec.class); + + public RemoveNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(RemoveNetworkCmd command) { + WebTarget webTarget = getBaseResource().path("/networks/" + command.getNetworkId()); + + LOGGER.trace("DELETE: {}", webTarget); + webTarget.request().accept(MediaType.APPLICATION_JSON).delete(); + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveSecretCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveSecretCmdExec.java new file mode 100644 index 000000000..b9332b27b --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveSecretCmdExec.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.RemoveSecretCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class RemoveSecretCmdExec extends AbstrSyncDockerCmdExec implements + RemoveSecretCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(RemoveSecretCmdExec.class); + + public RemoveSecretCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(RemoveSecretCmd command) { + WebTarget webTarget = getBaseResource().path("/secrets/" + command.getSecretId()); + + LOGGER.trace("DELETE: {}", webTarget); + webTarget.request().accept(MediaType.APPLICATION_JSON).delete(); + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveServiceCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveServiceCmdExec.java new file mode 100644 index 000000000..648caf28f --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveServiceCmdExec.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.RemoveServiceCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class RemoveServiceCmdExec extends AbstrSyncDockerCmdExec implements + RemoveServiceCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(RemoveServiceCmdExec.class); + + public RemoveServiceCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(RemoveServiceCmd command) { + WebTarget webTarget = getBaseResource().path("/services/" + command.getServiceId()); + + LOGGER.trace("DELETE: {}", webTarget); + webTarget.request().accept(MediaType.APPLICATION_JSON).delete(); + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveSwarmNodeCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveSwarmNodeCmdExec.java new file mode 100644 index 000000000..a58954f84 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveSwarmNodeCmdExec.java @@ -0,0 +1,31 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.RemoveSwarmNodeCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + + +public class RemoveSwarmNodeCmdExec extends AbstrSyncDockerCmdExec implements + RemoveSwarmNodeCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(RemoveSwarmNodeCmdExec.class); + + public RemoveSwarmNodeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(RemoveSwarmNodeCmd command) { + WebTarget webTarget = getBaseResource().path("/nodes/" + command.getSwarmNodeId()); + + webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); + + LOGGER.trace("DELETE: {}", webTarget); + webTarget.request().accept(MediaType.APPLICATION_JSON).delete(); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveVolumeCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveVolumeCmdExec.java new file mode 100644 index 000000000..482401b07 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RemoveVolumeCmdExec.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.RemoveVolumeCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class RemoveVolumeCmdExec extends AbstrSyncDockerCmdExec implements + RemoveVolumeCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(RemoveVolumeCmdExec.class); + + public RemoveVolumeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(RemoveVolumeCmd command) { + WebTarget webTarget = getBaseResource().path("/volumes/" + command.getName()); + + LOGGER.trace("DELETE: {}", webTarget); + webTarget.request().accept(MediaType.APPLICATION_JSON).delete(); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RenameContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RenameContainerCmdExec.java new file mode 100644 index 000000000..fdb312f1f --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RenameContainerCmdExec.java @@ -0,0 +1,35 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.RenameContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +public class RenameContainerCmdExec extends AbstrSyncDockerCmdExec + implements RenameContainerCmd.Exec { + private static final Logger LOG = LoggerFactory.getLogger(RenameContainerCmdExec.class); + + public RenameContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(RenameContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/rename") + .resolveTemplate("id", command.getContainerId()) + .queryParam("name", command.getName()); + + LOG.trace("POST: {}", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ResizeContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ResizeContainerCmdExec.java new file mode 100644 index 000000000..4913bde79 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ResizeContainerCmdExec.java @@ -0,0 +1,36 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.ResizeContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +public class ResizeContainerCmdExec extends AbstrSyncDockerCmdExec implements ResizeContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ResizeContainerCmdExec.class); + + public ResizeContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(ResizeContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/resize") + .resolveTemplate("id", command.getContainerId()).queryParam("h", command.getHeight()) + .queryParam("w", command.getWidth()); + + LOGGER.trace("POST: {}", webResource); + + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(command).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ResizeExecCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ResizeExecCmdExec.java new file mode 100644 index 000000000..e799a95d5 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/ResizeExecCmdExec.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.ResizeExecCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + + +public class ResizeExecCmdExec extends AbstrSyncDockerCmdExec implements ResizeExecCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(ResizeExecCmdExec.class); + + public ResizeExecCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(ResizeExecCmd command) { + WebTarget webResource = getBaseResource().path("/exec/{id}/resize") + .resolveTemplate("id", command.getExecId()).queryParam("h", command.getHeight()).queryParam("w", command.getWidth()); + + LOGGER.trace("POST: {}", webResource); + + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RestartContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RestartContainerCmdExec.java new file mode 100644 index 000000000..42f2579e6 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/RestartContainerCmdExec.java @@ -0,0 +1,45 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.RestartContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +import java.io.IOException; + +public class RestartContainerCmdExec extends AbstrSyncDockerCmdExec implements + RestartContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(RestartContainerCmdExec.class); + + public RestartContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(RestartContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/restart").resolveTemplate("id", + command.getContainerId()); + + if (command.getSignal() != null) { + webResource = webResource.queryParam("signal", command.getSignal()); + } + + if (command.getTimeout() != null) { + webResource = webResource.queryParam("t", String.valueOf(command.getTimeout())); + } + + LOGGER.trace("POST: {}", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/SaveImageCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/SaveImageCmdExec.java new file mode 100644 index 000000000..94001bd5c --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/SaveImageCmdExec.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.SaveImageCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import com.google.common.base.Strings; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.InputStream; + +public class SaveImageCmdExec extends AbstrSyncDockerCmdExec implements SaveImageCmd.Exec { + private static final Logger LOGGER = LoggerFactory.getLogger(SaveImageCmdExec.class); + + public SaveImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected InputStream execute(SaveImageCmd command) { + + String name = command.getName(); + if (!Strings.isNullOrEmpty(command.getTag())) { + name += ":" + command.getTag(); + } + + WebTarget webResource = getBaseResource(). + path("/images/" + name + "/get"); + + LOGGER.trace("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON).get(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/SaveImagesCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/SaveImagesCmdExec.java new file mode 100644 index 000000000..a1bb47c05 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/SaveImagesCmdExec.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.SaveImagesCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import com.google.common.collect.ImmutableSet; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.InputStream; +import java.util.List; + +public class SaveImagesCmdExec extends AbstrSyncDockerCmdExec implements SaveImagesCmd.Exec { + private static final Logger LOGGER = LoggerFactory.getLogger(SaveImagesCmdExec.class); + + public SaveImagesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected InputStream execute(SaveImagesCmd command) { + + final List images = command.getImages(); + if (images.isEmpty()) { + LOGGER.warn("No images specified for " + SaveImagesCmd.class.getName() + "."); + } + final ImmutableSet.Builder queryParamSet = ImmutableSet.builder(); + for (SaveImagesCmd.TaggedImage image : images) { + queryParamSet.add(image.asString()); + } + final WebTarget webResource = getBaseResource() + .path("/images/get") + .queryParamsSet("names", queryParamSet.build()); + + LOGGER.trace("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON).get(); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/SearchImagesCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/SearchImagesCmdExec.java new file mode 100644 index 000000000..0e92c9f97 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/SearchImagesCmdExec.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.core.exec; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.SearchImagesCmd; +import com.github.dockerjava.api.model.SearchItem; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class SearchImagesCmdExec extends AbstrSyncDockerCmdExec> implements + SearchImagesCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(SearchImagesCmdExec.class); + + public SearchImagesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected List execute(SearchImagesCmd command) { + WebTarget webResource = getBaseResource().path("/images/search") + .queryParam("term", command.getTerm()); + + if (command.getLimit() != null) { + webResource = webResource.queryParam("limit", command.getLimit()); + } + + LOGGER.trace("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference>() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/StartContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/StartContainerCmdExec.java new file mode 100644 index 000000000..774a85c3d --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/StartContainerCmdExec.java @@ -0,0 +1,40 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.StartContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +import java.io.IOException; + +public class StartContainerCmdExec extends AbstrSyncDockerCmdExec implements + StartContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(StartContainerCmdExec.class); + + public StartContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(StartContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/start").resolveTemplate("id", + command.getContainerId()); + + LOGGER.trace("POST: {}", webResource); + try { + webResource.request() + .accept(MediaType.APPLICATION_JSON) + .post(null) + .close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/StatsCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/StatsCmdExec.java new file mode 100644 index 000000000..6668ace04 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/StatsCmdExec.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.StatsCmd; +import com.github.dockerjava.api.model.Statistics; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; + +public class StatsCmdExec extends AbstrAsyncDockerCmdExec implements StatsCmd.Exec { + private static final Logger LOGGER = LoggerFactory.getLogger(StatsCmdExec.class); + + public StatsCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute0(StatsCmd command, ResultCallback resultCallback) { + + WebTarget webTarget = getBaseResource().path("/containers/{id}/stats").resolveTemplate("id", + command.getContainerId()); + + if (Boolean.TRUE.equals(command.hasNoStream())) { + webTarget = webTarget.queryParam("stream", "0"); + } + + LOGGER.trace("GET: {}", webTarget); + + webTarget.request().get(new TypeReference() { + }, resultCallback); + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/StopContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/StopContainerCmdExec.java new file mode 100644 index 000000000..85b139e93 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/StopContainerCmdExec.java @@ -0,0 +1,40 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.StopContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +import java.io.IOException; + +public class StopContainerCmdExec extends AbstrSyncDockerCmdExec implements + StopContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(StopContainerCmdExec.class); + + public StopContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(StopContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/stop").resolveTemplate("id", + command.getContainerId()); + + if (command.getTimeout() != null) { + webResource = webResource.queryParam("t", String.valueOf(command.getTimeout())); + } + + LOGGER.trace("POST: {}", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/TagImageCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/TagImageCmdExec.java new file mode 100644 index 000000000..05ed13859 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/TagImageCmdExec.java @@ -0,0 +1,36 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.TagImageCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; + +import java.io.IOException; + +public class TagImageCmdExec extends AbstrSyncDockerCmdExec implements TagImageCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(TagImageCmdExec.class); + + public TagImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(TagImageCmd command) { + WebTarget webTarget = getBaseResource().path("/images/" + command.getImageId() + "/tag") + .queryParam("repo", command.getRepository()).queryParam("tag", command.getTag()); + + webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); + + LOGGER.trace("POST: {}", webTarget); + try { + webTarget.request().post(null).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/TopContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/TopContainerCmdExec.java new file mode 100644 index 000000000..f0ce0f71a --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/TopContainerCmdExec.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.core.exec; + +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.TopContainerCmd; +import com.github.dockerjava.api.command.TopContainerResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class TopContainerCmdExec extends AbstrSyncDockerCmdExec implements + TopContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(TopContainerCmdExec.class); + + public TopContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected TopContainerResponse execute(TopContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/top").resolveTemplate("id", + command.getContainerId()); + + if (!StringUtils.isEmpty(command.getPsArgs())) { + webResource = webResource.queryParam("ps_args", command.getPsArgs()); + } + + LOGGER.trace("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { + }); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UnpauseContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UnpauseContainerCmdExec.java new file mode 100644 index 000000000..af23d9cb4 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UnpauseContainerCmdExec.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.dockerjava.api.command.UnpauseContainerCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +import java.io.IOException; + +public class UnpauseContainerCmdExec extends AbstrSyncDockerCmdExec implements + UnpauseContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(UnpauseContainerCmdExec.class); + + public UnpauseContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(UnpauseContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/unpause").resolveTemplate("id", + command.getContainerId()); + + LOGGER.trace("POST: {}", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateContainerCmdExec.java new file mode 100644 index 000000000..411430d56 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateContainerCmdExec.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.core.exec; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.UpdateContainerCmd; +import com.github.dockerjava.api.model.UpdateContainerResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + + +/** + * @author Kanstantsin Shautsou + */ +public class UpdateContainerCmdExec extends AbstrSyncDockerCmdExec + implements UpdateContainerCmd.Exec { + private static final Logger LOGGER = LoggerFactory.getLogger(UpdateContainerCmdExec.class); + + public UpdateContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected UpdateContainerResponse execute(UpdateContainerCmd command) { + WebTarget webResource = getBaseResource().path("/containers/{id}/update") + .resolveTemplate("id", command.getContainerId()); + + LOGGER.trace("POST: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON) + .post(command, new TypeReference() { + }); + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateServiceCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateServiceCmdExec.java new file mode 100644 index 000000000..61eb8b271 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateServiceCmdExec.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.command.UpdateServiceCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +/** + * Update service settings. + */ +public class UpdateServiceCmdExec extends AbstrSyncDockerCmdExec + implements UpdateServiceCmd.Exec { + private static final Logger LOGGER = LoggerFactory.getLogger(UpdateServiceCmdExec.class); + + public UpdateServiceCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(UpdateServiceCmd command) { + WebTarget webResource = getBaseResource().path("/services/{id}/update") + .resolveTemplate("id", command.getServiceId()) + .queryParam("version", command.getVersion()); + + LOGGER.trace("POST: {}", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(command.getServiceSpec()).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateSwarmCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateSwarmCmdExec.java new file mode 100644 index 000000000..59b912dce --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateSwarmCmdExec.java @@ -0,0 +1,37 @@ +package com.github.dockerjava.core.exec; + + +import com.github.dockerjava.api.command.UpdateSwarmCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +public class UpdateSwarmCmdExec extends AbstrSyncDockerCmdExec + implements UpdateSwarmCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(UpdateSwarmCmdExec.class); + + public UpdateSwarmCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(UpdateSwarmCmd command) { + WebTarget webResource = getBaseResource().path("/swarm/update") + .queryParam("version", command.getVersion()); + webResource = booleanQueryParam(webResource, "rotateManagerToken", command.getRotateManagerToken()); + webResource = booleanQueryParam(webResource, "rotateWorkertoken", command.getRotateWorkerToken()); + + LOGGER.trace("POST: {} ", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(command.getSwarmSpec()).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateSwarmNodeCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateSwarmNodeCmdExec.java new file mode 100644 index 000000000..b050ba2bd --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/UpdateSwarmNodeCmdExec.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.core.exec; + + +import com.github.dockerjava.api.command.UpdateSwarmNodeCmd; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +/** + * Update swarmNode spec + */ +public class UpdateSwarmNodeCmdExec extends AbstrSyncDockerCmdExec + implements UpdateSwarmNodeCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(UpdateSwarmNodeCmdExec.class); + + public UpdateSwarmNodeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute(UpdateSwarmNodeCmd command) { + WebTarget webResource = getBaseResource().path("/nodes/{id}/update") + .resolveTemplate("id", command.getSwarmNodeId()) + .queryParam("version", command.getVersion()); + + LOGGER.trace("POST: {}", webResource); + try { + webResource.request().accept(MediaType.APPLICATION_JSON).post(command.getSwarmNodeSpec()).close(); + } catch (IOException e) { + throw new RuntimeException(e); + } + return null; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/VersionCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/VersionCmdExec.java new file mode 100644 index 000000000..c673d284d --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/VersionCmdExec.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.exec; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.command.VersionCmd; +import com.github.dockerjava.api.model.Version; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class VersionCmdExec extends AbstrSyncDockerCmdExec implements VersionCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(VersionCmdExec.class); + + public VersionCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Version execute(VersionCmd command) { + WebTarget webResource = getBaseResource().path("/version"); + + LOGGER.trace("GET: {}", webResource); + return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { + }); + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/exec/WaitContainerCmdExec.java b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/WaitContainerCmdExec.java new file mode 100644 index 000000000..3d2b2fa8e --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/exec/WaitContainerCmdExec.java @@ -0,0 +1,42 @@ +package com.github.dockerjava.core.exec; + +import com.github.dockerjava.api.model.WaitContainerCondition; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.WaitContainerCmd; +import com.github.dockerjava.api.model.WaitResponse; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.MediaType; +import com.github.dockerjava.core.WebTarget; + +public class WaitContainerCmdExec extends AbstrAsyncDockerCmdExec implements + WaitContainerCmd.Exec { + + private static final Logger LOGGER = LoggerFactory.getLogger(WaitContainerCmdExec.class); + + public WaitContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { + super(baseResource, dockerClientConfig); + } + + @Override + protected Void execute0(WaitContainerCmd command, ResultCallback resultCallback) { + WebTarget webTarget = getBaseResource().path("/containers/{id}/wait").resolveTemplate("id", + command.getContainerId()); + + WaitContainerCondition condition = command.getCondition(); + if (condition != null) { + webTarget = webTarget.queryParam("condition", condition.getValue()); + } + + LOGGER.trace("POST: {}", webTarget); + + webTarget.request().accept(MediaType.APPLICATION_JSON).post((Object) null, new TypeReference() { + }, resultCallback); + + return null; + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/util/CacheFromEncoder.java b/docker-java-core/src/main/java/com/github/dockerjava/core/util/CacheFromEncoder.java new file mode 100644 index 000000000..b5753bf57 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/util/CacheFromEncoder.java @@ -0,0 +1,26 @@ +package com.github.dockerjava.core.util; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.ObjectMapper; + +import java.util.Collection; + +/** + * JSON Encoder for the docker --cache-from parameter. + */ +public class CacheFromEncoder { + + private CacheFromEncoder() { + } + + // This instance MUST NOT be used for domain-specific serialization of the docker-java types + private static final ObjectMapper MAPPER = new ObjectMapper(); + + public static String jsonEncode(Collection imageIds) { + try { + return MAPPER.writeValueAsString(imageIds); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/util/CertificateUtils.java b/docker-java-core/src/main/java/com/github/dockerjava/core/util/CertificateUtils.java new file mode 100644 index 000000000..7cb1a19d4 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/util/CertificateUtils.java @@ -0,0 +1,194 @@ +package com.github.dockerjava.core.util; + +import java.io.BufferedReader; +import java.io.File; +import java.io.IOException; +import java.io.Reader; +import java.io.StringReader; +import java.security.KeyStore; +import java.security.KeyStoreException; +import java.security.NoSuchAlgorithmException; +import java.security.PrivateKey; +import java.security.cert.Certificate; +import java.security.cert.CertificateException; +import java.security.spec.InvalidKeySpecException; +import java.util.ArrayList; +import java.util.List; + +import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; +import org.bouncycastle.asn1.ASN1ObjectIdentifier; +import org.bouncycastle.asn1.pkcs.PrivateKeyInfo; +import org.bouncycastle.cert.X509CertificateHolder; +import org.bouncycastle.cert.jcajce.JcaX509CertificateConverter; +import org.bouncycastle.jce.provider.BouncyCastleProvider; +import org.bouncycastle.openssl.PEMKeyPair; +import org.bouncycastle.openssl.PEMParser; +import org.bouncycastle.openssl.jcajce.JcaPEMKeyConverter; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.annotation.CheckForNull; + +import static java.util.Objects.requireNonNull; + +public class CertificateUtils { + private static final Logger LOG = LoggerFactory.getLogger(CertificateUtils.class); + + private CertificateUtils() { + // utility class + } + + public static boolean verifyCertificatesExist(String dockerCertPath) { + String[] files = {"ca.pem", "cert.pem", "key.pem"}; + boolean result = true; + for (String file : files) { + File path = new File(dockerCertPath, file); + result &= path.exists(); + } + + return result; + } + + @SuppressFBWarnings(value = "NP_NULL_ON_SOME_PATH_FROM_RETURN_VALUE") + public static KeyStore createKeyStore(final String keypem, final String certpem) throws NoSuchAlgorithmException, + InvalidKeySpecException, IOException, CertificateException, KeyStoreException { + PrivateKey privateKey = loadPrivateKey(keypem); + requireNonNull(privateKey); + List privateCertificates = loadCertificates(certpem); + + KeyStore keyStore = KeyStore.getInstance("JKS"); + keyStore.load(null); + + keyStore.setKeyEntry("docker", + privateKey, + "docker".toCharArray(), + privateCertificates.toArray(new Certificate[privateCertificates.size()]) + ); + + return keyStore; + } + + /** + * from "cert.pem" String + */ + public static List loadCertificates(final String certpem) throws IOException, + CertificateException { + final StringReader certReader = new StringReader(certpem); + try (BufferedReader reader = new BufferedReader(certReader)) { + return loadCertificates(reader); + } + } + + /** + * "cert.pem" from reader + */ + public static List loadCertificates(final Reader reader) throws IOException, + CertificateException { + try (PEMParser pemParser = new PEMParser(reader)) { + List certificates = new ArrayList<>(); + + JcaX509CertificateConverter certificateConverter = new JcaX509CertificateConverter() + .setProvider(BouncyCastleProvider.PROVIDER_NAME); + Object certObj; + + while ((certObj = pemParser.readObject()) != null) { + if (certObj instanceof X509CertificateHolder) { + X509CertificateHolder certificateHolder = (X509CertificateHolder) certObj; + certificates.add(certificateConverter.getCertificate(certificateHolder)); + } + } + + return certificates; + } + } + + + /** + * Return private key ("key.pem") from Reader + */ + @CheckForNull + public static PrivateKey loadPrivateKey(final Reader reader) throws IOException, NoSuchAlgorithmException, + InvalidKeySpecException { + try (PEMParser pemParser = new PEMParser(reader)) { + Object readObject = pemParser.readObject(); + while (readObject != null) { + PrivateKeyInfo privateKeyInfo = getPrivateKeyInfoOrNull(readObject); + if (privateKeyInfo != null) { + return new JcaPEMKeyConverter().getPrivateKey(privateKeyInfo); + } + readObject = pemParser.readObject(); + } + } + + return null; + } + + /** + * Find a PrivateKeyInfo in the PEM object details. Returns null if the PEM object type is unknown. + */ + @CheckForNull + private static PrivateKeyInfo getPrivateKeyInfoOrNull(Object pemObject) throws NoSuchAlgorithmException { + PrivateKeyInfo privateKeyInfo = null; + if (pemObject instanceof PEMKeyPair) { + PEMKeyPair pemKeyPair = (PEMKeyPair) pemObject; + privateKeyInfo = pemKeyPair.getPrivateKeyInfo(); + } else if (pemObject instanceof PrivateKeyInfo) { + privateKeyInfo = (PrivateKeyInfo) pemObject; + } else if (pemObject instanceof ASN1ObjectIdentifier) { + // no idea how it can be used + final ASN1ObjectIdentifier asn1ObjectIdentifier = (ASN1ObjectIdentifier) pemObject; + LOG.trace("Ignoring asn1ObjectIdentifier {}", asn1ObjectIdentifier); + } else { + LOG.warn("Unknown object '{}' from PEMParser", pemObject); + } + return privateKeyInfo; + } + + /** + * Return KeyPair from "key.pem" + */ + @CheckForNull + public static PrivateKey loadPrivateKey(final String keypem) throws IOException, NoSuchAlgorithmException, + InvalidKeySpecException { + try (StringReader certReader = new StringReader(keypem); + BufferedReader reader = new BufferedReader(certReader)) { + return loadPrivateKey(reader); + } + } + + /** + * "ca.pem" from String + */ + public static KeyStore createTrustStore(String capem) throws IOException, CertificateException, + KeyStoreException, NoSuchAlgorithmException { + try (Reader certReader = new StringReader(capem)) { + return createTrustStore(certReader); + } + } + + /** + * "ca.pem" from Reader + */ + public static KeyStore createTrustStore(final Reader certReader) throws IOException, CertificateException, + KeyStoreException, NoSuchAlgorithmException { + try (PEMParser pemParser = new PEMParser(certReader)) { + + KeyStore trustStore = KeyStore.getInstance("JKS"); + trustStore.load(null); + + int index = 1; + Object pemCert; + + while ((pemCert = pemParser.readObject()) != null) { + Certificate caCertificate = new JcaX509CertificateConverter() + .setProvider(BouncyCastleProvider.PROVIDER_NAME) + .getCertificate((X509CertificateHolder) pemCert); + trustStore.setCertificateEntry("ca-" + index, caCertificate); + index++; + } + + return trustStore; + } + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/util/CompressArchiveUtil.java b/docker-java-core/src/main/java/com/github/dockerjava/core/util/CompressArchiveUtil.java new file mode 100644 index 000000000..eb7b90aca --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/util/CompressArchiveUtil.java @@ -0,0 +1,114 @@ +package com.github.dockerjava.core.util; + +import static com.github.dockerjava.core.util.FilePathUtil.relativize; + +import java.io.BufferedInputStream; +import java.io.BufferedOutputStream; +import java.io.File; +import java.io.FileNotFoundException; +import java.io.FileOutputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.util.zip.GZIPOutputStream; + +import org.apache.commons.compress.archivers.tar.TarArchiveEntry; +import org.apache.commons.compress.archivers.tar.TarArchiveOutputStream; +import org.apache.commons.compress.compressors.gzip.GzipCompressorOutputStream; +import org.apache.commons.io.FileUtils; + +import com.google.common.io.ByteStreams; + +public class CompressArchiveUtil { + private CompressArchiveUtil() { + // utility class + } + + static void addFileToTar(TarArchiveOutputStream tarArchiveOutputStream, Path file, String entryName) + throws IOException { + if (Files.isSymbolicLink(file)) { + TarArchiveEntry tarArchiveEntry = new TarArchiveEntry(entryName, TarArchiveEntry.LF_SYMLINK); + tarArchiveEntry.setLinkName(Files.readSymbolicLink(file).toString()); + tarArchiveOutputStream.putArchiveEntry(tarArchiveEntry); + } else { + TarArchiveEntry tarArchiveEntry = (TarArchiveEntry) tarArchiveOutputStream.createArchiveEntry(file.toFile(), + entryName); + if (file.toFile().canExecute()) { + tarArchiveEntry.setMode(tarArchiveEntry.getMode() | 0755); + } + tarArchiveOutputStream.putArchiveEntry(tarArchiveEntry); + if (file.toFile().isFile()) { + try (InputStream input = new BufferedInputStream(Files.newInputStream(file))) { + ByteStreams.copy(input, tarArchiveOutputStream); + } + } + } + tarArchiveOutputStream.closeArchiveEntry(); + } + + private static TarArchiveOutputStream buildTarStream(Path outputPath, boolean gZipped) throws IOException { + OutputStream outputStream = new BufferedOutputStream(Files.newOutputStream(outputPath)); + if (gZipped) { + outputStream = new GzipCompressorOutputStream(outputStream); + } + TarArchiveOutputStream tarArchiveOutputStream = new TarArchiveOutputStream(outputStream); + tarArchiveOutputStream.setLongFileMode(TarArchiveOutputStream.LONGFILE_POSIX); + tarArchiveOutputStream.setBigNumberMode(TarArchiveOutputStream.BIGNUMBER_POSIX); + return tarArchiveOutputStream; + } + + /** + * Recursively tar file + * + * @param inputPath + * file path can be directory + * @param outputPath + * where to put the archived file + * @param childrenOnly + * if inputPath is directory and if childrenOnly is true, the archive will contain all of its children, else the archive + * contains unique entry which is the inputPath itself + * @param gZipped + * compress with gzip algorithm + */ + public static void tar(Path inputPath, Path outputPath, boolean gZipped, boolean childrenOnly) throws IOException { + if (!Files.exists(inputPath)) { + throw new FileNotFoundException("File not found " + inputPath); + } + FileUtils.touch(outputPath.toFile()); + + try (TarArchiveOutputStream tarArchiveOutputStream = buildTarStream(outputPath, gZipped)) { + if (!Files.isDirectory(inputPath)) { + addFileToTar(tarArchiveOutputStream, inputPath, inputPath.getFileName().toString()); + } else { + Path sourcePath = inputPath; + if (!childrenOnly) { + // In order to have the dossier as the root entry + sourcePath = inputPath.getParent(); + } + Files.walkFileTree(inputPath, new TarDirWalker(sourcePath, tarArchiveOutputStream)); + } + tarArchiveOutputStream.flush(); + } + } + + public static File archiveTARFiles(File base, Iterable files, String archiveNameWithOutExtension) + throws IOException { + File tarFile = new File(FileUtils.getTempDirectoryPath(), archiveNameWithOutExtension + ".tar"); + tarFile.deleteOnExit(); + try (TarArchiveOutputStream tos = new TarArchiveOutputStream(new GZIPOutputStream(new BufferedOutputStream( + new FileOutputStream(tarFile))))) { + tos.setLongFileMode(TarArchiveOutputStream.LONGFILE_POSIX); + tos.setBigNumberMode(TarArchiveOutputStream.BIGNUMBER_POSIX); + for (File file : files) { + // relativize with method using Path otherwise method with File resolves the symlinks + // and this is not want we want. If the file is a symlink, the relativized path should + // keep the symlink name and not the target it points to. + addFileToTar(tos, file.toPath(), relativize(base.toPath(), file.toPath())); + } + } + + return tarFile; + } +} diff --git a/src/main/java/com/github/dockerjava/core/util/FilePathUtil.java b/docker-java-core/src/main/java/com/github/dockerjava/core/util/FilePathUtil.java similarity index 100% rename from src/main/java/com/github/dockerjava/core/util/FilePathUtil.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/util/FilePathUtil.java diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/util/FiltersBuilder.java b/docker-java-core/src/main/java/com/github/dockerjava/core/util/FiltersBuilder.java new file mode 100644 index 000000000..0d6c1d268 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/util/FiltersBuilder.java @@ -0,0 +1,157 @@ +package com.github.dockerjava.core.util; + +import com.google.common.base.Strings; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collection; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Map.Entry; +import java.util.regex.Pattern; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +/** + * Representation of Docker filters. + * + * @author Carlos Sanchez + * + */ +public class FiltersBuilder { + + private static final Pattern UNTIL_TIMESTAMP_PATTERN = + Pattern.compile("^\\d{1,10}$"); + private static final Pattern UNTIL_DATETIME_PATTERN = + Pattern.compile("^([0-9]+)-(0[1-9]|1[012])-(0[1-9]|[12][0-9]|3[01])([Tt]([01][0-9]|2[0-3]):([0-5][0-9]):" + + "([0-5][0-9]|60)(\\.[0-9]+)?)?(([Zz])|([\\+|\\-]([01][0-9]|2[0-3]):[0-5][0-9]))?$"); + private static final Pattern UNTIL_GO_PATTERN = + Pattern.compile("^([1-9][0-9]*h)?([1-9][0-9]*m)?([1-9][0-9]*s)?$"); + + private Map> filters = new HashMap<>(); + + public FiltersBuilder() { + } + + public FiltersBuilder withFilter(String key, String... value) { + filters.put(key, Arrays.asList(value)); + return this; + } + + public FiltersBuilder withFilter(String key, Collection value) { + filters.put(key, value instanceof List ? (List) value : new ArrayList<>(value)); + return this; + } + + public List getFilter(String key) { + return filters.get(key); + } + + public FiltersBuilder withImages(String... image) { + withFilter("image", image); + return this; + } + + public List getImage() { + return getFilter("image"); + } + + public FiltersBuilder withContainers(String... container) { + withFilter("container", container); + return this; + } + + public List getContainer() { + return getFilter("container"); + } + + /** + * Filter by event types + * + * @param eventTypes an array of event types + */ + public FiltersBuilder withEventTypes(String... eventTypes) { + withFilter("type", Stream.of(eventTypes).collect(Collectors.toList())); + return this; + } + + /** + * Filter by labels + * + * @param labels + * string array in the form ["key"] or ["key=value"] or a mix of both + */ + public FiltersBuilder withLabels(String... labels) { + withFilter("label", labels); + return this; + } + + /** + * Filter by labels + * + * @param labels + * {@link Map} of labels that contains label keys and values + */ + public FiltersBuilder withLabels(Map labels) { + withFilter("label", labelsMapToList(labels).toArray(new String[labels.size()])); + return this; + } + + public FiltersBuilder withUntil(String until) throws NumberFormatException { + if (!isValidUntil(until)) { + throw new NumberFormatException("Not valid format of 'until': " + until); + } + + return withFilter("until", until); + } + + private static List labelsMapToList(Map labels) { + List result = new ArrayList<>(); + for (Entry entry : labels.entrySet()) { + String rest = (entry.getValue() != null & !entry.getValue().isEmpty()) ? "=" + entry.getValue() : ""; + + String label = entry.getKey() + rest; + + result.add(label); + } + return result; + } + + private boolean isValidUntil(String until) { + if (UNTIL_DATETIME_PATTERN.matcher(until).matches()) { + return true; + } else if (!Strings.isNullOrEmpty(until) && UNTIL_GO_PATTERN.matcher(until).matches()) { + return true; + } else if (UNTIL_TIMESTAMP_PATTERN.matcher(until).matches()) { + return true; + } + + return false; + } + + // CHECKSTYLE:OFF + @Override + public boolean equals(Object o) { + if (this == o) + return true; + if (o == null || getClass() != o.getClass()) + return false; + + FiltersBuilder filters1 = (FiltersBuilder) o; + + return filters.equals(filters1.filters); + + } + + // CHECKSTYLE:ON + + @Override + public int hashCode() { + return filters.hashCode(); + } + + public Map> build() { + return filters; + } +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/util/FiltersEncoder.java b/docker-java-core/src/main/java/com/github/dockerjava/core/util/FiltersEncoder.java new file mode 100644 index 000000000..975d65aa3 --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/util/FiltersEncoder.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.core.util; + +import java.util.List; +import java.util.Map; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.ObjectMapper; + +/** + * JSON Encoder for docker filters. + * + * @author Carlos Sanchez + */ +public class FiltersEncoder { + + private FiltersEncoder() { + } + + // This instance MUST NOT be used for domain-specific serialization of the docker-java types + private static final ObjectMapper MAPPER = new ObjectMapper(); + + public static String jsonEncode(Map> mapStringListString) { + try { + return MAPPER.writeValueAsString(mapStringListString); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + } + +} diff --git a/docker-java-core/src/main/java/com/github/dockerjava/core/util/SwarmNodesFiltersBuilder.java b/docker-java-core/src/main/java/com/github/dockerjava/core/util/SwarmNodesFiltersBuilder.java new file mode 100644 index 000000000..02a4bc8fc --- /dev/null +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/util/SwarmNodesFiltersBuilder.java @@ -0,0 +1,94 @@ +package com.github.dockerjava.core.util; + + +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +/** + * Representation of filters to SwarmNodes lists + */ +public class SwarmNodesFiltersBuilder { + + private Map> filters = new HashMap<>(); + + public SwarmNodesFiltersBuilder() { + + } + + public SwarmNodesFiltersBuilder withFilter(String key, String... value) { + filters.put(key, Arrays.asList(value)); + return this; + } + + public SwarmNodesFiltersBuilder withFilter(String key, List value) { + filters.put(key, value); + return this; + } + + public List getFilter(String key) { + return filters.get(key); + } + + public SwarmNodesFiltersBuilder withIds(List ids) { + withFilter("id", ids); + return this; + } + + public List getIds() { + return getFilter("id"); + } + + public SwarmNodesFiltersBuilder withNames(List names) { + withFilter("name", names); + return this; + } + + public SwarmNodesFiltersBuilder withMemberships(List memberships) { + withFilter("membership", memberships); + return this; + } + + public List getMemberships() { + return getFilter("membership"); + } + + public SwarmNodesFiltersBuilder withRoles(List roles) { + withFilter("role", roles); + return this; + } + + public List getRoles() { + return getFilter("role"); + } + + public List getNames() { + return getFilter("names"); + } + + // CHECKSTYLE:OFF + @Override + public boolean equals(Object o) { + if (this == o) + return true; + if (o == null || getClass() != o.getClass()) + return false; + + SwarmNodesFiltersBuilder filters1 = (SwarmNodesFiltersBuilder) o; + + return filters.equals(filters1.filters); + + } + + // CHECKSTYLE:ON + + @Override + public int hashCode() { + return filters.hashCode(); + } + + public Map> build() { + return filters; + } +} diff --git a/src/main/java/com/github/dockerjava/core/util/TarDirWalker.java b/docker-java-core/src/main/java/com/github/dockerjava/core/util/TarDirWalker.java similarity index 83% rename from src/main/java/com/github/dockerjava/core/util/TarDirWalker.java rename to docker-java-core/src/main/java/com/github/dockerjava/core/util/TarDirWalker.java index bbe834ee6..a2b87d7bc 100644 --- a/src/main/java/com/github/dockerjava/core/util/TarDirWalker.java +++ b/docker-java-core/src/main/java/com/github/dockerjava/core/util/TarDirWalker.java @@ -33,11 +33,7 @@ public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) th @Override public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException { - TarArchiveEntry tarEntry = new TarArchiveEntry(FilePathUtil.relativize(basePath, file)); - if (file.toFile().canExecute()) { - tarEntry.setMode(tarEntry.getMode() | 0755); - } - CompressArchiveUtil.putTarEntry(tarArchiveOutputStream, tarEntry, file); + CompressArchiveUtil.addFileToTar(tarArchiveOutputStream, file, FilePathUtil.relativize(basePath, file)); return FileVisitResult.CONTINUE; } diff --git a/docker-java-transport-httpclient5/pom.xml b/docker-java-transport-httpclient5/pom.xml new file mode 100644 index 000000000..52cf66de2 --- /dev/null +++ b/docker-java-transport-httpclient5/pom.xml @@ -0,0 +1,72 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java-transport-httpclient5 + jar + + docker-java-transport-httpclient5 + https://github.com/docker-java/docker-java + Java API Client for Docker + + + com.github.dockerjava.transport.httpclient5 + + + + + ${project.groupId} + docker-java-transport + ${project.version} + + + + org.apache.httpcomponents.client5 + httpclient5 + 5.5.1 + + + + net.java.dev.jna + jna + 5.18.1 + + + + ${project.groupId} + docker-java-transport-tck + ${project.version} + test + + + + + + + com.github.siom79.japicmp + japicmp-maven-plugin + + + true + + + + + org.apache.felix + maven-bundle-plugin + true + + + com.github.dockerjava.httpclient5.* + + + + + + diff --git a/docker-java-transport-httpclient5/src/main/java/com/github/dockerjava/httpclient5/ApacheDockerHttpClient.java b/docker-java-transport-httpclient5/src/main/java/com/github/dockerjava/httpclient5/ApacheDockerHttpClient.java new file mode 100644 index 000000000..68e0eeddf --- /dev/null +++ b/docker-java-transport-httpclient5/src/main/java/com/github/dockerjava/httpclient5/ApacheDockerHttpClient.java @@ -0,0 +1,58 @@ +package com.github.dockerjava.httpclient5; + +import com.github.dockerjava.transport.SSLConfig; + +import java.net.URI; +import java.time.Duration; +import java.util.Objects; + +public final class ApacheDockerHttpClient extends ApacheDockerHttpClientImpl { + + public static final class Builder { + + private URI dockerHost = null; + + private SSLConfig sslConfig = null; + + private int maxConnections = Integer.MAX_VALUE; + + private Duration connectionTimeout; + + private Duration responseTimeout; + + public Builder dockerHost(URI value) { + this.dockerHost = Objects.requireNonNull(value, "dockerHost"); + return this; + } + + public Builder sslConfig(SSLConfig value) { + this.sslConfig = value; + return this; + } + + public Builder maxConnections(int value) { + this.maxConnections = value; + return this; + } + + public Builder connectionTimeout(Duration connectionTimeout) { + this.connectionTimeout = connectionTimeout; + return this; + } + + public Builder responseTimeout(Duration responseTimeout) { + this.responseTimeout = responseTimeout; + return this; + } + + public ApacheDockerHttpClient build() { + Objects.requireNonNull(dockerHost, "dockerHost"); + return new ApacheDockerHttpClient(dockerHost, sslConfig, maxConnections, connectionTimeout, responseTimeout); + } + } + + private ApacheDockerHttpClient(URI dockerHost, SSLConfig sslConfig, int maxConnections, Duration connectionTimeout, + Duration responseTimeout) { + super(dockerHost, sslConfig, maxConnections, connectionTimeout, responseTimeout); + } +} diff --git a/docker-java-transport-httpclient5/src/main/java/com/github/dockerjava/httpclient5/ApacheDockerHttpClientImpl.java b/docker-java-transport-httpclient5/src/main/java/com/github/dockerjava/httpclient5/ApacheDockerHttpClientImpl.java new file mode 100644 index 000000000..c97a2bc45 --- /dev/null +++ b/docker-java-transport-httpclient5/src/main/java/com/github/dockerjava/httpclient5/ApacheDockerHttpClientImpl.java @@ -0,0 +1,262 @@ +package com.github.dockerjava.httpclient5; + +import com.github.dockerjava.transport.DockerHttpClient; +import com.github.dockerjava.transport.NamedPipeSocket; +import com.github.dockerjava.transport.SSLConfig; +import com.github.dockerjava.transport.UnixSocket; + +import org.apache.hc.client5.http.SystemDefaultDnsResolver; +import org.apache.hc.client5.http.classic.methods.HttpUriRequestBase; +import org.apache.hc.client5.http.config.ConnectionConfig; +import org.apache.hc.client5.http.config.RequestConfig; +import org.apache.hc.client5.http.impl.DefaultSchemePortResolver; +import org.apache.hc.client5.http.impl.classic.CloseableHttpClient; +import org.apache.hc.client5.http.impl.classic.HttpClients; +import org.apache.hc.client5.http.impl.io.DefaultHttpClientConnectionOperator; +import org.apache.hc.client5.http.impl.io.ManagedHttpClientConnectionFactory; +import org.apache.hc.client5.http.impl.io.PoolingHttpClientConnectionManager; +import org.apache.hc.client5.http.io.HttpClientConnectionOperator; +import org.apache.hc.client5.http.ssl.DefaultClientTlsStrategy; +import org.apache.hc.client5.http.ssl.TlsSocketStrategy; +import org.apache.hc.core5.http.ClassicHttpResponse; +import org.apache.hc.core5.http.ConnectionClosedException; +import org.apache.hc.core5.http.ContentLengthStrategy; +import org.apache.hc.core5.http.Header; +import org.apache.hc.core5.http.HttpHeaders; +import org.apache.hc.core5.http.HttpHost; +import org.apache.hc.core5.http.NameValuePair; +import org.apache.hc.core5.http.impl.DefaultContentLengthStrategy; +import org.apache.hc.core5.http.io.SocketConfig; +import org.apache.hc.core5.http.io.entity.ByteArrayEntity; +import org.apache.hc.core5.http.io.entity.EmptyInputStream; +import org.apache.hc.core5.http.io.entity.InputStreamEntity; +import org.apache.hc.core5.http.protocol.HttpContext; +import org.apache.hc.core5.http.protocol.HttpCoreContext; +import org.apache.hc.core5.net.URIAuthority; +import org.apache.hc.core5.util.TimeValue; +import org.apache.hc.core5.util.Timeout; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.net.ssl.SSLContext; +import java.io.IOException; +import java.io.InputStream; +import java.net.Socket; +import java.net.URI; +import java.time.Duration; +import java.util.List; +import java.util.Map; +import java.util.concurrent.TimeUnit; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +class ApacheDockerHttpClientImpl implements DockerHttpClient { + + private final CloseableHttpClient httpClient; + private final HttpHost host; + private final String pathPrefix; + + protected ApacheDockerHttpClientImpl( + URI dockerHost, + SSLConfig sslConfig, + int maxConnections, + Duration connectionTimeout, + Duration responseTimeout + ) { + SSLContext sslContext; + try { + sslContext = sslConfig != null ? sslConfig.getSSLContext() : null; + } catch (Exception e) { + throw new RuntimeException(e); + } + HttpClientConnectionOperator connectionOperator = createConnectionOperator(dockerHost, sslContext); + + switch (dockerHost.getScheme()) { + case "unix": + case "npipe": + pathPrefix = ""; + host = new HttpHost(dockerHost.getScheme(), "localhost", 2375); + break; + case "tcp": + String rawPath = dockerHost.getRawPath(); + pathPrefix = rawPath.endsWith("/") + ? rawPath.substring(0, rawPath.length() - 1) + : rawPath; + host = new HttpHost( + sslContext != null ? "https" : "http", + dockerHost.getHost(), + dockerHost.getPort() + ); + break; + default: + throw new IllegalArgumentException("Unsupported protocol scheme: " + dockerHost); + } + + PoolingHttpClientConnectionManager connectionManager = new PoolingHttpClientConnectionManager( + connectionOperator, + null, + null, + null, + new ManagedHttpClientConnectionFactory( + null, + null, + null, + null, + message -> { + Header transferEncodingHeader = message.getFirstHeader(HttpHeaders.TRANSFER_ENCODING); + if (transferEncodingHeader != null) { + if ("identity".equalsIgnoreCase(transferEncodingHeader.getValue())) { + return ContentLengthStrategy.UNDEFINED; + } + } + return DefaultContentLengthStrategy.INSTANCE.determineLength(message); + }, + null + ) + ); + // See https://github.com/docker-java/docker-java/pull/1590#issuecomment-870581289 + connectionManager.setDefaultSocketConfig( + SocketConfig.copy(SocketConfig.DEFAULT) + .setSoTimeout(Timeout.ZERO_MILLISECONDS) + .build() + ); + connectionManager.setMaxTotal(maxConnections); + connectionManager.setDefaultMaxPerRoute(maxConnections); + connectionManager.setDefaultConnectionConfig(ConnectionConfig.custom() + .setValidateAfterInactivity(TimeValue.NEG_ONE_SECOND) + .setConnectTimeout(connectionTimeout != null ? Timeout.of(connectionTimeout.toNanos(), TimeUnit.NANOSECONDS) : null) + .build()); + + httpClient = HttpClients.custom() + .setRequestExecutor(new HijackingHttpRequestExecutor(null)) + .setConnectionManager(connectionManager) + .setDefaultRequestConfig(RequestConfig.custom() + .setResponseTimeout(responseTimeout != null ? Timeout.of(responseTimeout.toNanos(), TimeUnit.NANOSECONDS) : null) + .build()) + .disableConnectionState() + .build(); + } + + private HttpClientConnectionOperator createConnectionOperator( + URI dockerHost, + SSLContext sslContext + ) { + String dockerHostScheme = dockerHost.getScheme(); + String dockerHostPath = dockerHost.getPath(); + TlsSocketStrategy tlsSocketStrategy = sslContext != null ? + new DefaultClientTlsStrategy(sslContext) : DefaultClientTlsStrategy.createSystemDefault(); + return new DefaultHttpClientConnectionOperator( + socksProxy -> { + if ("unix".equalsIgnoreCase(dockerHostScheme)) { + return UnixSocket.get(dockerHostPath); + } else if ("npipe".equalsIgnoreCase(dockerHostScheme)) { + return new NamedPipeSocket(dockerHostPath); + } else { + return socksProxy == null ? new Socket() : new Socket(socksProxy); + } + }, + DefaultSchemePortResolver.INSTANCE, + SystemDefaultDnsResolver.INSTANCE, + name -> "https".equalsIgnoreCase(name) ? tlsSocketStrategy : null); + } + + @Override + public Response execute(Request request) { + HttpContext context = new HttpCoreContext(); + HttpUriRequestBase httpUriRequest = new HttpUriRequestBase(request.method(), URI.create(pathPrefix + request.path())); + httpUriRequest.setScheme(host.getSchemeName()); + httpUriRequest.setAuthority(new URIAuthority(host.getHostName(), host.getPort())); + + request.headers().forEach(httpUriRequest::addHeader); + + byte[] bodyBytes = request.bodyBytes(); + if (bodyBytes != null) { + httpUriRequest.setEntity(new ByteArrayEntity(bodyBytes, null)); + } else { + InputStream body = request.body(); + if (body != null) { + httpUriRequest.setEntity(new InputStreamEntity(body, null)); + } + } + + if (request.hijackedInput() != null) { + context.setAttribute(HijackingHttpRequestExecutor.HIJACKED_INPUT_ATTRIBUTE, request.hijackedInput()); + httpUriRequest.setHeader("Upgrade", "tcp"); + httpUriRequest.setHeader("Connection", "Upgrade"); + } + + try { + ClassicHttpResponse response = httpClient.executeOpen(host, httpUriRequest, context); + + return new ApacheResponse(httpUriRequest, response); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Override + public void close() throws IOException { + httpClient.close(); + } + + static class ApacheResponse implements Response { + + private static final Logger LOGGER = LoggerFactory.getLogger(ApacheResponse.class); + + private final HttpUriRequestBase request; + + private final ClassicHttpResponse response; + + ApacheResponse(HttpUriRequestBase httpUriRequest, ClassicHttpResponse response) { + this.request = httpUriRequest; + this.response = response; + } + + @Override + public int getStatusCode() { + return response.getCode(); + } + + @Override + public Map> getHeaders() { + return Stream.of(response.getHeaders()).collect(Collectors.groupingBy( + NameValuePair::getName, + Collectors.mapping(NameValuePair::getValue, Collectors.toList()) + )); + } + + @Override + public String getHeader(String name) { + Header firstHeader = response.getFirstHeader(name); + return firstHeader != null ? firstHeader.getValue() : null; + } + + @Override + public InputStream getBody() { + try { + return response.getEntity() != null + ? response.getEntity().getContent() + : EmptyInputStream.INSTANCE; + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Override + public void close() { + try { + request.abort(); + } catch (Exception e) { + LOGGER.debug("Failed to abort the request", e); + } + + try { + response.close(); + } catch (ConnectionClosedException e) { + LOGGER.trace("Failed to close the response", e); + } catch (Exception e) { + LOGGER.debug("Failed to close the response", e); + } + } + } +} diff --git a/docker-java-transport-httpclient5/src/main/java/com/github/dockerjava/httpclient5/HijackingHttpRequestExecutor.java b/docker-java-transport-httpclient5/src/main/java/com/github/dockerjava/httpclient5/HijackingHttpRequestExecutor.java new file mode 100644 index 000000000..df8fbd059 --- /dev/null +++ b/docker-java-transport-httpclient5/src/main/java/com/github/dockerjava/httpclient5/HijackingHttpRequestExecutor.java @@ -0,0 +1,146 @@ +package com.github.dockerjava.httpclient5; + +import org.apache.hc.core5.http.ClassicHttpRequest; +import org.apache.hc.core5.http.ClassicHttpResponse; +import org.apache.hc.core5.http.ConnectionReuseStrategy; +import org.apache.hc.core5.http.HttpException; +import org.apache.hc.core5.http.HttpHeaders; +import org.apache.hc.core5.http.HttpStatus; +import org.apache.hc.core5.http.ProtocolException; +import org.apache.hc.core5.http.ProtocolVersion; +import org.apache.hc.core5.http.impl.io.HttpRequestExecutor; +import org.apache.hc.core5.http.io.HttpClientConnection; +import org.apache.hc.core5.http.io.HttpResponseInformationCallback; +import org.apache.hc.core5.http.io.entity.AbstractHttpEntity; +import org.apache.hc.core5.http.message.BasicClassicHttpRequest; +import org.apache.hc.core5.http.message.StatusLine; +import org.apache.hc.core5.http.protocol.HttpContext; +import org.apache.hc.core5.http.protocol.HttpCoreContext; +import org.apache.hc.core5.io.Closer; + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.util.Objects; + +class HijackingHttpRequestExecutor extends HttpRequestExecutor { + + static final String HIJACKED_INPUT_ATTRIBUTE = "com.github.docker-java.hijackedInput"; + + HijackingHttpRequestExecutor(ConnectionReuseStrategy connectionReuseStrategy) { + super(connectionReuseStrategy); + } + + @Override + public ClassicHttpResponse execute( + ClassicHttpRequest request, + HttpClientConnection conn, + HttpResponseInformationCallback informationCallback, + HttpContext context + ) throws IOException, HttpException { + Objects.requireNonNull(request, "HTTP request"); + Objects.requireNonNull(conn, "Client connection"); + Objects.requireNonNull(context, "HTTP context"); + + InputStream hijackedInput = (InputStream) context.getAttribute(HIJACKED_INPUT_ATTRIBUTE); + if (hijackedInput != null) { + return executeHijacked(request, conn, (HttpCoreContext) context, hijackedInput); + } + + return super.execute(request, conn, informationCallback, context); + } + + private ClassicHttpResponse executeHijacked( + ClassicHttpRequest request, + HttpClientConnection conn, + HttpCoreContext context, + InputStream hijackedInput + ) throws HttpException, IOException { + try { + context.setSSLSession(conn.getSSLSession()); + context.setEndpointDetails(conn.getEndpointDetails()); + final ProtocolVersion transportVersion = request.getVersion(); + if (transportVersion != null) { + context.setProtocolVersion(transportVersion); + } + + conn.sendRequestHeader(request); + conn.sendRequestEntity(request); + conn.flush(); + + ClassicHttpResponse response = conn.receiveResponseHeader(); + if (response.getCode() != HttpStatus.SC_SWITCHING_PROTOCOLS) { + conn.terminateRequest(request); + throw new ProtocolException("Expected 101 Switching Protocols, got: " + new StatusLine(response)); + } + + Thread thread = new Thread(() -> { + try { + BasicClassicHttpRequest fakeRequest = new BasicClassicHttpRequest("POST", "/"); + fakeRequest.setHeader(HttpHeaders.CONTENT_LENGTH, Long.MAX_VALUE); + fakeRequest.setEntity(new HijackedEntity(hijackedInput)); + conn.sendRequestEntity(fakeRequest); + } catch (Exception e) { + throw new RuntimeException(e); + } + }); + thread.setName("docker-java-httpclient5-hijacking-stream-" + System.identityHashCode(request)); + thread.setDaemon(true); + thread.start(); + + // 101 -> 200 + response.setCode(200); + conn.receiveResponseEntity(response); + return response; + + } catch (final HttpException | IOException | RuntimeException ex) { + Closer.closeQuietly(conn); + throw ex; + } + } + + private static class HijackedEntity extends AbstractHttpEntity { + + private final InputStream inStream; + + HijackedEntity(InputStream inStream) { + super((String) null, null, false); + this.inStream = inStream; + } + + @Override + public void writeTo(OutputStream outStream) throws IOException { + byte[] buffer = new byte[1024]; + int read; + while ((read = inStream.read(buffer)) != -1) { + outStream.write(buffer, 0, read); + outStream.flush(); + } + } + + @Override + public InputStream getContent() { + return inStream; + } + + @Override + public boolean isStreaming() { + return true; + } + + @Override + public boolean isRepeatable() { + return false; + } + + @Override + public void close() throws IOException { + inStream.close(); + } + + @Override + public long getContentLength() { + return -1; + } + } +} diff --git a/docker-java-transport-httpclient5/src/test/java/com/github/dockerjava/transport/HttpClient5Tests.java b/docker-java-transport-httpclient5/src/test/java/com/github/dockerjava/transport/HttpClient5Tests.java new file mode 100644 index 000000000..d83621f78 --- /dev/null +++ b/docker-java-transport-httpclient5/src/test/java/com/github/dockerjava/transport/HttpClient5Tests.java @@ -0,0 +1,16 @@ +package com.github.dockerjava.transport; + +import com.github.dockerjava.httpclient5.ApacheDockerHttpClient; + +import java.net.URI; + +public class HttpClient5Tests extends DockerHttpClientTCK { + + @Override + protected DockerHttpClient createDockerHttpClient(URI dockerHost, SSLConfig sslConfig) { + return new ApacheDockerHttpClient.Builder() + .dockerHost(dockerHost) + .sslConfig(sslConfig) + .build(); + } +} diff --git a/docker-java-transport-jersey/pom.xml b/docker-java-transport-jersey/pom.xml new file mode 100644 index 000000000..a600c208d --- /dev/null +++ b/docker-java-transport-jersey/pom.xml @@ -0,0 +1,127 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java-transport-jersey + jar + + docker-java-transport-jersey + https://github.com/docker-java/docker-java + Java API Client for Docker + + + com.github.dockerjava.transport.jersey + + + + + ${project.groupId} + docker-java-core + ${project.version} + + + + org.glassfish.jersey.connectors + jersey-apache-connector + ${jersey.version} + + + org.apache.httpcomponents + httpcore + 4.4.13 + + + org.apache.httpcomponents + httpclient + ${httpclient.version} + + + commons-logging + commons-logging + + + + + org.glassfish.jersey.core + jersey-client + ${jersey.version} + + + org.glassfish.jersey.inject + jersey-hk2 + ${jersey.version} + + + com.kohlschutter.junixsocket + junixsocket-common + ${junixsocket.version} + + + com.kohlschutter.junixsocket + junixsocket-native-common + ${junixsocket.version} + + + + org.slf4j + jcl-over-slf4j + 1.7.30 + test + + + + ${project.groupId} + docker-java-transport-tck + ${project.version} + test + + + + + + + + com.github.siom79.japicmp + japicmp-maven-plugin + + + + + com.github.dockerjava.jaxrs.ApacheUnixSocket + com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier + com.github.dockerjava.jaxrs.async.GETCallbackNotifier + com.github.dockerjava.jaxrs.async.GETCallbackNotifier + com.github.dockerjava.jaxrs.async.POSTCallbackNotifier + com.github.dockerjava.jaxrs.UnixConnectionSocketFactory + com.github.dockerjava.jaxrs.JerseyDockerCmdExecFactory#releaseConnection(long) + + + + SUPERCLASS_REMOVED + true + true + + + + + + + + org.apache.felix + maven-bundle-plugin + true + + + com.github.dockerjava.jaxrs.* + + + + + + diff --git a/src/main/java/com/github/dockerjava/jaxrs/ApacheUnixSocket.java b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/ApacheUnixSocket.java similarity index 80% rename from src/main/java/com/github/dockerjava/jaxrs/ApacheUnixSocket.java rename to docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/ApacheUnixSocket.java index a1b860b28..72a851560 100644 --- a/src/main/java/com/github/dockerjava/jaxrs/ApacheUnixSocket.java +++ b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/ApacheUnixSocket.java @@ -41,13 +41,13 @@ * * This class also noop's any calls to setReuseAddress, which is called by the Apache client but isn't supported by AFUnixSocket. */ -public class ApacheUnixSocket extends Socket { +class ApacheUnixSocket extends Socket { private final AFUNIXSocket inner; - private final Queue optionsToSet = new ArrayDeque(); + private final Queue optionsToSet = new ArrayDeque<>(); - public ApacheUnixSocket() throws IOException { + ApacheUnixSocket() throws IOException { this.inner = AFUNIXSocket.newInstance(); } @@ -132,12 +132,7 @@ private void setAllSocketOptions() throws SocketException { @Override public void setTcpNoDelay(final boolean on) throws SocketException { - setSocketOption(new SocketOptionSetter() { - @Override - public void run() throws SocketException { - inner.setTcpNoDelay(on); - } - }); + setSocketOption(() -> inner.setTcpNoDelay(on)); } @Override @@ -147,12 +142,7 @@ public boolean getTcpNoDelay() throws SocketException { @Override public void setSoLinger(final boolean on, final int linger) throws SocketException { - setSocketOption(new SocketOptionSetter() { - @Override - public void run() throws SocketException { - inner.setSoLinger(on, linger); - } - }); + setSocketOption(() -> inner.setSoLinger(on, linger)); } @Override @@ -167,12 +157,7 @@ public void sendUrgentData(final int data) throws IOException { @Override public void setOOBInline(final boolean on) throws SocketException { - setSocketOption(new SocketOptionSetter() { - @Override - public void run() throws SocketException { - inner.setOOBInline(on); - } - }); + setSocketOption(() -> inner.setOOBInline(on)); } @Override @@ -182,12 +167,7 @@ public boolean getOOBInline() throws SocketException { @Override public synchronized void setSoTimeout(final int timeout) throws SocketException { - setSocketOption(new SocketOptionSetter() { - @Override - public void run() throws SocketException { - inner.setSoTimeout(timeout); - } - }); + setSocketOption(() -> inner.setSoTimeout(timeout)); } @Override @@ -197,12 +177,7 @@ public synchronized int getSoTimeout() throws SocketException { @Override public synchronized void setSendBufferSize(final int size) throws SocketException { - setSocketOption(new SocketOptionSetter() { - @Override - public void run() throws SocketException { - inner.setSendBufferSize(size); - } - }); + setSocketOption(() -> inner.setSendBufferSize(size)); } @Override @@ -212,12 +187,7 @@ public synchronized int getSendBufferSize() throws SocketException { @Override public synchronized void setReceiveBufferSize(final int size) throws SocketException { - setSocketOption(new SocketOptionSetter() { - @Override - public void run() throws SocketException { - inner.setReceiveBufferSize(size); - } - }); + setSocketOption(() -> inner.setReceiveBufferSize(size)); } @Override @@ -227,12 +197,7 @@ public synchronized int getReceiveBufferSize() throws SocketException { @Override public void setKeepAlive(final boolean on) throws SocketException { - setSocketOption(new SocketOptionSetter() { - @Override - public void run() throws SocketException { - inner.setKeepAlive(on); - } - }); + setSocketOption(() -> inner.setKeepAlive(on)); } @Override @@ -242,12 +207,7 @@ public boolean getKeepAlive() throws SocketException { @Override public void setTrafficClass(final int tc) throws SocketException { - setSocketOption(new SocketOptionSetter() { - @Override - public void run() throws SocketException { - inner.setTrafficClass(tc); - } - }); + setSocketOption(() -> inner.setTrafficClass(tc)); } @Override diff --git a/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/JerseyDockerCmdExecFactory.java b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/JerseyDockerCmdExecFactory.java new file mode 100644 index 000000000..6298cae3b --- /dev/null +++ b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/JerseyDockerCmdExecFactory.java @@ -0,0 +1,97 @@ +package com.github.dockerjava.jaxrs; + +import com.github.dockerjava.api.command.DelegatingDockerCmdExecFactory; +import com.github.dockerjava.api.command.DockerCmdExecFactory; +import com.github.dockerjava.core.DefaultDockerCmdExecFactory; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.DockerClientConfigAware; +import com.github.dockerjava.core.DockerClientImpl; +import com.github.dockerjava.transport.DockerHttpClient; +import org.glassfish.jersey.client.RequestEntityProcessing; + +import javax.ws.rs.client.ClientRequestFilter; +import javax.ws.rs.client.ClientResponseFilter; + +//import org.glassfish.jersey.apache.connector.ApacheConnectorProvider; +// see https://github.com/docker-java/docker-java/issues/196 +/** + * @deprecated use {@link JerseyDockerHttpClient} with {@link DockerClientImpl#getInstance(DockerClientConfig, DockerHttpClient)} + */ +@Deprecated +public class JerseyDockerCmdExecFactory extends DelegatingDockerCmdExecFactory implements DockerClientConfigAware { + + private JerseyDockerHttpClient.Builder clientBuilder = new JerseyDockerHttpClient.Builder(); + + @Deprecated + protected Integer connectTimeout; + + @Deprecated + protected Integer readTimeout; + + private DefaultDockerCmdExecFactory dockerCmdExecFactory; + + @Override + public final DockerCmdExecFactory getDockerCmdExecFactory() { + return dockerCmdExecFactory; + } + + @Override + public void init(DockerClientConfig dockerClientConfig) { + clientBuilder = clientBuilder + .dockerHost(dockerClientConfig.getDockerHost()) + .sslConfig(dockerClientConfig.getSSLConfig()); + dockerCmdExecFactory = new DefaultDockerCmdExecFactory( + clientBuilder.build(), + dockerClientConfig.getObjectMapper() + ); + dockerCmdExecFactory.init(dockerClientConfig); + } + + /** + * Configure connection timeout in milliseconds + */ + public JerseyDockerCmdExecFactory withConnectTimeout(Integer connectTimeout) { + clientBuilder = clientBuilder.connectTimeout(connectTimeout); + this.connectTimeout = connectTimeout; + return this; + } + + /** + * Configure read timeout in milliseconds + */ + public JerseyDockerCmdExecFactory withReadTimeout(Integer readTimeout) { + clientBuilder = clientBuilder.readTimeout(readTimeout); + this.readTimeout = readTimeout; + return this; + } + + public JerseyDockerCmdExecFactory withMaxTotalConnections(Integer maxTotalConnections) { + clientBuilder = clientBuilder.maxTotalConnections(maxTotalConnections); + return this; + } + + public JerseyDockerCmdExecFactory withMaxPerRouteConnections(Integer maxPerRouteConnections) { + clientBuilder = clientBuilder.maxPerRouteConnections(maxPerRouteConnections); + return this; + } + + public JerseyDockerCmdExecFactory withConnectionRequestTimeout(Integer connectionRequestTimeout) { + clientBuilder = clientBuilder.connectionRequestTimeout(connectionRequestTimeout); + return this; + } + + public JerseyDockerCmdExecFactory withClientResponseFilters(ClientResponseFilter... clientResponseFilter) { + clientBuilder = clientBuilder.clientResponseFilters(clientResponseFilter); + return this; + } + + public JerseyDockerCmdExecFactory withClientRequestFilters(ClientRequestFilter... clientRequestFilters) { + clientBuilder = clientBuilder.clientRequestFilters(clientRequestFilters); + return this; + } + + public JerseyDockerCmdExecFactory withRequestEntityProcessing(RequestEntityProcessing requestEntityProcessing) { + clientBuilder = clientBuilder.requestEntityProcessing(requestEntityProcessing); + return this; + } +} diff --git a/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/JerseyDockerHttpClient.java b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/JerseyDockerHttpClient.java new file mode 100644 index 000000000..78a65cf63 --- /dev/null +++ b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/JerseyDockerHttpClient.java @@ -0,0 +1,396 @@ +package com.github.dockerjava.jaxrs; + +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.transport.DockerHttpClient; +import com.github.dockerjava.transport.SSLConfig; +import com.github.dockerjava.jaxrs.filter.ResponseStatusExceptionFilter; +import com.github.dockerjava.jaxrs.filter.SelectiveLoggingFilter; +import org.apache.http.client.config.RequestConfig; +import org.apache.http.config.Registry; +import org.apache.http.config.RegistryBuilder; +import org.apache.http.conn.socket.ConnectionSocketFactory; +import org.apache.http.conn.socket.PlainConnectionSocketFactory; +import org.apache.http.conn.ssl.SSLConnectionSocketFactory; +import org.apache.http.impl.conn.PoolingHttpClientConnectionManager; +import org.apache.http.impl.io.EmptyInputStream; +import org.glassfish.jersey.CommonProperties; +import org.glassfish.jersey.apache.connector.ApacheClientProperties; +import org.glassfish.jersey.apache.connector.ApacheConnectorProvider; +import org.glassfish.jersey.client.ClientConfig; +import org.glassfish.jersey.client.ClientProperties; +import org.glassfish.jersey.client.RequestEntityProcessing; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.net.ssl.SSLContext; +import javax.ws.rs.ProcessingException; +import javax.ws.rs.client.Client; +import javax.ws.rs.client.ClientBuilder; +import javax.ws.rs.client.ClientRequestFilter; +import javax.ws.rs.client.ClientResponseFilter; +import javax.ws.rs.client.Entity; +import javax.ws.rs.client.Invocation; +import javax.ws.rs.core.MediaType; +import java.io.InputStream; +import java.net.InetSocketAddress; +import java.net.Proxy; +import java.net.ProxySelector; +import java.net.URI; +import java.net.URISyntaxException; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +/** + * @deprecated use Apache HttpClient 5-based transport + */ +@Deprecated +public final class JerseyDockerHttpClient implements DockerHttpClient { + + public static final class Builder { + + private URI dockerHost = null; + + private SSLConfig sslConfig = null; + + private Integer readTimeout = null; + + private Integer connectTimeout = null; + + private Integer maxTotalConnections = Integer.MAX_VALUE; + + private Integer maxPerRouteConnections = Integer.MAX_VALUE; + + private Integer connectionRequestTimeout = null; + + private ClientRequestFilter[] clientRequestFilters = null; + + private ClientResponseFilter[] clientResponseFilters = null; + + private RequestEntityProcessing requestEntityProcessing; + + public Builder dockerHost(URI value) { + this.dockerHost = Objects.requireNonNull(value, "dockerHost"); + return this; + } + + public Builder sslConfig(SSLConfig value) { + this.sslConfig = value; + return this; + } + + public Builder readTimeout(Integer value) { + this.readTimeout = value; + return this; + } + + public Builder connectTimeout(Integer value) { + this.connectTimeout = value; + return this; + } + + public Builder maxTotalConnections(Integer value) { + this.maxTotalConnections = value; + return this; + } + + public Builder maxPerRouteConnections(Integer value) { + this.maxPerRouteConnections = value; + return this; + } + + public Builder connectionRequestTimeout(Integer value) { + this.connectionRequestTimeout = value; + return this; + } + + public Builder clientResponseFilters(ClientResponseFilter[] value) { + this.clientResponseFilters = value; + return this; + } + + public Builder clientRequestFilters(ClientRequestFilter[] value) { + this.clientRequestFilters = value; + return this; + } + + public Builder requestEntityProcessing(RequestEntityProcessing value) { + this.requestEntityProcessing = value; + return this; + } + + public JerseyDockerHttpClient build() { + return new JerseyDockerHttpClient( + dockerHost, + sslConfig, + maxTotalConnections, + maxPerRouteConnections, + connectionRequestTimeout, + readTimeout, + connectTimeout, + clientRequestFilters, + clientResponseFilters, + requestEntityProcessing + ); + } + } + + private static final Logger LOGGER = LoggerFactory.getLogger(JerseyDockerHttpClient.class.getName()); + + private final Client client; + + private final PoolingHttpClientConnectionManager connManager; + + private final URI originalUri; + + private JerseyDockerHttpClient( + URI dockerHost, + SSLConfig sslConfig, + Integer maxTotalConnections, + Integer maxPerRouteConnections, + Integer connectionRequestTimeout, + Integer readTimeout, + Integer connectTimeout, + ClientRequestFilter[] clientRequestFilters, + ClientResponseFilter[] clientResponseFilters, + RequestEntityProcessing requestEntityProcessing + ) { + ClientConfig clientConfig = new ClientConfig(); + clientConfig.connectorProvider(new ApacheConnectorProvider()); + clientConfig.property(CommonProperties.FEATURE_AUTO_DISCOVERY_DISABLE, true); + + if (requestEntityProcessing != null) { + clientConfig.property(ClientProperties.REQUEST_ENTITY_PROCESSING, requestEntityProcessing); + } + + clientConfig.register(new ResponseStatusExceptionFilter()); + // clientConfig.register(JsonClientFilter.class); + RequestConfig.Builder requestConfigBuilder = RequestConfig.custom(); + + // logging may disabled via log level + clientConfig.register(new SelectiveLoggingFilter(LOGGER, false)); + + if (readTimeout != null) { + requestConfigBuilder.setSocketTimeout(readTimeout); + clientConfig.property(ClientProperties.READ_TIMEOUT, readTimeout); + } + + if (connectTimeout != null) { + requestConfigBuilder.setConnectTimeout(connectTimeout); + clientConfig.property(ClientProperties.CONNECT_TIMEOUT, connectTimeout); + } + + if (clientResponseFilters != null) { + for (ClientResponseFilter clientResponseFilter : clientResponseFilters) { + if (clientResponseFilter != null) { + clientConfig.register(clientResponseFilter); + } + } + } + + if (clientRequestFilters != null) { + for (ClientRequestFilter clientRequestFilter : clientRequestFilters) { + if (clientRequestFilter != null) { + clientConfig.register(clientRequestFilter); + } + } + } + + SSLContext sslContext = null; + + try { + if (sslConfig != null) { + sslContext = sslConfig.getSSLContext(); + } + } catch (Exception ex) { + throw new DockerClientException("Error in SSL Configuration", ex); + } + + final String protocol = sslContext != null ? "https" : "http"; + + switch (dockerHost.getScheme()) { + case "unix": + break; + case "tcp": + try { + dockerHost = new URI(dockerHost.toString().replaceFirst("tcp", protocol)); + } catch (URISyntaxException e) { + throw new RuntimeException(e); + } + + configureProxy(clientConfig, dockerHost, protocol); + break; + default: + throw new IllegalArgumentException("Unsupported protocol scheme: " + dockerHost); + } + + connManager = new PoolingHttpClientConnectionManager(getSchemeRegistry(dockerHost, sslContext)) { + + @Override + public void close() { + super.shutdown(); + } + + @Override + public void shutdown() { + // Disable shutdown of the pool. This will be done later, when this factory is closed + // This is a workaround for finalize method on jerseys ClientRuntime which + // closes the client and shuts down the connection pool when it is garbage collected + } + }; + + if (maxTotalConnections != null) { + connManager.setMaxTotal(maxTotalConnections); + } + if (maxPerRouteConnections != null) { + connManager.setDefaultMaxPerRoute(maxPerRouteConnections); + } + + clientConfig.property(ApacheClientProperties.CONNECTION_MANAGER, connManager); + + // Configure connection pool timeout + if (connectionRequestTimeout != null) { + requestConfigBuilder.setConnectionRequestTimeout(connectionRequestTimeout); + } + clientConfig.property(ApacheClientProperties.REQUEST_CONFIG, requestConfigBuilder.build()); + ClientBuilder clientBuilder = ClientBuilder.newBuilder().withConfig(clientConfig); + + if (sslContext != null) { + clientBuilder.sslContext(sslContext); + } + + client = clientBuilder.build(); + + this.originalUri = dockerHost; + } + + private URI sanitizeUrl(URI originalUri) { + if (originalUri.getScheme().equals("unix")) { + return UnixConnectionSocketFactory.sanitizeUri(originalUri); + } + return originalUri; + } + + private Registry getSchemeRegistry(URI originalUri, SSLContext sslContext) { + RegistryBuilder registryBuilder = RegistryBuilder.create(); + registryBuilder.register("http", PlainConnectionSocketFactory.getSocketFactory()); + if (sslContext != null) { + registryBuilder.register("https", new SSLConnectionSocketFactory(sslContext)); + } + registryBuilder.register("unix", new UnixConnectionSocketFactory(originalUri)); + return registryBuilder.build(); + } + + @Override + public Response execute(Request request) { + if (request.hijackedInput() != null) { + throw new UnsupportedOperationException("Does not support hijacking"); + } + String url = sanitizeUrl(originalUri).toString(); + if (url.endsWith("/") && request.path().startsWith("/")) { + url = url.substring(0, url.length() - 1); + } + + Invocation.Builder builder = client.target(url + request.path()).request(); + + request.headers().forEach(builder::header); + + try { + return new JerseyResponse( + builder.build(request.method(), toEntity(request)).invoke() + ); + } catch (ProcessingException e) { + if (e.getCause() instanceof DockerException) { + throw (DockerException) e.getCause(); + } + throw e; + } + } + + private Entity toEntity(Request request) { + byte[] bodyBytes = request.bodyBytes(); + if (bodyBytes != null) { + return Entity.json(bodyBytes); + } + InputStream body = request.body(); + if (body != null) { + return Entity.entity(body, MediaType.APPLICATION_JSON_TYPE); + } + switch (request.method()) { + case "POST": + return Entity.json(null); + default: + return null; + } + } + + private void configureProxy(ClientConfig clientConfig, URI originalUri, String protocol) { + List proxies = ProxySelector.getDefault().select(originalUri); + + for (Proxy proxy : proxies) { + InetSocketAddress address = (InetSocketAddress) proxy.address(); + if (address != null) { + String hostname = address.getHostName(); + int port = address.getPort(); + + clientConfig.property(ClientProperties.PROXY_URI, "http://" + hostname + ":" + port); + + String httpProxyUser = System.getProperty(protocol + ".proxyUser"); + if (httpProxyUser != null) { + clientConfig.property(ClientProperties.PROXY_USERNAME, httpProxyUser); + String httpProxyPassword = System.getProperty(protocol + ".proxyPassword"); + if (httpProxyPassword != null) { + clientConfig.property(ClientProperties.PROXY_PASSWORD, httpProxyPassword); + } + } + } + } + } + + @Override + public void close() { + if (client != null) { + client.close(); + } + + if (connManager != null) { + connManager.close(); + } + } + + private static class JerseyResponse implements Response { + + private final javax.ws.rs.core.Response response; + + JerseyResponse(javax.ws.rs.core.Response response) { + this.response = response; + } + + @Override + public int getStatusCode() { + return response.getStatus(); + } + + @Override + public Map> getHeaders() { + return response.getStringHeaders(); + } + + @Override + public InputStream getBody() { + return response.hasEntity() + ? response.readEntity(InputStream.class) + : EmptyInputStream.INSTANCE; + } + + @Override + public void close() { + try { + response.close(); + } catch (Exception e) { + LOGGER.debug("Failed to close the response", e); + } + } + } +} diff --git a/src/main/java/com/github/dockerjava/jaxrs/UnixConnectionSocketFactory.java b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/UnixConnectionSocketFactory.java similarity index 95% rename from src/main/java/com/github/dockerjava/jaxrs/UnixConnectionSocketFactory.java rename to docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/UnixConnectionSocketFactory.java index 0c5400ecd..84a72f077 100644 --- a/src/main/java/com/github/dockerjava/jaxrs/UnixConnectionSocketFactory.java +++ b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/UnixConnectionSocketFactory.java @@ -40,11 +40,11 @@ * Provides a ConnectionSocketFactory for connecting Apache HTTP clients to Unix sockets. */ @Contract(threading = ThreadingBehavior.IMMUTABLE_CONDITIONAL) -public class UnixConnectionSocketFactory implements ConnectionSocketFactory { +class UnixConnectionSocketFactory implements ConnectionSocketFactory { private File socketFile; - public UnixConnectionSocketFactory(final URI socketUri) { + UnixConnectionSocketFactory(final URI socketUri) { super(); final String filename = socketUri.toString().replaceAll("^unix:///", "unix://localhost/") diff --git a/src/main/java/com/github/dockerjava/jaxrs/filter/FollowRedirectsFilter.java b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/FollowRedirectsFilter.java similarity index 99% rename from src/main/java/com/github/dockerjava/jaxrs/filter/FollowRedirectsFilter.java rename to docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/FollowRedirectsFilter.java index 4a7765d97..64f5e88f8 100644 --- a/src/main/java/com/github/dockerjava/jaxrs/filter/FollowRedirectsFilter.java +++ b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/FollowRedirectsFilter.java @@ -17,6 +17,7 @@ * * This filter allows arbitrary redirection for other methods. */ +@Deprecated public class FollowRedirectsFilter implements ClientResponseFilter { @Override diff --git a/src/main/java/com/github/dockerjava/jaxrs/filter/JsonClientFilter.java b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/JsonClientFilter.java similarity index 98% rename from src/main/java/com/github/dockerjava/jaxrs/filter/JsonClientFilter.java rename to docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/JsonClientFilter.java index 2abc20618..a62034d27 100644 --- a/src/main/java/com/github/dockerjava/jaxrs/filter/JsonClientFilter.java +++ b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/JsonClientFilter.java @@ -12,6 +12,7 @@ * @author Konstantin Pelykh (kpelykh@gmail.com) * */ +@Deprecated public class JsonClientFilter implements ClientResponseFilter { @Override diff --git a/src/main/java/com/github/dockerjava/jaxrs/filter/LoggingFilter.java b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/LoggingFilter.java similarity index 97% rename from src/main/java/com/github/dockerjava/jaxrs/filter/LoggingFilter.java rename to docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/LoggingFilter.java index 8c9848318..986b4c10a 100644 --- a/src/main/java/com/github/dockerjava/jaxrs/filter/LoggingFilter.java +++ b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/LoggingFilter.java @@ -82,6 +82,7 @@ @PreMatching @Priority(Integer.MIN_VALUE) @SuppressWarnings("ClassWithMultipleLoggers") +@Deprecated public class LoggingFilter implements ContainerRequestFilter, ClientRequestFilter, ContainerResponseFilter, ClientResponseFilter, WriterInterceptor { @@ -95,13 +96,7 @@ public class LoggingFilter implements ContainerRequestFilter, ClientRequestFilte private static final String ENTITY_LOGGER_PROPERTY = LoggingFilter.class.getName() + ".entityLogger"; - private static final Comparator>> COMPARATOR = new Comparator>>() { - - @Override - public int compare(final Map.Entry> o1, final Map.Entry> o2) { - return o1.getKey().compareToIgnoreCase(o2.getKey()); - } - }; + private static final Comparator>> COMPARATOR = (o1, o2) -> o1.getKey().compareToIgnoreCase(o2.getKey()); private static final int DEFAULT_MAX_ENTITY_SIZE = 8 * 1024; @@ -202,7 +197,7 @@ private void printPrefixedHeaders(final StringBuilder b, final long id, final St } private Set>> getSortedHeaders(final Set>> headers) { - final TreeSet>> sortedHeaders = new TreeSet>>( + final TreeSet>> sortedHeaders = new TreeSet<>( COMPARATOR); sortedHeaders.addAll(headers); return sortedHeaders; diff --git a/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/ResponseStatusExceptionFilter.java b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/ResponseStatusExceptionFilter.java new file mode 100644 index 000000000..8cc0a0746 --- /dev/null +++ b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/ResponseStatusExceptionFilter.java @@ -0,0 +1,113 @@ +package com.github.dockerjava.jaxrs.filter; + +import com.fasterxml.jackson.databind.JsonNode; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.dockerjava.api.exception.BadRequestException; +import com.github.dockerjava.api.exception.ConflictException; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.InternalServerErrorException; +import com.github.dockerjava.api.exception.NotAcceptableException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.exception.NotModifiedException; +import com.github.dockerjava.api.exception.UnauthorizedException; +import org.apache.commons.io.IOUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.ws.rs.client.ClientRequestContext; +import javax.ws.rs.client.ClientResponseContext; +import javax.ws.rs.client.ClientResponseFilter; +import javax.ws.rs.core.MediaType; +import java.io.IOException; +import java.io.InputStream; +import java.nio.charset.Charset; + +/** + * This {@link ClientResponseFilter} implementation detects http status codes and throws {@link DockerException}s + * + * @author Marcus Linke + * + */ +@Deprecated +public class ResponseStatusExceptionFilter implements ClientResponseFilter { + + private static final Logger LOG = LoggerFactory.getLogger(ResponseStatusExceptionFilter.class); + + private final ObjectMapper objectMapper; + + @Deprecated + public ResponseStatusExceptionFilter() { + this(new ObjectMapper()); + } + + public ResponseStatusExceptionFilter(ObjectMapper objectMapper) { + this.objectMapper = objectMapper; + } + + @Override + public void filter(ClientRequestContext requestContext, ClientResponseContext responseContext) throws IOException { + int status = responseContext.getStatus(); + switch (status) { + case 200: + case 201: + case 204: + return; + case 304: + throw new NotModifiedException(getBodyAsMessage(responseContext)); + case 400: + throw new BadRequestException(getBodyAsMessage(responseContext)); + case 401: + throw new UnauthorizedException(getBodyAsMessage(responseContext)); + case 404: + throw new NotFoundException(getBodyAsMessage(responseContext)); + case 406: + throw new NotAcceptableException(getBodyAsMessage(responseContext)); + case 409: + throw new ConflictException(getBodyAsMessage(responseContext)); + case 500: + throw new InternalServerErrorException(getBodyAsMessage(responseContext)); + default: + throw new DockerException(getBodyAsMessage(responseContext), status); + } + } + + private String getBodyAsMessage(ClientResponseContext responseContext) { + if (responseContext.hasEntity()) { + try (InputStream entityStream = responseContext.getEntityStream()) { + Charset charset = null; + MediaType mediaType = responseContext.getMediaType(); + if (mediaType != null) { + String charsetName = mediaType.getParameters().get("charset"); + if (charsetName != null) { + try { + charset = Charset.forName(charsetName); + } catch (Exception ignored) { } + } + } + + if (charset == null) { + charset = Charset.defaultCharset(); + } + + String message = IOUtils.toString(entityStream, charset); + + if (MediaType.APPLICATION_JSON_TYPE.equals(mediaType)) { + try { + JsonNode node = objectMapper.readTree(message); + if (node != null) { + JsonNode messageNode = node.get("message"); + if (messageNode != null && messageNode.isTextual()) { + message = messageNode.textValue(); + } + } + } catch (IOException e) { + // ignore parsing errors and return the message as is + LOG.debug("Failed to unwrap error message: {}", e.getMessage(), e); + } + } + return message; + } catch (Exception ignored) { } + } + return null; + } +} diff --git a/src/main/java/com/github/dockerjava/jaxrs/filter/SelectiveLoggingFilter.java b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/SelectiveLoggingFilter.java similarity index 93% rename from src/main/java/com/github/dockerjava/jaxrs/filter/SelectiveLoggingFilter.java rename to docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/SelectiveLoggingFilter.java index 2251d8fad..862bdf0a9 100644 --- a/src/main/java/com/github/dockerjava/jaxrs/filter/SelectiveLoggingFilter.java +++ b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/filter/SelectiveLoggingFilter.java @@ -16,14 +16,16 @@ * * @author sfitts */ +@Deprecated public class SelectiveLoggingFilter extends LoggingFilter { // Immutable'ish private static final Set SKIPPED_CONTENT; static { - Set s = new HashSet(); + Set s = new HashSet<>(); s.add(MediaType.APPLICATION_OCTET_STREAM); s.add("application/tar"); + s.add("application/x-tar"); SKIPPED_CONTENT = Collections.unmodifiableSet(s); } diff --git a/src/main/java/com/github/dockerjava/jaxrs/util/WrappedResponseInputStream.java b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/util/WrappedResponseInputStream.java similarity index 96% rename from src/main/java/com/github/dockerjava/jaxrs/util/WrappedResponseInputStream.java rename to docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/util/WrappedResponseInputStream.java index afe0dce92..11e9f8684 100644 --- a/src/main/java/com/github/dockerjava/jaxrs/util/WrappedResponseInputStream.java +++ b/docker-java-transport-jersey/src/main/java/com/github/dockerjava/jaxrs/util/WrappedResponseInputStream.java @@ -11,6 +11,7 @@ * * @author Marcus Linke */ +@Deprecated public class WrappedResponseInputStream extends InputStream { private Response response; @@ -53,9 +54,12 @@ public int available() throws IOException { } public void close() throws IOException { + if (closed) { + return; + } closed = true; - response.close(); delegate.close(); + response.close(); } public void mark(int readlimit) { diff --git a/docker-java-transport-jersey/src/test/java/com/github/dockerjava/transport/JerseyTests.java b/docker-java-transport-jersey/src/test/java/com/github/dockerjava/transport/JerseyTests.java new file mode 100644 index 000000000..64dfe3966 --- /dev/null +++ b/docker-java-transport-jersey/src/test/java/com/github/dockerjava/transport/JerseyTests.java @@ -0,0 +1,26 @@ +package com.github.dockerjava.transport; + +import com.github.dockerjava.jaxrs.JerseyDockerHttpClient; +import org.junit.Ignore; +import org.junit.Test; + +import java.net.URI; + +public class JerseyTests extends DockerHttpClientTCK { + + @Override + protected DockerHttpClient createDockerHttpClient(URI dockerHost, SSLConfig sslConfig) { + return new JerseyDockerHttpClient.Builder() + .dockerHost(dockerHost) + .sslConfig(sslConfig) + .connectTimeout(30 * 1000) + .build(); + } + + @Test + @Ignore("does not support hijacking") + @Override + public void testHijacking() throws Exception { + super.testHijacking(); + } +} diff --git a/docker-java-transport-netty/pom.xml b/docker-java-transport-netty/pom.xml new file mode 100644 index 000000000..42fdd34b7 --- /dev/null +++ b/docker-java-transport-netty/pom.xml @@ -0,0 +1,72 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java-transport-netty + jar + + docker-java-transport-netty + https://github.com/docker-java/docker-java + Java API Client for Docker + + + com.github.dockerjava.transport.netty + + + + + ${project.groupId} + docker-java-core + ${project.version} + + + + io.netty + netty-codec-http + ${netty.version} + + + io.netty + netty-handler + ${netty.version} + + + io.netty + netty-handler-proxy + ${netty.version} + + + io.netty + netty-transport-native-epoll + ${netty.version} + linux-x86_64 + + + io.netty + netty-transport-native-kqueue + ${netty.version} + osx-x86_64 + + + + + + + org.apache.felix + maven-bundle-plugin + true + + + com.github.dockerjava.netty.* + + + + + + diff --git a/src/main/java/com/github/dockerjava/netty/ChannelProvider.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/ChannelProvider.java similarity index 100% rename from src/main/java/com/github/dockerjava/netty/ChannelProvider.java rename to docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/ChannelProvider.java diff --git a/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/NettyDockerCmdExecFactory.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/NettyDockerCmdExecFactory.java new file mode 100644 index 000000000..ed66a6db8 --- /dev/null +++ b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/NettyDockerCmdExecFactory.java @@ -0,0 +1,337 @@ +package com.github.dockerjava.netty; + +import static java.util.Objects.nonNull; + +import java.io.IOException; +import java.net.InetAddress; +import java.net.InetSocketAddress; +import java.net.SocketAddress; +import java.net.SocketTimeoutException; +import java.security.Security; +import java.util.Objects; +import java.util.concurrent.TimeUnit; + +import javax.net.ssl.SSLEngine; +import javax.net.ssl.SSLParameters; + +import com.github.dockerjava.core.AbstractDockerCmdExecFactory; +import com.github.dockerjava.core.WebTarget; +import org.apache.commons.lang3.SystemUtils; + +import com.github.dockerjava.api.command.DockerCmdExecFactory; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.SSLConfig; +import io.netty.bootstrap.Bootstrap; +import io.netty.channel.Channel; +import io.netty.channel.ChannelConfig; +import io.netty.channel.ChannelFactory; +import io.netty.channel.ChannelHandlerContext; +import io.netty.channel.ChannelInitializer; +import io.netty.channel.EventLoopGroup; +import io.netty.channel.epoll.EpollDomainSocketChannel; +import io.netty.channel.epoll.EpollEventLoopGroup; +import io.netty.channel.kqueue.KQueueDomainSocketChannel; +import io.netty.channel.kqueue.KQueueEventLoopGroup; +import io.netty.channel.nio.NioEventLoopGroup; +import io.netty.channel.socket.DuplexChannel; +import io.netty.channel.socket.SocketChannel; +import io.netty.channel.socket.nio.NioSocketChannel; +import io.netty.channel.unix.DomainSocketAddress; +import io.netty.channel.unix.UnixChannel; +import io.netty.handler.codec.http.HttpClientCodec; +import io.netty.handler.codec.http.HttpContentDecompressor; +import io.netty.handler.logging.LoggingHandler; +import io.netty.handler.ssl.SslHandler; +import io.netty.handler.timeout.IdleState; +import io.netty.handler.timeout.IdleStateEvent; +import io.netty.handler.timeout.IdleStateHandler; +import io.netty.util.concurrent.DefaultThreadFactory; +import org.bouncycastle.jce.provider.BouncyCastleProvider; + + +/** + * Experimental implementation of {@link DockerCmdExecFactory} that supports http connection hijacking that is needed to pass STDIN to the + * container. + *

+ * To use it just pass an instance via {@link com.github.dockerjava.core.DockerClientImpl#withDockerCmdExecFactory(DockerCmdExecFactory)} + * + * @author Marcus Linke + * @see https://docs.docker.com/engine/reference/api/docker_remote_api_v1.21/#attach-to-a-container + * @see https://docs.docker.com/engine/reference/api/docker_remote_api_v1.21/#exec-start + */ +public class NettyDockerCmdExecFactory extends AbstractDockerCmdExecFactory { + + private static String threadPrefix = "dockerjava-netty"; + + /* + * useful links: + * + * http://stackoverflow.com/questions/33296749/netty-connect-to-unix-domain-socket-failed + * http://netty.io/wiki/native-transports.html + * https://github.com/netty/netty/blob/master/example/src/main/java/io/netty/example/http/snoop/HttpSnoopClient.java + * https://github.com/slandelle/netty-request-chunking/blob/master/src/test/java/slandelle/ChunkingTest.java + */ + + private Bootstrap bootstrap; + + private EventLoopGroup eventLoopGroup; + + private NettyInitializer nettyInitializer; + + private WebTarget baseResource; + + private ChannelProvider channelProvider = new ChannelProvider() { + @Override + public DuplexChannel getChannel() { + DuplexChannel channel = connect(); + channel.pipeline().addLast(new LoggingHandler(getClass())); + return channel; + } + }; + + @Override + public void init(DockerClientConfig dockerClientConfig) { + super.init(dockerClientConfig); + + bootstrap = new Bootstrap(); + + String scheme = dockerClientConfig.getDockerHost().getScheme(); + String host = ""; + + switch (scheme) { + case "unix": + nettyInitializer = new UnixDomainSocketInitializer(); + host = "DUMMY"; + break; + case "tcp": + nettyInitializer = new InetSocketInitializer(); + host = dockerClientConfig.getDockerHost().getHost() + ":" + + Integer.toString(dockerClientConfig.getDockerHost().getPort()); + break; + default: + throw new IllegalArgumentException("Unsupported protocol scheme: " + dockerClientConfig.getDockerHost()); + } + + eventLoopGroup = nettyInitializer.init(bootstrap, dockerClientConfig); + + baseResource = new NettyWebTarget(dockerClientConfig.getObjectMapper(), channelProvider, host) + .path(dockerClientConfig.getApiVersion().asWebPathPart()); + } + + + private DuplexChannel connect() { + try { + return connect(bootstrap); + } catch (InterruptedException e) { + throw new RuntimeException(e); + } + } + + private DuplexChannel connect(final Bootstrap bootstrap) throws InterruptedException { + return nettyInitializer.connect(bootstrap); + } + + private interface NettyInitializer { + EventLoopGroup init(final Bootstrap bootstrap, DockerClientConfig dockerClientConfig); + + DuplexChannel connect(final Bootstrap bootstrap) throws InterruptedException; + } + + private class UnixDomainSocketInitializer implements NettyInitializer { + @Override + public EventLoopGroup init(Bootstrap bootstrap, DockerClientConfig dockerClientConfig) { + if (SystemUtils.IS_OS_LINUX) { + return epollGroup(); + } else if (SystemUtils.IS_OS_MAC_OSX) { + return kqueueGroup(); + } + throw new RuntimeException("Unsupported OS"); + } + + public EventLoopGroup epollGroup() { + EventLoopGroup epollEventLoopGroup = new EpollEventLoopGroup(0, new DefaultThreadFactory(threadPrefix)); + + ChannelFactory factory = () -> configure(new EpollDomainSocketChannel()); + + bootstrap.group(epollEventLoopGroup).channelFactory(factory).handler(new ChannelInitializer() { + @Override + protected void initChannel(final UnixChannel channel) throws Exception { + channel.pipeline().addLast(new HttpClientCodec()); + channel.pipeline().addLast(new HttpContentDecompressor()); + } + }); + return epollEventLoopGroup; + } + + public EventLoopGroup kqueueGroup() { + EventLoopGroup nioEventLoopGroup = new KQueueEventLoopGroup(0, new DefaultThreadFactory(threadPrefix)); + + bootstrap.group(nioEventLoopGroup).channel(KQueueDomainSocketChannel.class) + .handler(new ChannelInitializer() { + @Override + protected void initChannel(final KQueueDomainSocketChannel channel) throws Exception { + channel.pipeline().addLast(new LoggingHandler(getClass())); + channel.pipeline().addLast(new HttpClientCodec()); + channel.pipeline().addLast(new HttpContentDecompressor()); + } + }); + + return nioEventLoopGroup; + } + + @Override + public DuplexChannel connect(Bootstrap bootstrap) throws InterruptedException { + DockerClientConfig dockerClientConfig = getDockerClientConfig(); + String path = dockerClientConfig.getDockerHost().getPath(); + + return (DuplexChannel) bootstrap.connect(new DomainSocketAddress(path)).sync().channel(); + } + } + + private class InetSocketInitializer implements NettyInitializer { + @Override + public EventLoopGroup init(Bootstrap bootstrap, final DockerClientConfig dockerClientConfig) { + EventLoopGroup nioEventLoopGroup = new NioEventLoopGroup(0, new DefaultThreadFactory(threadPrefix)); + + InetAddress addr = InetAddress.getLoopbackAddress(); + + final SocketAddress proxyAddress = new InetSocketAddress(addr, 8008); + + Security.addProvider(new BouncyCastleProvider()); + + ChannelFactory factory = () -> configure(new NioSocketChannel()); + + bootstrap.group(nioEventLoopGroup).channelFactory(factory) + .handler(new ChannelInitializer() { + @Override + protected void initChannel(final SocketChannel channel) throws Exception { + // channel.pipeline().addLast(new + // HttpProxyHandler(proxyAddress)); + channel.pipeline().addLast(new HttpClientCodec()); + channel.pipeline().addLast(new HttpContentDecompressor()); + } + }); + + return nioEventLoopGroup; + } + + @Override + public DuplexChannel connect(Bootstrap bootstrap) throws InterruptedException { + DockerClientConfig dockerClientConfig = getDockerClientConfig(); + String host = dockerClientConfig.getDockerHost().getHost(); + int port = dockerClientConfig.getDockerHost().getPort(); + + if (port == -1) { + throw new RuntimeException("no port configured for " + host); + } + + final DuplexChannel channel = (DuplexChannel) bootstrap.connect(host, port).sync().channel(); + + final SslHandler ssl = initSsl(dockerClientConfig); + + if (ssl != null) { + channel.pipeline().addFirst(ssl); + + // https://tools.ietf.org/html/rfc5246#section-7.2.1 + // TLS has its own special message about connection termination. Because TLS is a + // session-level protocol, it can be covered by any transport-level protocol like + // TCP, UTP and so on. But we know exactly that data being transferred over TCP and + // that other side will never send any byte into this TCP connection, so this + // channel should be closed. + // RFC says that we must notify opposite side about closing. This could be done only + // in sun.security.ssl.SSLEngineImpl and unfortunately it does not send this + // message. On the other hand RFC does not enforce the opposite side to wait for + // such message. + ssl.sslCloseFuture().addListener(future -> channel.eventLoop().execute(channel::close)); + } + + return channel; + } + + private SslHandler initSsl(DockerClientConfig dockerClientConfig) { + SslHandler ssl = null; + + try { + String host = dockerClientConfig.getDockerHost().getHost(); + int port = dockerClientConfig.getDockerHost().getPort(); + + final SSLConfig sslConfig = dockerClientConfig.getSSLConfig(); + + if (sslConfig != null && sslConfig.getSSLContext() != null) { + + SSLEngine engine = sslConfig.getSSLContext().createSSLEngine(host, port); + engine.setUseClientMode(true); + engine.setSSLParameters(enableHostNameVerification(engine.getSSLParameters())); + + // in the future we may use HostnameVerifier like here: + // https://github.com/AsyncHttpClient/async-http-client/blob/1.8.x/src/main/java/com/ning/http/client/providers/netty/NettyConnectListener.java#L76 + + ssl = new SslHandler(engine); + } + + } catch (Exception e) { + throw new RuntimeException(e); + } + + return ssl; + } + } + + public SSLParameters enableHostNameVerification(SSLParameters sslParameters) { + sslParameters.setEndpointIdentificationAlgorithm("HTTPS"); + return sslParameters; + } + + @Override + public void close() throws IOException { + Objects.requireNonNull(eventLoopGroup, "Factory not initialized. You probably forgot to call init()!"); + + eventLoopGroup.shutdownGracefully(); + } + + private T configure(T channel) { + ChannelConfig channelConfig = channel.config(); + + if (nonNull(connectTimeout)) { + channelConfig.setConnectTimeoutMillis(connectTimeout); + } + if (nonNull(readTimeout)) { + channel.pipeline().addLast("readTimeoutHandler", new ReadTimeoutHandler()); + } + + return channel; + } + + private final class ReadTimeoutHandler extends IdleStateHandler { + private boolean alreadyTimedOut; + + ReadTimeoutHandler() { + super(readTimeout, 0, 0, TimeUnit.MILLISECONDS); + } + + /** + * Called when a read timeout was detected. + */ + @Override + protected synchronized void channelIdle(ChannelHandlerContext ctx, IdleStateEvent evt) throws Exception { + assert evt.state() == IdleState.READER_IDLE; + final Channel channel = ctx.channel(); + if (channel == null || !channel.isActive() || alreadyTimedOut) { + return; + } + DockerClientConfig dockerClientConfig = getDockerClientConfig(); + final Object dockerAPIEndpoint = dockerClientConfig.getDockerHost(); + final String msg = "Read timed out: No data received within " + readTimeout + + "ms. Perhaps the docker API (" + dockerAPIEndpoint + + ") is not responding normally, or perhaps you need to increase the readTimeout value."; + final Exception ex = new SocketTimeoutException(msg); + ctx.fireExceptionCaught(ex); + alreadyTimedOut = true; + } + } + + protected WebTarget getBaseResource() { + Objects.requireNonNull(baseResource, "Factory not initialized, baseResource not set. You probably forgot to call init()!"); + return baseResource; + } +} diff --git a/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/NettyInvocationBuilder.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/NettyInvocationBuilder.java new file mode 100644 index 000000000..ab13dc7b7 --- /dev/null +++ b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/NettyInvocationBuilder.java @@ -0,0 +1,499 @@ +package com.github.dockerjava.netty; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.InvocationBuilder; +import com.github.dockerjava.core.async.ResultCallbackTemplate; +import com.github.dockerjava.netty.handler.FramedResponseStreamHandler; +import com.github.dockerjava.netty.handler.HttpConnectionHijackHandler; +import com.github.dockerjava.netty.handler.HttpRequestProvider; +import com.github.dockerjava.netty.handler.HttpResponseHandler; +import com.github.dockerjava.netty.handler.HttpResponseStreamHandler; +import com.github.dockerjava.netty.handler.JsonResponseCallbackHandler; +import io.netty.buffer.Unpooled; +import io.netty.channel.Channel; +import io.netty.channel.ChannelFuture; +import io.netty.channel.ChannelFutureListener; +import io.netty.channel.socket.DuplexChannel; +import io.netty.handler.codec.http.DefaultFullHttpRequest; +import io.netty.handler.codec.http.DefaultHttpRequest; +import io.netty.handler.codec.http.FullHttpRequest; +import io.netty.handler.codec.http.HttpClientCodec; +import io.netty.handler.codec.http.HttpClientUpgradeHandler; +import io.netty.handler.codec.http.HttpHeaderNames; +import io.netty.handler.codec.http.HttpHeaderValues; +import io.netty.handler.codec.http.HttpMethod; +import io.netty.handler.codec.http.HttpRequest; +import io.netty.handler.codec.http.HttpVersion; +import io.netty.handler.codec.http.LastHttpContent; +import io.netty.handler.codec.json.JsonObjectDecoder; +import io.netty.handler.stream.ChunkedStream; +import io.netty.handler.stream.ChunkedWriteHandler; + +import java.io.BufferedInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.util.HashMap; +import java.util.Map; + +/** + * This class is basically a replacement of javax.ws.rs.client.Invocation.Builder to allow simpler migration of JAX-RS code to a netty based + * implementation. + * + * @author Marcus Linke + */ +public class NettyInvocationBuilder implements InvocationBuilder { + + public class ResponseCallback extends ResultCallbackTemplate, T> { + + private T result = null; + + public T awaitResult() { + try { + awaitCompletion(); + } catch (InterruptedException e) { + throw new RuntimeException(e); + } + return result; + } + + @Override + public void onNext(T object) { + result = object; + } + } + + public class SkipResultCallback extends ResultCallbackTemplate, Void> { + @Override + public void onNext(Void object) { + } + } + + private ChannelProvider channelProvider; + + private String resource; + + private Map headers = new HashMap<>(); + + private final ObjectMapper objectMapper; + + @Deprecated + public NettyInvocationBuilder(ChannelProvider channelProvider, String resource) { + this( + DockerClientConfig.getDefaultObjectMapper(), + channelProvider, + resource + ); + } + + public NettyInvocationBuilder(ObjectMapper objectMapper, ChannelProvider channelProvider, String resource) { + this.objectMapper = objectMapper; + this.channelProvider = channelProvider; + this.resource = resource; + } + + @Override + public InvocationBuilder accept(com.github.dockerjava.core.MediaType mediaType) { + return header(HttpHeaderNames.ACCEPT.toString(), mediaType.getMediaType()); + } + + public NettyInvocationBuilder header(String name, String value) { + headers.put(name, value); + return this; + } + + public void delete() { + + HttpRequestProvider requestProvider = httpDeleteRequestProvider(); + + try (ResponseCallback callback = new ResponseCallback<>()) { + + HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, callback); + + Channel channel = getChannel(); + + channel.pipeline().addLast(responseHandler); + + sendRequest(requestProvider, channel); + + callback.awaitResult(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + public void get(ResultCallback resultCallback) { + + HttpRequestProvider requestProvider = httpGetRequestProvider(); + + HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); + + FramedResponseStreamHandler streamHandler = new FramedResponseStreamHandler(resultCallback); + + Channel channel = getChannel(); + + channel.pipeline().addLast(responseHandler); + channel.pipeline().addLast(streamHandler); + + sendRequest(requestProvider, channel); + } + + public T get(TypeReference typeReference) { + try (ResponseCallback callback = new ResponseCallback<>()) { + get(typeReference, callback); + + return callback.awaitResult(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + public void get(TypeReference typeReference, ResultCallback resultCallback) { + + HttpRequestProvider requestProvider = httpGetRequestProvider(); + + Channel channel = getChannel(); + + JsonResponseCallbackHandler jsonResponseHandler = new JsonResponseCallbackHandler<>( + objectMapper, + typeReference, + resultCallback); + + HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); + + channel.pipeline().addLast(responseHandler); + channel.pipeline().addLast(new JsonObjectDecoder(3 * 1024 * 1024)); + channel.pipeline().addLast(jsonResponseHandler); + + sendRequest(requestProvider, channel); + + return; + } + + private DuplexChannel getChannel() { + return channelProvider.getChannel(); + } + + private HttpRequestProvider httpDeleteRequestProvider() { + return this::prepareDeleteRequest; + } + + private HttpRequestProvider httpGetRequestProvider() { + return this::prepareGetRequest; + } + + private HttpRequestProvider httpPostRequestProvider(final Object entity) { + return uri -> preparePostRequest(uri, entity); + } + + private HttpRequestProvider httpPutRequestProvider(final Object entity) { + return uri -> preparePutRequest(uri, entity); + } + + public InputStream post(final Object entity) { + + HttpRequestProvider requestProvider = httpPostRequestProvider(entity); + + Channel channel = getChannel(); + + AsyncResultCallback callback = new AsyncResultCallback<>(); + + HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, callback); + HttpResponseStreamHandler streamHandler = new HttpResponseStreamHandler(callback); + + channel.pipeline().addLast(responseHandler); + channel.pipeline().addLast(streamHandler); + + sendRequest(requestProvider, channel); + + return callback.awaitResult(); + } + + public void post(final Object entity, final InputStream stdin, final ResultCallback resultCallback) { + + HttpRequestProvider requestProvider = httpPostRequestProvider(entity); + + FramedResponseStreamHandler streamHandler = new FramedResponseStreamHandler(resultCallback); + + final DuplexChannel channel = getChannel(); + + // result callback's close() method must be called when the servers closes the connection + channel.closeFuture().addListener(future -> resultCallback.onComplete()); + + HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); + + HttpConnectionHijackHandler hijackHandler = new HttpConnectionHijackHandler(responseHandler); + + HttpClientCodec httpClientCodec = channel.pipeline().get(HttpClientCodec.class); + + channel.pipeline().addLast( + new HttpClientUpgradeHandler(httpClientCodec, hijackHandler, Integer.MAX_VALUE)); + channel.pipeline().addLast(streamHandler); + + sendRequest(requestProvider, channel); + + // wait for successful http upgrade procedure + hijackHandler.awaitUpgrade(); + + if (stdin != null) { + // now we can start a new thread that reads from stdin and writes to the channel + new Thread(new Runnable() { + + private int read(InputStream is, byte[] buf) { + try { + return is.read(buf); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Override + public void run() { + + byte[] buffer = new byte[1024]; + + int read; + while ((read = read(stdin, buffer)) != -1) { + channel.writeAndFlush(Unpooled.copiedBuffer(buffer, 0, read)); + } + + // we close the writing side of the socket, but keep the read side open to transfer stdout/stderr + channel.shutdownOutput(); + + } + }).start(); + } + } + + public T post(final Object entity, TypeReference typeReference) { + try (ResponseCallback callback = new ResponseCallback<>()) { + post(entity, typeReference, callback); + + return callback.awaitResult(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + public void post(final Object entity, TypeReference typeReference, final ResultCallback resultCallback) { + + HttpRequestProvider requestProvider = httpPostRequestProvider(entity); + + Channel channel = getChannel(); + + JsonResponseCallbackHandler jsonResponseHandler = new JsonResponseCallbackHandler<>( + objectMapper, + typeReference, + resultCallback); + + HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); + + channel.pipeline().addLast(responseHandler); + channel.pipeline().addLast(new JsonObjectDecoder(3 * 1024 * 1024)); + channel.pipeline().addLast(jsonResponseHandler); + + sendRequest(requestProvider, channel); + + return; + } + + private HttpRequest prepareDeleteRequest(String uri) { + + FullHttpRequest request = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, HttpMethod.DELETE, uri); + + setDefaultHeaders(request); + + return request; + } + + private FullHttpRequest prepareGetRequest(String uri) { + + FullHttpRequest request = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, HttpMethod.GET, uri); + + setDefaultHeaders(request); + + return request; + } + + private HttpRequest preparePostRequest(String uri, Object entity) { + return prepareEntityRequest(uri, entity, HttpMethod.POST); + } + + private HttpRequest preparePutRequest(String uri, Object entity) { + return prepareEntityRequest(uri, entity, HttpMethod.PUT); + } + + private HttpRequest prepareEntityRequest(String uri, Object entity, HttpMethod httpMethod) { + + HttpRequest request = null; + + if (entity != null) { + + FullHttpRequest fullRequest = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, httpMethod, uri); + + byte[] bytes; + try { + bytes = objectMapper.writeValueAsBytes(entity); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + + fullRequest.headers().set(HttpHeaderNames.CONTENT_TYPE, "application/json"); + fullRequest.content().clear().writeBytes(Unpooled.copiedBuffer(bytes)); + fullRequest.headers().set(HttpHeaderNames.CONTENT_LENGTH, bytes.length); + + request = fullRequest; + } else { + request = new DefaultHttpRequest(HttpVersion.HTTP_1_1, httpMethod, uri); + request.headers().set(HttpHeaderNames.CONTENT_LENGTH, 0); + } + + setDefaultHeaders(request); + + return request; + } + + private void sendRequest(HttpRequestProvider requestProvider, Channel channel) { + + ChannelFuture channelFuture = channel.writeAndFlush(requestProvider.getHttpRequest(resource)); + + channelFuture.addListener((ChannelFutureListener) future -> { + }); + } + + private void setDefaultHeaders(HttpRequest request) { + request.headers().set(HttpHeaderNames.HOST, ""); + request.headers().set(HttpHeaderNames.CONNECTION, HttpHeaderValues.KEEP_ALIVE); + request.headers().set(HttpHeaderNames.ACCEPT_ENCODING, HttpHeaderValues.GZIP); + + for (Map.Entry entry : headers.entrySet()) { + request.headers().set((CharSequence) entry.getKey(), entry.getValue()); + } + } + + public T post(TypeReference typeReference, InputStream body) { + try (ResponseCallback callback = new ResponseCallback<>()) { + post(typeReference, callback, body); + + return callback.awaitResult(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + public void post(TypeReference typeReference, ResultCallback resultCallback, InputStream body) { + HttpRequestProvider requestProvider = httpPostRequestProvider(null); + + Channel channel = getChannel(); + + JsonResponseCallbackHandler jsonResponseHandler = new JsonResponseCallbackHandler<>( + objectMapper, + typeReference, + resultCallback); + + HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); + + channel.pipeline().addLast(new ChunkedWriteHandler()); + channel.pipeline().addLast(responseHandler); + channel.pipeline().addLast(new JsonObjectDecoder(3 * 1024 * 1024)); + channel.pipeline().addLast(jsonResponseHandler); + + postChunkedStreamRequest(requestProvider, channel, body); + } + + public void postStream(InputStream body) { + SkipResultCallback resultCallback = new SkipResultCallback(); + + HttpRequestProvider requestProvider = httpPostRequestProvider(null); + + Channel channel = getChannel(); + + HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); + + channel.pipeline().addLast(new ChunkedWriteHandler()); + channel.pipeline().addLast(responseHandler); + + postChunkedStreamRequest(requestProvider, channel, body); + + try { + resultCallback.awaitCompletion(); + } catch (InterruptedException e) { + throw new RuntimeException(e); + } + } + + private void postChunkedStreamRequest(HttpRequestProvider requestProvider, Channel channel, InputStream body) { + HttpRequest request = requestProvider.getHttpRequest(resource); + + // don't accept FullHttpRequest here + if (request instanceof FullHttpRequest) { + throw new DockerClientException("fatal: request is instance of FullHttpRequest"); + } + + request.headers().set(HttpHeaderNames.TRANSFER_ENCODING, HttpHeaderValues.CHUNKED); + request.headers().remove(HttpHeaderNames.CONTENT_LENGTH); + + channel.write(request); + + channel.write(new ChunkedStream(new BufferedInputStream(body, 1024 * 1024), 1024 * 1024)); + channel.write(LastHttpContent.EMPTY_LAST_CONTENT); + channel.flush(); + } + + public InputStream get() { + HttpRequestProvider requestProvider = httpGetRequestProvider(); + + Channel channel = getChannel(); + + AsyncResultCallback resultCallback = new AsyncResultCallback<>(); + + HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); + + HttpResponseStreamHandler streamHandler = new HttpResponseStreamHandler(resultCallback); + + channel.pipeline().addLast(responseHandler); + channel.pipeline().addLast(streamHandler); + + sendRequest(requestProvider, channel); + + return resultCallback.awaitResult(); + } + + @Override + public void put(InputStream body, com.github.dockerjava.core.MediaType mediaType) { + HttpRequestProvider requestProvider = httpPutRequestProvider(null); + + Channel channel = getChannel(); + + try (ResponseCallback resultCallback = new ResponseCallback<>()) { + HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); + + channel.pipeline().addLast(new ChunkedWriteHandler()); + channel.pipeline().addLast(responseHandler); + + HttpRequest request = requestProvider.getHttpRequest(resource); + + // don't accept FullHttpRequest here + if (request instanceof FullHttpRequest) { + throw new DockerClientException("fatal: request is instance of FullHttpRequest"); + } + + request.headers().set(HttpHeaderNames.TRANSFER_ENCODING, HttpHeaderValues.CHUNKED); + request.headers().remove(HttpHeaderNames.CONTENT_LENGTH); + request.headers().set(HttpHeaderNames.CONTENT_TYPE, mediaType.getMediaType()); + + channel.write(request); + channel.write(new ChunkedStream(new BufferedInputStream(body, 1024 * 1024))); + channel.writeAndFlush(LastHttpContent.EMPTY_LAST_CONTENT); + + resultCallback.awaitResult(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } +} diff --git a/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/NettyWebTarget.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/NettyWebTarget.java new file mode 100644 index 000000000..8f2ffce27 --- /dev/null +++ b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/NettyWebTarget.java @@ -0,0 +1,210 @@ +package com.github.dockerjava.netty; + +import java.io.IOException; +import java.io.UnsupportedEncodingException; +import java.net.URLEncoder; +import java.nio.charset.Charset; +import java.nio.charset.UnsupportedCharsetException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.Map; +import java.util.Objects; +import java.util.Set; + +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.WebTarget; +import com.google.common.collect.ImmutableSet; +import io.netty.handler.codec.http.HttpConstants; +import org.apache.commons.lang3.StringUtils; + +import com.google.common.collect.ImmutableList; +import com.google.common.collect.ImmutableMap; + +/** + * This class is basically a replacement of {@link javax.ws.rs.client.WebTarget} to allow simpler migration of JAX-RS code to a netty based + * implementation. + * + * @author Marcus Linke + */ +public class NettyWebTarget implements WebTarget { + + private final ChannelProvider channelProvider; + + private final String host; + + private final ImmutableList path; + + private final ImmutableMap queryParams; + + /** + * Multiple values for the same name param. + */ + private final ImmutableMap> queryParamsSet; + + private static final String PATH_SEPARATOR = "/"; + + private final ObjectMapper objectMapper; + + @Deprecated + public NettyWebTarget(ChannelProvider channelProvider, String host) { + this( + DockerClientConfig.getDefaultObjectMapper(), + channelProvider, + host, + ImmutableList.of(), + ImmutableMap.of(), + ImmutableMap.of() + ); + } + + public NettyWebTarget(ObjectMapper objectMapper, ChannelProvider channelProvider, String host) { + this( + objectMapper, + channelProvider, + host, + ImmutableList.of(), + ImmutableMap.of(), + ImmutableMap.of() + ); + } + + private NettyWebTarget( + ObjectMapper objectMapper, + ChannelProvider channelProvider, + String host, + ImmutableList path, + ImmutableMap queryParams, + ImmutableMap> queryParamsSet + ) { + this.objectMapper = objectMapper; + this.channelProvider = channelProvider; + this.host = host; + this.path = path; + this.queryParams = queryParams; + this.queryParamsSet = queryParamsSet; + } + + public NettyWebTarget path(String... components) { + ImmutableList.Builder newPath = ImmutableList.builder().addAll(this.path); + + for (String component : components) { + newPath.addAll(Arrays.asList(StringUtils.split(component, PATH_SEPARATOR))); + } + + return new NettyWebTarget(objectMapper, channelProvider, host, newPath.build(), queryParams, queryParamsSet); + } + + public NettyInvocationBuilder request() { + String resource = PATH_SEPARATOR + StringUtils.join(path, PATH_SEPARATOR); + + List params = new ArrayList<>(); + for (Map.Entry entry : queryParams.entrySet()) { + params.add(entry.getKey() + "=" + encodeComponent(entry.getValue(), HttpConstants.DEFAULT_CHARSET)); + } + + for (Map.Entry> entry : queryParamsSet.entrySet()) { + for (String entryValueValue : entry.getValue()) { + params.add(entry.getKey() + "=" + encodeComponent(entryValueValue, HttpConstants.DEFAULT_CHARSET)); + } + } + + if (!params.isEmpty()) { + resource = resource + "?" + StringUtils.join(params, "&"); + } + + return new NettyInvocationBuilder(objectMapper, channelProvider, resource) + .header("Host", host); + } + + /** + * @see io.netty.handler.codec.http.QueryStringEncoder + */ + private static String encodeComponent(String s, Charset charset) { + // TODO: Optimize me. + try { + return URLEncoder.encode(s, charset.name()).replace("+", "%20"); + } catch (UnsupportedEncodingException ignored) { + throw new UnsupportedCharsetException(charset.name()); + } + } + + public NettyWebTarget resolveTemplate(String name, Object value) { + ImmutableList.Builder newPath = ImmutableList.builder(); + for (String component : path) { + component = component.replaceAll("\\{" + name + "\\}", value.toString()); + newPath.add(component); + } + return new NettyWebTarget(objectMapper, channelProvider, host, newPath.build(), queryParams, queryParamsSet); + } + + public NettyWebTarget queryParam(String name, Object value) { + ImmutableMap.Builder builder = ImmutableMap.builder().putAll(queryParams); + if (value != null) { + builder.put(name, value.toString()); + } + return new NettyWebTarget(objectMapper, channelProvider, host, path, builder.build(), queryParamsSet); + } + + public NettyWebTarget queryParamsSet(String name, Set values) { + ImmutableMap.Builder> builder = ImmutableMap.>builder().putAll(queryParamsSet); + if (values != null) { + ImmutableSet.Builder valueBuilder = ImmutableSet.builder(); + for (Object value : values) { + valueBuilder.add(value.toString()); + } + builder.put(name, valueBuilder.build()); + } + return new NettyWebTarget(objectMapper, channelProvider, host, path, queryParams, builder.build()); + } + + public NettyWebTarget queryParamsJsonMap(String name, Map values) { + if (values != null && !values.isEmpty()) { + try { + // when param value is JSON string + return queryParam(name, objectMapper.writeValueAsString(values)); + } catch (IOException e) { + throw new RuntimeException(e); + } + } else { + return this; + } + } + + @Override + public boolean equals(Object o) { + if (this == o) { + return true; + } + if (o == null || getClass() != o.getClass()) { + return false; + } + + NettyWebTarget webTarget = (NettyWebTarget) o; + + if (!Objects.equals(channelProvider, webTarget.channelProvider)) { + return false; + } + if (!Objects.equals(path, webTarget.path)) { + return false; + } + if (!Objects.equals(queryParams, webTarget.queryParams)) { + return false; + } + if (!Objects.equals(queryParamsSet, webTarget.queryParamsSet)) { + return false; + } + + return true; + } + + @Override + public int hashCode() { + int result = channelProvider != null ? channelProvider.hashCode() : 0; + result = 31 * result + (path != null ? path.hashCode() : 0); + result = 31 * result + (queryParams != null ? queryParams.hashCode() : 0); + result = 31 * result + (queryParamsSet != null ? queryParamsSet.hashCode() : 0); + return result; + } +} diff --git a/src/main/java/com/github/dockerjava/netty/handler/FramedResponseStreamHandler.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/FramedResponseStreamHandler.java similarity index 100% rename from src/main/java/com/github/dockerjava/netty/handler/FramedResponseStreamHandler.java rename to docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/FramedResponseStreamHandler.java diff --git a/src/main/java/com/github/dockerjava/netty/handler/HttpConnectionHijackHandler.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/HttpConnectionHijackHandler.java similarity index 100% rename from src/main/java/com/github/dockerjava/netty/handler/HttpConnectionHijackHandler.java rename to docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/HttpConnectionHijackHandler.java diff --git a/src/main/java/com/github/dockerjava/netty/handler/HttpRequestProvider.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/HttpRequestProvider.java similarity index 100% rename from src/main/java/com/github/dockerjava/netty/handler/HttpRequestProvider.java rename to docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/HttpRequestProvider.java diff --git a/src/main/java/com/github/dockerjava/netty/handler/HttpResponseHandler.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/HttpResponseHandler.java similarity index 95% rename from src/main/java/com/github/dockerjava/netty/handler/HttpResponseHandler.java rename to docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/HttpResponseHandler.java index 9986e10e9..548742875 100644 --- a/src/main/java/com/github/dockerjava/netty/handler/HttpResponseHandler.java +++ b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/HttpResponseHandler.java @@ -11,7 +11,6 @@ import io.netty.handler.codec.http.HttpResponse; import io.netty.handler.codec.http.LastHttpContent; -import java.io.Closeable; import java.nio.charset.Charset; import com.github.dockerjava.api.async.ResultCallback; @@ -52,12 +51,7 @@ protected void channelRead0(final ChannelHandlerContext ctx, HttpObject msg) thr response = (HttpResponse) msg; - resultCallback.onStart(new Closeable() { - @Override - public void close() { - ctx.channel().close(); - } - }); + resultCallback.onStart(() -> ctx.channel().close()); } else if (msg instanceof HttpContent) { diff --git a/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandler.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandler.java new file mode 100644 index 000000000..596334640 --- /dev/null +++ b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandler.java @@ -0,0 +1,177 @@ +package com.github.dockerjava.netty.handler; + +import io.netty.buffer.ByteBuf; +import io.netty.channel.ChannelHandlerContext; +import io.netty.channel.SimpleChannelInboundHandler; + +import java.io.IOException; +import java.io.InputStream; + +import com.github.dockerjava.api.async.ResultCallback; + +/** + * Handler that converts an incoming byte stream to an {@link InputStream}. + * + * @author marcus + */ +public class HttpResponseStreamHandler extends SimpleChannelInboundHandler { + + private ResultCallback resultCallback; + + private final HttpResponseInputStream stream = new HttpResponseInputStream(); + + public HttpResponseStreamHandler(ResultCallback resultCallback) { + this.resultCallback = resultCallback; + } + + @Override + protected void channelRead0(ChannelHandlerContext ctx, ByteBuf msg) throws Exception { + invokeCallbackOnFirstRead(); + + stream.write(msg.copy()); + } + + private void invokeCallbackOnFirstRead() { + if (resultCallback != null) { + resultCallback.onNext(stream); + resultCallback = null; + } + } + + @Override + public void channelInactive(ChannelHandlerContext ctx) throws Exception { + stream.writeComplete(); + + super.channelInactive(ctx); + } + + public static class HttpResponseInputStream extends InputStream { + + private boolean writeCompleted = false; + + private boolean closed = false; + + private ByteBuf current = null; + + private final Object lock = new Object(); + + public void write(ByteBuf byteBuf) throws InterruptedException { + synchronized (lock) { + if (closed) { + return; + } + while (current != null) { + lock.wait(); + + if (closed) { + return; + } + } + current = byteBuf; + + lock.notifyAll(); + } + } + + public void writeComplete() { + synchronized (lock) { + writeCompleted = true; + + lock.notifyAll(); + } + } + + @Override + public void close() throws IOException { + synchronized (lock) { + closed = true; + releaseCurrent(); + + lock.notifyAll(); + } + } + + @Override + public int available() throws IOException { + synchronized (lock) { + poll(0); + return readableBytes(); + } + } + + private int readableBytes() { + if (current != null) { + return current.readableBytes(); + } else { + return 0; + } + } + + @Override + public int read() throws IOException { + byte[] b = new byte[1]; + int n = read(b, 0, 1); + return n != -1 ? b[0] : -1; + } + + @Override + public int read(byte[] b, int off, int len) throws IOException { + synchronized (lock) { + off = poll(off); + + if (current == null) { + return -1; + } else { + int availableBytes = Math.min(len, current.readableBytes() - off); + current.readBytes(b, off, availableBytes); + return availableBytes; + } + } + } + + private int poll(int off) throws IOException { + synchronized (lock) { + while (readableBytes() <= off) { + try { + if (closed) { + throw new IOException("Stream closed"); + } + + off -= releaseCurrent(); + if (writeCompleted) { + return off; + } + while (current == null) { + lock.wait(); + + if (closed) { + throw new IOException("Stream closed"); + } + if (writeCompleted && current == null) { + return off; + } + } + } catch (InterruptedException e) { + throw new RuntimeException(e); + } + } + return off; + } + } + + private int releaseCurrent() { + synchronized (lock) { + if (current != null) { + int n = current.readableBytes(); + current.release(); + current = null; + + lock.notifyAll(); + + return n; + } + return 0; + } + } + } +} diff --git a/src/main/java/com/github/dockerjava/netty/handler/JsonRequestHandler.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/JsonRequestHandler.java similarity index 76% rename from src/main/java/com/github/dockerjava/netty/handler/JsonRequestHandler.java rename to docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/JsonRequestHandler.java index cb6f60678..b122c5090 100644 --- a/src/main/java/com/github/dockerjava/netty/handler/JsonRequestHandler.java +++ b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/JsonRequestHandler.java @@ -1,19 +1,22 @@ package com.github.dockerjava.netty.handler; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.dockerjava.core.DockerClientConfig; import io.netty.buffer.ByteBuf; import io.netty.channel.ChannelHandlerContext; import io.netty.handler.codec.MessageToByteEncoder; -import com.fasterxml.jackson.databind.ObjectMapper; - /** * Handler that encodes an outgoing object to JSON. * * @author Marcus Linke + * + * @deprecated unused in docker-java */ +@Deprecated public class JsonRequestHandler extends MessageToByteEncoder { - private ObjectMapper mapper = new ObjectMapper(); + private ObjectMapper mapper = DockerClientConfig.getDefaultObjectMapper(); @Override protected void encode(ChannelHandlerContext ctx, Object msg, ByteBuf out) throws Exception { diff --git a/src/main/java/com/github/dockerjava/netty/handler/JsonResponseCallbackHandler.java b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/JsonResponseCallbackHandler.java similarity index 77% rename from src/main/java/com/github/dockerjava/netty/handler/JsonResponseCallbackHandler.java rename to docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/JsonResponseCallbackHandler.java index db20d6216..f6e8af3c3 100644 --- a/src/main/java/com/github/dockerjava/netty/handler/JsonResponseCallbackHandler.java +++ b/docker-java-transport-netty/src/main/java/com/github/dockerjava/netty/handler/JsonResponseCallbackHandler.java @@ -1,5 +1,6 @@ package com.github.dockerjava.netty.handler; +import com.github.dockerjava.core.DockerClientConfig; import io.netty.buffer.ByteBuf; import io.netty.channel.ChannelHandlerContext; import io.netty.channel.SimpleChannelInboundHandler; @@ -15,13 +16,23 @@ */ public class JsonResponseCallbackHandler extends SimpleChannelInboundHandler { - private static ObjectMapper objectMapper = new ObjectMapper(); + private final ObjectMapper objectMapper; private TypeReference typeReference; private ResultCallback callback; + @Deprecated public JsonResponseCallbackHandler(TypeReference typeReference, ResultCallback callback) { + this( + DockerClientConfig.getDefaultObjectMapper(), + typeReference, + callback + ); + } + + public JsonResponseCallbackHandler(ObjectMapper objectMapper, TypeReference typeReference, ResultCallback callback) { + this.objectMapper = objectMapper; this.typeReference = typeReference; this.callback = callback; } diff --git a/docker-java-transport-okhttp/pom.xml b/docker-java-transport-okhttp/pom.xml new file mode 100644 index 000000000..2a0ae4227 --- /dev/null +++ b/docker-java-transport-okhttp/pom.xml @@ -0,0 +1,82 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java-transport-okhttp + jar + + docker-java-transport-okhttp + https://github.com/docker-java/docker-java + Java API Client for Docker + + + com.github.dockerjava.transport.okhttp + + + + + ${project.groupId} + docker-java-core + ${project.version} + + + + com.squareup.okhttp3 + okhttp + 3.14.9 + + + + net.java.dev.jna + jna + 5.18.1 + + + + ${project.groupId} + docker-java-transport-tck + ${project.version} + test + + + + + + + com.github.siom79.japicmp + japicmp-maven-plugin + + + + + SUPERCLASS_REMOVED + true + true + + + + com.github.dockerjava.okhttp.UnixDomainSocket$SockAddr + + + + + + + org.apache.felix + maven-bundle-plugin + true + + + com.github.dockerjava.okhttp.* + + + + + + diff --git a/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/HijackingInterceptor.java b/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/HijackingInterceptor.java new file mode 100644 index 000000000..275d8290b --- /dev/null +++ b/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/HijackingInterceptor.java @@ -0,0 +1,61 @@ +package com.github.dockerjava.okhttp; + +import com.github.dockerjava.transport.DockerHttpClient; +import okhttp3.Interceptor; +import okhttp3.Request; +import okhttp3.Response; +import okhttp3.internal.connection.Exchange; +import okhttp3.internal.http.RealInterceptorChain; +import okhttp3.internal.ws.RealWebSocket; +import okio.BufferedSink; + +import java.io.IOException; +import java.io.InputStream; + +class HijackingInterceptor implements Interceptor { + + @Override + public Response intercept(Chain chain) throws IOException { + Request request = chain.request(); + Response response = chain.proceed(request); + if (!response.isSuccessful()) { + return response; + } + + DockerHttpClient.Request originalRequest = request.tag(DockerHttpClient.Request.class); + + if (originalRequest == null) { + // WTF? + return response; + } + + InputStream stdin = originalRequest.hijackedInput(); + + if (stdin == null) { + return response; + } + + chain.call().timeout().clearTimeout().clearDeadline(); + + Exchange exchange = ((RealInterceptorChain) chain).exchange(); + RealWebSocket.Streams streams = exchange.newWebSocketStreams(); + Thread thread = new Thread(() -> { + try (BufferedSink sink = streams.sink) { + while (sink.isOpen()) { + int aByte = stdin.read(); + if (aByte < 0) { + break; + } + sink.writeByte(aByte); + sink.emit(); + } + } catch (Exception e) { + throw new RuntimeException(e); + } + }); + thread.setName("okhttp-hijack-streaming-" + System.identityHashCode(request)); + thread.setDaemon(true); + thread.start(); + return response; + } +} diff --git a/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/NamedPipeSocketFactory.java b/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/NamedPipeSocketFactory.java new file mode 100644 index 000000000..066ae7ce8 --- /dev/null +++ b/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/NamedPipeSocketFactory.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.okhttp; + +import com.github.dockerjava.transport.NamedPipeSocket; + +import javax.net.SocketFactory; +import java.net.InetAddress; +import java.net.Socket; + +class NamedPipeSocketFactory extends SocketFactory { + + final String socketFileName; + + NamedPipeSocketFactory(String socketFileName) { + this.socketFileName = socketFileName; + } + + @Override + public Socket createSocket() { + return new NamedPipeSocket(socketFileName); + } + + @Override + public Socket createSocket(String s, int i) { + throw new UnsupportedOperationException(); + } + + @Override + public Socket createSocket(String s, int i, InetAddress inetAddress, int i1) { + throw new UnsupportedOperationException(); + } + + @Override + public Socket createSocket(InetAddress inetAddress, int i) { + throw new UnsupportedOperationException(); + } + + @Override + public Socket createSocket(InetAddress inetAddress, int i, InetAddress inetAddress1, int i1) { + throw new UnsupportedOperationException(); + } +} diff --git a/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/OkDockerHttpClient.java b/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/OkDockerHttpClient.java new file mode 100644 index 000000000..ee58acb09 --- /dev/null +++ b/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/OkDockerHttpClient.java @@ -0,0 +1,314 @@ +package com.github.dockerjava.okhttp; + +import com.github.dockerjava.transport.DockerHttpClient; +import com.github.dockerjava.transport.SSLConfig; +import okhttp3.Call; +import okhttp3.ConnectionPool; +import okhttp3.Dns; +import okhttp3.HttpUrl; +import okhttp3.MediaType; +import okhttp3.OkHttpClient; +import okhttp3.RequestBody; +import okhttp3.ResponseBody; +import okio.BufferedSink; +import okio.Okio; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.net.ssl.SSLContext; +import javax.net.ssl.X509TrustManager; +import java.io.IOException; +import java.io.InputStream; +import java.io.UncheckedIOException; +import java.net.InetAddress; +import java.net.URI; +import java.security.cert.X509Certificate; +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.Objects; +import java.util.concurrent.TimeUnit; + +public final class OkDockerHttpClient implements DockerHttpClient { + + private static final Logger LOGGER = LoggerFactory.getLogger(OkDockerHttpClient.class); + + public static final class Builder { + + private URI dockerHost = null; + + private SSLConfig sslConfig = null; + + private Integer readTimeout = null; + + private Integer connectTimeout = null; + + private Boolean retryOnConnectionFailure = null; + + public Builder dockerHost(URI value) { + this.dockerHost = Objects.requireNonNull(value, "dockerHost"); + return this; + } + + public Builder sslConfig(SSLConfig value) { + this.sslConfig = value; + return this; + } + + public Builder readTimeout(Integer value) { + this.readTimeout = value; + return this; + } + + public Builder connectTimeout(Integer value) { + this.connectTimeout = value; + return this; + } + + Builder retryOnConnectionFailure(Boolean value) { + this.retryOnConnectionFailure = value; + return this; + } + + public OkDockerHttpClient build() { + Objects.requireNonNull(dockerHost, "dockerHost"); + return new OkDockerHttpClient( + dockerHost, + sslConfig, + readTimeout, + connectTimeout, + retryOnConnectionFailure + ); + } + } + + private static final String SOCKET_SUFFIX = ".socket"; + + final OkHttpClient client; + + final OkHttpClient streamingClient; + + private final HttpUrl baseUrl; + + private OkDockerHttpClient( + URI dockerHost, + SSLConfig sslConfig, + Integer readTimeout, + Integer connectTimeout, + Boolean retryOnConnectionFailure + ) { + okhttp3.OkHttpClient.Builder clientBuilder = new okhttp3.OkHttpClient.Builder() + .addNetworkInterceptor(new HijackingInterceptor()) + .readTimeout(0, TimeUnit.MILLISECONDS) + .retryOnConnectionFailure(true); + + if (readTimeout != null) { + clientBuilder.readTimeout(readTimeout, TimeUnit.MILLISECONDS); + } + + if (connectTimeout != null) { + clientBuilder.connectTimeout(connectTimeout, TimeUnit.MILLISECONDS); + } + + if (retryOnConnectionFailure != null) { + clientBuilder.retryOnConnectionFailure(retryOnConnectionFailure); + } + + switch (dockerHost.getScheme()) { + case "unix": + case "npipe": + String socketPath = dockerHost.getPath(); + + if ("unix".equals(dockerHost.getScheme())) { + clientBuilder.socketFactory(new UnixSocketFactory(socketPath)); + } else { + clientBuilder.socketFactory(new NamedPipeSocketFactory(socketPath)); + } + + clientBuilder + .connectionPool(new ConnectionPool(0, 1, TimeUnit.SECONDS)) + .dns(hostname -> { + if (hostname.endsWith(SOCKET_SUFFIX)) { + return Collections.singletonList(InetAddress.getByAddress(hostname, new byte[]{0, 0, 0, 0})); + } else { + return Dns.SYSTEM.lookup(hostname); + } + }); + break; + default: + } + + boolean isSSL = false; + if (sslConfig != null) { + try { + SSLContext sslContext = sslConfig.getSSLContext(); + if (sslContext != null) { + isSSL = true; + clientBuilder.sslSocketFactory(sslContext.getSocketFactory(), new TrustAllX509TrustManager()); + } + } catch (Exception e) { + throw new RuntimeException(e); + } + } + + client = clientBuilder.build(); + + streamingClient = client.newBuilder().build(); + + HttpUrl.Builder baseUrlBuilder; + + switch (dockerHost.getScheme()) { + case "unix": + case "npipe": + baseUrlBuilder = new HttpUrl.Builder() + .scheme("http") + .host("docker" + SOCKET_SUFFIX); + break; + case "tcp": + baseUrlBuilder = new HttpUrl.Builder() + .scheme(isSSL ? "https" : "http") + .host(dockerHost.getHost()) + .port(dockerHost.getPort()); + + if (dockerHost.getPath().length() > 0) { + baseUrlBuilder = baseUrlBuilder.encodedPath(dockerHost.getPath()); + } + break; + default: + throw new IllegalArgumentException("Unsupported protocol scheme: " + dockerHost); + } + baseUrl = baseUrlBuilder.build(); + } + + private RequestBody toRequestBody(Request request) { + byte[] bodyBytes = request.bodyBytes(); + if (bodyBytes != null) { + return RequestBody.create(null, bodyBytes); + } + + InputStream body = request.body(); + if (body != null) { + return new RequestBody() { + @Override + public MediaType contentType() { + return null; + } + + @Override + public void writeTo(BufferedSink sink) throws IOException { + sink.writeAll(Okio.source(body)); + } + }; + } + switch (request.method()) { + case "POST": + return RequestBody.create(null, ""); + default: + return null; + } + } + + @Override + public Response execute(Request request) { + String url = baseUrl.toString(); + if (url.endsWith("/") && request.path().startsWith("/")) { + url = url.substring(0, url.length() - 1); + } + okhttp3.Request.Builder requestBuilder = new okhttp3.Request.Builder() + .url(url + request.path()) + .tag(Request.class, request) + .method(request.method(), toRequestBody(request)); + + request.headers().forEach(requestBuilder::header); + + final OkHttpClient clientToUse; + + if (request.hijackedInput() == null) { + clientToUse = client; + } else { + clientToUse = streamingClient; + } + + Call call = clientToUse.newCall(requestBuilder.build()); + try { + return new OkResponse(call); + } catch (IOException e) { + call.cancel(); + throw new UncheckedIOException("Error while executing " + request, e); + } + } + + @Override + public void close() throws IOException { + for (OkHttpClient clientToClose : new OkHttpClient[]{client, streamingClient}) { + clientToClose.dispatcher().cancelAll(); + clientToClose.dispatcher().executorService().shutdown(); + clientToClose.connectionPool().evictAll(); + } + } + + static class OkResponse implements Response { + + static final ThreadLocal CLOSING = ThreadLocal.withInitial(() -> false); + + private final Call call; + + private final okhttp3.Response response; + + OkResponse(Call call) throws IOException { + this.call = call; + this.response = call.execute(); + } + + @Override + public int getStatusCode() { + return response.code(); + } + + @Override + public Map> getHeaders() { + return response.headers().toMultimap(); + } + + @Override + public InputStream getBody() { + ResponseBody body = response.body(); + if (body == null) { + return null; + } + + return body.source().inputStream(); + } + + @Override + public void close() { + boolean previous = CLOSING.get(); + CLOSING.set(true); + try { + call.cancel(); + response.close(); + } catch (Exception | AssertionError e) { + LOGGER.debug("Failed to close the response", e); + } finally { + CLOSING.set(previous); + } + } + } + + static class TrustAllX509TrustManager implements X509TrustManager { + @Override + public void checkClientTrusted(X509Certificate[] x509Certificates, String s) { + + } + + @Override + public void checkServerTrusted(X509Certificate[] x509Certificates, String s) { + + } + + @Override + public X509Certificate[] getAcceptedIssuers() { + return new X509Certificate[0]; + } + } +} diff --git a/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/OkHttpDockerCmdExecFactory.java b/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/OkHttpDockerCmdExecFactory.java new file mode 100644 index 000000000..a824e9954 --- /dev/null +++ b/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/OkHttpDockerCmdExecFactory.java @@ -0,0 +1,66 @@ +package com.github.dockerjava.okhttp; + +import com.github.dockerjava.api.command.DelegatingDockerCmdExecFactory; +import com.github.dockerjava.api.command.DockerCmdExecFactory; +import com.github.dockerjava.core.DefaultDockerCmdExecFactory; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.DockerClientConfigAware; +import com.github.dockerjava.core.DockerClientImpl; +import com.github.dockerjava.transport.DockerHttpClient; + +/** + * @deprecated use {@link OkDockerHttpClient} with {@link DockerClientImpl#getInstance(DockerClientConfig, DockerHttpClient)} + */ +@Deprecated +public class OkHttpDockerCmdExecFactory extends DelegatingDockerCmdExecFactory implements DockerClientConfigAware { + + private OkDockerHttpClient.Builder clientBuilder = new OkDockerHttpClient.Builder(); + + @Deprecated + protected Integer connectTimeout; + + @Deprecated + protected Integer readTimeout; + + private DefaultDockerCmdExecFactory dockerCmdExecFactory; + + /** + * Configure connection timeout in milliseconds + */ + public OkHttpDockerCmdExecFactory withConnectTimeout(Integer connectTimeout) { + clientBuilder = clientBuilder.connectTimeout(connectTimeout); + this.connectTimeout = connectTimeout; + return this; + } + + /** + * Configure read timeout in milliseconds + */ + public OkHttpDockerCmdExecFactory withReadTimeout(Integer readTimeout) { + clientBuilder = clientBuilder.readTimeout(readTimeout); + this.readTimeout = readTimeout; + return this; + } + + public OkHttpDockerCmdExecFactory setRetryOnConnectionFailure(Boolean retryOnConnectionFailure) { + this.clientBuilder = clientBuilder.retryOnConnectionFailure(retryOnConnectionFailure); + return this; + } + + @Override + public final DockerCmdExecFactory getDockerCmdExecFactory() { + return dockerCmdExecFactory; + } + + @Override + public void init(DockerClientConfig dockerClientConfig) { + clientBuilder = clientBuilder + .dockerHost(dockerClientConfig.getDockerHost()) + .sslConfig(dockerClientConfig.getSSLConfig()); + dockerCmdExecFactory = new DefaultDockerCmdExecFactory( + clientBuilder.build(), + dockerClientConfig.getObjectMapper() + ); + dockerCmdExecFactory.init(dockerClientConfig); + } +} diff --git a/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/UnixSocketFactory.java b/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/UnixSocketFactory.java new file mode 100644 index 000000000..d25bcb3d3 --- /dev/null +++ b/docker-java-transport-okhttp/src/main/java/com/github/dockerjava/okhttp/UnixSocketFactory.java @@ -0,0 +1,46 @@ +package com.github.dockerjava.okhttp; + +import com.github.dockerjava.transport.UnixSocket; + +import javax.net.SocketFactory; +import java.io.IOException; +import java.net.InetAddress; +import java.net.Socket; + +class UnixSocketFactory extends SocketFactory { + + private final String socketPath; + + UnixSocketFactory(String socketPath) { + this.socketPath = socketPath; + } + + @Override + public Socket createSocket() { + try { + return UnixSocket.get(socketPath); + } catch (IOException e) { + throw new RuntimeException("Failed create socket with path " + socketPath, e); + } + } + + @Override + public Socket createSocket(String s, int i) { + throw new UnsupportedOperationException(); + } + + @Override + public Socket createSocket(String s, int i, InetAddress inetAddress, int i1) { + throw new UnsupportedOperationException(); + } + + @Override + public Socket createSocket(InetAddress inetAddress, int i) { + throw new UnsupportedOperationException(); + } + + @Override + public Socket createSocket(InetAddress inetAddress, int i, InetAddress inetAddress1, int i1) { + throw new UnsupportedOperationException(); + } +} diff --git a/docker-java-transport-okhttp/src/test/java/com/github/dockerjava/transport/OkHttpClientTests.java b/docker-java-transport-okhttp/src/test/java/com/github/dockerjava/transport/OkHttpClientTests.java new file mode 100644 index 000000000..9a5b77ff3 --- /dev/null +++ b/docker-java-transport-okhttp/src/test/java/com/github/dockerjava/transport/OkHttpClientTests.java @@ -0,0 +1,17 @@ +package com.github.dockerjava.transport; + +import com.github.dockerjava.okhttp.OkDockerHttpClient; + +import java.net.URI; + +public class OkHttpClientTests extends DockerHttpClientTCK { + + @Override + protected DockerHttpClient createDockerHttpClient(URI dockerHost, SSLConfig sslConfig) { + return new OkDockerHttpClient.Builder() + .dockerHost(dockerHost) + .sslConfig(sslConfig) + .connectTimeout(30 * 100) + .build(); + } +} diff --git a/docker-java-transport-tck/pom.xml b/docker-java-transport-tck/pom.xml new file mode 100644 index 000000000..d7253c786 --- /dev/null +++ b/docker-java-transport-tck/pom.xml @@ -0,0 +1,70 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java-transport-tck + jar + + docker-java-transport-tck + https://github.com/docker-java/docker-java + Java API Client for Docker + + + com.github.dockerjava.transport.tck + + + + + ${project.groupId} + docker-java-core + ${project.version} + + + ${project.groupId} + docker-java-transport + ${project.version} + + + + org.assertj + assertj-core + 3.27.6 + + + + com.squareup.okhttp3 + mockwebserver + 3.14.9 + + + + org.testcontainers + testcontainers + 2.0.2 + + + + org.slf4j + slf4j-jdk14 + 1.7.35 + + + + + + + org.apache.maven.plugins + maven-deploy-plugin + + true + + + + + diff --git a/docker-java-transport-tck/src/main/java/com/github/dockerjava/transport/DockerHttpClientTCK.java b/docker-java-transport-tck/src/main/java/com/github/dockerjava/transport/DockerHttpClientTCK.java new file mode 100644 index 000000000..f90973be6 --- /dev/null +++ b/docker-java-transport-tck/src/main/java/com/github/dockerjava/transport/DockerHttpClientTCK.java @@ -0,0 +1,148 @@ +package com.github.dockerjava.transport; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.core.DefaultDockerClientConfig; +import com.github.dockerjava.core.DockerClientImpl; +import com.github.dockerjava.transport.DockerHttpClient.Request; +import com.github.dockerjava.transport.DockerHttpClient.Request.Method; +import com.github.dockerjava.transport.DockerHttpClient.Response; +import okhttp3.mockwebserver.MockResponse; +import okhttp3.mockwebserver.MockWebServer; +import org.junit.Test; +import org.testcontainers.DockerClientFactory; +import org.testcontainers.containers.GenericContainer; +import org.testcontainers.dockerclient.TransportConfig; + +import java.io.PipedInputStream; +import java.io.PipedOutputStream; +import java.net.URI; + +import static java.util.concurrent.TimeUnit.SECONDS; +import static org.assertj.core.api.Assertions.assertThat; + +public abstract class DockerHttpClientTCK { + + protected abstract DockerHttpClient createDockerHttpClient(URI dockerHost, SSLConfig sslConfig); + + @Test + public void testHijacking() throws Exception { + try ( + DockerClient client = createDockerClient(); + + PipedOutputStream out = new PipedOutputStream(); + PipedInputStream in = new PipedInputStream(out); + + AttachContainerTestCallback callback = new AttachContainerTestCallback(); + + AttacheableContainer container = new AttacheableContainer() { + @Override + protected void containerIsCreated(String containerId) { + client.attachContainerCmd(containerId) + .withStdOut(true) + .withFollowStream(true) + .withStdIn(in) + .exec(callback); + } + }; + ) { + container.start(); + assertThat(callback.awaitStarted(5, SECONDS)).as("attached").isTrue(); + + String snippet = "hello world"; + out.write((snippet + "\n").getBytes()); + out.flush(); + + assertThat(callback.awaitCompletion(15, SECONDS)).as("completed").isTrue(); + assertThat(callback.toString()).contains("STDOUT: " + snippet); + } + } + + /** + * Test that docker-java supports path in DOCKER_HOST + * + * @see valid values + */ + @Test + public final void testPath() throws Exception { + try (MockWebServer server = new MockWebServer()) { + String dockerHost = server.url("/%20some/path/").toString() + .replace("http://", "tcp://"); + + try (DockerHttpClient client = createDockerHttpClient(dockerHost)) { + server.enqueue(new MockResponse().setResponseCode(200)); + ping(client); + assertThat(server.takeRequest().getPath()) + .as("recorded path") + .isEqualTo("/%20some/path/_ping"); + } + } + } + + private DockerHttpClient createDockerHttpClient() { + // Use Testcontainers to detect Docker environment + TransportConfig transportConfig = DockerClientFactory.instance().getTransportConfig(); + return createDockerHttpClient(transportConfig.getDockerHost(), transportConfig.getSslConfig()); + } + + private DockerHttpClient createDockerHttpClient(String dockerHost) { + return createDockerHttpClient(URI.create(dockerHost), null); + } + + private DockerClient createDockerClient() { + return createDockerClient(createDockerHttpClient()); + } + + private DockerClient createDockerClient(DockerHttpClient dockerHttpClient) { + return DockerClientImpl.getInstance( + DefaultDockerClientConfig.createDefaultConfigBuilder().build(), + dockerHttpClient + ); + } + + private void ping(DockerHttpClient client) { + Request pingRequest = Request.builder() + .method(Method.GET) + .path("/_ping") + .build(); + + try (Response response = client.execute(pingRequest)) { + assertThat(response.getStatusCode()) + .as("status code") + .isEqualTo(200); + } + } + + private static class AttachContainerTestCallback extends ResultCallback.Adapter { + + private final StringBuffer log = new StringBuffer(); + + @Override + public void onNext(Frame item) { + log.append(item.toString()); + super.onNext(item); + } + + @Override + public String toString() { + return log.toString(); + } + } + + private static class AttacheableContainer extends GenericContainer { + + private AttacheableContainer() { + super("busybox:1.35.0"); + + withCommand("/bin/sh", "-c", "read line && echo $line"); + withCreateContainerCmdModifier(it -> { + it.withTty(false); + it.withAttachStdin(true); + it.withAttachStdout(true); + it.withAttachStderr(true); + it.withStdinOpen(true); + }); + } + } +} diff --git a/docker-java-transport-zerodep/pom.xml b/docker-java-transport-zerodep/pom.xml new file mode 100644 index 000000000..3cccafa33 --- /dev/null +++ b/docker-java-transport-zerodep/pom.xml @@ -0,0 +1,108 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java-transport-zerodep + jar + + docker-java-transport-zerodep + https://github.com/docker-java/docker-java + Java API Client for Docker + + + com.github.dockerjava.transport.zerodep + + + + + ${project.groupId} + docker-java-transport-httpclient5 + ${project.version} + + + + + + + com.github.siom79.japicmp + japicmp-maven-plugin + + + true + + + + + org.apache.felix + maven-bundle-plugin + true + + + com.github.dockerjava.zerodep.* + + + + + + org.apache.maven.plugins + maven-shade-plugin + + true + true + true + + + + com.github.docker-java:docker-java-transport + net.java.dev.jna:jna-platform + net.java.dev.jna:* + org.slf4j:slf4j-api + + + + + com.github.docker-java:docker-java-transport-httpclient5 + + com/github/dockerjava/httpclient5/ApacheDockerHttpClient.class + com/github/dockerjava/httpclient5/ApacheDockerHttpClient$* + + + + org.apache.httpcomponents.client5:httpclient5 + + mozilla/* + + + + + + org.apache + com.github.dockerjava.zerodep.shaded.org.apache + + + com.github.dockerjava.httpclient5 + com.github.dockerjava.zerodep + + + + + + + + + package + + shade + + + + + + + diff --git a/docker-java-transport-zerodep/src/main/java/com/github/dockerjava/httpclient5/ZerodepDockerHttpClient.java b/docker-java-transport-zerodep/src/main/java/com/github/dockerjava/httpclient5/ZerodepDockerHttpClient.java new file mode 100644 index 000000000..fcacc6d1b --- /dev/null +++ b/docker-java-transport-zerodep/src/main/java/com/github/dockerjava/httpclient5/ZerodepDockerHttpClient.java @@ -0,0 +1,58 @@ +package com.github.dockerjava.httpclient5; + +import java.net.URI; +import java.time.Duration; +import java.util.Objects; +import com.github.dockerjava.transport.SSLConfig; + +@SuppressWarnings("unused") +public final class ZerodepDockerHttpClient extends ApacheDockerHttpClientImpl { + + public static final class Builder { + + private URI dockerHost = null; + + private SSLConfig sslConfig = null; + + private int maxConnections = Integer.MAX_VALUE; + + private Duration connectionTimeout; + + private Duration responseTimeout; + + public Builder dockerHost(URI value) { + this.dockerHost = Objects.requireNonNull(value, "dockerHost"); + return this; + } + + public Builder sslConfig(SSLConfig value) { + this.sslConfig = value; + return this; + } + + public Builder maxConnections(int value) { + this.maxConnections = value; + return this; + } + + public Builder connectionTimeout(Duration connectionTimeout) { + this.connectionTimeout = connectionTimeout; + return this; + } + + public Builder responseTimeout(Duration responseTimeout) { + this.responseTimeout = responseTimeout; + return this; + } + + public ZerodepDockerHttpClient build() { + Objects.requireNonNull(dockerHost, "dockerHost"); + return new ZerodepDockerHttpClient(dockerHost, sslConfig, maxConnections, connectionTimeout, responseTimeout); + } + } + + private ZerodepDockerHttpClient(URI dockerHost, SSLConfig sslConfig, int maxConnections, Duration connectionTimeout, + Duration responseTimeout) { + super(dockerHost, sslConfig, maxConnections, connectionTimeout, responseTimeout); + } +} diff --git a/docker-java-transport/pom.xml b/docker-java-transport/pom.xml new file mode 100644 index 000000000..8be456dd1 --- /dev/null +++ b/docker-java-transport/pom.xml @@ -0,0 +1,59 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java-transport + jar + + docker-java-transport + https://github.com/docker-java/docker-java + Java API Client for Docker + + + com.github.dockerjava.transport + + + + + com.google.code.findbugs + annotations + 3.0.1u2 + provided + + + + org.immutables + value + 2.10.1 + provided + + + + net.java.dev.jna + jna + 5.18.1 + provided + + + + + + + org.apache.felix + maven-bundle-plugin + true + + + com.github.dockerjava.transport.* + + + + + + diff --git a/docker-java-transport/src/main/java/com/github/dockerjava/transport/AbstractSocket.java b/docker-java-transport/src/main/java/com/github/dockerjava/transport/AbstractSocket.java new file mode 100644 index 000000000..37a538bc9 --- /dev/null +++ b/docker-java-transport/src/main/java/com/github/dockerjava/transport/AbstractSocket.java @@ -0,0 +1,87 @@ +/* + * Copyright 2012-2020 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 + * + * https://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.github.dockerjava.transport; + +import java.io.IOException; +import java.net.InetAddress; +import java.net.Socket; +import java.net.SocketAddress; + +/** + * Abstract base class for custom socket implementation. + * + * @author Phillip Webb + */ +class AbstractSocket extends Socket { + + @Override + public void connect(SocketAddress endpoint) throws IOException { + } + + @Override + public void connect(SocketAddress endpoint, int timeout) throws IOException { + } + + @Override + public boolean isConnected() { + return true; + } + + @Override + public boolean isBound() { + return true; + } + + @Override + public void shutdownInput() throws IOException { + throw new UnsupportedSocketOperationException(); + } + + @Override + public void shutdownOutput() throws IOException { + throw new UnsupportedSocketOperationException(); + } + + @Override + public InetAddress getInetAddress() { + return null; + } + + @Override + public InetAddress getLocalAddress() { + return null; + } + + @Override + public SocketAddress getLocalSocketAddress() { + return null; + } + + @Override + public SocketAddress getRemoteSocketAddress() { + return null; + } + + private static class UnsupportedSocketOperationException extends UnsupportedOperationException { + + UnsupportedSocketOperationException() { + super("Unsupported socket operation"); + } + + } + +} diff --git a/docker-java-transport/src/main/java/com/github/dockerjava/transport/BsdDomainSocket.java b/docker-java-transport/src/main/java/com/github/dockerjava/transport/BsdDomainSocket.java new file mode 100644 index 000000000..12d2004e6 --- /dev/null +++ b/docker-java-transport/src/main/java/com/github/dockerjava/transport/BsdDomainSocket.java @@ -0,0 +1,83 @@ +/* + * Copyright 2012-2020 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 + * + * https://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.github.dockerjava.transport; + +import java.io.IOException; +import java.nio.charset.StandardCharsets; +import java.util.Arrays; +import java.util.List; + +import com.sun.jna.LastErrorException; +import com.sun.jna.Native; +import com.sun.jna.Platform; +import com.sun.jna.Structure; + +/** + * {@link DomainSocket} implementation for BSD based platforms. + * + * @author Phillip Webb + */ +class BsdDomainSocket extends DomainSocket { + + private static final int MAX_PATH_LENGTH = 104; + + static { + Native.register(Platform.C_LIBRARY_NAME); + } + + BsdDomainSocket(String path) throws IOException { + super(path); + } + + @Override + protected void connect(String path, int handle) { + SockaddrUn address = new SockaddrUn(AF_LOCAL, path.getBytes(StandardCharsets.UTF_8)); + connect(handle, address, address.size()); + } + + private native int connect(int fd, SockaddrUn address, int addressLen) throws LastErrorException; + + /** + * Native {@code sockaddr_un} structure as defined in {@code sys/un.h}. + */ + public static class SockaddrUn extends Structure implements Structure.ByReference { + + public byte sunLen; + + public byte sunFamily; + + public byte[] sunPath = new byte[MAX_PATH_LENGTH]; + + private SockaddrUn(byte sunFamily, byte[] path) { + if (path.length > MAX_PATH_LENGTH) { + throw new IllegalArgumentException("Path cannot exceed " + MAX_PATH_LENGTH + " bytes"); + } + System.arraycopy(path, 0, this.sunPath, 0, path.length); + this.sunPath[path.length] = 0; + this.sunLen = (byte) (fieldOffset("sunPath") + path.length); + this.sunFamily = sunFamily; + allocateMemory(); + } + + @Override + protected List getFieldOrder() { + return Arrays.asList("sunLen", "sunFamily", "sunPath"); + } + + } + +} diff --git a/docker-java-transport/src/main/java/com/github/dockerjava/transport/DockerHttpClient.java b/docker-java-transport/src/main/java/com/github/dockerjava/transport/DockerHttpClient.java new file mode 100644 index 000000000..7b780cb06 --- /dev/null +++ b/docker-java-transport/src/main/java/com/github/dockerjava/transport/DockerHttpClient.java @@ -0,0 +1,90 @@ +package com.github.dockerjava.transport; + +import org.immutables.value.Value; + +import javax.annotation.Nonnull; +import javax.annotation.Nullable; +import java.io.ByteArrayInputStream; +import java.io.Closeable; +import java.io.InputStream; +import java.util.List; +import java.util.Map; + +public interface DockerHttpClient extends Closeable { + + Response execute(Request request); + + interface Response extends Closeable { + + int getStatusCode(); + + Map> getHeaders(); + + InputStream getBody(); + + @Override + void close(); + + @Nullable + default String getHeader(@Nonnull String name) { + for (Map.Entry> entry : getHeaders().entrySet()) { + if (name.equalsIgnoreCase(entry.getKey())) { + List values = entry.getValue(); + return values.isEmpty() ? null : values.get(0); + } + } + + return null; + } + } + + @Value.Immutable + @Value.Style( + visibility = Value.Style.ImplementationVisibility.PACKAGE, + overshadowImplementation = true, + depluralize = true + ) + abstract class Request { + + public enum Method { + GET, + POST, + PUT, + DELETE, + OPTIONS, + PATCH, + } + + public static class Builder extends ImmutableRequest.Builder { + + public Builder method(Method method) { + return method(method.name()); + } + } + + public static Builder builder() { + return new Builder(); + } + + public abstract String method(); + + public abstract String path(); + + @Nullable + @Value.Default + public InputStream body() { + byte[] bodyBytes = bodyBytes(); + return bodyBytes != null + ? new ByteArrayInputStream(bodyBytes) + : null; + } + + @Nullable + public abstract byte[] bodyBytes(); + + @Nullable + public abstract InputStream hijackedInput(); + + public abstract Map headers(); + } +} diff --git a/docker-java-transport/src/main/java/com/github/dockerjava/transport/DomainSocket.java b/docker-java-transport/src/main/java/com/github/dockerjava/transport/DomainSocket.java new file mode 100644 index 000000000..a2a3503f5 --- /dev/null +++ b/docker-java-transport/src/main/java/com/github/dockerjava/transport/DomainSocket.java @@ -0,0 +1,191 @@ +/* + * Copyright 2012-2020 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 + * + * https://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.github.dockerjava.transport; + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.net.Socket; +import java.nio.ByteBuffer; + +import com.github.dockerjava.transport.FileDescriptor.Handle; +import com.sun.jna.LastErrorException; +import com.sun.jna.Native; +import com.sun.jna.Platform; + +/** + * A {@link Socket} implementation for Linux of BSD domain sockets. + * + * @author Phillip Webb + */ +public abstract class DomainSocket extends AbstractSocket { + + private static final int SHUT_RD = 0; + + private static final int SHUT_WR = 1; + + protected static final int PF_LOCAL = 1; + + protected static final byte AF_LOCAL = 1; + + protected static final int SOCK_STREAM = 1; + + private final FileDescriptor fileDescriptor; + + private final InputStream inputStream; + + private final OutputStream outputStream; + + static { + Native.register(Platform.C_LIBRARY_NAME); + } + + DomainSocket(String path) throws IOException { + try { + this.fileDescriptor = open(path); + this.inputStream = new DomainSocketInputStream(); + this.outputStream = new DomainSocketOutputStream(); + } catch (LastErrorException ex) { + throw new IOException(ex); + } + } + + private FileDescriptor open(String path) { + int handle = socket(PF_LOCAL, SOCK_STREAM, 0); + connect(path, handle); + return new FileDescriptor(handle, this::close); + } + + private int read(ByteBuffer buffer) throws IOException { + try (Handle handle = this.fileDescriptor.acquire()) { + if (handle.isClosed()) { + return -1; + } + try { + return read(handle.intValue(), buffer, buffer.remaining()); + } catch (LastErrorException ex) { + throw new IOException(ex); + } + } + } + + public void write(ByteBuffer buffer) throws IOException { + try (Handle handle = this.fileDescriptor.acquire()) { + if (!handle.isClosed()) { + try { + write(handle.intValue(), buffer, buffer.remaining()); + } catch (LastErrorException ex) { + throw new IOException(ex); + } + } + } + } + + @Override + public InputStream getInputStream() { + return this.inputStream; + } + + @Override + public OutputStream getOutputStream() { + return this.outputStream; + } + + @Override + public void close() throws IOException { + super.close(); + try { + this.fileDescriptor.close(); + } catch (LastErrorException ex) { + throw new IOException(ex); + } + } + + protected abstract void connect(String path, int handle); + + private native int socket(int domain, int type, int protocol) throws LastErrorException; + + private native int read(int fd, ByteBuffer buffer, int count) throws LastErrorException; + + private native int write(int fd, ByteBuffer buffer, int count) throws LastErrorException; + + private native int close(int fd) throws LastErrorException; + + /** + * Return a new {@link DomainSocket} for the given path. + * @param path the path to the domain socket + * @return a {@link DomainSocket} instance + * @throws IOException if the socket cannot be opened + * @deprecated use {@link UnixSocket#get(String)} + */ + @Deprecated + public static DomainSocket get(String path) throws IOException { + if (Platform.isMac() || isBsdPlatform()) { + return new BsdDomainSocket(path); + } + return new LinuxDomainSocket(path); + } + + private static boolean isBsdPlatform() { + return Platform.isFreeBSD() || Platform.iskFreeBSD() || Platform.isNetBSD() || Platform.isOpenBSD(); + } + + /** + * {@link InputStream} returned from the {@link DomainSocket}. + */ + private class DomainSocketInputStream extends InputStream { + + @Override + public int read() throws IOException { + ByteBuffer buffer = ByteBuffer.allocate(1); + int amountRead = DomainSocket.this.read(buffer); + return (amountRead != 1) ? -1 : buffer.get() & 0xFF; + } + + @Override + public int read(byte[] b, int off, int len) throws IOException { + if (len == 0) { + return 0; + } + int amountRead = DomainSocket.this.read(ByteBuffer.wrap(b, off, len)); + return (amountRead > 0) ? amountRead : -1; + } + + } + + /** + * {@link OutputStream} returned from the {@link DomainSocket}. + */ + private class DomainSocketOutputStream extends OutputStream { + + @Override + public void write(int b) throws IOException { + ByteBuffer buffer = ByteBuffer.allocate(1); + buffer.put(0, (byte) (b & 0xFF)); + DomainSocket.this.write(buffer); + } + + @Override + public void write(byte[] b, int off, int len) throws IOException { + if (len != 0) { + DomainSocket.this.write(ByteBuffer.wrap(b, off, len)); + } + } + + } + +} diff --git a/docker-java-transport/src/main/java/com/github/dockerjava/transport/FileDescriptor.java b/docker-java-transport/src/main/java/com/github/dockerjava/transport/FileDescriptor.java new file mode 100644 index 000000000..31960f949 --- /dev/null +++ b/docker-java-transport/src/main/java/com/github/dockerjava/transport/FileDescriptor.java @@ -0,0 +1,121 @@ +/* + * Copyright 2012-2020 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 + * + * https://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.github.dockerjava.transport; + +import java.io.Closeable; +import java.io.IOException; +import java.util.function.IntConsumer; + +/** + * Provides access to the underlying file system representation of an open file. + * + * @author Phillip Webb + * @see #acquire() + */ +class FileDescriptor { + + private final Handle openHandle; + + private final Handle closedHandler; + + private final IntConsumer closer; + + private Status status = Status.OPEN; + + private int referenceCount; + + FileDescriptor(int handle, IntConsumer closer) { + this.openHandle = new Handle(handle); + this.closedHandler = new Handle(-1); + this.closer = closer; + } + + @Override + protected void finalize() throws Throwable { + close(); + } + + /** + * Acquire an instance of the actual {@link Handle}. The caller must + * {@link Handle#close() close} the resulting handle when done. + * @return the handle + */ + synchronized Handle acquire() { + this.referenceCount++; + return (this.status != Status.OPEN) ? this.closedHandler : this.openHandle; + } + + private synchronized void release() { + this.referenceCount--; + if (this.referenceCount == 0 && this.status == Status.CLOSE_PENDING) { + this.closer.accept(this.openHandle.value); + this.status = Status.CLOSED; + } + } + + /** + * Close the underlying file when all handles have been released. + */ + synchronized void close() { + if (this.status == Status.OPEN) { + if (this.referenceCount == 0) { + this.closer.accept(this.openHandle.value); + this.status = Status.CLOSED; + } else { + this.status = Status.CLOSE_PENDING; + } + } + } + + /** + * The status of the file descriptor. + */ + private enum Status { + + OPEN, CLOSE_PENDING, CLOSED + + } + + /** + * Provides access to the actual file descriptor handle. + */ + final class Handle implements Closeable { + + private final int value; + + private Handle(int value) { + this.value = value; + } + + boolean isClosed() { + return this.value == -1; + } + + int intValue() { + return this.value; + } + + @Override + public void close() throws IOException { + if (!isClosed()) { + release(); + } + } + + } + +} diff --git a/docker-java-transport/src/main/java/com/github/dockerjava/transport/LinuxDomainSocket.java b/docker-java-transport/src/main/java/com/github/dockerjava/transport/LinuxDomainSocket.java new file mode 100644 index 000000000..e1467858a --- /dev/null +++ b/docker-java-transport/src/main/java/com/github/dockerjava/transport/LinuxDomainSocket.java @@ -0,0 +1,80 @@ +/* + * Copyright 2012-2020 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 + * + * https://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.github.dockerjava.transport; + +import java.io.IOException; +import java.nio.charset.StandardCharsets; +import java.util.Arrays; +import java.util.List; + +import com.sun.jna.LastErrorException; +import com.sun.jna.Native; +import com.sun.jna.Platform; +import com.sun.jna.Structure; + +/** + * {@link DomainSocket} implementation for Linux based platforms. + * + * @author Phillip Webb + */ +class LinuxDomainSocket extends DomainSocket { + + static { + Native.register(Platform.C_LIBRARY_NAME); + } + + LinuxDomainSocket(String path) throws IOException { + super(path); + } + + private static final int MAX_PATH_LENGTH = 108; + + @Override + protected void connect(String path, int handle) { + SockaddrUn address = new SockaddrUn(AF_LOCAL, path.getBytes(StandardCharsets.UTF_8)); + connect(handle, address, address.size()); + } + + private native int connect(int fd, SockaddrUn address, int addressLen) throws LastErrorException; + + /** + * Native {@code sockaddr_un} structure as defined in {@code sys/un.h}. + */ + public static class SockaddrUn extends Structure implements Structure.ByReference { + + public short sunFamily; + + public byte[] sunPath = new byte[MAX_PATH_LENGTH]; + + private SockaddrUn(byte sunFamily, byte[] path) { + if (path.length > MAX_PATH_LENGTH) { + throw new IllegalArgumentException("Path cannot exceed " + MAX_PATH_LENGTH + " bytes"); + } + System.arraycopy(path, 0, this.sunPath, 0, path.length); + this.sunPath[path.length] = 0; + this.sunFamily = sunFamily; + allocateMemory(); + } + + @Override + protected List getFieldOrder() { + return Arrays.asList("sunFamily", "sunPath"); + } + + } + +} diff --git a/docker-java-transport/src/main/java/com/github/dockerjava/transport/NamedPipeSocket.java b/docker-java-transport/src/main/java/com/github/dockerjava/transport/NamedPipeSocket.java new file mode 100644 index 000000000..e4aa315eb --- /dev/null +++ b/docker-java-transport/src/main/java/com/github/dockerjava/transport/NamedPipeSocket.java @@ -0,0 +1,158 @@ +package com.github.dockerjava.transport; + +import com.sun.jna.Native; +import com.sun.jna.win32.StdCallLibrary; +import com.sun.jna.win32.W32APIOptions; + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.net.Socket; +import java.net.SocketAddress; +import java.nio.ByteBuffer; +import java.nio.channels.AsynchronousByteChannel; +import java.nio.channels.AsynchronousCloseException; +import java.nio.channels.AsynchronousFileChannel; +import java.nio.channels.Channels; +import java.nio.channels.CompletionHandler; +import java.nio.file.FileSystemException; +import java.nio.file.Paths; +import java.nio.file.StandardOpenOption; +import java.util.concurrent.CompletableFuture; +import java.util.concurrent.Future; + +public class NamedPipeSocket extends Socket { + + private final String socketFileName; + + private AsynchronousFileByteChannel channel; + + public NamedPipeSocket(String socketFileName) { + this.socketFileName = socketFileName; + } + + @Override + public void close() throws IOException { + if (channel != null) { + channel.close(); + } + } + + @Override + public void connect(SocketAddress endpoint) throws IOException { + connect(endpoint, 0); + } + + @Override + public void connect(SocketAddress endpoint, int timeout) throws IOException { + long startedAt = System.currentTimeMillis(); + timeout = Math.max(timeout, 10_000); + while (true) { + try { + channel = new AsynchronousFileByteChannel( + AsynchronousFileChannel.open( + Paths.get(socketFileName), + StandardOpenOption.READ, + StandardOpenOption.WRITE + ) + ); + break; + } catch (FileSystemException e) { + if (System.currentTimeMillis() - startedAt >= timeout) { + throw new RuntimeException(e); + } else { + Kernel32.INSTANCE.WaitNamedPipe(socketFileName, 100); + } + } + } + } + + @Override + public InputStream getInputStream() { + return Channels.newInputStream(channel); + } + + @Override + public OutputStream getOutputStream() { + return Channels.newOutputStream(channel); + } + + interface Kernel32 extends StdCallLibrary { + + Kernel32 INSTANCE = Native.load("kernel32", Kernel32.class, W32APIOptions.DEFAULT_OPTIONS); + + @SuppressWarnings("checkstyle:methodname") + boolean WaitNamedPipe(String lpNamedPipeName, int nTimeOut); + } + + private static class AsynchronousFileByteChannel implements AsynchronousByteChannel { + private final AsynchronousFileChannel fileChannel; + + AsynchronousFileByteChannel(AsynchronousFileChannel fileChannel) { + this.fileChannel = fileChannel; + } + + @Override + public void read(ByteBuffer dst, A attachment, CompletionHandler handler) { + fileChannel.read(dst, 0, attachment, new CompletionHandler() { + @Override + public void completed(Integer read, A attachment) { + handler.completed(read > 0 ? read : -1, attachment); + } + + @Override + public void failed(Throwable exc, A attachment) { + if (exc instanceof AsynchronousCloseException) { + handler.completed(-1, attachment); + return; + } + handler.failed(exc, attachment); + } + }); + } + + @Override + public Future read(ByteBuffer dst) { + CompletableFutureHandler future = new CompletableFutureHandler(); + fileChannel.read(dst, 0, null, future); + return future; + } + + @Override + public void write(ByteBuffer src, A attachment, CompletionHandler handler) { + fileChannel.write(src, 0, attachment, handler); + } + + @Override + public Future write(ByteBuffer src) { + return fileChannel.write(src, 0); + } + + @Override + public void close() throws IOException { + fileChannel.close(); + } + + @Override + public boolean isOpen() { + return fileChannel.isOpen(); + } + + private static class CompletableFutureHandler extends CompletableFuture implements CompletionHandler { + + @Override + public void completed(Integer read, Object attachment) { + complete(read > 0 ? read : -1); + } + + @Override + public void failed(Throwable exc, Object attachment) { + if (exc instanceof AsynchronousCloseException) { + complete(-1); + return; + } + completeExceptionally(exc); + } + } + } +} diff --git a/docker-java-transport/src/main/java/com/github/dockerjava/transport/SSLConfig.java b/docker-java-transport/src/main/java/com/github/dockerjava/transport/SSLConfig.java new file mode 100644 index 000000000..a2840cb5f --- /dev/null +++ b/docker-java-transport/src/main/java/com/github/dockerjava/transport/SSLConfig.java @@ -0,0 +1,21 @@ +package com.github.dockerjava.transport; + +import javax.net.ssl.SSLContext; +import java.security.KeyManagementException; +import java.security.KeyStoreException; +import java.security.NoSuchAlgorithmException; +import java.security.UnrecoverableKeyException; + +/** + * Get an SSL Config. Allows for various different implementations. + */ +public interface SSLConfig { + + /** + * Get the SSL Context, from wherever it comes (file, keystore). + * + * @return an SSL context. + */ + SSLContext getSSLContext() throws KeyManagementException, UnrecoverableKeyException, NoSuchAlgorithmException, + KeyStoreException; +} diff --git a/docker-java-transport/src/main/java/com/github/dockerjava/transport/UnixSocket.java b/docker-java-transport/src/main/java/com/github/dockerjava/transport/UnixSocket.java new file mode 100644 index 000000000..eb7a49b51 --- /dev/null +++ b/docker-java-transport/src/main/java/com/github/dockerjava/transport/UnixSocket.java @@ -0,0 +1,111 @@ +package com.github.dockerjava.transport; + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.lang.reflect.InvocationTargetException; +import java.net.Socket; +import java.net.SocketAddress; +import java.net.SocketException; +import java.nio.ByteBuffer; +import java.nio.channels.Channels; +import java.nio.channels.SocketChannel; +import java.nio.channels.WritableByteChannel; + +public class UnixSocket extends AbstractSocket { + + /** + * Return a new {@link Socket} for the given path. Will use JDK's {@link java.net.UnixDomainSocketAddress} + * if available and fallback to {@link DomainSocket} otherwise. + * + * @param path the path to the domain socket + * @return a {@link Socket} instance + * @throws IOException if the socket cannot be opened + */ + public static Socket get(String path) throws IOException { + try { + return new UnixSocket(path); + } catch (ClassNotFoundException | NoSuchMethodException | InvocationTargetException | + IllegalAccessException e) { + //noinspection deprecation + return DomainSocket.get(path); + } + } + + private final SocketAddress socketAddress; + + private final SocketChannel socketChannel; + + private UnixSocket(String path) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, + IllegalAccessException, IOException { + Class unixDomainSocketAddress = Class.forName("java.net.UnixDomainSocketAddress"); + this.socketAddress = + (SocketAddress) unixDomainSocketAddress.getMethod("of", String.class) + .invoke(null, path); + this.socketChannel = SocketChannel.open(this.socketAddress); + } + + @Override + public InputStream getInputStream() throws IOException { + if (isClosed()) { + throw new SocketException("Socket is closed"); + } + if (!isConnected()) { + throw new SocketException("Socket is not connected"); + } + if (isInputShutdown()) { + throw new SocketException("Socket input is shutdown"); + } + + return Channels.newInputStream(socketChannel); + } + + @Override + public OutputStream getOutputStream() throws IOException { + if (isClosed()) { + throw new SocketException("Socket is closed"); + } + if (!isConnected()) { + throw new SocketException("Socket is not connected"); + } + if (isOutputShutdown()) { + throw new SocketException("Socket output is shutdown"); + } + + return Channels.newOutputStream(new WrappedWritableByteChannel()); + } + + @Override + public SocketAddress getLocalSocketAddress() { + return socketAddress; + } + + @Override + public SocketAddress getRemoteSocketAddress() { + return socketAddress; + } + + @Override + public void close() throws IOException { + super.close(); + this.socketChannel.close(); + } + + private class WrappedWritableByteChannel implements WritableByteChannel { + + @Override + public int write(ByteBuffer src) throws IOException { + return UnixSocket.this.socketChannel.write(src); + } + + @Override + public boolean isOpen() { + return UnixSocket.this.socketChannel.isOpen(); + } + + @Override + public void close() throws IOException { + UnixSocket.this.socketChannel.close(); + } + } +} diff --git a/docker-java/pom.xml b/docker-java/pom.xml new file mode 100644 index 000000000..3cfd7f255 --- /dev/null +++ b/docker-java/pom.xml @@ -0,0 +1,194 @@ + + 4.0.0 + + + com.github.docker-java + docker-java-parent + 0-SNAPSHOT + ../pom.xml + + + docker-java + jar + + docker-java + https://github.com/docker-java/docker-java + Java API Client for Docker + + + com.github.dockerjava + + + + + ${project.groupId} + docker-java-core + ${project.version} + + + ${project.groupId} + docker-java-transport-jersey + ${project.version} + + + ${project.groupId} + docker-java-transport-netty + ${project.version} + + + + org.slf4j + jcl-over-slf4j + ${slf4j-api.version} + + + + + ${project.groupId} + docker-java-transport-okhttp + ${project.version} + test + + + ${project.groupId} + docker-java-transport-httpclient5 + ${project.version} + test + + + ch.qos.logback + logback-core + ${logback.version} + test + + + + ch.qos.logback + logback-classic + ${logback.version} + test + + + + org.hamcrest + hamcrest-library + ${hamcrest.library.version} + test + + + + com.googlecode.lambdaj + lambdaj + ${lambdaj.version} + test + + + org.hamcrest + hamcrest-all + + + + + + org.testinfected.hamcrest-matchers + jpa-matchers + ${hamcrest.jpa-matchers} + test + + + + org.mockito + mockito-core + ${mockito.version} + test + + + + com.google.code.findbugs + annotations + 3.0.1u2 + provided + + + junit + junit + 4.13 + test + + + org.awaitility + awaitility + 4.3.0 + test + + + + com.fasterxml.jackson.core + jackson-databind + + 2.20.1 + test + + + + com.fasterxml.jackson.core + jackson-annotations + + 2.20 + test + + + + org.projectlombok + lombok + 1.18.38 + provided + + + + + + + + org.apache.maven.plugins + maven-surefire-plugin + + false + 3 + com.github.dockerjava.junit.category.Integration + + + + + org.apache.maven.plugins + maven-failsafe-plugin + ${maven-failsafe-plugin.version} + + + + integration-test + verify + + + + + false + 5 + com.github.dockerjava.junit.category.Integration + + + + + org.apache.felix + maven-bundle-plugin + true + + + !com.github.dockerjava.jaxrs.*,!com.github.dockerjava.netty.*,com.github.dockerjava.* + org.newsclub.net.unix;resolution:="optional",* + + + + + + diff --git a/docker-java/src/main/java/com/github/dockerjava/core/DockerClientBuilder.java b/docker-java/src/main/java/com/github/dockerjava/core/DockerClientBuilder.java new file mode 100644 index 000000000..8100f285e --- /dev/null +++ b/docker-java/src/main/java/com/github/dockerjava/core/DockerClientBuilder.java @@ -0,0 +1,110 @@ +package com.github.dockerjava.core; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.DockerCmdExecFactory; +import com.github.dockerjava.jaxrs.JerseyDockerCmdExecFactory; +import com.github.dockerjava.jaxrs.JerseyDockerHttpClient; +import com.github.dockerjava.transport.DockerHttpClient; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class DockerClientBuilder { + + private final DockerClientConfig dockerClientConfig; + + private DockerCmdExecFactory dockerCmdExecFactory = null; + + private DockerHttpClient dockerHttpClient = null; + + private DockerClientBuilder(DockerClientConfig dockerClientConfig) { + this.dockerClientConfig = dockerClientConfig; + } + + public static DockerClientBuilder getInstance() { + return new DockerClientBuilder( + DefaultDockerClientConfig.createDefaultConfigBuilder().build() + ); + } + + /** + * + * @deprecated use {@link #getInstance(DockerClientConfig)} + */ + @Deprecated + public static DockerClientBuilder getInstance(DefaultDockerClientConfig.Builder dockerClientConfigBuilder) { + return getInstance(dockerClientConfigBuilder.build()); + } + + public static DockerClientBuilder getInstance(DockerClientConfig dockerClientConfig) { + return new DockerClientBuilder(dockerClientConfig); + } + + /** + * + * @deprecated use {@link DefaultDockerClientConfig.Builder#withDockerHost(String)} + */ + @Deprecated + public static DockerClientBuilder getInstance(String serverUrl) { + return new DockerClientBuilder( + DefaultDockerClientConfig.createDefaultConfigBuilder() + .withDockerHost(serverUrl) + .build() + ); + } + + /** + * + * @deprecated no replacement, use one of {@link DockerHttpClient} + */ + @Deprecated + public static DockerCmdExecFactory getDefaultDockerCmdExecFactory() { + return new JerseyDockerCmdExecFactory(); + } + + /** + * Note that this method overrides {@link DockerHttpClient} if it was previously set + * + * @deprecated use {@link #withDockerHttpClient(DockerHttpClient)} + */ + @Deprecated + public DockerClientBuilder withDockerCmdExecFactory(DockerCmdExecFactory dockerCmdExecFactory) { + this.dockerCmdExecFactory = dockerCmdExecFactory; + this.dockerHttpClient = null; + return this; + } + + /** + * Note that this method overrides {@link DockerCmdExecFactory} if it was previously set + */ + public DockerClientBuilder withDockerHttpClient(DockerHttpClient dockerHttpClient) { + this.dockerCmdExecFactory = null; + this.dockerHttpClient = dockerHttpClient; + return this; + } + + public DockerClient build() { + if (dockerHttpClient != null) { + return DockerClientImpl.getInstance( + dockerClientConfig, + dockerHttpClient + ); + } else if (dockerCmdExecFactory != null) { + return DockerClientImpl.getInstance(dockerClientConfig) + .withDockerCmdExecFactory(dockerCmdExecFactory); + } else { + Logger log = LoggerFactory.getLogger(DockerClientBuilder.class); + log.warn( + "'dockerHttpClient' should be set. " + + "Falling back to Jersey, will be an error in future releases." + ); + + return DockerClientImpl.getInstance( + dockerClientConfig, + new JerseyDockerHttpClient.Builder() + .dockerHost(dockerClientConfig.getDockerHost()) + .sslConfig(dockerClientConfig.getSSLConfig()) + .build() + ); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/ModelsSerializableTest.java b/docker-java/src/test/java/com/github/dockerjava/api/ModelsSerializableTest.java new file mode 100644 index 000000000..1c7c1de6c --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/ModelsSerializableTest.java @@ -0,0 +1,77 @@ +package com.github.dockerjava.api; + +import com.github.dockerjava.api.model.Binds; +import com.github.dockerjava.api.model.BuildResponseItem; +import com.github.dockerjava.api.model.DockerObject; +import com.github.dockerjava.api.model.DockerObjectAccessor; +import com.github.dockerjava.api.model.PullResponseItem; +import com.github.dockerjava.api.model.PushResponseItem; +import com.github.dockerjava.api.model.ResponseItem; +import com.google.common.reflect.ClassPath.ClassInfo; +import org.apache.commons.lang3.reflect.FieldUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; +import java.io.Serializable; +import java.util.Arrays; +import java.util.List; + +import static com.google.common.reflect.ClassPath.from; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.instanceOf; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.object.IsCompatibleType.typeCompatibleWith; + +/** + * TODO use ArchUnit + * + * @author Kanstantsin Shautsou + */ +public class ModelsSerializableTest { + private static final Logger LOG = LoggerFactory.getLogger(ModelsSerializableTest.class); + + private List excludeClasses = Arrays.asList( + Binds.class.getName(), + BuildResponseItem.class.getName(), + PullResponseItem.class.getName(), + PushResponseItem.class.getName(), + ResponseItem.class.getName(), + ResponseItem.ErrorDetail.class.getName(), + ResponseItem.ProgressDetail.class.getName() + ); + + @Test + public void allModelsSerializable() throws IOException, NoSuchFieldException, IllegalAccessException { + final ClassLoader contextClassLoader = Thread.currentThread().getContextClassLoader(); + for (ClassInfo classInfo : from(contextClassLoader).getAllClasses()) { + if (!classInfo.getPackageName().equals("com.github.dockerjava.api.model")) { + continue; + } + + if ( + classInfo.getName().endsWith("Test") + || DockerObject.class.getName().equals(classInfo.getName()) + || DockerObjectAccessor.class.getName().equals(classInfo.getName()) + ) { + continue; + } + + final Class aClass = classInfo.load(); + if (aClass.getProtectionDomain().getCodeSource().getLocation().getPath().endsWith("test-classes/") + || aClass.isEnum()) { + continue; + } + + LOG.debug("Checking: {}", aClass); + assertThat(aClass, typeCompatibleWith(Serializable.class)); + + final Object serialVersionUID = FieldUtils.readDeclaredStaticField(aClass, "serialVersionUID", true); + if (!excludeClasses.contains(aClass.getName())) { + assertThat(serialVersionUID, instanceOf(Long.class)); + assertThat("Follow devel docs for " + aClass, (Long) serialVersionUID, is(1L)); + } + } + } +} diff --git a/src/test/java/com/github/dockerjava/api/command/CommandJSONSamples.java b/docker-java/src/test/java/com/github/dockerjava/api/command/CommandJSONSamples.java similarity index 80% rename from src/test/java/com/github/dockerjava/api/command/CommandJSONSamples.java rename to docker-java/src/test/java/com/github/dockerjava/api/command/CommandJSONSamples.java index 7ee892487..23ef4b7f2 100644 --- a/src/test/java/com/github/dockerjava/api/command/CommandJSONSamples.java +++ b/docker-java/src/test/java/com/github/dockerjava/api/command/CommandJSONSamples.java @@ -24,7 +24,13 @@ */ public enum CommandJSONSamples implements JSONResourceRef { - inspectContainerResponse_full, inspectContainerResponse_full_1_21, inspectContainerResponse_empty; + inspectContainerResponse_full, + inspectContainerResponse_full_1_21, + inspectContainerResponse_full_1_26a, + inspectContainerResponse_full_1_26b, + inspectContainerResponse_empty, + updateContainerResponse_empty, + updateContainerResponse_warnings; @Override public String getFileName() { diff --git a/docker-java/src/test/java/com/github/dockerjava/api/command/InspectContainerResponseTest.java b/docker-java/src/test/java/com/github/dockerjava/api/command/InspectContainerResponseTest.java new file mode 100644 index 000000000..12105e8a1 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/command/InspectContainerResponseTest.java @@ -0,0 +1,195 @@ +/* + * Copyright 2015 CloudBees Inc., Oleg Nenashev. + * + * 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.github.dockerjava.api.command; + +import com.fasterxml.jackson.databind.JavaType; +import com.github.dockerjava.api.model.ContainerNetwork; +import com.github.dockerjava.api.model.Isolation; +import com.github.dockerjava.api.model.Volume; +import com.github.dockerjava.core.RemoteApiVersion; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import java.io.IOException; +import java.util.Collections; +import java.util.List; + +import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; +import static com.github.dockerjava.test.serdes.JSONTestHelper.testRoundTrip; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.Matchers.nullValue; +import static org.hamcrest.core.IsNot.not; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertTrue; + +/** + * Tests for {@link InspectContainerResponse}. + * + * @author Oleg Nenashev + */ +public class InspectContainerResponseTest { + + @Test + public void roundTrip_full() throws IOException { + InspectContainerResponse[] responses = testRoundTrip(CommandJSONSamples.inspectContainerResponse_full, + InspectContainerResponse[].class); + assertEquals(1, responses.length); + final InspectContainerResponse response = responses[0]; + + // Check volumes: https://github.com/docker-java/docker-java/issues/211 + assertEquals(2, response.getVolumes().length); + assertEquals(2, response.getVolumesRW().length); + assertEquals("/bar/foo/myvol2" ,response.getVolumes()[1].getContainerPath()); + assertEquals("/path2", response.getVolumes()[1].getHostPath()); + assertEquals("/bar/foo/myvol2", response.getVolumesRW()[1].getVolume().getPath()); + assertFalse(response.getVolumesRW()[1].getAccessMode().toBoolean()); + assertTrue(response.getVolumesRW()[0].getAccessMode().toBoolean()); + assertThat(response.getLogPath(), is("/mnt/sda1/var/lib/docker/containers/469e5edd8d5b33e3c905a7ffc97360ec6ee211d6782815fbcd144568045819e1/469e5edd8d5b33e3c905a7ffc97360ec6ee211d6782815fbcd144568045819e1-json.log")); + } + + @Test + public void roundTrip_full_healthcheck() throws IOException { + + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(InspectContainerResponse.class); + + final InspectContainerResponse response = testRoundTrip(RemoteApiVersion.VERSION_1_24, + "/containers/inspect/1.json", + type + ); + + assertEquals("healthy", response.getState().getHealth().getStatus()); + assertEquals(new Integer(0), response.getState().getHealth().getFailingStreak()); + assertEquals(2, response.getState().getHealth().getLog().size()); + assertEquals("Hello", response.getState().getHealth().getLog().get(0).getOutput()); + assertEquals("World", response.getState().getHealth().getLog().get(1).getOutput()); + } + + @Test + public void roundTrip_1_21_full() throws IOException { + InspectContainerResponse[] responses = testRoundTrip(CommandJSONSamples.inspectContainerResponse_full_1_21, + InspectContainerResponse[].class); + assertEquals(1, responses.length); + final InspectContainerResponse response = responses[0]; + final InspectContainerResponse.ContainerState state = response.getState(); + assertThat(state, not(nullValue())); + + assertFalse(state.getDead()); + assertThat(state.getStatus(), containsString("running")); + assertFalse(state.getRestarting()); + assertFalse(state.getOOMKilled()); + assertThat(state.getError(), is(emptyString())); + } + + @Test + public void roundTrip_1_26a_full() throws IOException { + InspectContainerResponse[] responses = testRoundTrip(CommandJSONSamples.inspectContainerResponse_full_1_26a, + InspectContainerResponse[].class); + + assertEquals(1, responses.length); + final InspectContainerResponse response = responses[0]; + + final List mounts = response.getMounts(); + assertEquals(1, mounts.size()); + + final InspectContainerResponse.Mount mount = mounts.get(0); + final Volume volume = mount.getDestination(); + assertEquals("/var/lib/postgresql/data", volume.getPath()); + } + + @Test + public void roundTrip_1_26b_full() throws IOException { + InspectContainerResponse[] responses = testRoundTrip(CommandJSONSamples.inspectContainerResponse_full_1_26b, + InspectContainerResponse[].class); + + assertEquals(1, responses.length); + final InspectContainerResponse response = responses[0]; + + final List mounts = response.getMounts(); + assertEquals(1, mounts.size()); + + final InspectContainerResponse.Mount mount = mounts.get(0); + final Volume volume = mount.getDestination(); + assertEquals("/srv/test", volume.getPath()); + } + + @Test + public void roundTrip_empty() throws IOException { + testRoundTrip(CommandJSONSamples.inspectContainerResponse_empty, InspectContainerResponse[].class); + } + + @Test + public void inspect_windows_container() throws IOException { + + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(InspectContainerResponse.class); + + final InspectContainerResponse response = testRoundTrip(RemoteApiVersion.VERSION_1_38, + "/containers/inspect/lcow.json", + type + ); + + assertThat(response, notNullValue()); + + assertThat(response.getConfig(), notNullValue()); + assertThat(response.getConfig().getCmd(), is(new String[]{"cmd"})); + assertThat(response.getConfig().getImage(), is("microsoft/nanoserver")); + + assertThat(response.getDriver(), is("windowsfilter")); + + assertThat(response.getGraphDriver(), notNullValue()); + assertThat(response.getGraphDriver().getName(), is("windowsfilter")); + assertThat(response.getGraphDriver().getData(), is(new GraphData().withDir( + "C:\\ProgramData\\Docker\\windowsfilter\\35da02ca897bd378ee52be3066c847fee396ba1a28a00b4be36f42c6686bf556" + ))); + + assertThat(response.getHostConfig(), notNullValue()); + assertThat(response.getHostConfig().getIsolation(), is(Isolation.HYPERV)); + + assertThat(response.getImageId(), is("sha256:1381511ec0122f197b6abff5bc0692bef19943ddafd6680eff41197afa3a6dda")); + assertThat(response.getLogPath(), is( + "C:\\ProgramData\\Docker\\containers\\35da02ca897bd378ee52be3066c847fee396ba1a28a00b4be36f42c6686bf556" + + "\\35da02ca897bd378ee52be3066c847fee396ba1a28a00b4be36f42c6686bf556-json.log" + )); + assertThat(response.getName(), is("/cranky_clarke")); + + assertThat(response.getNetworkSettings(), notNullValue()); + assertThat(response.getNetworkSettings().getNetworks(), is(Collections.singletonMap("nat", + new ContainerNetwork() + .withEndpointId("493b77d6fe7e3b92435b1eb01461fde669781330deb84a9cbada360db8997ebc") + .withGateway("172.17.18.1") + .withGlobalIPv6Address("") + .withGlobalIPv6PrefixLen(0) + .withIpv4Address("172.17.18.123") + .withIpPrefixLen(16) + .withIpV6Gateway("") + .withMacAddress("00:aa:ff:cf:dd:09") + .withNetworkID("398c0e206dd677ed4a6566f9de458311f5767d8c7a8b963275490ab64c5d10a7") + ))); + + assertThat(response.getPath(), is("cmd")); + assertThat(response.getPlatform(), is("windows")); + } + + @Test + public void equals() { + assertThat(new InspectContainerResponse(), equalTo(new InspectContainerResponse())); + } +} diff --git a/src/test/java/com/github/dockerjava/api/command/InspectExecResponseTest.java b/docker-java/src/test/java/com/github/dockerjava/api/command/InspectExecResponseTest.java similarity index 81% rename from src/test/java/com/github/dockerjava/api/command/InspectExecResponseTest.java rename to docker-java/src/test/java/com/github/dockerjava/api/command/InspectExecResponseTest.java index cb9264d9f..2449d39be 100644 --- a/src/test/java/com/github/dockerjava/api/command/InspectExecResponseTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/api/command/InspectExecResponseTest.java @@ -1,15 +1,15 @@ package com.github.dockerjava.api.command; import com.fasterxml.jackson.databind.JavaType; -import com.fasterxml.jackson.databind.ObjectMapper; import com.github.dockerjava.core.RemoteApiVersion; -import org.testng.annotations.Test; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.Matchers.hasSize; import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; +import static org.hamcrest.Matchers.emptyString; import static org.hamcrest.core.IsNull.notNullValue; import static org.hamcrest.core.IsNull.nullValue; @@ -20,8 +20,7 @@ public class InspectExecResponseTest { @Test public void test_1_22_SerDer1() throws Exception { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().uncheckedSimpleType(InspectExecResponse.class); + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(InspectExecResponse.class); final InspectExecResponse execResponse = testRoundTrip(RemoteApiVersion.VERSION_1_22, "/exec/ID/1.json", @@ -41,7 +40,7 @@ public void test_1_22_SerDer1() throws Exception { assertThat(processConfig.getEntryPoint(), is("/bin/bash")); assertThat(processConfig.getArguments(), hasSize(0)); assertThat(processConfig.isPrivileged(), is(false)); - assertThat(processConfig.getUser(), isEmptyString()); + assertThat(processConfig.getUser(), is(emptyString())); assertThat(execResponse.isOpenStdin(), is(false)); @@ -51,6 +50,6 @@ public void test_1_22_SerDer1() throws Exception { assertThat(execResponse.getContainerID(), is("ffa39805f089af3099e36452a985481f96170a9dff40be69d34d1722c7660d38")); - assertThat(execResponse.getDetachKeys(), isEmptyString()); + assertThat(execResponse.getDetachKeys(), is(emptyString())); } } diff --git a/docker-java/src/test/java/com/github/dockerjava/api/command/InspectImageResponseTest.java b/docker-java/src/test/java/com/github/dockerjava/api/command/InspectImageResponseTest.java new file mode 100644 index 000000000..a7ec3a78d --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/command/InspectImageResponseTest.java @@ -0,0 +1,250 @@ +package com.github.dockerjava.api.command; + +import com.fasterxml.jackson.databind.JavaType; +import com.github.dockerjava.api.model.ContainerConfig; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import java.io.IOException; +import java.util.Collections; + +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_22; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_25; +import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.contains; +import static org.hamcrest.Matchers.containsInAnyOrder; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.hasItem; +import static org.hamcrest.Matchers.hasSize; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.Matchers.nullValue; + +/** + * @author Kanstantsin Shautsou + */ +public class InspectImageResponseTest { + @Test + public void serder1_22Json() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(InspectImageResponse.class); + + final InspectImageResponse inspectImage = testRoundTrip(VERSION_1_22, + "images/image1/inspect1.json", + type + ); + + final ContainerConfig config = new ContainerConfig() + .withAttachStderr(false) + .withAttachStdin(false) + .withAttachStdout(false) + .withCmd(null) + .withDomainName("") + .withEntrypoint(null) + .withEnv(new String[]{"HOME=/", "PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin"}) + .withExposedPorts(null) + .withHostName("aee9ba801acc") + .withImage("511136ea3c5a64f264b78b5433614aec563103b4d4702f3ba7d4d2698e22c158") + .withLabels(null) + .withMacAddress(null) + .withNetworkDisabled(null) + .withOnBuild(new String[]{}) + .withStdinOpen(false) + .withPortSpecs(null) + .withStdInOnce(false) + .withTty(false) + .withUser("") + .withVolumes(null) + .withWorkingDir(""); + + final ContainerConfig containerConfig = new ContainerConfig() + .withAttachStderr(false) + .withAttachStdin(false) + .withAttachStdout(false) + .withCmd(new String[]{"/bin/sh", "-c", "#(nop) MAINTAINER hack@worldticket.net"}) + .withDomainName("") + .withEntrypoint(null) + .withEnv(new String[]{"HOME=/", "PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin"}) + .withExposedPorts(null) + .withHostName("aee9ba801acc") + .withImage("511136ea3c5a64f264b78b5433614aec563103b4d4702f3ba7d4d2698e22c158") + .withLabels(null) + .withMacAddress(null) + .withNetworkDisabled(null) + .withOnBuild(new String[]{}) + .withStdinOpen(false) + .withPortSpecs(null) + .withStdInOnce(false) + .withTty(false) + .withUser("") + .withVolumes(null) + .withWorkingDir(""); + + assertThat(inspectImage, notNullValue()); + assertThat(inspectImage.getArch(), is("amd64")); + assertThat(inspectImage.getAuthor(), is("hack@worldticket.net")); + assertThat(inspectImage.getComment(), is(emptyString())); + + assertThat(inspectImage.getConfig(), notNullValue()); + assertThat(inspectImage.getConfig(), equalTo(config)); + + assertThat(inspectImage.getCreated(), is("2014-04-29T19:59:10.84997669Z")); + assertThat(inspectImage.getContainer(), is("aee9ba801acca0e648ffd91df204ba82ae85d97608a4864a019e2004d7e1b133")); + + assertThat(inspectImage.getContainerConfig(), notNullValue()); + assertThat(inspectImage.getContainerConfig(), equalTo(containerConfig)); + + assertThat(inspectImage.getDockerVersion(), is("0.8.1")); + assertThat(inspectImage.getId(), is("sha256:ee45fe0d1fcdf1a0f9c2d1e36c6f4b3202bbb2032f14d7c9312b27bfcf6aee24")); + assertThat(inspectImage.getOs(), is("linux")); + assertThat(inspectImage.getParent(), is(emptyString())); + assertThat(inspectImage.getSize(), is(0L)); + + assertThat(inspectImage.getRepoTags(), hasSize(1)); + assertThat(inspectImage.getRepoTags(), hasItem("hackmann/empty:latest")); + + final GraphDriver aufsGraphDriver = new GraphDriver().withName("aufs"); + final GraphDriver graphDriver = inspectImage.getGraphDriver(); + assertThat(graphDriver, notNullValue()); + assertThat(graphDriver, equalTo(aufsGraphDriver)); + assertThat(graphDriver.getName(), is("aufs")); + assertThat(graphDriver.getData(), nullValue()); + + assertThat(inspectImage.getVirtualSize(), is(0L)); + + + final InspectImageResponse inspectImageResponse = new InspectImageResponse().withArch("amd64") + .withAuthor("hack@worldticket.net") + .withComment("") + .withConfig(config) + .withContainer("aee9ba801acca0e648ffd91df204ba82ae85d97608a4864a019e2004d7e1b133") + .withContainerConfig(containerConfig) + .withCreated("2014-04-29T19:59:10.84997669Z") + .withDockerVersion("0.8.1") + .withId("sha256:ee45fe0d1fcdf1a0f9c2d1e36c6f4b3202bbb2032f14d7c9312b27bfcf6aee24") + .withOs("linux") + .withParent("") + .withSize(0L) + .withRepoTags(Collections.singletonList("hackmann/empty:latest")) + .withRepoDigests(Collections.emptyList()) + .withVirtualSize(0L) + .withGraphDriver(aufsGraphDriver); + + assertThat(inspectImage, equalTo(inspectImageResponse)); + } + + + @Test + public void serder1_22_doc() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(InspectImageResponse.class); + + final InspectImageResponse inspectImage = testRoundTrip(VERSION_1_22, + "images/docImage/doc.json", + type + ); + + assertThat(inspectImage, notNullValue()); + + assertThat(inspectImage.getRepoDigests(), hasSize(1)); + assertThat(inspectImage.getRepoDigests(), + contains("localhost:5000/test/busybox/example@" + + "sha256:cbbf2f9a99b47fc460d422812b6a5adff7dfee951d8fa2e4a98caa0382cfbdbf") + ); + + assertThat(inspectImage.getRepoTags(), hasSize(3)); + assertThat(inspectImage.getRepoTags(), containsInAnyOrder( + "example:1.0", + "example:latest", + "example:stable" + )); + } + + @Test + public void serder1_22_inspect_doc() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(InspectImageResponse.class); + + final InspectImageResponse inspectImage = testRoundTrip(VERSION_1_22, + "images/docImage/inspect_doc.json", + type + ); + + GraphData newGraphData = new GraphData() + .withDeviceId("5") + .withDeviceName("docker-253:1-2763198-d2cc496561d6d520cbc0236b4ba88c362c446a7619992123f11c809cded25b47") + .withDeviceSize("171798691840"); + + assertThat(inspectImage, notNullValue()); + GraphDriver graphDriver = inspectImage.getGraphDriver(); + assertThat(graphDriver, notNullValue()); + GraphData data = graphDriver.getData(); + + assertThat(data, is(newGraphData)); + + assertThat(data.getDeviceId(), is("5")); + assertThat(data.getDeviceName(), + is("docker-253:1-2763198-d2cc496561d6d520cbc0236b4ba88c362c446a7619992123f11c809cded25b47")); + assertThat(data.getDeviceSize(), + is("171798691840")); + } + + @Test + public void testOverlayNetworkRootDir() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(InspectImageResponse.class); + + final InspectImageResponse inspectImage = testRoundTrip(VERSION_1_22, "images/overlay/inspectOverlay.json", type); + + final GraphData overlayGraphData = new GraphData() + .withRootDir("/var/lib/docker/overlay/7e8d362d6b78d47eafe4863fd129cbcada35dbd419d7188cc1dbf1233d505576/root"); + final GraphDriver overlayGraphDriver = new GraphDriver().withName("overlay").withData(overlayGraphData); + final GraphDriver graphDriver = inspectImage.getGraphDriver(); + assertThat(graphDriver, notNullValue()); + assertThat(graphDriver, equalTo(overlayGraphDriver)); + assertThat(graphDriver.getName(), is("overlay")); + assertThat(graphDriver.getData(), equalTo(overlayGraphData)); + } + + @Test + public void inspectWindowsImage() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(InspectImageResponse.class); + + final InspectImageResponse inspectImage = testRoundTrip(VERSION_1_25, + "images/windowsImage/doc.json", + type + ); + + assertThat(inspectImage, notNullValue()); + + assertThat(inspectImage.getRepoTags(), hasSize(1)); + assertThat(inspectImage.getRepoTags(), contains( + "microsoft/nanoserver:latest" + )); + + assertThat(inspectImage.getRepoDigests(), hasSize(1)); + assertThat(inspectImage.getRepoDigests(), contains("microsoft/nanoserver@" + + "sha256:aee7d4330fe3dc5987c808f647441c16ed2fa1c7d9c6ef49d6498e5c9860b50b") + ); + + assertThat(inspectImage.getConfig(), notNullValue()); + assertThat(inspectImage.getConfig().getCmd(), is(new String[]{"c:\\windows\\system32\\cmd.exe"})); + + assertThat(inspectImage.getOs(), is("windows")); + assertThat(inspectImage.getOsVersion(), is("10.0.14393")); + assertThat(inspectImage.getSize(), is(651862727L)); + assertThat(inspectImage.getVirtualSize(), is(651862727L)); + + assertThat(inspectImage.getGraphDriver(), notNullValue()); + assertThat(inspectImage.getGraphDriver().getName(), is("windowsfilter")); + assertThat(inspectImage.getGraphDriver().getData(), notNullValue()); + assertThat(inspectImage.getGraphDriver().getData().getDir(), is("C:\\control\\windowsfilter\\" + + "6fe6a289b98276a6a5ca0345156ca61d7b38f3da6bb49ef95af1d0f1ac37e5bf" + )); + + assertThat(inspectImage.getRootFS(), notNullValue()); + assertThat(inspectImage.getRootFS().getType(), is("layers")); + assertThat(inspectImage.getRootFS().getLayers(), hasSize(1)); + assertThat(inspectImage.getRootFS().getLayers(), contains( + "sha256:342d4e407550c52261edd20cd901b5ce438f0b1e940336de3978210612365063" + )); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/command/UpdateContainerResponseTest.java b/docker-java/src/test/java/com/github/dockerjava/api/command/UpdateContainerResponseTest.java new file mode 100644 index 000000000..300a5c324 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/command/UpdateContainerResponseTest.java @@ -0,0 +1,33 @@ +package com.github.dockerjava.api.command; + +import com.github.dockerjava.api.model.UpdateContainerResponse; +import org.junit.Test; + +import java.io.IOException; +import java.util.List; + +import static com.github.dockerjava.test.serdes.JSONTestHelper.testRoundTrip; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertNull; + +public class UpdateContainerResponseTest { + @Test + public void roundTrip_empty() throws IOException { + UpdateContainerResponse response = testRoundTrip(CommandJSONSamples.updateContainerResponse_empty, UpdateContainerResponse.class); + assertNull(response.getWarnings()); + } + + @Test + public void roundTrip_warnings() throws IOException { + UpdateContainerResponse response = testRoundTrip(CommandJSONSamples.updateContainerResponse_warnings, + UpdateContainerResponse.class); + + List warnings = response.getWarnings(); + assertNotNull(warnings); + assertEquals(1, warnings.size()); + + final String warning = warnings.get(0); + assertEquals("Published ports are discarded when using host network mode", warning); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/AccessModeTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/AccessModeTest.java new file mode 100644 index 000000000..d5ff7044a --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/AccessModeTest.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.api.model; + +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; + +import static com.github.dockerjava.api.model.AccessMode.rw; +import static org.junit.Assert.assertEquals; + + +public class AccessModeTest { + + @Rule + public ExpectedException expectedEx = ExpectedException.none(); + + @Test + public void defaultAccessMode() { + assertEquals(rw, AccessMode.DEFAULT); + } + + @Test + public void stringify() { + assertEquals("rw", AccessMode.rw.toString()); + } + + @Test + public void fromString() { + assertEquals(rw, AccessMode.valueOf("rw")); + } + + @Test + public void fromIllegalString() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("No enum const"); + + AccessMode.valueOf("xx"); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/AuthConfigTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/AuthConfigTest.java new file mode 100644 index 000000000..ae3e4a91b --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/AuthConfigTest.java @@ -0,0 +1,110 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.databind.JavaType; +import com.github.dockerjava.core.RemoteApiVersion; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import java.io.IOException; + +import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; +import static org.hamcrest.CoreMatchers.*; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.notNullValue; +import static org.junit.Assert.assertEquals; + +public class AuthConfigTest { + + @Test + public void defaultServerAddress() { + assertEquals(new AuthConfig().getRegistryAddress(), "https://index.docker.io/v1/"); + } + + @Test + public void serderDocs1() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(AuthConfig.class); + + final AuthConfig authConfig = testRoundTrip(RemoteApiVersion.VERSION_1_22, + "/other/AuthConfig/docs1.json", + type + ); + + assertThat(authConfig, notNullValue()); + assertThat(authConfig.getUsername(), is("jdoe")); + assertThat(authConfig.getPassword(), is("secret")); + assertThat(authConfig.getEmail(), is("jdoe@acme.com")); + + final AuthConfig authConfig1 = new AuthConfig().withUsername("jdoe") + .withPassword("secret") + .withEmail("jdoe@acme.com"); + + assertThat(authConfig1, equalTo(authConfig)); + } + + @Test + public void serderDocs2() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(AuthConfig.class); + + final AuthConfig authConfig = testRoundTrip(RemoteApiVersion.VERSION_1_22, + "/other/AuthConfig/docs2.json", + type + ); + + assertThat(authConfig, notNullValue()); + assertThat(authConfig.getRegistrytoken(), is("9cbaf023786cd7...")); + + + final AuthConfig authConfig1 = new AuthConfig().withRegistrytoken("9cbaf023786cd7..."); + + assertThat(authConfig1, equalTo(authConfig)); + } + + @Test + public void compatibleWithIdentitytoken() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(AuthConfig.class); + final AuthConfig authConfig = testRoundTrip(RemoteApiVersion.VERSION_1_23, + "/other/AuthConfig/docs1.json", + type + ); + String auth = "YWRtaW46"; + String identitytoken = "1cba468e-8cbe-4c55-9098-2c2ed769e885"; + assertThat(authConfig, notNullValue()); + assertThat(authConfig.getAuth(), is(auth)); + assertThat(authConfig.getIdentitytoken(), is(identitytoken)); + final AuthConfig authConfig1 = new AuthConfig().withAuth(auth).withIdentityToken(identitytoken); + assertThat(authConfig1, equalTo(authConfig)); + } + + @Test + public void shouldNotFailWithStackOrchestratorInConfig() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(AuthConfig.class); + final AuthConfig authConfig = testRoundTrip(RemoteApiVersion.VERSION_1_25, + "/other/AuthConfig/orchestrators.json", + type + ); + assertThat(authConfig, notNullValue()); + assertThat(authConfig.getAuth(), is(nullValue())); + assertThat(authConfig.getStackOrchestrator(), is("kubernetes")); + } + + @Test + public void toStringDoesNotContainSensitiveStrings() { + AuthConfig authConfig = new AuthConfig() + .withAuth("authValue") + .withEmail("emailValue") + .withPassword("passwordValue") + .withIdentityToken("identityTokenValue") + .withRegistrytoken("registryTokenValue") + .withRegistryAddress("registryAddressValue"); + String toStringValue = authConfig.toString(); + + assertThat(toStringValue, not(containsString("authValue"))); + assertThat(toStringValue, not(containsString("passwordValue"))); + assertThat(toStringValue, not(containsString("identityTokenValue"))); + assertThat(toStringValue, not(containsString("registryTokenValue"))); + + assertThat(toStringValue, containsString("emailValue")); + assertThat(toStringValue, containsString("registryAddressValue")); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/BindPropagationTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/BindPropagationTest.java new file mode 100644 index 000000000..0fe65c048 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/BindPropagationTest.java @@ -0,0 +1,24 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import static org.hamcrest.Matchers.is; +import static org.hamcrest.MatcherAssert.assertThat; + +public class BindPropagationTest { + + @Test + public void toJson() throws Exception { + String value = JSONTestHelper.getMapper().writeValueAsString(BindPropagation.R_PRIVATE); + + assertThat(value, is("\"rprivate\"")); + } + + @Test + public void fromJson() throws Exception { + BindPropagation value = JSONTestHelper.getMapper().readValue("\"rprivate\"", BindPropagation.class); + + assertThat(value, is(BindPropagation.R_PRIVATE)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/BindTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/BindTest.java new file mode 100644 index 000000000..d31a66dde --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/BindTest.java @@ -0,0 +1,365 @@ +package com.github.dockerjava.api.model; + +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; + +import static com.github.dockerjava.api.model.AccessMode.ro; +import static com.github.dockerjava.api.model.AccessMode.rw; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.nullValue; +import static org.hamcrest.core.Is.is; + +public class BindTest { + + @Rule + public ExpectedException expectedEx = ExpectedException.none(); + + @Test + public void parseUsingDefaultAccessMode() { + Bind bind = Bind.parse("/host:/container"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(AccessMode.DEFAULT)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseReadWriteWindows() { + Bind bind = Bind.parse("C:\\host:/container:rw"); + assertThat(bind.getPath(), is("C:\\host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseReadWriteNoCopyWindows() { + Bind bind = Bind.parse("C:\\host:/container:rw,nocopy"); + assertThat(bind.getPath(), is("C:\\host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), is(true)); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseReadWriteSharedWindows() { + Bind bind = Bind.parse("C:\\host:/container:rw,shared"); + assertThat(bind.getPath(), is("C:\\host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.SHARED)); + } + + @Test + public void parseReadWriteSlaveWindows() { + Bind bind = Bind.parse("C:\\host:/container:rw,slave"); + assertThat(bind.getPath(), is("C:\\host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.SLAVE)); + } + + @Test + public void parseReadWritePrivateWindows() { + Bind bind = Bind.parse("C:\\host:/container:rw,private"); + assertThat(bind.getPath(), is("C:\\host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.PRIVATE)); + } + + @Test + public void parseReadOnlyWindows() { + Bind bind = Bind.parse("C:\\host:/container:ro"); + assertThat(bind.getPath(), is("C:\\host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(ro)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseSELOnlyWindows() { + Bind bind = Bind.parse("C:\\host:/container:Z"); + assertThat(bind.getPath(), is("C:\\host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(AccessMode.DEFAULT)); + assertThat(bind.getSecMode(), is(SELContext.single)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + + bind = Bind.parse("C:\\host:/container:z"); + assertThat(bind.getPath(), is("C:\\host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(AccessMode.DEFAULT)); + assertThat(bind.getSecMode(), is(SELContext.shared)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseReadWriteSELWindows() { + Bind bind = Bind.parse("C:\\host:/container:rw,Z"); + assertThat(bind.getPath(), is("C:\\host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.single)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseReadOnlySELWindows() { + Bind bind = Bind.parse("C:\\host:/container:ro,z"); + assertThat(bind.getPath(), is("C:\\host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(ro)); + assertThat(bind.getSecMode(), is(SELContext.shared)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseReadWrite() { + Bind bind = Bind.parse("/host:/container:rw"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseReadWriteNoCopy() { + Bind bind = Bind.parse("/host:/container:rw,nocopy"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), is(true)); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseReadWriteShared() { + Bind bind = Bind.parse("/host:/container:rw,shared"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.SHARED)); + } + + @Test + public void parseReadWriteRshared() { + Bind bind = Bind.parse("/host:/container:rw,rshared"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.RSHARED)); + } + + @Test + public void parseReadWriteSlave() { + Bind bind = Bind.parse("/host:/container:rw,slave"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.SLAVE)); + } + + @Test + public void parseReadWriteRslave() { + Bind bind = Bind.parse("/host:/container:rw,rslave"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.RSLAVE)); + } + + @Test + public void parseReadWritePrivate() { + Bind bind = Bind.parse("/host:/container:rw,private"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.PRIVATE)); + } + + @Test + public void parseReadWriteRprivate() { + Bind bind = Bind.parse("/host:/container:rw,rprivate"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.RPRIVATE)); + } + + @Test + public void parseReadOnly() { + Bind bind = Bind.parse("/host:/container:ro"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(ro)); + assertThat(bind.getSecMode(), is(SELContext.none)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseSELOnly() { + Bind bind = Bind.parse("/host:/container:Z"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(AccessMode.DEFAULT)); + assertThat(bind.getSecMode(), is(SELContext.single)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + + bind = Bind.parse("/host:/container:z"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(AccessMode.DEFAULT)); + assertThat(bind.getSecMode(), is(SELContext.shared)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseReadWriteSEL() { + Bind bind = Bind.parse("/host:/container:rw,Z"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(rw)); + assertThat(bind.getSecMode(), is(SELContext.single)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseReadOnlySEL() { + Bind bind = Bind.parse("/host:/container:ro,z"); + assertThat(bind.getPath(), is("/host")); + assertThat(bind.getVolume().getPath(), is("/container")); + assertThat(bind.getAccessMode(), is(ro)); + assertThat(bind.getSecMode(), is(SELContext.shared)); + assertThat(bind.getNoCopy(), nullValue()); + assertThat(bind.getPropagationMode(), is(PropagationMode.DEFAULT_MODE)); + } + + @Test + public void parseInvalidAccessMode() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage( "Error parsing Bind"); + + Bind.parse("/host:/container:xx"); + } + + @Test + public void parseInvalidInput() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing Bind 'nonsense'"); + + Bind.parse("nonsense"); + } + + @Test + public void parseNull() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing Bind 'null'"); + + Bind.parse(null); + } + + @Test + public void toStringReadOnly() { + assertThat(Bind.parse("/host:/container:ro").toString(), is("/host:/container:ro")); + } + + @Test + public void toStringReadWrite() { + assertThat(Bind.parse("/host:/container:rw").toString(), is("/host:/container:rw")); + } + + @Test + public void toStringReadWriteNoCopy() { + assertThat(Bind.parse("/host:/container:rw,nocopy").toString(), is("/host:/container:rw,nocopy")); + } + + @Test + public void toStringReadWriteShared() { + assertThat(Bind.parse("/host:/container:rw,shared").toString(), is("/host:/container:rw,shared")); + } + + @Test + public void toStringReadWriteRshared() { + assertThat(Bind.parse("/host:/container:rw,rshared").toString(), is("/host:/container:rw,rshared")); + } + + @Test + public void toStringReadWriteSlave() { + assertThat(Bind.parse("/host:/container:rw,slave").toString(), is("/host:/container:rw,slave")); + } + + @Test + public void toStringReadWriteRslave() { + assertThat(Bind.parse("/host:/container:rw,rslave").toString(), is("/host:/container:rw,rslave")); + } + + @Test + public void toStringReadWritePrivate() { + assertThat(Bind.parse("/host:/container:rw,private").toString(), is("/host:/container:rw,private")); + } + + @Test + public void toStringReadWriteRprivate() { + assertThat(Bind.parse("/host:/container:rw,rprivate").toString(), is("/host:/container:rw,rprivate")); + } + + @Test + public void toStringDefaultAccessMode() { + assertThat(Bind.parse("/host:/container").toString(), is("/host:/container:rw")); + } + + @Test + public void toStringReadOnlySEL() { + assertThat(Bind.parse("/host:/container:ro,Z").toString(), is("/host:/container:ro,Z")); + } + + @Test + public void toStringReadWriteSEL() { + assertThat(Bind.parse("/host:/container:rw,z").toString(), is("/host:/container:rw,z")); + } + + @Test + public void toStringDefaultSEL() { + assertThat(Bind.parse("/host:/container:Z").toString(), is("/host:/container:rw,Z")); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/BindingTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/BindingTest.java new file mode 100644 index 000000000..1d7458d85 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/BindingTest.java @@ -0,0 +1,70 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.api.model.Ports.Binding; +import org.junit.Ignore; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; + +import static org.junit.Assert.assertEquals; + + +public class BindingTest { + + @Rule + public ExpectedException expectedEx = ExpectedException.none(); + + @Test + public void parseIpAndPort() { + assertEquals(Binding.parse("127.0.0.1:80"), Binding.bindIpAndPort("127.0.0.1", 80)); + } + + @Test + public void parsePortOnly() { + assertEquals(Binding.parse("80"), Binding.bindPort(80)); + } + + @Test + public void parseIPOnly() { + assertEquals(Binding.parse("127.0.0.1"), Binding.bindIp("127.0.0.1")); + } + + @Test + public void parseEmptyString() { + assertEquals(Binding.parse(""), Binding.empty()); + } + + // Strings can be used since it can be a range. Let the docker daemon do the validation. + @Test + @Ignore + public void parseInvalidInput() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing Binding 'nonsense'"); + + Binding.parse("nonsense"); + } + + @Test + public void parseNull() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing Binding 'null'"); + + Binding.parse(null); + } + + @Test + public void toStringIpAndHost() { + assertEquals(Binding.parse("127.0.0.1:80").toString(), "127.0.0.1:80"); + } + + @Test + public void toStringPortOnly() { + assertEquals(Binding.parse("80").toString(), "80"); + } + + @Test + public void toStringIpOnly() { + assertEquals(Binding.parse("127.0.0.1").toString(), "127.0.0.1"); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/BindsTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/BindsTest.java new file mode 100644 index 000000000..83c5ba788 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/BindsTest.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import static org.hamcrest.Matchers.arrayContaining; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.MatcherAssert.assertThat; + +public class BindsTest { + + @Test + public void usesToJson() throws Exception { + Binds binds = new Binds( + Bind.parse("/foo:/bar:rw"), + Bind.parse("/bip:/bop:ro") + ); + String json = JSONTestHelper.getMapper().writeValueAsString(binds); + + assertThat(json, is("[\"/foo:/bar:rw\",\"/bip:/bop:ro\"]")); + } + + @Test + public void usesFromJson() throws Exception { + Binds binds = JSONTestHelper.getMapper().readValue("[\"/foo:/bar:rw\",\"/bip:/bop:ro\"]", Binds.class); + + assertThat(binds, notNullValue()); + assertThat(binds.getBinds(), arrayContaining( + Bind.parse("/foo:/bar:rw"), + Bind.parse("/bip:/bop:ro") + )); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/CapabilityTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/CapabilityTest.java new file mode 100644 index 000000000..b0652d945 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/CapabilityTest.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.databind.JsonMappingException; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import static org.junit.Assert.assertEquals; + +public class CapabilityTest { + + @Test + public void serializeCapability() throws Exception { + String json = JSONTestHelper.getMapper().writeValueAsString(Capability.ALL); + assertEquals("\"ALL\"", json); + } + + @Test + public void deserializeCapability() throws Exception { + Capability capability = JSONTestHelper.getMapper().readValue("\"ALL\"", Capability.class); + assertEquals(Capability.ALL, capability); + + Capability compatibleCapability = JSONTestHelper.getMapper().readValue("\"CAP_ALL\"", Capability.class); + assertEquals(Capability.ALL, compatibleCapability); + } + + @Test(expected = JsonMappingException.class) + public void deserializeInvalidCapability() throws Exception { + JSONTestHelper.getMapper().readValue("\"nonsense\"", Capability.class); + } +} diff --git a/src/test/java/com/github/dockerjava/api/model/ContainerTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/ContainerTest.java similarity index 82% rename from src/test/java/com/github/dockerjava/api/model/ContainerTest.java rename to docker-java/src/test/java/com/github/dockerjava/api/model/ContainerTest.java index a143e6c57..7b1e22f0d 100644 --- a/src/test/java/com/github/dockerjava/api/model/ContainerTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/ContainerTest.java @@ -1,9 +1,9 @@ package com.github.dockerjava.api.model; -import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.type.CollectionType; import com.github.dockerjava.core.RemoteApiVersion; -import org.testng.annotations.Test; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; import java.io.IOException; import java.util.List; @@ -20,8 +20,7 @@ public class ContainerTest { @Test public void serderJson1() throws IOException { - final ObjectMapper mapper = new ObjectMapper(); - final CollectionType type = mapper.getTypeFactory().constructCollectionType(List.class, Container.class); + final CollectionType type = JSONTestHelper.getMapper().getTypeFactory().constructCollectionType(List.class, Container.class); final List containers = testRoundTrip(RemoteApiVersion.VERSION_1_22, "containers/json/filter1.json", diff --git a/src/test/java/com/github/dockerjava/api/model/DeviceTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/DeviceTest.java similarity index 91% rename from src/test/java/com/github/dockerjava/api/model/DeviceTest.java rename to docker-java/src/test/java/com/github/dockerjava/api/model/DeviceTest.java index 5c48c4886..9d191fe52 100644 --- a/src/test/java/com/github/dockerjava/api/model/DeviceTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/DeviceTest.java @@ -1,21 +1,17 @@ package com.github.dockerjava.api.model; -import org.testng.annotations.Test; +import org.junit.Test; -import java.util.ArrayList; import java.util.Arrays; -import java.util.Collections; import java.util.HashMap; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; -import static junit.framework.Assert.fail; import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.contains; import static org.hamcrest.Matchers.containsString; import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.is; +import static org.junit.Assert.fail; /** * @author Kanstantsin Shautsou @@ -58,7 +54,7 @@ public class DeviceTest { }}; @Test - public void testParse() throws Exception { + public void testParse() { assertThat(Device.parse("/dev/sda:/dev/xvdc:r"), equalTo(new Device("r", "/dev/xvdc", "/dev/sda"))); diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/EventsTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/EventsTest.java new file mode 100644 index 000000000..0f0b2456d --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/EventsTest.java @@ -0,0 +1,63 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.databind.JavaType; +import com.github.dockerjava.core.RemoteApiVersion; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import java.io.IOException; +import java.util.HashMap; + +import static com.github.dockerjava.api.model.EventType.CONTAINER; +import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; +import static org.hamcrest.CoreMatchers.equalTo; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.CoreMatchers.nullValue; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.notNullValue; + +/** + * @author Kanstantsin Shautsou + */ +public class EventsTest { + + @Test + public void serderDocs1() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(Event.class); + + final Event event = testRoundTrip(RemoteApiVersion.VERSION_1_24, + "/events/docs1.json", + type + ); + + assertThat(event, notNullValue()); + assertThat(event.getType(), is(CONTAINER)); + assertThat(event.getAction(), is("create")); + assertThat(event.getId(), is("ede54ee1afda366ab42f824e8a5ffd195155d853ceaec74a927f249ea270c743")); + assertThat(event.getFrom(), is("alpine")); + assertThat(event.getTime(), is(1461943101L)); + assertThat(event.getNode(), nullValue()); + assertThat(event.getTimeNano(), is(1461943101381709551L)); + + final HashMap attributes = new HashMap<>(); + attributes.put("com.example.some-label", "some-label-value"); + attributes.put("image", "alpine"); + attributes.put("name", "my-container"); + + final EventActor actor = new EventActor() + .withId("ede54ee1afda366ab42f824e8a5ffd195155d853ceaec74a927f249ea270c743") + .withAttributes(attributes); + + final Event event1 = new Event() + .withType(CONTAINER) + .withStatus("create") + .withId("ede54ee1afda366ab42f824e8a5ffd195155d853ceaec74a927f249ea270c743") + .withFrom("alpine") + .withTime(1461943101L) + .withTimenano(1461943101381709551L) + .withAction("create") + .withEventActor(actor); + + assertThat(event1, equalTo(event)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/ExposedPortTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/ExposedPortTest.java new file mode 100644 index 000000000..fce761380 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/ExposedPortTest.java @@ -0,0 +1,49 @@ +package com.github.dockerjava.api.model; + +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; + +import static com.github.dockerjava.api.model.InternetProtocol.DEFAULT; +import static com.github.dockerjava.api.model.InternetProtocol.TCP; +import static org.junit.Assert.assertEquals; + +public class ExposedPortTest { + + @Rule + public ExpectedException expectedEx = ExpectedException.none(); + + @Test + public void parsePortAndProtocol() { + ExposedPort exposedPort = ExposedPort.parse("80/tcp"); + assertEquals(new ExposedPort(80, TCP), exposedPort); + } + + @Test + public void parsePortOnly() { + ExposedPort exposedPort = ExposedPort.parse("80"); + assertEquals(new ExposedPort(80, DEFAULT), exposedPort); + } + + @Test + public void parseInvalidInput() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing ExposedPort 'nonsense'"); + + ExposedPort.parse("nonsense"); + } + + @Test + public void parseNull() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing ExposedPort 'null'"); + + ExposedPort.parse(null); + } + + @Test + public void stringify() { + assertEquals("80/tcp", ExposedPort.parse("80/tcp").toString()); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/ExposedPortsTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/ExposedPortsTest.java new file mode 100644 index 000000000..f46dddc68 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/ExposedPortsTest.java @@ -0,0 +1,67 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.databind.JsonNode; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.apache.commons.compress.utils.Lists; +import org.junit.Test; + +import java.util.List; +import java.util.Map; +import java.util.Map.Entry; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.aMapWithSize; +import static org.hamcrest.Matchers.arrayContainingInAnyOrder; +import static org.hamcrest.Matchers.arrayWithSize; +import static org.hamcrest.Matchers.notNullValue; + +public class ExposedPortsTest { + + @Test + public void usesToJson() throws Exception { + ExposedPorts ports = new ExposedPorts( + new ExposedPort(80), + new ExposedPort(123, InternetProtocol.UDP), + new ExposedPort(3868, InternetProtocol.SCTP) + ); + String json = JSONTestHelper.getMapper().writeValueAsString(ports); + List> jsonEntries = getJsonEntries(json); + + String jsonExpected = "{\"80/tcp\":{},\"123/udp\":{},\"3868/sctp\":{}}"; + List> jsonEntriesExpected = getJsonEntries(jsonExpected); + + assertThat(jsonEntries.toArray(), arrayContainingInAnyOrder(jsonEntriesExpected.toArray())); + } + + private List> getJsonEntries(String json) throws Exception { + JsonNode jsonNode = JSONTestHelper.getMapper().readValue(json, JsonNode.class); + return Lists.newArrayList(jsonNode.fields()); + } + + @Test + public void usesFromJson() throws Exception { + ExposedPorts ports = JSONTestHelper.getMapper().readValue("{\"80/tcp\":{},\"123/udp\":{},\"3868/sctp\":{}}", ExposedPorts.class); + + assertThat(ports, notNullValue()); + assertThat(ports.getExposedPorts(), arrayContainingInAnyOrder( + new ExposedPort(80), + new ExposedPort(123, InternetProtocol.UDP), + new ExposedPort(3868, InternetProtocol.SCTP) + )); + } + + @Test + public void usesFromJsonWithDuplicate() { + ExposedPorts ports = new ExposedPorts( + new ExposedPort(80, InternetProtocol.UDP), + new ExposedPort(80), + new ExposedPort(80) + ); + + assertThat(ports, notNullValue()); + assertThat(ports.getExposedPorts(), arrayWithSize(3)); + + Map map = ports.toPrimitive(); + assertThat(map, aMapWithSize(2)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/HostConfigTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/HostConfigTest.java new file mode 100644 index 000000000..5e13103dd --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/HostConfigTest.java @@ -0,0 +1,15 @@ +package com.github.dockerjava.api.model; + +import org.junit.Test; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; + +public class HostConfigTest { + + @Test + public void testNewObjectsEqual() { + assertThat(HostConfig.newHostConfig(), + equalTo(HostConfig.newHostConfig())); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/IdentifierTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/IdentifierTest.java new file mode 100644 index 000000000..3b8efa2c5 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/IdentifierTest.java @@ -0,0 +1,42 @@ +package com.github.dockerjava.api.model; + + +import org.junit.Test; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertTrue; + +public class IdentifierTest { + + @Test + public void testFromCompoundString() throws Exception { + + Identifier i1 = Identifier.fromCompoundString("10.0.0.1/jim"); + Identifier i2 = Identifier.fromCompoundString("10.0.0.1/jim:123"); + Identifier i3 = Identifier.fromCompoundString("10.0.0.1:123/jim:124"); + Identifier i3A = Identifier.fromCompoundString("10.0.0.1:123/jim:latest"); + + assertTrue(!i1.tag.isPresent()); + assertEquals("10.0.0.1/jim", i1.repository.name); + + assertTrue(i2.tag.isPresent()); + assertEquals("123", i2.tag.get()); + assertEquals("10.0.0.1/jim", i2.repository.name); + + assertTrue(i3.tag.isPresent()); + assertEquals("124", i3.tag.get()); + assertEquals("10.0.0.1:123/jim", i3.repository.name); + assertEquals(123, i3.repository.getURL().getPort()); + assertEquals("latest", i3A.tag.get()); + + Identifier i4 = Identifier.fromCompoundString("centos:latest"); + assertTrue(i4.tag.isPresent()); + assertEquals("latest", i4.tag.get()); + + Identifier i5 = Identifier.fromCompoundString("busybox"); + assertTrue(!i5.tag.isPresent()); + + Identifier i6 = Identifier.fromCompoundString("10.0.0.1:5000/my-test-image:1234"); + assertEquals("my-test-image", i6.repository.getPath()); + } +} diff --git a/src/test/java/com/github/dockerjava/api/model/InfoTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/InfoTest.java similarity index 85% rename from src/test/java/com/github/dockerjava/api/model/InfoTest.java rename to docker-java/src/test/java/com/github/dockerjava/api/model/InfoTest.java index 806870ae6..71d4d60d4 100644 --- a/src/test/java/com/github/dockerjava/api/model/InfoTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/InfoTest.java @@ -1,12 +1,14 @@ package com.github.dockerjava.api.model; import com.fasterxml.jackson.databind.JavaType; -import com.fasterxml.jackson.databind.ObjectMapper; import com.github.dockerjava.api.model.InfoRegistryConfig.IndexConfig; +import com.github.dockerjava.core.RemoteApiVersion; +import com.github.dockerjava.test.serdes.JSONTestHelper; import org.hamcrest.CoreMatchers; -import org.testng.annotations.Test; +import org.junit.Test; import java.io.IOException; +import java.util.Arrays; import java.util.Collections; import java.util.LinkedHashMap; import java.util.List; @@ -22,7 +24,7 @@ import static org.hamcrest.Matchers.equalTo; import static org.hamcrest.Matchers.hasEntry; import static org.hamcrest.Matchers.hasSize; -import static org.hamcrest.Matchers.isEmptyString; +import static org.hamcrest.Matchers.emptyString; import static org.hamcrest.Matchers.notNullValue; import static org.hamcrest.Matchers.nullValue; @@ -33,8 +35,7 @@ public class InfoTest { @Test public void serder1Json() throws IOException { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().constructType(Info.class); + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(Info.class); final Info info = testRoundTrip(VERSION_1_22, "info/1.json", @@ -79,9 +80,9 @@ public void serder1Json() throws IOException { assertThat(info.getExperimentalBuild(), is(false)); - assertThat(info.getHttpProxy(), isEmptyString()); - assertThat(info.getHttpsProxy(), isEmptyString()); - assertThat(info.getNoProxy(), isEmptyString()); + assertThat(info.getHttpProxy(), is(emptyString())); + assertThat(info.getHttpsProxy(), is(emptyString())); + assertThat(info.getNoProxy(), is(emptyString())); assertThat(info.getOomKillDisable(), is(true)); assertThat(info.getOsType(), equalTo("linux")); @@ -173,8 +174,7 @@ public void serder1Json() throws IOException { @Test public void serder2Json() throws IOException { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().constructType(Info.class); + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(Info.class); final Info info = testRoundTrip(VERSION_1_22, "info/2.json", @@ -234,9 +234,9 @@ public void serder2Json() throws IOException { assertThat(info.getExperimentalBuild(), is(false)); - assertThat(info.getHttpProxy(), isEmptyString()); - assertThat(info.getHttpsProxy(), isEmptyString()); - assertThat(info.getNoProxy(), isEmptyString()); + assertThat(info.getHttpProxy(), is(emptyString())); + assertThat(info.getHttpsProxy(), is(emptyString())); + assertThat(info.getNoProxy(), is(emptyString())); assertThat(info.getOomKillDisable(), is(true)); assertThat(info.getOsType(), equalTo("linux")); @@ -327,4 +327,38 @@ public void serder2Json() throws IOException { assertThat(info, is(withInfo)); } + + @Test + public void info_1_38() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(Info.class); + + final Info info = testRoundTrip(RemoteApiVersion.VERSION_1_38, + "info/lcow.json", + type + ); + + assertThat(info, notNullValue()); + assertThat(info.getArchitecture(), is("x86_64")); + assertThat(info.getDockerRootDir(), is("C:\\ProgramData\\Docker")); + assertThat(info.getDriver(), is("windowsfilter (windows) lcow (linux)")); + + assertThat(info.getDriverStatuses(), equalTo(Arrays.asList( + Arrays.asList("Windows", ""), + Arrays.asList("LCOW", "") + ))); + + assertThat(info.getIsolation(), is("hyperv")); + assertThat(info.getKernelVersion(), is("10.0 17134 (17134.1.amd64fre.rs4_release.180410-1804)")); + assertThat(info.getOsType(), is("windows")); + assertThat(info.getOperatingSystem(), is("Windows 10 Pro Version 1803 (OS Build 17134.228)")); + + final Map> plugins = new LinkedHashMap<>(); + plugins.put("Authorization", null); + plugins.put("Log", asList("awslogs", "etwlogs", "fluentd", "gelf", "json-file", "logentries", "splunk", "syslog")); + plugins.put("Network", asList("ics", "l2bridge", "l2tunnel", "nat", "null", "overlay", "transparent")); + plugins.put("Volume", singletonList("local")); + assertThat(info.getPlugins(), equalTo(plugins)); + + assertThat(info.getServerVersion(), is("18.06.1-ce")); + } } diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/InternetProtocolTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/InternetProtocolTest.java new file mode 100644 index 000000000..0421e1411 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/InternetProtocolTest.java @@ -0,0 +1,50 @@ +package com.github.dockerjava.api.model; + +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; + +import static com.github.dockerjava.api.model.InternetProtocol.TCP; +import static org.junit.Assert.assertEquals; + +public class InternetProtocolTest { + @Rule + public ExpectedException expectedEx = ExpectedException.none(); + + @Test + public void defaultProtocol() { + assertEquals(TCP, InternetProtocol.DEFAULT); + } + + @Test + public void stringify() { + assertEquals("tcp", TCP.toString()); + } + + @Test + public void parseUpperCase() { + assertEquals(TCP, InternetProtocol.parse("TCP")); + } + + @Test + public void parseLowerCase() { + assertEquals(TCP, InternetProtocol.parse("tcp")); + } + + @Test + public void parseInvalidInput() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing Protocol 'xx'"); + + InternetProtocol.parse("xx"); + } + + @Test + public void parseNull() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing Protocol 'null'"); + + InternetProtocol.parse(null); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/LinkTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/LinkTest.java new file mode 100644 index 000000000..b780eb22f --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/LinkTest.java @@ -0,0 +1,49 @@ +package com.github.dockerjava.api.model; + +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; + +import static org.junit.Assert.assertEquals; + +public class LinkTest { + + @Rule + public ExpectedException expectedEx = ExpectedException.none(); + + @Test + public void parse() { + Link link = Link.parse("name:alias"); + assertEquals("name", link.getName()); + assertEquals("alias", link.getAlias()); + } + + @Test + public void parseWithContainerNames() { + Link link = Link.parse("/name:/conatiner/alias"); + assertEquals("name", link.getName()); + assertEquals("alias", link.getAlias()); + } + + @Test + public void parseInvalidInput() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing Link 'nonsense'"); + + Link.parse("nonsense"); + } + + @Test + public void parseNull() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing Link 'null'"); + + Link.parse(null); + } + + @Test + public void stringify() { + assertEquals("name:alias", Link.parse("name:alias").toString()); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/LinksTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/LinksTest.java new file mode 100644 index 000000000..0cf496412 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/LinksTest.java @@ -0,0 +1,35 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import static org.hamcrest.Matchers.arrayContaining; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.MatcherAssert.assertThat; + +public class LinksTest { + + @Test + public void usesToJson() throws Exception { + Links links = new Links( + new Link("/foo", "/bar"), + new Link("bip", "bop") + ); + String json = JSONTestHelper.getMapper().writeValueAsString(links); + + assertThat(json, is("[\"/foo:/bar\",\"bip:bop\"]")); + } + + @Test + public void usesFromJson() throws Exception { + Links links = JSONTestHelper.getMapper().readValue("[\"/foo:/bar\",\"bip:bop\"]", Links.class); + + assertThat(links, notNullValue()); + assertThat(links.getLinks(), arrayContaining( + new Link("foo", "bar"), + new Link("bip", "bop") + )); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/PortBindingTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/PortBindingTest.java new file mode 100644 index 000000000..9190cfda4 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/PortBindingTest.java @@ -0,0 +1,66 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.api.model.Ports.Binding; +import org.junit.Ignore; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; + +import static org.junit.Assert.assertEquals; + +public class PortBindingTest { + + private static final ExposedPort TCP_8080 = ExposedPort.tcp(8080); + + @Rule + public ExpectedException expectedEx = ExpectedException.none(); + + @Test + public void fullDefinition() { + assertEquals(PortBinding.parse("127.0.0.1:80:8080/tcp"), + new PortBinding(Binding.bindIpAndPort("127.0.0.1", 80), TCP_8080)); + } + + @Test + public void noProtocol() { + assertEquals(new PortBinding(Binding.bindIpAndPort("127.0.0.1", 80), TCP_8080), PortBinding.parse("127.0.0.1:80:8080")); + } + + @Test + public void noHostIp() { + assertEquals(new PortBinding(Binding.bindPort(80), TCP_8080), PortBinding.parse("80:8080/tcp")); + } + + @Test + public void portsOnly() { + assertEquals(new PortBinding(Binding.bindPort(80), TCP_8080), PortBinding.parse("80:8080")); + } + + @Test + public void exposedPortOnly() { + assertEquals(new PortBinding(Binding.empty(), TCP_8080), PortBinding.parse("8080")); + } + + @Test + public void dynamicHostPort() { + assertEquals(new PortBinding(Binding.bindIp("127.0.0.1"), TCP_8080), PortBinding.parse("127.0.0.1::8080")); + } + + @Test + @Ignore + public void parseInvalidInput() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing PortBinding 'nonsense'"); + + PortBinding.parse("nonsense"); + } + + @Test + public void parseNull() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing PortBinding 'null'"); + + PortBinding.parse(null); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/PortsAddBindingsTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/PortsAddBindingsTest.java new file mode 100644 index 000000000..484e5897f --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/PortsAddBindingsTest.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.api.model.Ports.Binding; +import org.junit.Before; +import org.junit.Test; + +import java.util.Map; + +import static org.junit.Assert.assertArrayEquals; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNull; + +/** + * As there may be several {@link Binding}s per {@link ExposedPort}, it makes a difference if you add {@link PortBinding}s for the same or + * different {@link ExposedPort}s to {@link Ports}. This test verifies that the Map in {@link Ports} is populated correctly in both cases. + */ +public class PortsAddBindingsTest { + private static final ExposedPort TCP_80 = ExposedPort.tcp(80); + + private static final ExposedPort TCP_90 = ExposedPort.tcp(90); + + private static final Binding BINDING_8080 = Binding.bindPort(8080); + + private static final Binding BINDING_9090 = Binding.bindPort(9090); + + private Ports ports; + + @Before + public void setup() { + ports = new Ports(); + } + + @Test + public void addTwoBindingsForDifferentExposedPorts() { + ports.add(new PortBinding(BINDING_8080, TCP_80), new PortBinding(BINDING_9090, TCP_90)); + + Map bindings = ports.getBindings(); + // two keys with one value each + assertEquals(2, bindings.size()); + assertArrayEquals(new Binding[] {BINDING_8080}, bindings.get(TCP_80)); + assertArrayEquals(new Binding[] {BINDING_9090}, bindings.get(TCP_90)); + } + + @Test + public void addTwoBindingsForSameExposedPort() { + ports.add(new PortBinding(BINDING_8080, TCP_80), new PortBinding(BINDING_9090, TCP_80)); + + Map bindings = ports.getBindings(); + // one key with two values + assertEquals(1, bindings.size()); + assertArrayEquals(new Binding[] {BINDING_8080, BINDING_9090}, bindings.get(TCP_80)); + } + + @Test + public void addNullBindings() { + ports.add(new PortBinding(null, TCP_80)); + Map bindings = ports.getBindings(); + // one key with two values + assertEquals(1, bindings.size()); + assertNull(bindings.get(TCP_80)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/PortsSerializingTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/PortsSerializingTest.java new file mode 100644 index 000000000..2f528556f --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/PortsSerializingTest.java @@ -0,0 +1,59 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.api.model.Ports.Binding; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import java.util.Map; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNull; + +public class PortsSerializingTest { + + private final String jsonWithDoubleBindingForOnePort = "{\"80/tcp\":[{\"HostIp\":\"10.0.0.1\",\"HostPort\":\"80\"},{\"HostIp\":\"10.0.0.2\",\"HostPort\":\"80\"}]}"; + + private final String jsonWithNullBindingForOnePort = "{\"80/tcp\":null}"; + + @Test + public void deserializingPortWithMultipleBindings() throws Exception { + Ports ports = JSONTestHelper.getMapper().readValue(jsonWithDoubleBindingForOnePort, Ports.class); + Map map = ports.getBindings(); + assertEquals(1, map.size()); + + Binding[] bindings = map.get(ExposedPort.tcp(80)); + assertEquals(2, bindings.length); + assertEquals(new Binding("10.0.0.1", "80"), bindings[0]); + assertEquals(new Binding("10.0.0.2", "80"), bindings[1]); + } + + @Test + public void serializingPortWithMultipleBindings() throws Exception { + Ports ports = new Ports(); + ports.bind(ExposedPort.tcp(80), new Binding("10.0.0.1", "80")); + ports.bind(ExposedPort.tcp(80), new Binding("10.0.0.2", "80")); + assertEquals(jsonWithDoubleBindingForOnePort, JSONTestHelper.getMapper().writeValueAsString(ports)); + } + + @Test + public void serializingEmptyBinding() throws Exception { + Ports ports = new Ports(ExposedPort.tcp(80), new Binding(null, null)); + assertEquals("{\"80/tcp\":[{\"HostIp\":\"\",\"HostPort\":\"\"}]}", JSONTestHelper.getMapper().writeValueAsString(ports)); + } + + @Test + public void deserializingPortWithNullBindings() throws Exception { + Ports ports = JSONTestHelper.getMapper().readValue(jsonWithNullBindingForOnePort, Ports.class); + Map map = ports.getBindings(); + assertEquals(1, map.size()); + + assertNull(map.get(ExposedPort.tcp(80))); + } + + @Test + public void serializingWithNullBindings() throws Exception { + Ports ports = new Ports(); + ports.bind(ExposedPort.tcp(80), null); + assertEquals(jsonWithNullBindingForOnePort, JSONTestHelper.getMapper().writeValueAsString(ports)); + } +} diff --git a/src/test/java/com/github/dockerjava/api/model/PullResponseItemTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/PullResponseItemTest.java similarity index 83% rename from src/test/java/com/github/dockerjava/api/model/PullResponseItemTest.java rename to docker-java/src/test/java/com/github/dockerjava/api/model/PullResponseItemTest.java index 1850d5c42..f73036864 100644 --- a/src/test/java/com/github/dockerjava/api/model/PullResponseItemTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/PullResponseItemTest.java @@ -15,13 +15,13 @@ */ package com.github.dockerjava.api.model; -import static com.github.dockerjava.test.serdes.JSONTestHelper.testRoundTrip; -import static org.testng.Assert.assertFalse; -import static org.testng.Assert.assertTrue; +import org.junit.Test; import java.io.IOException; -import org.testng.annotations.Test; +import static com.github.dockerjava.test.serdes.JSONTestHelper.testRoundTrip; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertTrue; /** * Tests logic of PullResponseItem's error/success handling by simulating a JSON response. @@ -29,6 +29,14 @@ * @author Zach Marshall */ public class PullResponseItemTest { + @Test + public void imageAlreadyExists() throws IOException { + PullResponseItem response = testRoundTrip(PullResponseJSONSamples.pullImageResponse_alreadyExists, + PullResponseItem.class); + assertTrue(response.isPullSuccessIndicated()); + assertFalse(response.isErrorIndicated()); + } + @Test public void pullNewerImage() throws IOException { PullResponseItem response = testRoundTrip(PullResponseJSONSamples.pullImageResponse_newerImage, diff --git a/src/test/java/com/github/dockerjava/api/model/PullResponseJSONSamples.java b/docker-java/src/test/java/com/github/dockerjava/api/model/PullResponseJSONSamples.java similarity index 87% rename from src/test/java/com/github/dockerjava/api/model/PullResponseJSONSamples.java rename to docker-java/src/test/java/com/github/dockerjava/api/model/PullResponseJSONSamples.java index 31cdf0f3b..4997a390a 100644 --- a/src/test/java/com/github/dockerjava/api/model/PullResponseJSONSamples.java +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/PullResponseJSONSamples.java @@ -23,7 +23,9 @@ * @author Zach Marshall */ public enum PullResponseJSONSamples implements JSONResourceRef { - pullImageResponse_legacy, pullImageResponse_error, pullImageResponse_newerImage, pullImageResponse_upToDate; + pullImageResponse_legacy, pullImageResponse_error, + pullImageResponse_newerImage, pullImageResponse_upToDate, + pullImageResponse_alreadyExists; @Override public String getFileName() { diff --git a/src/test/java/com/github/dockerjava/api/model/RepositoryTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/RepositoryTest.java similarity index 81% rename from src/test/java/com/github/dockerjava/api/model/RepositoryTest.java rename to docker-java/src/test/java/com/github/dockerjava/api/model/RepositoryTest.java index c78b61a85..835eefee7 100644 --- a/src/test/java/com/github/dockerjava/api/model/RepositoryTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/RepositoryTest.java @@ -1,8 +1,11 @@ package com.github.dockerjava.api.model; -import junit.framework.TestCase; +import org.junit.Test; -public class RepositoryTest extends TestCase { +import static org.junit.Assert.assertEquals; + +public class RepositoryTest { + @Test public void testRepository() throws Exception { Repository repo = new Repository("10.0.0.1/jim"); diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/RestartPolicyParsingTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/RestartPolicyParsingTest.java new file mode 100644 index 000000000..57dbc1b9f --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/RestartPolicyParsingTest.java @@ -0,0 +1,55 @@ +package com.github.dockerjava.api.model; + +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; + +import static org.junit.Assert.assertEquals; + +public class RestartPolicyParsingTest { + + @Rule + public ExpectedException expectedEx = ExpectedException.none(); + + + @Test + public void noRestart() { + assertEquals(RestartPolicy.noRestart(), RestartPolicy.parse("no")); + } + + @Test + public void alwaysRestart() { + assertEquals(RestartPolicy.alwaysRestart(), RestartPolicy.parse("always")); + } + + @Test + public void unlessStoppedRestart() { + assertEquals(RestartPolicy.unlessStoppedRestart(), RestartPolicy.parse("unless-stopped")); + } + + @Test + public void onFailureRestart() { + assertEquals(RestartPolicy.onFailureRestart(0), RestartPolicy.parse("on-failure")); + } + + @Test + public void onFailureRestartWithCount() { + assertEquals(RestartPolicy.onFailureRestart(2), RestartPolicy.parse("on-failure:2")); + } + + @Test + public void illegalSyntax() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing RestartPolicy 'nonsense'"); + + RestartPolicy.parse("nonsense"); + } + + @Test + public void illegalRetryCount() { + expectedEx.expect(IllegalArgumentException.class); + expectedEx.expectMessage("Error parsing RestartPolicy 'on-failure:X'"); + + RestartPolicy.parse("on-failure:X"); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/RestartPolicySerializingTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/RestartPolicySerializingTest.java new file mode 100644 index 000000000..c9c6a897d --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/RestartPolicySerializingTest.java @@ -0,0 +1,49 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import static org.junit.Assert.assertEquals; + +/** + * Compares serialization results of various {@link RestartPolicy}s with what Docker (as of 1.3.3) actually sends when executing + * docker run --restart xxx. + */ +public class RestartPolicySerializingTest { + + @Test + // --restart no + public void noRestart() throws Exception { + String json = JSONTestHelper.getMapper().writeValueAsString(RestartPolicy.noRestart()); + assertEquals("{\"MaximumRetryCount\":0,\"Name\":\"\"}", json); + } + + @Test + // --restart always + public void alwaysRestart() throws Exception { + String json = JSONTestHelper.getMapper().writeValueAsString(RestartPolicy.alwaysRestart()); + assertEquals("{\"MaximumRetryCount\":0,\"Name\":\"always\"}", json); + } + + @Test + // --restart unless-stopped + public void unlessStoppedRestart() throws Exception { + String json = JSONTestHelper.getMapper().writeValueAsString(RestartPolicy.unlessStoppedRestart()); + assertEquals("{\"MaximumRetryCount\":0,\"Name\":\"unless-stopped\"}", json); + } + + @Test + // --restart on-failure + public void onFailureRestart() throws Exception { + String json = JSONTestHelper.getMapper().writeValueAsString(RestartPolicy.onFailureRestart(0)); + assertEquals("{\"MaximumRetryCount\":0,\"Name\":\"on-failure\"}", json); + } + + @Test + // --restart on-failure:2 + public void onFailureRestartWithCount() throws Exception { + String json = JSONTestHelper.getMapper().writeValueAsString(RestartPolicy.onFailureRestart(2)); + assertEquals("{\"MaximumRetryCount\":2,\"Name\":\"on-failure\"}", json); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/RestartPolicyToStringTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/RestartPolicyToStringTest.java new file mode 100644 index 000000000..e32f97341 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/RestartPolicyToStringTest.java @@ -0,0 +1,25 @@ +package com.github.dockerjava.api.model; + +import org.junit.Test; +import org.junit.runner.RunWith; +import org.junit.runners.Parameterized; + +import static org.junit.Assert.assertEquals; + +@RunWith(Parameterized.class) +public class RestartPolicyToStringTest { + + @Parameterized.Parameters(name = "{0}") + public static Object[][] restartPolicies() { + return new Object[][] { {"no"}, {"always"}, {"unless-stopped"}, {"on-failure"}, {"on-failure:2"}}; + } + + @Parameterized.Parameter + public String policy; + + @Test + public void serializationWithoutCount() { + assertEquals(policy, RestartPolicy.parse(policy).toString()); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/StatisticsTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/StatisticsTest.java new file mode 100644 index 000000000..9734c5e14 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/StatisticsTest.java @@ -0,0 +1,141 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.databind.JavaType; +import com.github.dockerjava.core.RemoteApiVersion; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.hamcrest.Matchers; +import org.junit.Test; + +import java.io.IOException; +import java.util.Arrays; + +import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.core.IsEqual.equalTo; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.Matchers.empty; + +/** + * @author Yuting Liu + */ +public class StatisticsTest { + + @Test + public void serderJson1() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(Statistics.class); + + final Statistics statistics = testRoundTrip(RemoteApiVersion.VERSION_1_27, + "containers/container/stats/stats1.json", + type + ); + + assertThat(statistics.getRead(), equalTo("2017-12-06T00:42:03.8352972Z")); + + final StatisticNetworksConfig network = statistics.getNetworks().get("eth0"); + assertThat(network.getRxBytes(), is(1230L)); + assertThat(network.getRxPackets(), is(19L)); + assertThat(network.getRxErrors(), is(0L)); + assertThat(network.getRxDropped(), is(0L)); + assertThat(network.getTxBytes(), is(0L)); + assertThat(network.getTxPackets(), is(0L)); + assertThat(network.getTxErrors(), is(0L)); + assertThat(network.getTxDropped(), is(0L)); + + final MemoryStatsConfig memoryStats = statistics.getMemoryStats(); + assertThat(memoryStats.getUsage(), is(647168L)); + assertThat(memoryStats.getMaxUsage(), is(1703936L)); + + final StatsConfig stats = memoryStats.getStats(); + assertThat(stats.getActiveAnon(), is(102400L)); + assertThat(stats.getActiveFile(), is(0L)); + assertThat(stats.getCache(), is(0L)); + assertThat(stats.getDirty(), is(0L)); + assertThat(stats.getHierarchicalMemoryLimit(), is(9223372036854771712L)); + assertThat(stats.getHierarchicalMemswLimit(), is(9223372036854771712L)); + assertThat(stats.getInactiveAnon(), is(0L)); + assertThat(stats.getInactiveFile(), is(0L)); + assertThat(stats.getMappedFile(), is(0L)); + assertThat(stats.getPgfault(), is(9656L)); + assertThat(stats.getPgmajfault(), is(0L)); + assertThat(stats.getPgpgin(), is(3425L)); + assertThat(stats.getPgpgout(), is(3400L)); + assertThat(stats.getRss(), is(102400L)); + assertThat(stats.getRssHuge(), is(0L)); + assertThat(stats.getSwap(), is(0L)); + assertThat(stats.getTotalActiveAnon(), is(102400L)); + assertThat(stats.getTotalActiveFile(), is(0L)); + assertThat(stats.getTotalCache(), is(0L)); + assertThat(stats.getTotalDirty(), is(0L)); + assertThat(stats.getTotalInactiveAnon(), is(0L)); + assertThat(stats.getTotalInactiveFile(), is(0L)); + assertThat(stats.getTotalMappedFile(), is(0L)); + assertThat(stats.getTotalPgfault(), is(9656L)); + assertThat(stats.getTotalPgmajfault(), is(0L)); + assertThat(stats.getTotalPgpgin(), is(3425L)); + assertThat(stats.getTotalPgpgout(), is(3400L)); + assertThat(stats.getTotalRss(), is(102400L)); + assertThat(stats.getTotalRssHuge(), is(0L)); + assertThat(stats.getTotalSwap(), is(0L)); + assertThat(stats.getTotalUnevictable(), is(0L)); + assertThat(stats.getTotalWriteback(), is(0L)); + assertThat(stats.getUnevictable(), is(0L)); + assertThat(stats.getWriteback(), is(0L)); + + assertThat(memoryStats.getLimit(), is(2095874048L)); + assertThat(memoryStats.getFailcnt(), is(0L)); + + final BlkioStatsConfig blkioStats = statistics.getBlkioStats(); + assertThat(blkioStats.getIoServiceBytesRecursive(), Matchers.hasSize(5)); + assertThat(blkioStats.getIoServiceBytesRecursive(), equalTo(Arrays.asList( + new BlkioStatEntry().withMajor(259L).withMinor(0L).withOp("Read").withValue(823296L), + new BlkioStatEntry().withMajor(259L).withMinor(0L).withOp("Write").withValue(122880L), + new BlkioStatEntry().withMajor(259L).withMinor(0L).withOp("Sync").withValue(835584L), + new BlkioStatEntry().withMajor(259L).withMinor(0L).withOp("Async").withValue(110592L), + new BlkioStatEntry().withMajor(259L).withMinor(0L).withOp("Total").withValue(946176L) + ))); + + assertThat(blkioStats.getIoServicedRecursive(), Matchers.hasSize(5)); + assertThat(blkioStats.getIoServicedRecursive(), equalTo(Arrays.asList( + new BlkioStatEntry().withMajor(259L).withMinor(0L).withOp("Read").withValue(145L), + new BlkioStatEntry().withMajor(259L).withMinor(0L).withOp("Write").withValue(4L), + new BlkioStatEntry().withMajor(259L).withMinor(0L).withOp("Sync").withValue(148L), + new BlkioStatEntry().withMajor(259L).withMinor(0L).withOp("Async").withValue(1L), + new BlkioStatEntry().withMajor(259L).withMinor(0L).withOp("Total").withValue(149L) + ))); + assertThat(blkioStats.getIoQueueRecursive(), is(empty())); + assertThat(blkioStats.getIoServiceTimeRecursive(), is(empty())); + assertThat(blkioStats.getIoWaitTimeRecursive(), is(empty())); + assertThat(blkioStats.getIoMergedRecursive(), is(empty())); + assertThat(blkioStats.getIoTimeRecursive(), is(empty())); + assertThat(blkioStats.getSectorsRecursive(), is(empty())); + + final CpuStatsConfig cpuStats = statistics.getCpuStats(); + final CpuUsageConfig cpuUsage = cpuStats.getCpuUsage(); + assertThat(cpuUsage.getTotalUsage(), is(212198028L)); + assertThat(cpuUsage.getPercpuUsage(), equalTo(Arrays.asList(71592953L, 42494761L, 59298344L, 38811970L))); + assertThat(cpuUsage.getUsageInKernelmode(), is(170000000L)); + assertThat(cpuUsage.getUsageInUsermode(), is(20000000L)); + assertThat(cpuStats.getSystemCpuUsage(), is(545941980000000L)); + assertThat(cpuStats.getOnlineCpus(), is(4L)); + final ThrottlingDataConfig throttlingData = cpuStats.getThrottlingData(); + assertThat(throttlingData.getPeriods(), is(0L)); + assertThat(throttlingData.getThrottledPeriods(), is(0L)); + assertThat(throttlingData.getThrottledTime(), is(0L)); + + final CpuStatsConfig preCpuStats = statistics.getPreCpuStats(); + final CpuUsageConfig preCpuUsage = preCpuStats.getCpuUsage(); + assertThat(preCpuUsage.getTotalUsage(), is(211307214L)); + assertThat(preCpuUsage.getPercpuUsage(), equalTo(Arrays.asList(71451389L, 42097782L, 59298344L, 38459699L))); + assertThat(preCpuUsage.getUsageInKernelmode(), is(170000000L)); + assertThat(preCpuUsage.getUsageInUsermode(), is(20000000L)); + assertThat(preCpuStats.getSystemCpuUsage(), is(545937990000000L)); + assertThat(preCpuStats.getOnlineCpus(), is(4L)); + final ThrottlingDataConfig preThrottlingData = preCpuStats.getThrottlingData(); + assertThat(preThrottlingData.getPeriods(), is(0L)); + assertThat(preThrottlingData.getThrottledPeriods(), is(0L)); + assertThat(preThrottlingData.getThrottledTime(), is(0L)); + + final PidsStatsConfig pidsStats = statistics.getPidsStats(); + assertThat(pidsStats.getCurrent(), is(2L)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/UlimitsTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/UlimitsTest.java new file mode 100644 index 000000000..12d1a0d36 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/UlimitsTest.java @@ -0,0 +1,36 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import static org.hamcrest.Matchers.arrayContaining; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.MatcherAssert.assertThat; + +public class UlimitsTest { + + @Test + public void usesToJson() throws Exception { + Ulimit[] ulimits = new Ulimit[]{ + new Ulimit("nproc", 709, 1026), + new Ulimit("nofile", 1024, 4096), + new Ulimit("core", 99999999998L, 99999999999L) + }; + String json = JSONTestHelper.getMapper().writeValueAsString(ulimits); + + assertThat(json, is("[{\"Name\":\"nproc\",\"Soft\":709,\"Hard\":1026},{\"Name\":\"nofile\",\"Soft\":1024,\"Hard\":4096},{\"Name\":\"core\",\"Soft\":99999999998,\"Hard\":99999999999}]")); + } + + @Test + public void usesFromJson() throws Exception { + Ulimit[] ulimits = JSONTestHelper.getMapper().readValue("[{\"Name\":\"nproc\",\"Soft\":709,\"Hard\":1026},{\"Name\":\"nofile\",\"Soft\":1024,\"Hard\":4096},{\"Name\":\"core\",\"Soft\":99999999998,\"Hard\":99999999999}]", Ulimit[].class); + + assertThat(ulimits, notNullValue()); + assertThat(ulimits, arrayContaining( + new Ulimit("nproc", 709, 1026), + new Ulimit("nofile", 1024, 4096), + new Ulimit("core", 99999999998L, 99999999999L) + )); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/VersionTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/VersionTest.java new file mode 100644 index 000000000..77fafac37 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/VersionTest.java @@ -0,0 +1,86 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.databind.JavaType; +import com.github.dockerjava.core.RemoteApiVersion; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import java.util.Collections; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; + +import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.notNullValue; + +/** + * @author Kanstantsin Shautsou + */ +public class VersionTest { + + @Test + public void testSerDer1() throws Exception { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(Version.class); + + final Version version = testRoundTrip(RemoteApiVersion.VERSION_1_22, + "/version/1.json", + type + ); + + assertThat(version, notNullValue()); + + assertThat(version.getVersion(), is("1.10.1")); + assertThat(version.getApiVersion(), is("1.22")); + assertThat(version.getGitCommit(), is("9e83765")); + assertThat(version.getGoVersion(), is("go1.5.3")); + assertThat(version.getOperatingSystem(), is("linux")); + assertThat(version.getArch(), is("amd64")); + assertThat(version.getKernelVersion(), is("4.1.17-boot2docker")); + assertThat(version.getBuildTime(), is("2016-02-11T20:39:58.688092588+00:00")); + } + + @Test + public void version_1_38() throws Exception { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(Version.class); + + final Version version = testRoundTrip(RemoteApiVersion.VERSION_1_38, + "/version/lcow.json", + type + ); + + assertThat(version, notNullValue()); + assertThat(version.getApiVersion(), is("1.38")); + assertThat(version.getArch(), is("amd64")); + assertThat(version.getBuildTime(), is("2018-08-21T17:36:40.000000000+00:00")); + + Map details = new LinkedHashMap<>(); + details.put("ApiVersion", "1.38"); + details.put("Arch", "amd64"); + details.put("BuildTime", "2018-08-21T17:36:40.000000000+00:00"); + details.put("Experimental", "true"); + details.put("GitCommit", "e68fc7a"); + details.put("GoVersion", "go1.10.3"); + details.put("KernelVersion", "10.0 17134 (17134.1.amd64fre.rs4_release.180410-1804)"); + details.put("MinAPIVersion", "1.24"); + details.put("Os", "windows"); + + List components = Collections.singletonList(new VersionComponent() + .withDetails(details) + .withName("Engine") + .withVersion("18.06.1-ce") + ); + assertThat(version.getComponents(), equalTo(components)); + + assertThat(version.getExperimental(), is(true)); + assertThat(version.getGitCommit(), is("e68fc7a")); + assertThat(version.getGoVersion(), is("go1.10.3")); + assertThat(version.getKernelVersion(), is("10.0 17134 (17134.1.amd64fre.rs4_release.180410-1804)")); + assertThat(version.getMinAPIVersion(), is("1.24")); + assertThat(version.getOperatingSystem(), is("windows")); + assertThat(version.getPlatform(), equalTo(new VersionPlatform().withName(""))); + assertThat(version.getVersion(), is("18.06.1-ce")); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/VolumeBindsTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/VolumeBindsTest.java new file mode 100644 index 000000000..7f38eb0f0 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/VolumeBindsTest.java @@ -0,0 +1,42 @@ +package com.github.dockerjava.api.model; + +import com.fasterxml.jackson.databind.JsonMappingException; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import java.io.IOException; + +import static org.hamcrest.Matchers.is; +import static org.junit.Assert.assertEquals; +import static org.hamcrest.MatcherAssert.assertThat; + +public class VolumeBindsTest { + + @Test + public void usesToJson() throws Exception { + VolumeBinds binds = new VolumeBinds( + new VolumeBind("/bar", "/foo"), + new VolumeBind("/bop", "/bip") + ); + String json = JSONTestHelper.getMapper().writeValueAsString(binds); + + assertThat(json, is("{\"/foo\":\"/bar\",\"/bip\":\"/bop\"}")); + } + + @Test + public void t() throws IOException { + String s = "{\"/data\":\"/some/path\"}"; + VolumeBinds volumeBinds = JSONTestHelper.getMapper().readValue(s, VolumeBinds.class); + VolumeBind[] binds = volumeBinds.getBinds(); + assertEquals(1, binds.length); + assertEquals("/some/path", binds[0].getHostPath()); + assertEquals("/data", binds[0].getContainerPath()); + } + + @Test(expected = JsonMappingException.class) + public void t1() throws IOException { + String s = "{\"/data\": {} }"; + JSONTestHelper.getMapper().readValue(s, VolumeBinds.class); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/VolumeFromSerializingTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/VolumeFromSerializingTest.java new file mode 100644 index 000000000..6155f88e3 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/VolumeFromSerializingTest.java @@ -0,0 +1,24 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import static org.junit.Assert.assertEquals; + +public class VolumeFromSerializingTest { + + private final String json = "\"container1:ro\""; + + @Test + public void deserializing() throws Exception { + VolumesFrom volumeFrom = JSONTestHelper.getMapper().readValue(json, VolumesFrom.class); + assertEquals(volumeFrom, new VolumesFrom("container1", AccessMode.ro)); + } + + @Test + public void serializing() throws Exception { + VolumesFrom volumeFrom = new VolumesFrom("container1", AccessMode.ro); + assertEquals(json, JSONTestHelper.getMapper().writeValueAsString(volumeFrom)); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/VolumeTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/VolumeTest.java new file mode 100644 index 000000000..20e28a55d --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/VolumeTest.java @@ -0,0 +1,12 @@ +package com.github.dockerjava.api.model; + +import org.junit.Test; + +import static org.junit.Assert.assertEquals; + +public class VolumeTest { + @Test + public void getPath() { + assertEquals("/path", new Volume("/path").getPath()); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/VolumesRWTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/VolumesRWTest.java new file mode 100644 index 000000000..ed52ff82d --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/VolumesRWTest.java @@ -0,0 +1,35 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import static org.hamcrest.Matchers.arrayContaining; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.MatcherAssert.assertThat; + +public class VolumesRWTest { + + @Test + public void usesToJson() throws Exception { + VolumesRW volumes = new VolumesRW( + new VolumeRW(new Volume("/foo")), + new VolumeRW(new Volume("/bar"), AccessMode.ro) + ); + String json = JSONTestHelper.getMapper().writeValueAsString(volumes); + + assertThat(json, is("{\"/foo\":true,\"/bar\":false}")); + } + + @Test + public void usesFromJson() throws Exception { + VolumesRW volumes = JSONTestHelper.getMapper().readValue("{\"/foo\":true,\"/bar\":false}", VolumesRW.class); + + assertThat(volumes, notNullValue()); + assertThat(volumes.getVolumesRW(), arrayContaining( + new VolumeRW(new Volume("/foo")), + new VolumeRW(new Volume("/bar"), AccessMode.ro) + )); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/api/model/VolumesTest.java b/docker-java/src/test/java/com/github/dockerjava/api/model/VolumesTest.java new file mode 100644 index 000000000..d39e02583 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/api/model/VolumesTest.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.api.model; + +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; + +import static org.hamcrest.Matchers.arrayContaining; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.MatcherAssert.assertThat; + +public class VolumesTest { + + @Test + public void usesToJson() throws Exception { + Volumes volumes = new Volumes( + new Volume("/foo"), + new Volume("/bar") + ); + String json = JSONTestHelper.getMapper().writeValueAsString(volumes); + + assertThat(json, is("{\"/foo\":{},\"/bar\":{}}")); + } + + @Test + public void usesFromJson() throws Exception { + Volumes volumes = JSONTestHelper.getMapper().readValue("{\"/foo\":{},\"/bar\":{}}", Volumes.class); + + assertThat(volumes, notNullValue()); + assertThat(volumes.getVolumes(), arrayContaining( + new Volume("/foo"), + new Volume("/bar") + )); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/AttachContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/AttachContainerCmdIT.java new file mode 100644 index 000000000..dde47e2d8 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/AttachContainerCmdIT.java @@ -0,0 +1,246 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.api.model.StreamType; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.File; +import java.io.PipedInputStream; +import java.io.PipedOutputStream; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.TimeUnit; + +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static java.util.concurrent.TimeUnit.SECONDS; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.not; +import static org.junit.Assert.*; + +/** + * @author Kanstantsin Shautsou + */ +public class AttachContainerCmdIT extends CmdIT { + private static final Logger LOG = LoggerFactory.getLogger(AttachContainerCmdIT.class); + + + @Rule + public ExpectedException expectedException = ExpectedException.none(); + + @Test + public void attachContainerWithStdin() throws Exception { + DockerClient dockerClient = dockerRule.getClient(); + + String snippet = "hello world"; + + CreateContainerResponse container = dockerClient.createContainerCmd("busybox") + .withCmd("/bin/sh", "-c", "read line && echo $line") + .withTty(false) + .withAttachStdin(true) + .withAttachStdout(true) + .withAttachStderr(true) + .withStdinOpen(true) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + AttachContainerTestCallback callback = new AttachContainerTestCallback() { + @Override + public void onNext(Frame frame) { + assertEquals(StreamType.STDOUT, frame.getStreamType()); + super.onNext(frame); + } + }; + + try ( + PipedOutputStream out = new PipedOutputStream(); + PipedInputStream in = new PipedInputStream(out) + ) { + dockerClient.attachContainerCmd(container.getId()) + .withStdErr(true) + .withStdOut(true) + .withFollowStream(true) + .withStdIn(in) + .exec(callback); + + assertTrue("Processing of the response should start shortly after executing `attachContainerCmd`", + callback.awaitStarted(5, SECONDS)); + + dockerClient.startContainerCmd(container.getId()).exec(); + + out.write((snippet + "\n").getBytes()); + out.flush(); + + callback.awaitCompletion(15, SECONDS); + callback.close(); + } + + assertThat(callback.toString(), containsString(snippet)); + } + + @Test + public void attachContainerWithoutTTY() throws Exception { + DockerClient dockerClient = dockerRule.getClient(); + + String snippet = "hello world"; + + CreateContainerResponse container = dockerClient.createContainerCmd(DEFAULT_IMAGE) + .withCmd("echo", snippet) + .withTty(false) + .withAttachStdout(true) + .withAttachStderr(true) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + AttachContainerTestCallback callback = new AttachContainerTestCallback() { + @Override + public void onNext(Frame frame) { + assertThat(frame.getStreamType(), equalTo(StreamType.STDOUT)); + super.onNext(frame); + } + }; + + dockerClient.attachContainerCmd(container.getId()) + .withStdErr(true) + .withStdOut(true) + .withFollowStream(true) + .withLogs(true) + .exec(callback); + + assertTrue("Processing of the response should start shortly after executing `attachContainerCmd`", + callback.awaitStarted(5, SECONDS)); + + dockerClient.startContainerCmd(container.getId()).exec(); + + callback.awaitCompletion(30, TimeUnit.SECONDS); + callback.close(); + + assertThat(callback.toString(), containsString(snippet)); + } + + @Test + public void attachContainerWithTTY() throws Exception { + DockerClient dockerClient = dockerRule.getClient(); + + File baseDir = new File(Thread.currentThread().getContextClassLoader() + .getResource("attachContainerTestDockerfile").getFile()); + + String imageId = dockerRule.buildImage(baseDir); + + CreateContainerResponse container = dockerClient.createContainerCmd(imageId) + .withTty(true) + .withAttachStdout(true) + .withAttachStderr(true) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + + AttachContainerTestCallback callback = new AttachContainerTestCallback() { + @Override + public void onNext(Frame frame) { + assertThat(frame.getStreamType(), equalTo(StreamType.RAW)); + super.onNext(frame); + } + }; + + dockerClient.attachContainerCmd(container.getId()) + .withStdErr(true) + .withStdOut(true) + .withFollowStream(true) + .exec(callback); + + assertTrue("Processing of the response should start shortly after executing `attachContainerCmd`", + callback.awaitStarted(5, SECONDS)); + + dockerClient.startContainerCmd(container.getId()).exec(); + + callback.awaitCompletion(15, TimeUnit.SECONDS); + callback.close(); + + LOG.debug("log: {}", callback.toString()); + + // HexDump.dump(collectFramesCallback.toString().getBytes(), 0, System.out, 0); + assertThat(callback.toString(), containsString("stdout\r\nstderr")); + } + + /** + * {@link ResultCallback#onComplete()} should be called immediately after + * container exit. It was broken for Netty and TLS connection. + */ + @Test + public void attachContainerClosesStdoutWhenContainerExits() throws Exception { + DockerClient dockerClient = dockerRule.getClient(); + + CreateContainerResponse container = dockerClient.createContainerCmd(DEFAULT_IMAGE) + .withCmd("echo", "hello") + .withTty(false) + .withAttachStdout(true) + .withAttachStderr(true) + .exec(); + LOG.info("Created container: {}", container.toString()); + + CountDownLatch gotLine = new CountDownLatch(1); + try ( + ResultCallback.Adapter resultCallback = dockerClient.attachContainerCmd(container.getId()) + .withStdOut(true) + .withStdErr(true) + .withFollowStream(true) + .exec(new ResultCallback.Adapter() { + @Override + public void onNext(Frame item) { + LOG.info("Got frame: {}", item); + if (item.getStreamType() == StreamType.STDOUT) { + gotLine.countDown(); + } + super.onNext(item); + } + + @Override + public void onComplete() { + LOG.info("On complete"); + super.onComplete(); + } + }) + ) { + resultCallback.awaitStarted(5, SECONDS); + LOG.info("Attach started"); + + dockerClient.startContainerCmd(container.getId()).exec(); + LOG.info("Container started"); + + assertTrue("Should get first line quickly after the start", gotLine.await(15, SECONDS)); + + resultCallback.awaitCompletion(5, SECONDS); + } + } + + public static class AttachContainerTestCallback extends ResultCallback.Adapter { + private final StringBuffer log = new StringBuffer(); + + @Override + public void onNext(Frame item) { + log.append(new String(item.getPayload())); + super.onNext(item); + } + + @Override + public String toString() { + return log.toString(); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/AuthCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/AuthCmdIT.java new file mode 100644 index 000000000..d37bea819 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/AuthCmdIT.java @@ -0,0 +1,41 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.exception.UnauthorizedException; +import com.github.dockerjava.api.model.AuthResponse; +import com.github.dockerjava.core.DockerClientBuilder; +import org.junit.Ignore; +import org.junit.Test; + +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_22; +import static com.github.dockerjava.junit.DockerMatchers.apiVersionGreater; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.junit.Assert.assertThrows; +import static org.junit.Assume.assumeThat; + +/** + * @author Kanstantsin Shautsou + */ +public class AuthCmdIT extends CmdIT { + + @Test + public void testAuth() { + assumeThat("Fails on 1.22. Temporary disabled.", dockerRule, apiVersionGreater(VERSION_1_22)); + + AuthResponse response = dockerRule.getClient().authCmd().exec(); + + assertThat(response.getStatus(), is("Login Succeeded")); + } + + + @Ignore("Disabled because of 500/InternalServerException") + @Test + public void testAuthInvalid() { + assertThrows("Wrong login/password, please try again", UnauthorizedException.class, () -> { + DockerClientBuilder.getInstance(dockerRule.config("garbage")) + .build() + .authCmd() + .exec(); + }); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/BuildImageCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/BuildImageCmdIT.java new file mode 100644 index 000000000..d514ed59c --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/BuildImageCmdIT.java @@ -0,0 +1,369 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.BuildImageCmd; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectImageResponse; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.AuthConfigurations; +import com.github.dockerjava.core.util.CompressArchiveUtil; +import com.github.dockerjava.junit.PrivateRegistryRule; +import net.jcip.annotations.NotThreadSafe; +import org.apache.commons.io.FileUtils; +import org.apache.commons.io.filefilter.TrueFileFilter; +import org.junit.ClassRule; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.TemporaryFolder; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.File; +import java.io.FileInputStream; +import java.io.InputStream; +import java.util.Arrays; +import java.util.Collection; +import java.util.Collections; +import java.util.HashSet; +import java.util.UUID; +import java.util.concurrent.TimeUnit; + +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_21; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_23; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_27; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_28; +import static com.github.dockerjava.junit.DockerMatchers.isGreaterOrEqual; +import static org.apache.commons.io.FileUtils.writeStringToFile; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsInAnyOrder; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.endsWith; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.nullValue; +import static org.junit.Assume.assumeThat; + +/** + * @author Kanstantsin Shautsou + */ +@NotThreadSafe +public class BuildImageCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(BuildImageCmd.class); + + @ClassRule + public static PrivateRegistryRule REGISTRY = new PrivateRegistryRule(); + + @Rule + public TemporaryFolder folder = new TemporaryFolder(new File("target/")); + + @Test + public void author() throws Exception { + + String imageId = dockerRule.buildImage(fileFromBuildTestResource("AUTHOR")); + + InspectImageResponse inspectImageResponse = dockerRule.getClient().inspectImageCmd(imageId).exec(); + assertThat(inspectImageResponse, not(nullValue())); + LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + + assertThat(inspectImageResponse.getAuthor(), equalTo("Guillaume J. Charmes \"guillaume@dotcloud.com\"")); + } + + @Test + public void buildImageFromTar() throws Exception { + File baseDir = fileFromBuildTestResource("ADD/file"); + Collection files = FileUtils.listFiles(baseDir, TrueFileFilter.INSTANCE, TrueFileFilter.INSTANCE); + File tarFile = CompressArchiveUtil.archiveTARFiles(baseDir, files, UUID.randomUUID().toString()); + String response = dockerfileBuild(new FileInputStream(tarFile)); + assertThat(response, containsString("Successfully executed testrun.sh")); + } + + @Test + public void buildImageFromTarWithDockerfileNotInBaseDirectory() throws Exception { + File baseDir = fileFromBuildTestResource("dockerfileNotInBaseDirectory"); + Collection files = FileUtils.listFiles(baseDir, TrueFileFilter.INSTANCE, TrueFileFilter.INSTANCE); + File tarFile = CompressArchiveUtil.archiveTARFiles(baseDir, files, UUID.randomUUID().toString()); + String response = dockerfileBuild(new FileInputStream(tarFile), "dockerfileFolder/Dockerfile"); + assertThat(response, containsString("Successfully executed testrun.sh")); + } + + @Test + public void onBuild() throws Exception { + File baseDir = fileFromBuildTestResource("ONBUILD/parent"); + + dockerRule.getClient().buildImageCmd(baseDir) + .withNoCache(true) + .withTag("docker-java-onbuild") + .start() + .awaitImageId(); + + baseDir = fileFromBuildTestResource("ONBUILD/child"); + String response = dockerfileBuild(baseDir); + assertThat(response, containsString("Successfully executed testrun.sh")); + } + + @Test + public void addUrl() throws Exception { + File baseDir = fileFromBuildTestResource("ADD/url"); + String response = dockerfileBuild(baseDir); + assertThat(response, containsString("Example Domain")); + } + + @Test + public void addFileInSubfolder() throws Exception { + File baseDir = fileFromBuildTestResource("ADD/fileInSubfolder"); + String response = dockerfileBuild(baseDir); + assertThat(response, containsString("Successfully executed testrun.sh")); + } + + @Test + public void addFilesViaWildcard() throws Exception { + File baseDir = fileFromBuildTestResource("ADD/filesViaWildcard"); + String response = dockerfileBuild(baseDir); + assertThat(response, containsString("Successfully executed testinclude1.sh")); + assertThat(response, not(containsString("Successfully executed testinclude2.sh"))); + } + + @Test + public void addFolder() throws Exception { + File baseDir = fileFromBuildTestResource("ADD/folder"); + String response = dockerfileBuild(baseDir); + assertThat(response, containsString("Successfully executed testAddFolder.sh")); + } + + private String dockerfileBuild(InputStream tarInputStream, String dockerFilePath) throws Exception { + + return execBuild(dockerRule.getClient().buildImageCmd().withTarInputStream(tarInputStream).withDockerfilePath(dockerFilePath)); + } + + private String dockerfileBuild(InputStream tarInputStream) throws Exception { + + return execBuild(dockerRule.getClient().buildImageCmd().withTarInputStream(tarInputStream)); + } + + private String dockerfileBuild(File baseDir) throws Exception { + + return execBuild(dockerRule.getClient().buildImageCmd(baseDir)); + } + + private String execBuild(BuildImageCmd buildImageCmd) throws Exception { + String imageId = buildImageCmd.withNoCache(true).start().awaitImageId(); + + // Create container based on image + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(imageId).exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + dockerRule.getClient().waitContainerCmd(container.getId()).start().awaitStatusCode(); + + return dockerRule.containerLog(container.getId()); + } + + @Test(expected = DockerClientException.class) + public void dockerignoreDockerfileIgnored() { + File baseDir = fileFromBuildTestResource("dockerignore/DockerfileIgnored"); + + dockerRule.getClient().buildImageCmd(baseDir).withNoCache(true).start().awaitImageId(); + } + + @Test + public void dockerignoreDockerfileNotIgnored() { + File baseDir = fileFromBuildTestResource("dockerignore/DockerfileNotIgnored"); + + dockerRule.getClient().buildImageCmd(baseDir).withNoCache(true).start().awaitImageId(); + } + + @Test(expected = DockerClientException.class) + public void dockerignoreInvalidDockerIgnorePattern() { + File baseDir = fileFromBuildTestResource("dockerignore/InvalidDockerignorePattern"); + + dockerRule.getClient().buildImageCmd(baseDir).withNoCache(true).start().awaitImageId(); + } + + @Test + public void dockerignoreValidDockerIgnorePattern() throws Exception { + File baseDir = fileFromBuildTestResource("dockerignore/ValidDockerignorePattern"); + String response = dockerfileBuild(baseDir); + assertThat(response, containsString("/tmp/a/a /tmp/a/c /tmp/a/d")); + } + + @Test + public void env() throws Exception { + File baseDir = fileFromBuildTestResource("ENV"); + String response = dockerfileBuild(baseDir); + assertThat(response, containsString("testENVSubstitution successfully completed")); + } + + @Test + public void fromPrivateRegistry() throws Exception { + AuthConfig authConfig = REGISTRY.getAuthConfig(); + String imgName = authConfig.getRegistryAddress() + "/testuser/busybox"; + + File dockerfile = folder.newFile("Dockerfile"); + writeStringToFile(dockerfile, "FROM " + imgName); + + File baseDir; + InspectImageResponse inspectImageResponse; + + dockerRule.getClient().authCmd().withAuthConfig(authConfig).exec(); + dockerRule.getClient().tagImageCmd("busybox:latest", imgName, "latest") + .withForce() + .exec(); + + dockerRule.getClient().pushImageCmd(imgName) + .withTag("latest") + .withAuthConfig(authConfig) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + + dockerRule.getClient().removeImageCmd(imgName) + .withForce(true) + .exec(); + +// baseDir = fileFromBuildTestResource("FROM/privateRegistry"); + baseDir = folder.getRoot(); + + AuthConfigurations authConfigurations = new AuthConfigurations(); + authConfigurations.addConfig(authConfig); + + String imageId = dockerRule.getClient().buildImageCmd(baseDir) + .withNoCache(true) + .withBuildAuthConfigs(authConfigurations) + .start() + .awaitImageId(); + + inspectImageResponse = dockerRule.getClient().inspectImageCmd(imageId).exec(); + assertThat(inspectImageResponse, not(nullValue())); + LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + } + + @Test + public void buildArgs() { + File baseDir = fileFromBuildTestResource("buildArgs"); + + String imageId = dockerRule.getClient().buildImageCmd(baseDir).withNoCache(true).withBuildArg("testArg", "abc !@#$%^&*()_+") + .start() + .awaitImageId(); + + InspectImageResponse inspectImageResponse = dockerRule.getClient().inspectImageCmd(imageId).exec(); + assertThat(inspectImageResponse, not(nullValue())); + LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + + assertThat(inspectImageResponse.getConfig().getLabels().get("test"), equalTo("abc !@#$%^&*()_+")); + } + + @Test + public void labels() { + assumeThat("API version should be >= 1.23", dockerRule, isGreaterOrEqual(VERSION_1_23)); + + File baseDir = fileFromBuildTestResource("labels"); + + String imageId = dockerRule.getClient().buildImageCmd(baseDir).withNoCache(true) + .withLabels(Collections.singletonMap("test", "abc")) + .start() + .awaitImageId(); + + InspectImageResponse inspectImageResponse = dockerRule.getClient().inspectImageCmd(imageId).exec(); + assertThat(inspectImageResponse, not(nullValue())); + LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + + assertThat(inspectImageResponse.getConfig().getLabels().get("test"), equalTo("abc")); + } + + @Test + public void multipleTags() { + assumeThat("API version should be >= 1.23", dockerRule, isGreaterOrEqual(VERSION_1_21)); + + + File baseDir = fileFromBuildTestResource("labels"); + + String imageId = dockerRule.getClient().buildImageCmd(baseDir).withNoCache(true) + .withTag("fallback-when-withTags-not-called") + .withTags(new HashSet<>(Arrays.asList("docker-java-test:tag1", "docker-java-test:tag2"))) + .start() + .awaitImageId(); + + InspectImageResponse inspectImageResponse = dockerRule.getClient().inspectImageCmd(imageId).exec(); + assertThat(inspectImageResponse, not(nullValue())); + LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + + assertThat(inspectImageResponse.getRepoTags().size(), equalTo(2)); + assertThat(inspectImageResponse.getRepoTags(), containsInAnyOrder("docker-java-test:tag1", "docker-java-test:tag2")); + } + + @Test + public void cacheFrom() { + assumeThat(dockerRule, isGreaterOrEqual(VERSION_1_27)); + + File baseDir1 = fileFromBuildTestResource("CacheFrom/test1"); + String imageId1 = dockerRule.getClient().buildImageCmd(baseDir1) + .start() + .awaitImageId(); + InspectImageResponse inspectImageResponse1 = dockerRule.getClient().inspectImageCmd(imageId1).exec(); + assertThat(inspectImageResponse1, not(nullValue())); + + File baseDir2 = fileFromBuildTestResource("CacheFrom/test2"); + String imageId2 = dockerRule.getClient().buildImageCmd(baseDir2).withCacheFrom(new HashSet<>(Arrays.asList(imageId1))) + .start() + .awaitImageId(); + InspectImageResponse inspectImageResponse2 = dockerRule.getClient().inspectImageCmd(imageId2).exec(); + assertThat(inspectImageResponse2, not(nullValue())); + + // Compare whether the image2's parent layer is from image1 so that cache is used + assertThat(inspectImageResponse2.getParent(), equalTo(inspectImageResponse1.getId())); + + } + + @Test + public void quiet() { + File baseDir = fileFromBuildTestResource("labels"); + + String imageId = dockerRule.getClient() + .buildImageCmd(baseDir) + .withQuiet(true) + .start() + .awaitImageId(); + + InspectImageResponse inspectImageResponse = dockerRule.getClient().inspectImageCmd(imageId).exec(); + assertThat(inspectImageResponse, not(nullValue())); + assertThat(inspectImageResponse.getId(), endsWith(imageId)); + LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + } + + @Test + public void extraHosts() { + assumeThat(dockerRule, isGreaterOrEqual(VERSION_1_28)); + + File baseDir = fileFromBuildTestResource("labels"); + + String imageId = dockerRule.getClient() + .buildImageCmd(baseDir) + .withExtraHosts(new HashSet<>(Arrays.asList("host1"))) + .start() + .awaitImageId(); + + InspectImageResponse inspectImageResponse = dockerRule.getClient().inspectImageCmd(imageId).exec(); + assertThat(inspectImageResponse, not(nullValue())); + LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + } + + public void dockerfileNotInBaseDirectory() throws Exception { + File baseDirectory = fileFromBuildTestResource("dockerfileNotInBaseDirectory"); + File dockerfile = fileFromBuildTestResource("dockerfileNotInBaseDirectory/dockerfileFolder/Dockerfile"); + BuildImageCmd command = dockerRule.getClient().buildImageCmd() + .withBaseDirectory(baseDirectory) + .withDockerfile(dockerfile); + + String response = execBuild(command); + + assertThat(response, containsString("Successfully executed testrun.sh")); + } + + private File fileFromBuildTestResource(String resource) { + return new File(Thread.currentThread().getContextClassLoader() + .getResource("buildTests/" + resource).getFile()); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/CmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/CmdIT.java new file mode 100644 index 000000000..b01cb1e90 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/CmdIT.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.core.DockerClientBuilder; +import com.github.dockerjava.core.DockerClientConfig; +import com.github.dockerjava.core.DockerClientImpl; +import com.github.dockerjava.core.DockerRule; +import com.github.dockerjava.httpclient5.ApacheDockerHttpClient; +import com.github.dockerjava.junit.category.Integration; +import com.github.dockerjava.transport.DockerHttpClient; +import org.junit.Rule; +import org.junit.experimental.categories.Category; + +/** + * @author Kanstantsin Shautsou + */ +@Category(Integration.class) +public abstract class CmdIT { + + public static DockerHttpClient createDockerHttpClient(DockerClientConfig config) { + return new TrackingDockerHttpClient( + new ApacheDockerHttpClient.Builder() + .dockerHost(config.getDockerHost()) + .sslConfig(config.getSSLConfig()) + .build() + ); + } + + public static DockerClientImpl createDockerClient(DockerClientConfig config) { + return (DockerClientImpl) DockerClientBuilder.getInstance(config) + .withDockerHttpClient(createDockerHttpClient(config)) + .build(); + } + + @Rule + public DockerRule dockerRule = new DockerRule(); + + @Rule + public DockerHttpClientLeakDetector leakDetector = new DockerHttpClientLeakDetector(); +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/CommitCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/CommitCmdIT.java new file mode 100644 index 000000000..bd87d4aab --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/CommitCmdIT.java @@ -0,0 +1,95 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectImageResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.google.common.collect.ImmutableMap; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Map; + +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.greaterThanOrEqualTo; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.startsWith; +import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; + + +public class CommitCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(CommitCmdIT.class); + + @Test + public void commit() throws DockerException, InterruptedException { + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("touch", "/test") + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + LOG.info("Committing container: {}", container.toString()); + String imageId = dockerRule.getClient().commitCmd(container.getId()).exec(); + + //swarm needs some time to reflect new images + synchronized (this) { + wait(5000); + } + + InspectImageResponse inspectImageResponse = dockerRule.getClient().inspectImageCmd(imageId).exec(); + LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + + InspectImageResponse busyboxImg = dockerRule.getClient().inspectImageCmd("busybox").exec(); + + assertThat(inspectImageResponse.getParent(), equalTo(busyboxImg.getId())); + } + + @Test + public void commitWithLabels() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("touch", "/test") + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + Integer status = dockerRule.getClient().waitContainerCmd(container.getId()) + .start() + .awaitStatusCode(); + + assertThat(status, is(0)); + + LOG.info("Committing container: {}", container.toString()); + Map labels = ImmutableMap.of("label1", "abc", "label2", "123"); + String imageId = dockerRule.getClient().commitCmd(container.getId()) + .withLabels(labels) + .exec(); + + InspectImageResponse inspectImageResponse = dockerRule.getClient().inspectImageCmd(imageId).exec(); + LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + + //use config here since containerConfig contains the configuration of the container which was + //committed to the container + //https://stackoverflow.com/questions/36216220/what-is-different-of-config-and-containerconfig-of-docker-inspect + Map responseLabels = inspectImageResponse.getConfig().getLabels(); + //swarm will attach additional labels here + assertThat(responseLabels.size(), greaterThanOrEqualTo(2)); + assertThat(responseLabels.get("label1"), equalTo("abc")); + assertThat(responseLabels.get("label2"), equalTo("123")); + } + + @Test(expected = NotFoundException.class) + public void commitNonExistingContainer() throws DockerException { + + dockerRule.getClient().commitCmd("non-existent").exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/ConnectToNetworkCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/ConnectToNetworkCmdIT.java new file mode 100644 index 000000000..40b552611 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/ConnectToNetworkCmdIT.java @@ -0,0 +1,96 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.CreateNetworkResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.ContainerNetwork; +import com.github.dockerjava.api.model.Network; +import net.jcip.annotations.ThreadSafe; +import org.junit.Test; + +import static com.github.dockerjava.junit.DockerAssume.assumeNotSwarm; +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.hasItem; +import static org.hamcrest.core.Is.is; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertTrue; + +/** + * TODO fix parallel. + */ +@ThreadSafe +public class ConnectToNetworkCmdIT extends CmdIT { + + @Test + public void connectToNetwork() throws InterruptedException { + assumeNotSwarm("no network in swarm", dockerRule); + String networkName = "connectToNetwork"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("sleep", "9999").exec(); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + CreateNetworkResponse network = dockerRule.getClient().createNetworkCmd().withName(networkName).exec(); + + dockerRule.getClient().connectToNetworkCmd().withNetworkId(network.getId()).withContainerId(container.getId()).exec(); + + Network updatedNetwork = dockerRule.getClient().inspectNetworkCmd().withNetworkId(network.getId()).exec(); + + assertTrue(updatedNetwork.getContainers().containsKey(container.getId())); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertNotNull(inspectContainerResponse.getNetworkSettings().getNetworks().get(networkName)); + } + + @Test + public void connectToNetworkWithContainerNetwork() throws InterruptedException { + assumeNotSwarm("no network in swarm", dockerRule); + + final String subnetPrefix = "10.100.100"; + final String networkName = "ContainerWithNetwork"; + final String containerIp = subnetPrefix + ".100"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("sleep", "9999") + .exec(); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + try { + dockerRule.getClient().removeNetworkCmd(networkName).exec(); + } catch (DockerException ignore) { + } + + CreateNetworkResponse network = dockerRule.getClient().createNetworkCmd() + .withName(networkName) + .withIpam(new Network.Ipam() + .withConfig(new Network.Ipam.Config() + .withSubnet(subnetPrefix + ".0/24"))) + .exec(); + + dockerRule.getClient().connectToNetworkCmd() + .withNetworkId(network.getId()) + .withContainerId(container.getId()) + .withContainerNetwork(new ContainerNetwork() + .withAliases("aliasName") + .withIpamConfig(new ContainerNetwork.Ipam() + .withIpv4Address(containerIp))) + .exec(); + + Network updatedNetwork = dockerRule.getClient().inspectNetworkCmd().withNetworkId(network.getId()).exec(); + + Network.ContainerNetworkConfig containerNetworkConfig = updatedNetwork.getContainers().get(container.getId()); + assertNotNull(containerNetworkConfig); + assertThat(containerNetworkConfig.getIpv4Address(), is(containerIp + "/24")); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + ContainerNetwork testNetwork = inspectContainerResponse.getNetworkSettings().getNetworks().get(networkName); + assertNotNull(testNetwork); + assertThat(testNetwork.getAliases(), hasItem("aliasName")); + assertThat(testNetwork.getGateway(), is(subnetPrefix + ".1")); + assertThat(testNetwork.getIpAddress(), is(containerIp)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/ContainerDiffCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/ContainerDiffCmdIT.java new file mode 100644 index 000000000..7ff39f3fa --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/ContainerDiffCmdIT.java @@ -0,0 +1,51 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.ChangeLog; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +import static ch.lambdaj.Lambda.selectUnique; +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; + +public class ContainerDiffCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(ContainerDiffCmdIT.class); + + @Test + public void testContainerDiff() throws DockerException { + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("touch", "/test").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + int exitCode = dockerRule.getClient().waitContainerCmd(container.getId()).start() + .awaitStatusCode(); + assertThat(exitCode, equalTo(0)); + + List filesystemDiff = dockerRule.getClient().containerDiffCmd(container.getId()).exec(); + LOG.info("Container DIFF: {}", filesystemDiff.toString()); + + assertThat(filesystemDiff.size(), equalTo(1)); + ChangeLog testChangeLog = selectUnique(filesystemDiff, hasField("path", equalTo("/test"))); + + assertThat(testChangeLog, hasField("path", equalTo("/test"))); + assertThat(testChangeLog, hasField("kind", equalTo(1))); + } + + @Test(expected = NotFoundException.class) + public void testContainerDiffWithNonExistingContainer() throws DockerException { + + dockerRule.getClient().containerDiffCmd("non-existing").exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/CopyArchiveFromContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/CopyArchiveFromContainerCmdIT.java new file mode 100644 index 000000000..e0c2ca03e --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/CopyArchiveFromContainerCmdIT.java @@ -0,0 +1,89 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.core.util.CompressArchiveUtil; +import com.github.dockerjava.utils.TestUtils; +import org.apache.commons.compress.archivers.tar.TarArchiveEntry; +import org.apache.commons.compress.archivers.tar.TarArchiveInputStream; +import org.apache.commons.io.IOUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.InputStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.nio.file.StandardOpenOption; + +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.isEmptyOrNullString; +import static org.hamcrest.Matchers.not; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertTrue; + +public class CopyArchiveFromContainerCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(CopyArchiveFromContainerCmdIT.class); + + @Test + public void copyFromContainer() { + // TODO extract this into a shared method + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withName("copyFromContainer") + .withCmd("touch", "/copyFromContainer") + .exec(); + + LOG.info("Created container: {}", container); + assertThat(container.getId(), not(isEmptyOrNullString())); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InputStream response = dockerRule.getClient().copyArchiveFromContainerCmd(container.getId(), "/copyFromContainer").exec(); + + // read the stream fully. Otherwise, the underlying stream will not be closed. + String responseAsString = TestUtils.asString(response); + assertNotNull(responseAsString); + assertTrue(responseAsString.length() > 0); + } + + @Test(expected = NotFoundException.class) + public void copyFromNonExistingContainer() { + + dockerRule.getClient().copyArchiveFromContainerCmd("non-existing", "/test").exec(); + } + + @Test + public void copyFromContainerBinaryFile() throws Exception { + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withName("copyFromContainerBinaryFile") + .exec(); + + LOG.info("Created container: {}", container); + assertThat(container.getId(), not(isEmptyOrNullString())); + + Path temp = Files.createTempFile("", ".tar.gz"); + Path binaryFile = Paths.get(ClassLoader.getSystemResource("testCopyFromArchive/binary.dat").toURI()); + CompressArchiveUtil.tar(binaryFile, temp, true, false); + + try (InputStream uploadStream = Files.newInputStream(temp)) { + dockerRule.getClient().copyArchiveToContainerCmd(container.getId()).withTarInputStream(uploadStream).exec(); + } + + InputStream response = dockerRule.getClient().copyArchiveFromContainerCmd(container.getId(), "/binary.dat").exec(); + + try (TarArchiveInputStream tarInputStream = new TarArchiveInputStream(response)) { + TarArchiveEntry nextTarEntry = tarInputStream.getNextTarEntry(); + + assertEquals("binary.dat", nextTarEntry.getName()); + try (InputStream binaryFileInputStream = Files.newInputStream(binaryFile, StandardOpenOption.READ)) { + assertTrue(IOUtils.contentEquals(binaryFileInputStream, tarInputStream)); + } + + assertNull("Nothing except binary.dat is expected to be copied.", tarInputStream.getNextTarEntry()); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/CopyArchiveToContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/CopyArchiveToContainerCmdIT.java new file mode 100644 index 000000000..efce65c29 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/CopyArchiveToContainerCmdIT.java @@ -0,0 +1,220 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.core.util.CompressArchiveUtil; +import com.github.dockerjava.utils.LogContainerTestCallback; +import org.apache.commons.io.FileUtils; +import org.junit.Ignore; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; +import java.io.InputStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.concurrent.TimeUnit; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.isEmptyOrNullString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.notNullValue; +import static org.junit.Assert.assertTrue; +import static org.junit.Assume.assumeThat; + +public class CopyArchiveToContainerCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(CopyArchiveToContainerCmdIT.class); + + @Test + public void copyFileToContainer() throws Exception { + CreateContainerResponse container = prepareContainerForCopy("1"); + Path temp = Files.createTempFile("", ".tar.gz"); + CompressArchiveUtil.tar(Paths.get(ClassLoader.getSystemResource("testReadFile").toURI()), temp, true, false); + try (InputStream uploadStream = Files.newInputStream(temp)) { + dockerRule.getClient() + .copyArchiveToContainerCmd(container.getId()) + .withTarInputStream(uploadStream) + .exec(); + assertFileCopied(container); + } + } + + @Test + public void copyStreamToContainer() throws Exception { + CreateContainerResponse container = prepareContainerForCopy("2"); + dockerRule.getClient().copyArchiveToContainerCmd(container.getId()) + .withHostResource(Paths.get(ClassLoader.getSystemResource("testReadFile").toURI()).toString()) + .exec(); + assertFileCopied(container); + } + + @Test + public void copyStreamToContainerTwice() throws Exception { + CreateContainerResponse container = prepareContainerForCopy("rerun"); + CopyArchiveToContainerCmd copyArchiveToContainerCmd = dockerRule.getClient().copyArchiveToContainerCmd(container.getId()) + .withHostResource(Paths.get(ClassLoader.getSystemResource("testReadFile").toURI()).toString()); + copyArchiveToContainerCmd.exec(); + assertFileCopied(container); + //run again to make sure no DockerClientException + copyArchiveToContainerCmd.exec(); + } + + private CreateContainerResponse prepareContainerForCopy(String method) { + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withName("docker-java-itest-copyToContainer" + method) + .exec(); + LOG.info("Created container: {}", container); + assertThat(container.getId(), not(isEmptyOrNullString())); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + // Copy a folder to the container + return container; + } + + private void assertFileCopied(CreateContainerResponse container) throws IOException { + try (InputStream response = dockerRule.getClient().copyArchiveFromContainerCmd(container.getId(), "testReadFile").exec()) { + boolean bytesAvailable = response.read() != -1; + assertTrue( "The file was not copied to the container.", bytesAvailable); + } + } + + @Test(expected = NotFoundException.class) + public void copyToNonExistingContainer() throws Exception { + dockerRule.getClient().copyArchiveToContainerCmd("non-existing") + .withHostResource(Paths.get(ClassLoader.getSystemResource("testReadFile").toURI()).toString()).exec(); + } + + @Test + public void copyDirWithLastAddedTarEntryEmptyDir() throws Exception{ + // create a temp dir + Path localDir = Files.createTempDirectory(null); + localDir.toFile().deleteOnExit(); + // create empty sub-dir with name b + Files.createDirectory(localDir.resolve("b")); + // create sub-dir with name a + Path dirWithFile = Files.createDirectory(localDir.resolve("a")); + // create file in sub-dir b, name or conter are irrelevant + Files.createFile(dirWithFile.resolve("file")); + + // create a test container + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("sleep", "9999") + .exec(); + // start the container + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + // copy data from local dir to container + dockerRule.getClient().copyArchiveToContainerCmd(container.getId()) + .withHostResource(localDir.toString()) + .exec(); + + // cleanup dir + FileUtils.deleteDirectory(localDir.toFile()); + } + + @Test + public void copyFileWithExecutePermission() throws Exception { + // create script file, add permission to execute + Path scriptPath = Files.createTempFile("run", ".sh"); + boolean executable = scriptPath.toFile().setExecutable(true, false); + if (!executable){ + throw new Exception("Execute permission on file not set!"); + } + String snippet = "Running script with execute permission."; + String scriptTextStr = "#!/bin/sh\necho \"" + snippet + "\""; + // write content for created script + Files.write(scriptPath, scriptTextStr.getBytes()); + // create a test container which starts and waits 3 seconds for the + // script to be copied to the container's home dir and then executes it + String containerCmd = "sleep 3; /home/" + scriptPath.getFileName().toString(); + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withName("copyFileWithExecutivePerm") + .withCmd("/bin/sh", "-c", containerCmd) + .exec(); + // start the container + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + // copy script to container home dir + dockerRule.getClient().copyArchiveToContainerCmd(container.getId()) + .withRemotePath("/home") + .withHostResource(scriptPath.toString()) + .exec(); + // await exid code + int exitCode = dockerRule.getClient().waitContainerCmd(container.getId()) + .start() + .awaitStatusCode(); + // check result + assertThat(exitCode, equalTo(0)); + } + + @Ignore("Docker issue https://github.com/moby/moby/issues/46388") + @Test + public void copyFileWithUIDGID() throws Exception { + Path with = Files.createFile(Files.createTempDirectory("copyFileWithUIDGID").resolve("uidgid.with")); + Files.write(with, "with".getBytes()); + + Path without = Files.createFile(Files.createTempDirectory("copyFileWithUIDGID").resolve("uidgid.without")); + Files.write(without, "without".getBytes()); + + String containerCmd = "while [ ! -f /home/uidgid.with ]; do true; done && stat -c %n:%u /home/uidgid.with /home/uidgid.without"; + Long syncUserUid = 4L; // sync user in busybox uses uid=4 + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withName("copyFileWithUIDGID") + .withCmd("/bin/sh", "-c", containerCmd) + .withUser(syncUserUid.toString()) + .exec(); + // start the container + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + dockerRule.getClient().copyArchiveToContainerCmd(container.getId()) + .withRemotePath("/home/") + .withHostResource(without.toString()) + .withCopyUIDGID(false) + .exec(); + dockerRule.getClient().copyArchiveToContainerCmd(container.getId()) + .withRemotePath("/home/") + .withHostResource(with.toString()) + .withCopyUIDGID(true) + .exec(); + + // await exit code + int exitCode = dockerRule.getClient().waitContainerCmd(container.getId()) + .start() + .awaitStatusCode(); + // check result + assertThat(exitCode, equalTo(0)); + + LogContainerTestCallback loggingCallback = new LogContainerTestCallback(true); + + dockerRule.getClient().logContainerCmd(container.getId()) + .withStdOut(true) + .withTailAll() + .exec(loggingCallback); + + loggingCallback.awaitCompletion(3, TimeUnit.SECONDS); + String containerOutput = loggingCallback.toString(); + + assertThat(containerOutput, containsString(String.format("/home/uidgid.with:%d", syncUserUid))); + + Long hostUid = getHostUidIfPossible(); + assumeThat("could not get the uid on host platform", hostUid, notNullValue(Long.class)); + assertThat(containerOutput, containsString(String.format("/home/uidgid.without:%d", hostUid))); + } + + private static Long getHostUidIfPossible() { + try { + Class unixSystemClazz = Class.forName("com.sun.security.auth.module.UnixSystem"); + Object unixSystem = unixSystemClazz.newInstance(); + Object uid = unixSystemClazz.getMethod("getUid").invoke(unixSystem); + if (uid == null) { + return null; + } + + return uid instanceof Long ? (Long) uid : Long.parseLong(uid.toString()); + } catch (Exception e) { + return null; + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/CopyFileFromContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/CopyFileFromContainerCmdIT.java new file mode 100644 index 000000000..3864aa9e4 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/CopyFileFromContainerCmdIT.java @@ -0,0 +1,58 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.NotFoundException; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.InputStream; + +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_24; +import static com.github.dockerjava.junit.DockerAssume.assumeNotSwarm; +import static com.github.dockerjava.junit.DockerMatchers.isGreaterOrEqual; +import static com.github.dockerjava.utils.TestUtils.asString; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.isEmptyOrNullString; +import static org.hamcrest.Matchers.not; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertTrue; +import static org.junit.Assume.assumeThat; + +public class CopyFileFromContainerCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(CopyFileFromContainerCmdIT.class); + + @Test + public void copyFromContainer() { + assumeThat("Doesn't work since 1.24", dockerRule, not(isGreaterOrEqual(VERSION_1_24))); + + assumeNotSwarm("", dockerRule); + + String containerName = "copyFileFromContainer"; + dockerRule.ensureContainerRemoved(containerName); + + // TODO extract this into a shared method + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withName(containerName) + .withCmd("touch", "/copyFileFromContainer") + .exec(); + + LOG.info("Created container: {}", container); + assertThat(container.getId(), not(isEmptyOrNullString())); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InputStream response = dockerRule.getClient().copyFileFromContainerCmd(container.getId(), "/copyFileFromContainer").exec(); + + // read the stream fully. Otherwise, the underlying stream will not be closed. + String responseAsString = asString(response); + assertNotNull(responseAsString); + assertTrue(responseAsString.length() > 0); + } + + @Test(expected = NotFoundException.class) + public void copyFromNonExistingContainer() { + + dockerRule.getClient().copyFileFromContainerCmd("non-existing", "/test").exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/CreateContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/CreateContainerCmdIT.java new file mode 100644 index 000000000..99d5dd997 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/CreateContainerCmdIT.java @@ -0,0 +1,1176 @@ +package com.github.dockerjava.cmd; + +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.node.ObjectNode; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.CreateContainerCmd; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.CreateNetworkResponse; +import com.github.dockerjava.api.command.CreateVolumeResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.exception.ConflictException; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.core.DefaultDockerClientConfig; +import com.github.dockerjava.api.exception.InternalServerErrorException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.Bind; +import com.github.dockerjava.api.model.ContainerNetwork; +import com.github.dockerjava.api.model.Device; +import com.github.dockerjava.api.model.DockerObjectAccessor; +import com.github.dockerjava.api.model.ExposedPort; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.api.model.HostConfig; +import com.github.dockerjava.api.model.Link; +import com.github.dockerjava.api.model.LogConfig; +import com.github.dockerjava.api.model.Network; +import com.github.dockerjava.api.model.Ports; +import com.github.dockerjava.api.model.Ports.Binding; +import com.github.dockerjava.api.model.RestartPolicy; +import com.github.dockerjava.api.model.Ulimit; +import com.github.dockerjava.api.model.Volume; +import com.github.dockerjava.api.model.VolumesFrom; +import com.github.dockerjava.junit.DockerAssume; +import com.github.dockerjava.junit.PrivateRegistryRule; +import com.github.dockerjava.utils.TestUtils; +import net.jcip.annotations.NotThreadSafe; +import org.apache.commons.io.FileUtils; +import org.apache.commons.lang3.SystemUtils; +import org.junit.ClassRule; +import org.junit.Ignore; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; +import org.junit.rules.TemporaryFolder; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.File; +import java.security.SecureRandom; +import java.util.Arrays; +import java.util.Collections; +import java.util.HashMap; +import java.util.Map; +import java.util.UUID; +import java.util.concurrent.TimeUnit; + +import static com.github.dockerjava.api.model.Capability.MKNOD; +import static com.github.dockerjava.api.model.Capability.NET_ADMIN; +import static com.github.dockerjava.api.model.HostConfig.newHostConfig; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_23; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_24; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_44; +import static com.github.dockerjava.junit.DockerMatchers.isGreaterOrEqual; +import static com.github.dockerjava.junit.DockerMatchers.mountedVolumes; +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.allOf; +import static org.hamcrest.Matchers.contains; +import static org.hamcrest.Matchers.containsInAnyOrder; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.hasEntry; +import static org.hamcrest.Matchers.hasItem; +import static org.hamcrest.Matchers.hasItemInArray; +import static org.hamcrest.Matchers.instanceOf; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.Matchers.nullValue; +import static org.hamcrest.Matchers.startsWith; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertSame; +import static org.junit.Assume.assumeThat; + +@NotThreadSafe +public class CreateContainerCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(CreateContainerCmdIT.class); + + @ClassRule + public static PrivateRegistryRule REGISTRY = new PrivateRegistryRule(); + + @Rule + public TemporaryFolder tempDir = new TemporaryFolder(new File("target/")); + + @Rule + public ExpectedException exception = ExpectedException.none(); + + @Test(expected = ConflictException.class) + public void createContainerWithExistingName() throws DockerException { + + String containerName = "generated_" + new SecureRandom().nextInt(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("env") + .withName(containerName).exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("env").withName(containerName).exec(); + } + + @Test + public void createContainerWithVolume() throws DockerException { + + Volume volume = new Volume("/var/log"); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withVolumes(volume) + .withCmd("true").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + LOG.info("Inspect container {}", inspectContainerResponse.getConfig().getVolumes()); + + assertThat(inspectContainerResponse.getConfig().getVolumes().keySet(), contains("/var/log")); + + assertThat(inspectContainerResponse.getMounts().get(0).getDestination(), equalTo(volume)); + assertThat(inspectContainerResponse.getMounts().get(0).getMode(), equalTo("")); + assertThat(inspectContainerResponse.getMounts().get(0).getRW(), equalTo(true)); + } + + @Test + public void createContainerWithReadOnlyVolume() throws DockerException { + + Volume volume = new Volume("/srv/test"); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withVolumes(volume) + .withCmd("true").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + LOG.info("Inspect container {}", inspectContainerResponse.getConfig().getVolumes()); + + assertThat(inspectContainerResponse.getConfig().getVolumes().keySet(), contains("/srv/test")); + + assertThat(inspectContainerResponse.getMounts().get(0).getDestination(), equalTo(volume)); + // TODO: Create a read-only volume and test like this + // assertFalse(inspectContainerResponse.getMounts().get(0).getRW()); + } + + @Test + public void createContainerWithVolumesFrom() throws DockerException { + String container1Name = UUID.randomUUID().toString(); + CreateVolumeResponse volume1Info = dockerRule.getClient().createVolumeCmd().exec(); + CreateVolumeResponse volume2Info = dockerRule.getClient().createVolumeCmd().exec(); + + Volume volume1 = new Volume("/src/webapp1"); + Volume volume2 = new Volume("/src/webapp2"); + Bind bind1 = new Bind(volume1Info.getName(), volume1); + Bind bind2 = new Bind(volume2Info.getName(), volume2); + + // create a running container with bind mounts + CreateContainerResponse container1 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("sleep", "9999") + .withName(container1Name) + .withHostConfig(newHostConfig() + .withBinds(bind1, bind2)) + .exec(); + + LOG.info("Created container1 {}", container1.toString()); + + InspectContainerResponse inspectContainerResponse1 = dockerRule.getClient().inspectContainerCmd(container1.getId()) + .exec(); + + assertThat(Arrays.asList(inspectContainerResponse1.getHostConfig().getBinds()), containsInAnyOrder(bind1, bind2)); + + assertThat(inspectContainerResponse1, mountedVolumes(containsInAnyOrder(volume1, volume2))); + + // create a second container with volumes from first container + CreateContainerResponse container2 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("sleep", "9999") + .withHostConfig(newHostConfig() + .withVolumesFrom(new VolumesFrom(container1Name))) + .exec(); + + LOG.info("Created container2 {}", container2.toString()); + + InspectContainerResponse inspectContainerResponse2 = dockerRule.getClient().inspectContainerCmd(container2.getId()) + .exec(); + + // No volumes are created, the information is just stored in .HostConfig.VolumesFrom + assertThat(inspectContainerResponse2.getHostConfig().getVolumesFrom(), + hasItemInArray(new VolumesFrom(container1Name))); + assertThat(inspectContainerResponse1, mountedVolumes(containsInAnyOrder(volume1, volume2))); + + // To ensure that the information stored in VolumesFrom really is considered + // when starting the container, we start it and verify that it has the same + // bind mounts as the first container. + // This is somehow out of scope here, but it helped me to understand how the + // VolumesFrom feature really works. + dockerRule.getClient().startContainerCmd(container2.getId()).exec(); + LOG.info("Started container2 {}", container2.toString()); + + inspectContainerResponse2 = dockerRule.getClient().inspectContainerCmd(container2.getId()).exec(); + + assertThat(inspectContainerResponse2.getHostConfig().getVolumesFrom(), hasItemInArray(new VolumesFrom( + container1Name))); + + assertThat(inspectContainerResponse2, mountedVolumes(containsInAnyOrder(volume1, volume2))); + } + + @Test + public void createContainerWithEnv() throws Exception { + final String testVariable = "VARIABLE=success"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withEnv(testVariable) + .withCmd("env") + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), hasItem(testVariable)); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + assertThat(dockerRule.containerLog(container.getId()), containsString(testVariable)); + } + + @Test + public void createContainerWithEnvAdditive() throws Exception { + + final String testVariable1 = "VARIABLE1=success1"; + final String testVariable2 = "VARIABLE2=success2"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withEnv(testVariable1) + .withEnv(testVariable2) + .withCmd("env") + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), not(hasItem(testVariable1))); + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), hasItem(testVariable2)); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + assertThat(dockerRule.containerLog(container.getId()), not(containsString(testVariable1))); + assertThat(dockerRule.containerLog(container.getId()), containsString(testVariable2)); + } + + @Test + public void createContainerWithEnvAdditiveMap() throws Exception { + final String[] testVariables1 = {"VARIABLE1=success1", "VARIABLE2=success2"}; + final String[] testVariables2 = {"VARIABLE3=success3", "VARIABLE4=success4"}; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withEnv(testVariables1) + .withEnv(testVariables2) + .withCmd("env") + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), not(hasItem(testVariables1[0]))); + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), not(hasItem(testVariables1[1]))); + + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), hasItem(testVariables2[0])); + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), hasItem(testVariables2[1])); + + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + assertThat(dockerRule.containerLog(container.getId()), not(containsString(testVariables1[0]))); + assertThat(dockerRule.containerLog(container.getId()), not(containsString(testVariables1[1]))); + + assertThat(dockerRule.containerLog(container.getId()), containsString(testVariables2[0])); + assertThat(dockerRule.containerLog(container.getId()), containsString(testVariables2[1])); + } + + @Test + public void createContainerWithEnvAsVararg() throws Exception { + + final String testVariable1 = "VARIABLE1=success1"; + final String testVariable2 = "VARIABLE2=success2"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withEnv(testVariable1, testVariable2) + .withCmd("env") + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), hasItem(testVariable1)); + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), hasItem(testVariable2)); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + assertThat(dockerRule.containerLog(container.getId()), containsString(testVariable1)); + assertThat(dockerRule.containerLog(container.getId()), containsString(testVariable2)); + } + + @Test + public void createContainerWithEnvAsMap() throws Exception { + final String[] testVariables = {"VARIABLE1=success1", "VARIABLE2=success2"}; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withEnv(testVariables) + .withCmd("env") + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), hasItem(testVariables[0])); + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), hasItem(testVariables[1])); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + assertThat(dockerRule.containerLog(container.getId()), containsString(testVariables[0])); + assertThat(dockerRule.containerLog(container.getId()), containsString(testVariables[1])); + } + + @Test + public void createContainerWithHostname() throws Exception { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withHostName("docker-java") + .withCmd("env").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getConfig().getHostName(), equalTo("docker-java")); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + assertThat(dockerRule.containerLog(container.getId()), containsString("HOSTNAME=docker-java")); + } + + @Test(expected = ConflictException.class) + public void createContainerWithName() throws DockerException { + String containerName = "container_"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withName(containerName) + .withCmd("env").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getName(), equalTo("/" + containerName)); + + dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withName(containerName) + .withCmd("env") + .exec(); + } + + @Test + public void createContainerWithLink() throws DockerException { + String containerName1 = "containerWithlink_"; + String containerName2 = "container2Withlink_"; + + CreateContainerResponse container1 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("sleep", "9999") + .withName(containerName1).exec(); + LOG.info("Created container1 {}", container1.toString()); + assertThat(container1.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container1.getId()).exec(); + + InspectContainerResponse inspectContainerResponse1 = dockerRule.getClient().inspectContainerCmd(container1.getId()) + .exec(); + LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); + assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); + + CreateContainerResponse container2 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withName(containerName2) + .withCmd("env") + .withHostConfig(newHostConfig() + .withLinks(new Link(containerName1, "container1Link"))) + .exec(); + LOG.info("Created container {}", container2.toString()); + assertThat(container2.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse2 = dockerRule.getClient().inspectContainerCmd(container2.getId()) + .exec(); + assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[]{new Link(containerName1, + "container1Link")})); + } + + @Test + @Ignore + public void createContainerWithMemorySwappiness() throws DockerException { + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("sleep", "9999") + .withHostConfig(newHostConfig() + .withMemorySwappiness(42L)) + .exec(); + assertThat(container.getId(), not(is(emptyString()))); + LOG.info("Created container {}", container.toString()); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + LOG.info("Started container {}", container.toString()); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient() + .inspectContainerCmd(container.getId()) + .exec(); + LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); + assertSame(42L, inspectContainerResponse.getHostConfig().getMemorySwappiness()); + } + + @Test + public void createContainerWithLinkInCustomNetwork() throws DockerException { + String containerName1 = "containerCustomlink_"; + String containerName2 = "containerCustom2link_"; + String networkName = "linkNetcustom"; + + CreateNetworkResponse createNetworkResponse = dockerRule.getClient().createNetworkCmd() + .withName(networkName) + .withDriver("bridge") + .exec(); + + assertNotNull(createNetworkResponse.getId()); + + CreateContainerResponse container1 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withHostConfig(newHostConfig() + .withNetworkMode(networkName)) + .withCmd("sleep", "9999") + .withName(containerName1) + .exec(); + + assertThat(container1.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container1.getId()).exec(); + + InspectContainerResponse inspectContainerResponse1 = dockerRule.getClient().inspectContainerCmd(container1.getId()) + .exec(); + LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); + assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); + + CreateContainerResponse container2 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withHostConfig(newHostConfig() + .withLinks(new Link(containerName1, containerName1 + "Link")) + .withNetworkMode(networkName)) + .withName(containerName2) + .withCmd("env") + .exec(); + + LOG.info("Created container {}", container2.toString()); + assertThat(container2.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse2 = dockerRule.getClient().inspectContainerCmd(container2.getId()) + .exec(); + + ContainerNetwork linkNet = inspectContainerResponse2.getNetworkSettings().getNetworks().get(networkName); + assertNotNull(linkNet); + assertThat(linkNet.getLinks(), equalTo(new Link[]{new Link(containerName1, containerName1 + "Link")})); + } + + @Test + public void createContainerWithCustomIp() throws DockerException { + String containerName1 = "containerCustomIplink_"; + String networkName = "customIpNet"; + String subnetPrefix = "10.100.101"; + + CreateNetworkResponse createNetworkResponse = dockerRule.getClient().createNetworkCmd() + .withIpam(new Network.Ipam() + .withConfig(new Network.Ipam.Config() + .withSubnet(subnetPrefix + ".0/24"))) + .withDriver("bridge") + .withName(networkName) + .exec(); + + assertNotNull(createNetworkResponse.getId()); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withHostConfig(newHostConfig() + .withNetworkMode(networkName)) + .withCmd("sleep", "9999") + .withName(containerName1) + .withIpv4Address(subnetPrefix + ".100") + .exec(); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient() + .inspectContainerCmd(container.getId()) + .exec(); + + ContainerNetwork customIpNet = inspectContainerResponse.getNetworkSettings().getNetworks().get(networkName); + assertNotNull(customIpNet); + assertThat(customIpNet.getGateway(), is(subnetPrefix + ".1")); + assertThat(customIpNet.getIpAddress(), is(subnetPrefix + ".100")); + } + + @Test + public void createContainerWithAlias() throws DockerException { + String containerName1 = "containerAlias_"; + String networkName = "aliasNet"; + + CreateNetworkResponse createNetworkResponse = dockerRule.getClient().createNetworkCmd() + .withName(networkName) + .withDriver("bridge") + .exec(); + + assertNotNull(createNetworkResponse.getId()); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withHostConfig(newHostConfig() + .withNetworkMode(networkName)) + .withCmd("sleep", "9999") + .withName(containerName1) + .withAliases("server") + .exec(); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()) + .exec(); + + ContainerNetwork aliasNet = inspectContainerResponse.getNetworkSettings().getNetworks().get(networkName); + assertThat(aliasNet.getAliases(), hasItem("server")); + } + + @Test + public void createContainerWithCapAddAndCapDrop() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withHostConfig(newHostConfig() + .withCapAdd(NET_ADMIN) + .withCapDrop(MKNOD)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapAdd()), contains(NET_ADMIN)); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapDrop()), contains(MKNOD)); + } + + @Test + public void createContainerWithDns() throws DockerException { + + String aDnsServer = "8.8.8.8"; + String anotherDnsServer = "8.8.4.4"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("true") + .withHostConfig(newHostConfig() + .withDns(aDnsServer, anotherDnsServer)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDns()), + contains(aDnsServer, anotherDnsServer)); + } + + @Test + public void createContainerWithEntrypoint() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withName("containerEntrypoint") + .withEntrypoint("sleep", "9999").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEntrypoint()), contains("sleep", "9999")); + + } + + @Test + public void createContainerWithExtraHosts() throws DockerException { + + String[] extraHosts = {"dockerhost:127.0.0.1", "otherhost:10.0.0.1"}; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withName("containerextrahosts") + .withHostConfig(newHostConfig() + .withExtraHosts(extraHosts)).exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getExtraHosts()), + containsInAnyOrder("dockerhost:127.0.0.1", "otherhost:10.0.0.1")); + } + + @Test + public void createContainerWithDevices() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("sleep", "9999") + .withHostConfig(newHostConfig() + .withDevices(new Device("rwm", "/dev/nulo", "/dev/zero"))) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDevices()), contains(new Device("rwm", + "/dev/nulo", "/dev/zero"))); + } + + @Test + public void createContainerWithPortBindings() throws DockerException { + int baseport = 10_000; + + ExposedPort tcp22 = ExposedPort.tcp(22); + ExposedPort tcp23 = ExposedPort.tcp(23); + + Ports portBindings = new Ports(); + portBindings.bind(tcp22, Binding.bindPort(baseport + 22)); + portBindings.bind(tcp23, Binding.bindPort(baseport + 23)); + portBindings.bind(tcp23, Binding.bindPort(baseport + 24)); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("true") + .withExposedPorts(tcp22, tcp23) + .withHostConfig(newHostConfig() + .withPortBindings(portBindings)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getExposedPorts()), contains(tcp22, tcp23)); + + assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp22)[0], + is(equalTo(Binding.bindPort(baseport + 22)))); + + assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[0], + is(equalTo(Binding.bindPort(baseport + 23)))); + + assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[1], + is(equalTo(Binding.bindPort(baseport + 24)))); + + } + + @Test + public void createContainerWithLinking() throws DockerException { + String containerName1 = "containerWithlinking_"; + String containerName2 = "container2Withlinking_"; + + CreateContainerResponse container1 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("sleep", "9999") + .withName(containerName1).exec(); + + LOG.info("Created container1 {}", container1.toString()); + assertThat(container1.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container1.getId()).exec(); + + InspectContainerResponse inspectContainerResponse1 = dockerRule.getClient().inspectContainerCmd(container1.getId()) + .exec(); + LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); + + assertThat(inspectContainerResponse1.getConfig(), is(notNullValue())); + assertThat(inspectContainerResponse1.getId(), not(is(emptyString()))); + assertThat(inspectContainerResponse1.getId(), startsWith(container1.getId())); + assertThat(inspectContainerResponse1.getName(), equalTo("/" + containerName1)); + assertThat(inspectContainerResponse1.getImageId(), not(is(emptyString()))); + assertThat(inspectContainerResponse1.getState(), is(notNullValue())); + assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); + + if (!inspectContainerResponse1.getState().getRunning()) { + assertThat(inspectContainerResponse1.getState().getExitCode(), is(equalTo(0))); + } + + CreateContainerResponse container2 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("sleep", "9999") + .withName(containerName2) + .withHostConfig(newHostConfig() + .withLinks(new Link(containerName1, containerName1 + "Link"))) + .exec(); + + LOG.info("Created container2 {}", container2.toString()); + assertThat(container2.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse2 = dockerRule.getClient().inspectContainerCmd(container2.getId()) + .exec(); + LOG.info("Container2 Inspect: {}", inspectContainerResponse2.toString()); + + assertThat(inspectContainerResponse2.getConfig(), is(notNullValue())); + assertThat(inspectContainerResponse2.getId(), not(is(emptyString()))); + assertThat(inspectContainerResponse2.getHostConfig(), is(notNullValue())); + assertThat(inspectContainerResponse2.getHostConfig().getLinks(), is(notNullValue())); + assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[]{new Link(containerName1, + containerName1 + "Link")})); + assertThat(inspectContainerResponse2.getId(), startsWith(container2.getId())); + assertThat(inspectContainerResponse2.getName(), equalTo("/" + containerName2)); + assertThat(inspectContainerResponse2.getImageId(), not(is(emptyString()))); + + } + + @Test + public void createContainerWithRestartPolicy() throws DockerException { + + RestartPolicy restartPolicy = RestartPolicy.onFailureRestart(5); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("sleep", "9999") + .withHostConfig(newHostConfig().withRestartPolicy(restartPolicy)).exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getHostConfig().getRestartPolicy(), is(equalTo(restartPolicy))); + } + + @Test + public void createContainerWithPidMode() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("true") + .withHostConfig(newHostConfig().withPidMode("host")).exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getHostConfig().getPidMode(), is(equalTo("host"))); + } + + /** + * This tests support for --net option for the docker run command: --net="bridge" Set the Network mode for the container 'bridge': + * creates a new network stack for the container on the docker bridge 'none': no networking for this container 'container:': reuses + * another container network stack 'host': use the host network stack inside the container. Note: the host mode gives the container full + * access to local system services such as D-bus and is therefore considered insecure. + */ + @Test + public void createContainerWithNetworkMode() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("true") + .withHostConfig(newHostConfig() + .withNetworkMode("host")) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getHostConfig().getNetworkMode(), is(equalTo("host"))); + } + + @Test + public void createContainerWithMacAddress() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withMacAddress("00:80:41:ae:fd:7e").withCmd("true").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getConfig().getMacAddress(), is("00:80:41:ae:fd:7e")); + } + + @Test + public void createContainerWithULimits() throws DockerException { + String containerName = "containerulimit"; + Ulimit[] ulimits = {new Ulimit("nproc", 709, 1026), new Ulimit("nofile", 1024, 4096)}; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withName(containerName) + .withHostConfig(newHostConfig() + .withUlimits(ulimits)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getUlimits()), + containsInAnyOrder(new Ulimit("nproc", 709, 1026), new Ulimit("nofile", 1024, 4096))); + + } + + @Test + public void createContainerWithIntegerBoundsExceedingULimit() throws DockerException { + String containerName = "containercoreulimit"; + Ulimit[] ulimits = {new Ulimit("core", 99999999998L, 99999999999L)}; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withName(containerName) + .withHostConfig(newHostConfig() + .withUlimits(ulimits)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getUlimits()), + contains(new Ulimit("core", 99999999998L, 99999999999L))); + + } + + @Test + public void createContainerWithLabels() throws DockerException { + + Map labels = new HashMap<>(); + labels.put("com.github.dockerjava.null", null); + labels.put("com.github.dockerjava.Boolean", "true"); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("sleep", "9999") + .withLabels(labels).exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + // null becomes empty string + labels.put("com.github.dockerjava.null", ""); + + // swarm adds 3d label + assertThat(inspectContainerResponse.getConfig().getLabels(), allOf( + hasEntry("com.github.dockerjava.null", ""), + hasEntry("com.github.dockerjava.Boolean", "true") + )); + } + + @Test + public void createContainerWithLogConfig() throws DockerException { + + LogConfig logConfig = new LogConfig(LogConfig.LoggingType.NONE, null); + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withHostConfig(newHostConfig() + .withLogConfig(logConfig)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + // null becomes empty string + assertThat(inspectContainerResponse.getHostConfig().getLogConfig().type, is(logConfig.type)); + } + + /** + * https://github.com/calavera/docker/blob/3781cde61ff10b1d9114ae5b4c5c1d1b2c20a1ee/integration-cli/docker_cli_run_unix_test.go#L319-L333 + */ + @Test + public void testWithStopSignal() throws Exception { + Integer signal = 10; // SIGUSR1 in busybox + + CreateContainerResponse resp = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("/bin/sh", "-c", "trap 'echo \"exit trapped 10\"; exit 10' USR1; while true; do sleep 1; done") + .withAttachStdin(true) + .withTty(true) + .withStopSignal(signal.toString()) + .exec(); + final String containerId = resp.getId(); + assertThat(containerId, not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(containerId).exec(); + + InspectContainerResponse inspect = dockerRule.getClient().inspectContainerCmd(containerId).exec(); + assertThat(inspect.getState().getRunning(), is(true)); + + dockerRule.getClient().stopContainerCmd(containerId).exec(); + Thread.sleep(TimeUnit.SECONDS.toMillis(3)); + + inspect = dockerRule.getClient().inspectContainerCmd(containerId).exec(); + assertThat(inspect.getState().getRunning(), is(false)); + assertThat(inspect.getState().getExitCode(), is(signal)); + + StringBuilder stringBuilder = new StringBuilder(); + final StringBuilderLogReader callback = new StringBuilderLogReader(stringBuilder); + dockerRule.getClient().logContainerCmd(containerId) + .withStdErr(true) + .withStdOut(true) + .withTailAll() + .exec(callback) + .awaitCompletion(); + + String log = callback.builder.toString(); + assertThat(log.trim(), is("exit trapped 10")); + } + + private static class StringBuilderLogReader extends ResultCallback.Adapter { + public StringBuilder builder; + + public StringBuilderLogReader(StringBuilder builder) { + this.builder = builder; + } + + @Override + public void onNext(Frame item) { + builder.append(new String(item.getPayload())); + super.onNext(item); + } + } + + @Test + public void createContainerWithCgroupParent() throws DockerException { + assumeThat(!SystemUtils.IS_OS_LINUX, is(true)); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withHostConfig(newHostConfig() + .withCgroupParent("/parent")) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainer = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainer.getHostConfig().getCgroupParent(), is("/parent")); + } + + @SuppressWarnings("Duplicates") + @Test + public void createContainerWithShmSize() throws DockerException { + HostConfig hostConfig = new HostConfig().withShmSize(96 * FileUtils.ONE_MB); + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withHostConfig(hostConfig).withCmd("true").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getHostConfig().getShmSize(), is(hostConfig.getShmSize())); + } + + @SuppressWarnings("Duplicates") + @Test + public void createContainerWithShmPidsLimit() throws DockerException { + assumeThat("API version should be >= 1.23", dockerRule, isGreaterOrEqual(VERSION_1_23)); + + HostConfig hostConfig = new HostConfig().withPidsLimit(2L); + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withHostConfig(hostConfig).withCmd("true").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getHostConfig().getPidsLimit(), is(hostConfig.getPidsLimit())); + } + + @Test + public void createContainerWithNetworkID() { + assumeThat("API version should be >= 1.23", dockerRule, isGreaterOrEqual(VERSION_1_24)); + + String networkName = "net-" + UUID.randomUUID().toString(); + Map labels = new HashMap<>(); + labels.put("com.example.label", "test"); + CreateNetworkResponse createNetworkResponse = dockerRule.getClient().createNetworkCmd().withName(networkName) + .withLabels(labels).withAttachable(true).exec(); + String networkId = createNetworkResponse.getId(); + CreateContainerResponse createContainerResponse = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withLabels(labels).withCmd("true").exec(); + String containerId = createContainerResponse.getId(); + dockerRule.getClient().connectToNetworkCmd().withContainerId(containerId).withNetworkId(networkId).exec(); + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(containerId).exec(); + ContainerNetwork containerNetwork = inspectContainerResponse.getNetworkSettings().getNetworks().get(networkName); + if (containerNetwork == null) { + // swarm node used network id + containerNetwork = inspectContainerResponse.getNetworkSettings().getNetworks().get(networkId); + } + assertThat(containerNetwork, notNullValue()); + } + + @Test + public void createContainerFromPrivateRegistryWithValidAuth() throws Exception { + DockerAssume.assumeSwarm(dockerRule.getClient()); + + AuthConfig authConfig = REGISTRY.getAuthConfig(); + + String imgName = REGISTRY.createPrivateImage("create-container-with-valid-auth"); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(imgName) + .withAuthConfig(authConfig) + .exec(); + + assertThat(container.getId(), is(notNullValue())); + } + + @Test + public void createContainerFromPrivateRegistryWithNoAuth() throws Exception { + AuthConfig authConfig = REGISTRY.getAuthConfig(); + + String imgName = REGISTRY.createPrivateImage("create-container-with-no-auth"); + + if (TestUtils.isSwarm(dockerRule.getClient())) { + exception.expect(instanceOf(InternalServerErrorException.class)); + } else { + exception.expect(instanceOf(NotFoundException.class)); + } + + dockerRule.getClient().createContainerCmd(imgName) + .exec(); + } + + @Test + public void createContainerWithTmpFs() throws DockerException { + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("sleep", "9999") + .withHostConfig(new HostConfig().withTmpFs(Collections.singletonMap("/tmp", "rw,noexec,nosuid,size=50m"))).exec(); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + assertThat(inspectContainerResponse.getHostConfig().getTmpFs().get("/tmp"), equalTo("rw,noexec,nosuid,size=50m")); + } + + @Test + public void createContainerWithNanoCPUs() throws DockerException { + Long nanoCPUs = 1000000000L; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("sleep", "9999") + .withHostConfig(newHostConfig() + .withNanoCPUs(nanoCPUs)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getHostConfig().getNanoCPUs(), is(nanoCPUs)); + } + + @Test + public void overrideHostConfigWithRawValues() { + HostConfig hostConfig = new HostConfig() + .withNanoCPUs(1_000_000_000L); + + DockerObjectAccessor.overrideRawValue( + hostConfig, + "NanoCPUs", + 500_000_000L + ); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("sleep", "9999") + .withHostConfig(hostConfig) + .exec(); + + LOG.info("Created container {}", container.toString()); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getHostConfig().getNanoCPUs(), is(500_000_000L)); + } + + @Test + public void shouldNotEncodeAuth() { + CreateContainerCmd cmd = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withAuthConfig(new AuthConfig().withEmail("test@test.com")) + .withCmd("sleep", "9999"); + + ObjectMapper objectMapper = dockerRule.getConfig().getObjectMapper(); + + ObjectNode jsonNode = objectMapper.valueToTree(cmd); + + assertThat(jsonNode.get("authConfig"), nullValue()); + } + + @Test + public void shouldHandleANetworkAliasWithoutACustomNetworkGracefully() { + // Should not throw + dockerRule.getClient() + .createContainerCmd(DEFAULT_IMAGE) + .withAliases("hello-world") + .withHostConfig(newHostConfig()) + .withCmd("sleep", "9999") + .exec(); + } + + @Test + public void createContainerWithAnnotations() throws DockerException { + DefaultDockerClientConfig forcedConfig = DefaultDockerClientConfig.createDefaultConfigBuilder() + .withApiVersion(VERSION_1_44) + .withRegistryUrl("https://index.docker.io/v1/") + .build(); + + DockerClient forcedClient = CmdIT.createDockerClient(forcedConfig); + Map annotations = new HashMap<>(); + annotations.put("com.example.key1", "value1"); + annotations.put("com.example.key2", "value2"); + + CreateContainerResponse container = forcedClient.createContainerCmd(DEFAULT_IMAGE) + .withCmd("sleep", "9999") + .withHostConfig(newHostConfig() + .withAnnotations(annotations)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = forcedClient.inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getHostConfig().getAnnotations(), equalTo(annotations)); + assertThat(inspectContainerResponse.getHostConfig().getAnnotations().get("com.example.key1"), equalTo("value1")); + assertThat(inspectContainerResponse.getHostConfig().getAnnotations().get("com.example.key2"), equalTo("value2")); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/CreateNetworkCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/CreateNetworkCmdIT.java new file mode 100644 index 000000000..d60425a2a --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/CreateNetworkCmdIT.java @@ -0,0 +1,91 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateNetworkResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Network; +import net.jcip.annotations.NotThreadSafe; +import org.junit.Test; + +import java.util.HashMap; +import java.util.Map; + +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_21; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_25; +import static com.github.dockerjava.junit.DockerAssume.assumeNotSwarm; +import static com.github.dockerjava.junit.DockerMatchers.isGreaterOrEqual; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.CoreMatchers.notNullValue; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.greaterThan; +import static org.junit.Assert.assertTrue; +import static org.junit.Assume.assumeThat; + +@NotThreadSafe +public class CreateNetworkCmdIT extends CmdIT { + + @Test + public void createNetwork() throws DockerException { + assumeNotSwarm("no network in swarm", dockerRule); + + String networkName = "createNetwork"; + + CreateNetworkResponse createNetworkResponse = dockerRule.getClient().createNetworkCmd().withName(networkName).exec(); + + assertNotNull(createNetworkResponse.getId()); + + Network network = dockerRule.getClient().inspectNetworkCmd().withNetworkId(createNetworkResponse.getId()).exec(); + assertThat(network.getName(), is(networkName)); + assertThat(network.getDriver(), is("bridge")); + assertThat(network.getCreated().getTime(), greaterThan(0L)); + } + + @Test + public void createNetworkWithIpamConfig() throws DockerException { + assumeNotSwarm("no network in swarm", dockerRule); + + String networkName = "networkIpam"; + String subnet = "10.67.79.0/24"; + + Network.Ipam ipam = new Network.Ipam().withConfig(new Network.Ipam.Config().withSubnet(subnet)); + CreateNetworkResponse createNetworkResponse = dockerRule.getClient().createNetworkCmd().withName(networkName).withIpam(ipam).exec(); + + assertNotNull(createNetworkResponse.getId()); + + Network network = dockerRule.getClient().inspectNetworkCmd().withNetworkId(createNetworkResponse.getId()).exec(); + assertEquals(networkName, network.getName()); + assertEquals("bridge", network.getDriver()); + assertEquals(subnet, network.getIpam().getConfig().iterator().next().getSubnet()); + } + + @Test + public void createAttachableNetwork() throws DockerException { + assumeThat("API version should be > 1.24", dockerRule, isGreaterOrEqual(VERSION_1_25)); + + String networkName = "createAttachableNetwork"; + CreateNetworkResponse createNetworkResponse = dockerRule.getClient().createNetworkCmd() + .withName(networkName) + .withAttachable(true) + .exec(); + assertNotNull(createNetworkResponse.getId()); + Network network = dockerRule.getClient().inspectNetworkCmd().withNetworkId(createNetworkResponse.getId()).exec(); + assertThat(network, notNullValue()); + assertTrue(network.isAttachable()); + } + + @Test + public void createNetworkWithLabel() throws DockerException { + assumeNotSwarm("no network in swarm?", dockerRule); + assumeThat("API version should be >= 1.21", dockerRule, isGreaterOrEqual(VERSION_1_21)); + + String networkName = "createNetworkWithLabel"; + Map labels = new HashMap<>(); + labels.put("com.example.usage", "test"); + CreateNetworkResponse createNetworkResponse = dockerRule.getClient().createNetworkCmd().withName(networkName).withLabels(labels).exec(); + assertNotNull(createNetworkResponse.getId()); + Network network = dockerRule.getClient().inspectNetworkCmd().withNetworkId(createNetworkResponse.getId()).exec(); + assertEquals(labels, network.getLabels()); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/CreateVolumeCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/CreateVolumeCmdIT.java new file mode 100644 index 000000000..f59907afa --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/CreateVolumeCmdIT.java @@ -0,0 +1,50 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateVolumeResponse; +import com.github.dockerjava.api.exception.DockerException; +import org.junit.Test; + +import java.util.Collections; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.equalTo; + +public class CreateVolumeCmdIT extends CmdIT { + + @Test + public void createVolume() throws DockerException { + + String volumeName = "volume1"; + + CreateVolumeResponse createVolumeResponse = dockerRule.getClient().createVolumeCmd().withName(volumeName) + .withDriver("local").withLabels(Collections.singletonMap("is-timelord", "yes")).exec(); + + assertThat(createVolumeResponse.getName(), equalTo(volumeName)); + assertThat(createVolumeResponse.getDriver(), equalTo("local")); + assertThat(createVolumeResponse.getLabels(), equalTo(Collections.singletonMap("is-timelord", "yes"))); + assertThat(createVolumeResponse.getMountpoint(), containsString("/volume1/")); + } + + @Test + public void createVolumeWithExistingName() throws DockerException { + + String volumeName = "volume1"; + + CreateVolumeResponse createVolumeResponse1 = dockerRule.getClient().createVolumeCmd().withName(volumeName) + .withDriver("local").withLabels(Collections.singletonMap("is-timelord", "yes")).exec(); + + assertThat(createVolumeResponse1.getName(), equalTo(volumeName)); + assertThat(createVolumeResponse1.getDriver(), equalTo("local")); + assertThat(createVolumeResponse1.getLabels(), equalTo(Collections.singletonMap("is-timelord", "yes"))); + assertThat(createVolumeResponse1.getMountpoint(), containsString("/volume1/")); + + CreateVolumeResponse createVolumeResponse2 = dockerRule.getClient().createVolumeCmd().withName(volumeName) + .withDriver("local").withLabels(Collections.singletonMap("is-timelord", "yes")).exec(); + + assertThat(createVolumeResponse2.getName(), equalTo(volumeName)); + assertThat(createVolumeResponse2.getDriver(), equalTo("local")); + assertThat(createVolumeResponse2.getLabels(), equalTo(Collections.singletonMap("is-timelord", "yes"))); + assertThat(createVolumeResponse2.getMountpoint(), equalTo(createVolumeResponse1.getMountpoint())); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/CustomCommandIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/CustomCommandIT.java new file mode 100644 index 000000000..bf273a98c --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/CustomCommandIT.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.core.DockerRule; +import com.github.dockerjava.transport.DockerHttpClient; +import com.github.dockerjava.transport.DockerHttpClient.Request; +import com.github.dockerjava.transport.DockerHttpClient.Response; +import org.apache.commons.io.IOUtils; +import org.junit.Assume; +import org.junit.Test; + +import java.nio.charset.StandardCharsets; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; + +public class CustomCommandIT extends CmdIT { + + @Test + public void testCustomCommand() throws Exception { + DockerHttpClient httpClient = CmdIT.createDockerHttpClient(DockerRule.config(null)); + + Assume.assumeNotNull(httpClient); + + Request request = Request.builder() + .method(Request.Method.GET) + .path("/_ping") + .build(); + + try (Response response = httpClient.execute(request)) { + assertThat(response.getStatusCode(), equalTo(200)); + assertThat(IOUtils.toString(response.getBody(), StandardCharsets.UTF_8), equalTo("OK")); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/DisconnectFromNetworkCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/DisconnectFromNetworkCmdIT.java new file mode 100644 index 000000000..2d932cc24 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/DisconnectFromNetworkCmdIT.java @@ -0,0 +1,60 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.CreateNetworkResponse; +import com.github.dockerjava.api.model.Network; +import org.junit.Test; + +import static com.github.dockerjava.api.model.HostConfig.newHostConfig; +import static com.github.dockerjava.junit.DockerAssume.assumeNotSwarm; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertTrue; + +public class DisconnectFromNetworkCmdIT extends CmdIT { + + @Test + public void disconnectFromNetwork() { + assumeNotSwarm("no network in swarm", dockerRule); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + CreateNetworkResponse network = dockerRule.getClient().createNetworkCmd().withName("disconnectNetwork").exec(); + + dockerRule.getClient().connectToNetworkCmd().withNetworkId(network.getId()).withContainerId(container.getId()).exec(); + + Network updatedNetwork = dockerRule.getClient().inspectNetworkCmd().withNetworkId(network.getId()).exec(); + + assertTrue(updatedNetwork.getContainers().containsKey(container.getId())); + + dockerRule.getClient().disconnectFromNetworkCmd().withNetworkId(network.getId()).withContainerId(container.getId()).exec(); + + updatedNetwork = dockerRule.getClient().inspectNetworkCmd().withNetworkId(network.getId()).exec(); + + assertFalse(updatedNetwork.getContainers().containsKey(container.getId())); + } + + @Test + public void forceDisconnectFromNetwork() { + assumeNotSwarm("no network in swarm", dockerRule); + + CreateNetworkResponse network = dockerRule.getClient().createNetworkCmd().withName("testNetwork2").exec(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withHostConfig(newHostConfig() + .withNetworkMode("testNetwork2")) + .withCmd("sleep", "9999") + .exec(); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + dockerRule.getClient().disconnectFromNetworkCmd() + .withNetworkId(network.getId()) + .withContainerId(container.getId()) + .withForce(true) + .exec(); + + Network updatedNetwork = dockerRule.getClient().inspectNetworkCmd().withNetworkId(network.getId()).exec(); + assertFalse(updatedNetwork.getContainers().containsKey(container.getId())); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/DockerHttpClientLeakDetector.java b/docker-java/src/test/java/com/github/dockerjava/cmd/DockerHttpClientLeakDetector.java new file mode 100644 index 000000000..1da12f3e0 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/DockerHttpClientLeakDetector.java @@ -0,0 +1,31 @@ +package com.github.dockerjava.cmd; + +import org.junit.rules.ExternalResource; + +public class DockerHttpClientLeakDetector extends ExternalResource { + + @Override + protected void before() { + synchronized (TrackingDockerHttpClient.ACTIVE_RESPONSES) { + TrackingDockerHttpClient.ACTIVE_RESPONSES.clear(); + } + } + + @Override + protected void after() { + synchronized (TrackingDockerHttpClient.ACTIVE_RESPONSES) { + if (TrackingDockerHttpClient.ACTIVE_RESPONSES.isEmpty()) { + return; + } + + System.out.println("Leaked responses:"); + IllegalStateException exception = new IllegalStateException("Leaked responses!"); + exception.setStackTrace(new StackTraceElement[0]); + + TrackingDockerHttpClient.ACTIVE_RESPONSES.forEach(response -> { + exception.addSuppressed(response.allocatedAt); + }); + throw exception; + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/EventsCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/EventsCmdIT.java new file mode 100644 index 000000000..2a16e5474 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/EventsCmdIT.java @@ -0,0 +1,197 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.model.Event; +import com.github.dockerjava.api.model.EventType; +import com.github.dockerjava.utils.TestUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.ArrayList; +import java.util.List; +import java.util.concurrent.CopyOnWriteArrayList; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.TimeUnit; + +import static com.github.dockerjava.junit.DockerAssume.assumeNotSwarm; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.CoreMatchers.notNullValue; +import static org.hamcrest.CoreMatchers.nullValue; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.junit.Assert.assertTrue; + +/* + * NOTE: These tests may fail if there is a difference between local and daemon time + * (this is especially a problem when using boot2docker as time may not in sync + * with the virtualbox host system) + */ +public class EventsCmdIT extends CmdIT { + private static final Logger LOG = LoggerFactory.getLogger(EventsCmdIT.class); + + private static String getEpochTime() { + return String.valueOf(System.currentTimeMillis() / 1000); + } + + @Test + public void testEventStreamTimeBound() throws Exception { + //since until and filtering events is broken in swarm + //https://github.com/docker/swarm/issues/1203 + assumeNotSwarm("", dockerRule); + + String startTime = getEpochTime(); + int expectedEvents = generateEvents(); + String endTime = getEpochTime(); + + EventsTestCallback eventCallback = new EventsTestCallback(expectedEvents); + + dockerRule.getClient().eventsCmd() + .withSince(startTime) + .withUntil(endTime) + .exec(eventCallback); + + List events = eventCallback.awaitExpectedEvents(30, TimeUnit.SECONDS); + + // we may receive more events as expected + assertTrue("Received events: " + events, events.size() >= expectedEvents); + } + + @Test + public void testEventStreaming() throws Exception { + String startTime = getEpochTime(); + + int expectedEvents = generateEvents(); + + EventsTestCallback eventCallback = new EventsTestCallback(expectedEvents); + + dockerRule.getClient().eventsCmd() + .withSince(startTime) + .exec(eventCallback); + + generateEvents(); + + List events = eventCallback.awaitExpectedEvents(30, TimeUnit.SECONDS); + + // we may receive more events as expected + assertTrue("Received events: " + events, events.size() >= expectedEvents); + + for (Event event : events) { + if (TestUtils.isSwarm(dockerRule.getClient())) { + assertThat(event.getNode(), is(notNullValue())); + assertThat(event.getNode().getAddr(), is(notNullValue())); + assertThat(event.getNode().getId(), is(notNullValue())); + assertThat(event.getNode().getIp(), is(notNullValue())); + assertThat(event.getNode().getName(), is(notNullValue())); + } else { + assertThat(event.getNode(), is(nullValue())); + } + } + } + + @Test + public void testEventStreamingWithFilter() throws Exception { + //since until and filtering events is broken in swarm + //https://github.com/docker/swarm/issues/1203 + assumeNotSwarm("", dockerRule); + + String startTime = getEpochTime(); + int expectedEvents = 1; + + EventsTestCallback eventCallback = new EventsTestCallback(expectedEvents); + + dockerRule.getClient().eventsCmd() + .withSince(startTime) + .withEventFilter("start") + .exec(eventCallback); + + generateEvents(); + + List events = eventCallback.awaitExpectedEvents(30, TimeUnit.SECONDS); + + // we should only get "start" events here + for (Event event : events) { + assertThat("Received event: " + event, event.getAction(), is("start")); + } + } + + @Test + public void testEventStreamingWithEventTypeFilter() throws Exception { + assumeNotSwarm("", dockerRule); + + String startTime = getEpochTime(); + generateEvents(); + String endTime = getEpochTime(); + + for (EventType eventType : EventType.values()) { + List events = new CopyOnWriteArrayList<>(); + try ( + ResultCallback.Adapter eventCallback = dockerRule.getClient().eventsCmd() + .withSince(startTime) + .withUntil(endTime) + .withEventTypeFilter(eventType) + .exec(new ResultCallback.Adapter() { + @Override + public void onNext(Event event) { + events.add(event); + } + }) + ) { + eventCallback.awaitCompletion(30, TimeUnit.SECONDS); + + for (Event event : events) { + assertThat("Received event: " + event, event.getType(), is(eventType)); + } + } + } + } + + /** + * This method generates some events and returns the number of events being generated + */ + private int generateEvents() throws Exception { + String testImage = "busybox:latest"; + + dockerRule.getClient().pullImageCmd(testImage).start().awaitCompletion(); + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(testImage).withCmd("sleep", "9999").exec(); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + dockerRule.getClient().stopContainerCmd(container.getId()).withTimeout(1).exec(); + + // generates 5 events with remote api 1.24: + + // Event[status=pull,id=busybox:latest,from=,node=,type=IMAGE,action=pull,actor=com.github.dockerjava.api.model.EventActor@417db6d7[id=busybox:latest,attributes={name=busybox}],time=1473455186,timeNano=1473455186436681587] + // Event[status=create,id=6ec10182cde227040bfead8547b63105e6bbc4e94b99f6098bfad6e158ce0d3c,from=busybox:latest,node=,type=CONTAINER,action=create,actor=com.github.dockerjava.api.model.EventActor@40bcec[id=6ec10182cde227040bfead8547b63105e6bbc4e94b99f6098bfad6e158ce0d3c,attributes={image=busybox:latest, name=sick_lamport}],time=1473455186,timeNano=1473455186470713257] + // Event[status=,id=,from=,node=,type=NETWORK,action=connect,actor=com.github.dockerjava.api.model.EventActor@318a1b01[id=10870ceb13abb7cf841ea68868472da881b33c8ed08d2cde7dbb39d7c24d1d27,attributes={container=6ec10182cde227040bfead8547b63105e6bbc4e94b99f6098bfad6e158ce0d3c, name=bridge, type=bridge}],time=1473455186,timeNano=1473455186544318466] + // Event[status=start,id=6ec10182cde227040bfead8547b63105e6bbc4e94b99f6098bfad6e158ce0d3c,from=busybox:latest,node=,type=CONTAINER,action=start,actor=com.github.dockerjava.api.model.EventActor@606f43a3[id=6ec10182cde227040bfead8547b63105e6bbc4e94b99f6098bfad6e158ce0d3c,attributes={image=busybox:latest, name=sick_lamport}],time=1473455186,timeNano=1473455186786163819] + // Event[status=kill,id=6ec10182cde227040bfead8547b63105e6bbc4e94b99f6098bfad6e158ce0d3c,from=busybox:latest,node=,type=CONTAINER,action=kill,actor=com.github.dockerjava.api.model.EventActor@72a9ffcf[id=6ec10182cde227040bfead8547b63105e6bbc4e94b99f6098bfad6e158ce0d3c,attributes={image=busybox:latest, name=sick_lamport, signal=15}],time=1473455186,timeNano=1473455186792963392] + + return 5; + } + + private class EventsTestCallback extends ResultCallback.Adapter { + + private final CountDownLatch countDownLatch; + + private final List events = new ArrayList<>(); + + public EventsTestCallback(int expextedEvents) { + this.countDownLatch = new CountDownLatch(expextedEvents); + } + + public void onNext(Event event) { + LOG.info("Received event #{}: {}", countDownLatch.getCount(), event); + events.add(event); + countDownLatch.countDown(); + } + + public List awaitExpectedEvents(long timeout, TimeUnit unit) { + try { + countDownLatch.await(timeout, unit); + close(); + } catch (Exception e) { + throw new RuntimeException(e); + } + return new ArrayList<>(events); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/ExecCreateCmdImplIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/ExecCreateCmdImplIT.java new file mode 100644 index 000000000..2f18d7e85 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/ExecCreateCmdImplIT.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.ExecCreateCmdResponse; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.security.SecureRandom; + +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; + +public class ExecCreateCmdImplIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(ExecCreateCmdImplIT.class); + + + @Test + public void execCreateTest() { + String containerName = "generated_" + new SecureRandom().nextInt(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withUser("root").withCmd("top") + .withName(containerName).exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + ExecCreateCmdResponse execCreateCmdResponse = dockerRule.getClient().execCreateCmd(container.getId()) + .withCmd("touch", "file.log").exec(); + + assertThat(execCreateCmdResponse.getId(), not(is(emptyString()))); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/ExecStartCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/ExecStartCmdIT.java new file mode 100644 index 000000000..fc111f0e2 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/ExecStartCmdIT.java @@ -0,0 +1,100 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.ExecCreateCmdResponse; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.core.command.ExecStartResultCallback; +import org.junit.Ignore; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.InputStream; +import java.security.SecureRandom; + +import static com.github.dockerjava.junit.DockerAssume.assumeNotSwarm; +import static com.github.dockerjava.utils.TestUtils.asString; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertTrue; + +public class ExecStartCmdIT extends CmdIT { + private static final Logger LOG = LoggerFactory.getLogger(ExecStartCmdIT.class); + + @Test + public void execStart() throws Exception { + assumeNotSwarm("no network in swarm", dockerRule); + + String containerName = "generated_" + new SecureRandom().nextInt(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("top") + .withName(containerName).exec(); + LOG.info("Created container {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + ExecCreateCmdResponse execCreateCmdResponse = dockerRule.getClient().execCreateCmd(container.getId()) + .withAttachStdout(true) + .withCmd("touch", "/execStartTest.log") + .withUser("root") + .exec(); + dockerRule.getClient().execStartCmd(execCreateCmdResponse.getId()) + .exec(new ExecStartResultCallback(System.out, System.err)) + .awaitCompletion(); + + InputStream response = dockerRule.getClient().copyArchiveFromContainerCmd(container.getId(), "/execStartTest.log").exec(); + + // read the stream fully. Otherwise, the underlying stream will not be closed. + String responseAsString = asString(response); + assertNotNull(responseAsString); + assertTrue(responseAsString.length() > 0); + } + + @Test + public void execStartAttached() throws Exception { + String containerName = "generated_" + new SecureRandom().nextInt(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999") + .withName(containerName).exec(); + LOG.info("Created container {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + ExecCreateCmdResponse execCreateCmdResponse = dockerRule.getClient().execCreateCmd(container.getId()) + .withAttachStdout(true).withCmd("touch", "/execStartTest.log").exec(); + dockerRule.getClient().execStartCmd(execCreateCmdResponse.getId()).withDetach(false).withTty(true) + .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); + + InputStream response = dockerRule.getClient().copyArchiveFromContainerCmd(container.getId(), "/execStartTest.log").exec(); + + // read the stream fully. Otherwise, the underlying stream will not be closed. + String responseAsString = asString(response); + assertNotNull(responseAsString); + assertTrue(responseAsString.length() > 0); + } + + @Ignore + @Test(expected = NotFoundException.class) + public void execStartWithNonExistentUser() throws Exception { + String containerName = "generated_" + new SecureRandom().nextInt(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999") + .withName(containerName).exec(); + LOG.info("Created container {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + ExecCreateCmdResponse execCreateCmdResponse = dockerRule.getClient().execCreateCmd(container.getId()) + .withAttachStdout(true).withCmd("touch", "/execStartTest.log").withUser("NonExistentUser").exec(); + dockerRule.getClient().execStartCmd(execCreateCmdResponse.getId()).withDetach(false).withTty(true) + .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); + + dockerRule.getClient().copyArchiveFromContainerCmd(container.getId(), "/execStartTest.log").exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/HealthCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/HealthCmdIT.java new file mode 100644 index 000000000..bdca27572 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/HealthCmdIT.java @@ -0,0 +1,88 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.HealthStateLog; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.model.HealthCheck; +import com.github.dockerjava.core.RemoteApiVersion; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.time.ZonedDateTime; +import java.time.temporal.ChronoUnit; +import java.util.Arrays; +import java.util.List; +import java.util.concurrent.TimeUnit; + +import static com.github.dockerjava.junit.DockerMatchers.isGreaterOrEqual; +import static org.awaitility.Awaitility.await; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.greaterThanOrEqualTo; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.not; +import static org.junit.Assume.assumeThat; + +public class HealthCmdIT extends CmdIT { + private final Logger LOG = LoggerFactory.getLogger(HealthCmdIT.class); + + @Test + public void healthiness() { + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("nc", "-l", "-p", "8080") + .withHealthcheck(new HealthCheck() + .withTest(Arrays.asList("CMD", "sh", "-c", "netstat -ltn | grep 8080")) + .withInterval(TimeUnit.SECONDS.toNanos(1)) + .withTimeout(TimeUnit.MINUTES.toNanos(1)) + .withStartPeriod(TimeUnit.SECONDS.toNanos(30)) + .withRetries(10)) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + await().atMost(60L, TimeUnit.SECONDS).untilAsserted( + () -> { + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + assertThat(inspectContainerResponse.getState().getHealth().getStatus(), is(equalTo("healthy"))); + } + ); + } + + @Test + public void healthiness_startInterval() { + assumeThat("API version should be >= 1.44", dockerRule, isGreaterOrEqual(RemoteApiVersion.VERSION_1_44)); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("nc", "-l", "-p", "8080") + .withHealthcheck(new HealthCheck() + .withTest(Arrays.asList("CMD", "sh", "-c", "netstat -ltn | grep 8080")) + .withInterval(TimeUnit.SECONDS.toNanos(5)) + .withTimeout(TimeUnit.MINUTES.toNanos(1)) + .withStartPeriod(TimeUnit.SECONDS.toNanos(2)) + .withStartInterval(TimeUnit.SECONDS.toNanos(1)) + .withRetries(10)) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + await().atMost(60L, TimeUnit.SECONDS).untilAsserted( + () -> { + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + List healthStateLogs = inspectContainerResponse.getState().getHealth().getLog(); + assertThat(healthStateLogs.size(), is(greaterThanOrEqualTo(2))); + healthStateLogs.forEach(log -> LOG.info("Health log: {}", log.getStart())); + HealthStateLog log1 = healthStateLogs.get(healthStateLogs.size() - 1); + HealthStateLog log2 = healthStateLogs.get(healthStateLogs.size() - 2); + long diff = ChronoUnit.NANOS.between(ZonedDateTime.parse(log2.getStart()), ZonedDateTime.parse(log1.getStart())); + assertThat(diff, is(greaterThanOrEqualTo(inspectContainerResponse.getConfig().getHealthcheck().getInterval()))); + } + ); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/InfoCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/InfoCmdIT.java new file mode 100644 index 000000000..74fc2cbda --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/InfoCmdIT.java @@ -0,0 +1,59 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Info; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static com.github.dockerjava.utils.TestUtils.isNotSwarm; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.greaterThan; +import static org.hamcrest.Matchers.isEmptyOrNullString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.notNullValue; + +/** + * @author Kanstantsin Shautsou + */ +public class InfoCmdIT extends CmdIT { + private static final Logger LOG = LoggerFactory.getLogger(InfoCmdIT.class); + + @Test + public void infoTest() throws DockerException { + DockerClient dockerClient = dockerRule.getClient(); + // Make sure that there is at least one container for the assertion + // TODO extract this into a shared method + if (dockerClient.listContainersCmd().withShowAll(true).exec().size() == 0) { + CreateContainerResponse container = dockerClient.createContainerCmd(DEFAULT_IMAGE) + .withName("docker-java-itest-info") + .withCmd("touch", "/test") + .exec(); + + LOG.info("Created container: {}", container); + assertThat(container.getId(), not(isEmptyOrNullString())); + + dockerClient.startContainerCmd(container.getId()).exec(); + } + + Info dockerInfo = dockerClient.infoCmd().exec(); + LOG.info(dockerInfo.toString()); + + assertThat(dockerInfo.getContainers(), notNullValue()); + assertThat(dockerInfo.getContainers(), greaterThan(0)); + + assertThat(dockerInfo.getImages(), notNullValue()); + assertThat(dockerInfo.getImages(), greaterThan(0)); + assertThat(dockerInfo.getDebug(), notNullValue()); + assertThat(dockerInfo.getRuntimes(), notNullValue()); + + if (isNotSwarm(dockerClient)) { + assertThat(dockerInfo.getNFd(), greaterThan(0)); + assertThat(dockerInfo.getNGoroutines(), greaterThan(0)); + assertThat(dockerInfo.getNCPU(), greaterThan(0)); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/InspectContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/InspectContainerCmdIT.java new file mode 100644 index 000000000..e47a911d7 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/InspectContainerCmdIT.java @@ -0,0 +1,160 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectContainerCmd; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Container; +import org.junit.Ignore; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.security.SecureRandom; +import java.util.Collections; +import java.util.Map; +import java.util.UUID; + +import static com.github.dockerjava.utils.TestUtils.isNotSwarm; +import static com.github.dockerjava.utils.TestUtils.isSwarm; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.greaterThanOrEqualTo; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.Matchers.nullValue; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertTrue; + +public class InspectContainerCmdIT extends CmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(InspectContainerCmdIT.class); + + @Test + public void inspectContainer() throws DockerException { + + String containerName = "generated_" + new SecureRandom().nextInt(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("top") + .withName(containerName).exec(); + LOG.info("Created container {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse containerInfo = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + assertEquals(container.getId(), containerInfo.getId()); + + } + + @Test + public void inspectContainerNodeProperty() throws DockerException { + Map label = Collections.singletonMap("inspectContainerNodeProperty", UUID.randomUUID().toString()); + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withLabels(label) + .exec(); + + Container containerResult = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withLabelFilter(label) + .exec() + .get(0); + + String name = containerResult.getNames()[0]; + + InspectContainerResponse containerInfo = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + InspectContainerResponse.Node node = containerInfo.getNode(); + if (isSwarm(dockerRule.getClient())) { + assertThat(node, is(notNullValue())); + assertThat(node.getAddr(), is(notNullValue())); + assertThat(node.getId(), is(notNullValue())); + assertThat(node.getIp(), is(notNullValue())); + assertThat(node.getLabels(), is(notNullValue())); + assertThat(node.getLabels().get("com.github.dockerjava.test"), is("docker-java")); + assertThat(node.getCpus(), is(greaterThanOrEqualTo(1))); + assertThat(node.getMemory(), is(greaterThanOrEqualTo(64 * 1024 * 1024L))); + assertThat("/" + node.getName() + containerInfo.getName(), is(name)); + } else { + assertThat(node, is(nullValue())); + } + } + + @Test() + public void inspectContainerWithSize() throws DockerException { + + String containerName = "generated_" + new SecureRandom().nextInt(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("top") + .withName(containerName).exec(); + LOG.info("Created container {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerCmd command = dockerRule.getClient().inspectContainerCmd(container.getId()).withSize(true); + assertTrue(command.getSize()); + InspectContainerResponse containerInfo = command.exec(); + assertEquals(containerInfo.getId(), container.getId()); + + // TODO check swarm + if (isNotSwarm(dockerRule.getClient())) { + assertNotNull(containerInfo.getSizeRootFs()); + assertTrue(containerInfo.getSizeRootFs() > 0L); + assertNotNull(containerInfo.getSizeRw()); + assertEquals(4096, containerInfo.getSizeRw().longValue()); + } + } + + @Test(expected = NotFoundException.class) + public void inspectNonExistingContainer() throws DockerException { + dockerRule.getClient().inspectContainerCmd("non-existing").exec(); + } + + @Test + public void inspectContainerRestartCount() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("env").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getRestartCount(), equalTo(0)); + } + + @Test + @Ignore + public void inspectContainerNetworkSettings() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("env").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertFalse(inspectContainerResponse.getNetworkSettings().getHairpinMode()); + } + + @Test + public void inspectContainerNanoCPUs() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("env").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getHostConfig().getNanoCPUs(), is(0L)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/InspectExecCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/InspectExecCmdIT.java new file mode 100644 index 000000000..b256c6a7c --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/InspectExecCmdIT.java @@ -0,0 +1,122 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.ExecCreateCmdResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.command.InspectExecResponse; +import com.github.dockerjava.core.RemoteApiVersion; +import com.github.dockerjava.core.command.ExecStartResultCallback; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; +import java.security.SecureRandom; + +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_22; +import static com.github.dockerjava.utils.TestUtils.getVersion; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.Matchers.nullValue; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; + +public class InspectExecCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(InspectExecCmdIT.class); + + @Test + public void inspectExec() throws Exception { + String containerName = "generated_" + new SecureRandom().nextInt(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999") + .withName(containerName).exec(); + LOG.info("Created container {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + // Check that file does not exist + ExecCreateCmdResponse checkFileExec1 = dockerRule.getClient().execCreateCmd(container.getId()).withAttachStdout(true) + .withAttachStderr(true).withCmd("test", "-e", "/marker").exec(); + LOG.info("Created exec {}", checkFileExec1.toString()); + assertThat(checkFileExec1.getId(), not(is(emptyString()))); + dockerRule.getClient().execStartCmd(checkFileExec1.getId()).withDetach(false) + .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); + InspectExecResponse first = dockerRule.getClient().inspectExecCmd(checkFileExec1.getId()).exec(); + assertThat(first.isRunning(), is(false)); + assertThat(first.getExitCode(), is(1)); + + // Create the file + ExecCreateCmdResponse touchFileExec = dockerRule.getClient().execCreateCmd(container.getId()).withAttachStdout(true) + .withAttachStderr(true).withCmd("touch", "/marker").exec(); + LOG.info("Created exec {}", touchFileExec.toString()); + assertThat(touchFileExec.getId(), not(is(emptyString()))); + dockerRule.getClient().execStartCmd(touchFileExec.getId()).withDetach(false) + .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); + InspectExecResponse second = dockerRule.getClient().inspectExecCmd(touchFileExec.getId()).exec(); + assertThat(second.isRunning(), is(false)); + assertThat(second.getExitCode(), is(0)); + + // Check that file does exist now + ExecCreateCmdResponse checkFileExec2 = dockerRule.getClient().execCreateCmd(container.getId()).withAttachStdout(true) + .withAttachStderr(true).withCmd("test", "-e", "/marker").exec(); + LOG.info("Created exec {}", checkFileExec2.toString()); + assertThat(checkFileExec2.getId(), not(is(emptyString()))); + dockerRule.getClient().execStartCmd(checkFileExec2.getId()).withDetach(false) + .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); + InspectExecResponse third = dockerRule.getClient().inspectExecCmd(checkFileExec2.getId()).exec(); + assertThat(third.isRunning(), is(false)); + assertThat(third.getExitCode(), is(0)); + + // Get container info and check its roundtrip to ensure the consistency + InspectContainerResponse containerInfo = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + assertEquals(containerInfo.getId(), container.getId()); + JSONTestHelper.testRoundTrip(containerInfo); + } + + @Test + public void inspectExecNetworkSettings() { + final RemoteApiVersion apiVersion = getVersion(dockerRule.getClient()); + + String containerName = "generated_" + new SecureRandom().nextInt(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999") + .withName(containerName).exec(); + LOG.info("Created container {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + ExecCreateCmdResponse exec = dockerRule.getClient().execCreateCmd(container.getId()).withAttachStdout(true) + .withAttachStderr(true).withCmd("/bin/bash").exec(); + LOG.info("Created exec {}", exec.toString()); + assertThat(exec.getId(), not(is(emptyString()))); + + InspectExecResponse inspectExecResponse = dockerRule.getClient().inspectExecCmd(exec.getId()).exec(); + + if (apiVersion.isGreaterOrEqual(RemoteApiVersion.VERSION_1_22)) { + assertThat(inspectExecResponse.getExitCode(), is(nullValue())); + assertThat(inspectExecResponse.getCanRemove(), is(false)); + assertThat(inspectExecResponse.getContainerID(), is(container.getId())); + } else { + assertThat(inspectExecResponse.getExitCode(), is(0)); + assertNotNull(inspectExecResponse.getContainer().getNetworkSettings().getNetworks().get("bridge")); + } + + assertThat(inspectExecResponse.isOpenStdin(), is(false)); + assertThat(inspectExecResponse.isOpenStdout(), is(true)); + assertThat(inspectExecResponse.isRunning(), is(false)); + + final InspectExecResponse.Container inspectContainer = inspectExecResponse.getContainer(); + if (apiVersion.isGreaterOrEqual(VERSION_1_22)) { + assertThat(inspectContainer, nullValue()); + } else { + assertThat(inspectContainer, notNullValue()); + assertNotNull(inspectContainer.getNetworkSettings().getNetworks().get("bridge")); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/InspectNetworkCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/InspectNetworkCmdIT.java new file mode 100644 index 000000000..035d3d767 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/InspectNetworkCmdIT.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Network; +import org.junit.Test; + +import java.util.List; + +import static com.github.dockerjava.junit.DockerAssume.assumeNotSwarm; +import static com.github.dockerjava.utils.TestUtils.findNetwork; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.greaterThan; + +public class InspectNetworkCmdIT extends CmdIT { + + @Test + public void inspectNetwork() throws DockerException { + assumeNotSwarm("no network in swarm", dockerRule); + + List networks = dockerRule.getClient().listNetworksCmd().exec(); + + Network expected = findNetwork(networks, "bridge"); + + Network network = dockerRule.getClient().inspectNetworkCmd().withNetworkId(expected.getId()).exec(); + + assertThat(network.getName(), equalTo(expected.getName())); + assertThat(network.getScope(), equalTo(expected.getScope())); + assertThat(network.getDriver(), equalTo(expected.getDriver())); + assertThat(network.getIpam().getConfig().get(0).getSubnet(), equalTo(expected.getIpam().getConfig().get(0).getSubnet())); + assertThat(network.getIpam().getDriver(), equalTo(expected.getIpam().getDriver())); + assertThat(network.getCreated().getTime(), greaterThan(0L)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/InspectVolumeCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/InspectVolumeCmdIT.java new file mode 100644 index 000000000..2e9b806b2 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/InspectVolumeCmdIT.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.InspectVolumeResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import org.junit.Test; + +import java.util.Collections; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.equalTo; + +public class InspectVolumeCmdIT extends CmdIT { + + @Test + public void inspectVolume() throws DockerException { + + String volumeName = "volume1"; + + dockerRule.getClient().createVolumeCmd().withName(volumeName).withDriver("local") + .withLabels(Collections.singletonMap("is-timelord", "yes")).exec(); + + InspectVolumeResponse inspectVolumeResponse = dockerRule.getClient().inspectVolumeCmd(volumeName).exec(); + + assertThat(inspectVolumeResponse.getName(), equalTo("volume1")); + assertThat(inspectVolumeResponse.getDriver(), equalTo("local")); + assertThat(inspectVolumeResponse.getLabels(), equalTo(Collections.singletonMap("is-timelord", "yes"))); + assertThat(inspectVolumeResponse.getMountpoint(), containsString("/volume1/")); + } + + @Test(expected = NotFoundException.class) + public void inspectNonExistentVolume() throws DockerException { + + String volumeName = "non-existing"; + + dockerRule.getClient().inspectVolumeCmd(volumeName).exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/KillContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/KillContainerCmdIT.java new file mode 100644 index 000000000..617e547cb --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/KillContainerCmdIT.java @@ -0,0 +1,45 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; + +public class KillContainerCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(KillContainerCmdIT.class); + + @Test + public void killContainer() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + LOG.info("Killing container: {}", container.getId()); + dockerRule.getClient().killContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); + + assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); + assertThat(inspectContainerResponse.getState().getExitCode(), not(equalTo(0))); + + } + + @Test(expected = NotFoundException.class) + public void killNonExistingContainer() throws DockerException { + + dockerRule.getClient().killContainerCmd("non-existing").exec(); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/ListContainersCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/ListContainersCmdIT.java new file mode 100644 index 000000000..3490924c7 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/ListContainersCmdIT.java @@ -0,0 +1,394 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.model.Bind; +import com.github.dockerjava.api.model.Container; +import com.github.dockerjava.api.model.Volume; +import com.github.dockerjava.junit.DockerAssume; +import org.hamcrest.Matcher; +import org.junit.After; +import org.junit.Before; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.UUID; +import java.util.concurrent.TimeUnit; + +import static ch.lambdaj.Lambda.filter; +import static com.github.dockerjava.api.model.HostConfig.newHostConfig; +import static java.util.Arrays.asList; +import static java.util.Collections.singletonList; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.hasItem; +import static org.hamcrest.Matchers.hasSize; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.Matchers.oneOf; +import static org.hamcrest.Matchers.startsWith; +import static org.junit.Assert.assertEquals; +import static org.testinfected.hamcrest.jpa.PersistenceMatchers.hasField; + +public class ListContainersCmdIT extends CmdIT { + private static final Logger LOG = LoggerFactory.getLogger(ListContainersCmdIT.class); + private static final String DEFAULT_IMAGE = "busybox"; + private Map testLabel; + + @Before + public void setUp() { + //generate unique ids per test to isolate each test case from each other + testLabel = Collections.singletonMap("test", UUID.randomUUID().toString()); + } + + @After + public void tearDown() { + //remove all containers created by this test + List containers = dockerRule.getClient().listContainersCmd() + .withLabelFilter(testLabel) + .withShowAll(true) + .exec(); + + for (Container container : containers) { + removeContainer(container.getId()); + } + } + + @Test + public void testListContainers() { + List containers = dockerRule.getClient().listContainersCmd() + .withLabelFilter(testLabel) + .withShowAll(true) + .exec(); + assertThat(containers, notNullValue()); + LOG.info("Container List: {}", containers); + + int size = containers.size(); + + CreateContainerResponse container1 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .withCmd("echo") + .exec(); + assertThat(container1.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container1.getId()).exec(); + assertThat(inspectContainerResponse.getConfig().getImage(), is(equalTo(DEFAULT_IMAGE))); + + dockerRule.getClient().startContainerCmd(container1.getId()).exec(); + LOG.info("container id: " + container1.getId()); + + List containers2 = dockerRule.getClient().listContainersCmd() + .withLabelFilter(testLabel) + .withShowAll(true) + .exec(); + for (Container container : containers2) { + LOG.info("listContainer: id=" + container.getId() + " image=" + container.getImage()); + } + + assertThat(size + 1, is(equalTo(containers2.size()))); + Matcher matcher = hasItem(hasField("id", startsWith(container1.getId()))); + assertThat(containers2, matcher); + + List filteredContainers = filter(hasField("id", startsWith(container1.getId())), containers2); + assertThat(filteredContainers.size(), is(equalTo(1))); + + for (Container container : filteredContainers) { + LOG.info("filteredContainer: " + container); + } + + Container container2 = filteredContainers.get(0); + assertThat(container2.getCommand(), not(is(emptyString()))); + assertThat(container2.getImage(), startsWith(DEFAULT_IMAGE)); + } + + @Test + public void testListContainersWithLabelsFilter() { + // list with filter by Map label + dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE).withCmd("echo") + .withLabels(testLabel) + .exec(); + + List filteredContainersByMap = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withLabelFilter(testLabel) + .exec(); + + assertThat(filteredContainersByMap.size(), is(1)); + + Container container3 = filteredContainersByMap.get(0); + assertThat(container3.getCommand(), not(is(emptyString()))); + assertThat(container3.getImage(), startsWith(DEFAULT_IMAGE)); + + // List by string label + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withLabelFilter(singletonList("test=" + testLabel.get("test"))) + .exec(); + + assertThat(filteredContainers.size(), is(1)); + + container3 = filteredContainers.get(0); + assertThat(container3.getCommand(), not(is(emptyString()))); + assertThat(container3.getImage(), startsWith(DEFAULT_IMAGE)); + assertEquals(testLabel.get("test"), container3.getLabels().get("test")); + } + + @Test + public void testNameFilter() { + String testUUID = testLabel.get("test"); + + String id1, id2; + id1 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .withName("nameFilterTest1-" + testUUID) + .exec() + .getId(); + + id2 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .withName("nameFilterTest2-" + testUUID) + .exec() + .getId(); + + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withNameFilter(asList("nameFilterTest1-" + testUUID, "nameFilterTest2-" + testUUID)) + .exec(); + + assertThat(filteredContainers.size(), is(2)); + assertThat(filteredContainers.get(0).getId(), is(oneOf(id1, id2))); + assertThat(filteredContainers.get(1).getId(), is(oneOf(id1, id2))); + } + + @Test + public void testIdsFilter() { + String id1, id2; + id1 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .exec() + .getId(); + + id2 = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .exec() + .getId(); + + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withIdFilter(asList(id1, id2)) + .exec(); + + assertThat(filteredContainers.size(), is(2)); + assertThat(filteredContainers.get(0).getId(), is(oneOf(id1, id2))); + assertThat(filteredContainers.get(1).getId(), is(oneOf(id1, id2))); + } + + @Test + public void shouldFilterByCreatedStatus() { + String containerId = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .exec() + .getId(); + + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withLabelFilter(testLabel) + .withStatusFilter(singletonList("created")) + .exec(); + + assertThat(filteredContainers.size(), is(1)); + assertThat(filteredContainers.get(0).getId(), is(containerId)); + } + + @Test + public void shouldFilterByRunningStatus() { + String containerId = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .exec() + .getId(); + dockerRule.getClient().startContainerCmd(containerId).exec(); + + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withLabelFilter(testLabel) + .withStatusFilter(singletonList("running")) + .exec(); + + assertThat(filteredContainers, hasSize(1)); + assertThat(filteredContainers.get(0).getId(), is(containerId)); + } + + @Test + public void shouldFilterByPausedStatus() { + String containerId = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("sh", "-c", "sleep 99999") + .withLabels(testLabel) + .exec() + .getId(); + dockerRule.getClient().startContainerCmd(containerId).exec(); + dockerRule.getClient().pauseContainerCmd(containerId).exec(); + + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withLabelFilter(testLabel) + .withStatusFilter(singletonList("paused")) + .exec(); + + assertThat(filteredContainers, hasSize(1)); + assertThat(filteredContainers.get(0).getId(), is(containerId)); + } + + @Test + public void shouldFilterByExitedStatus() throws InterruptedException { + String containerId = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("sh", "-c", "sleep 99999") + .withLabels(testLabel) + .exec() + .getId(); + dockerRule.getClient().startContainerCmd(containerId).exec(); + dockerRule.getClient().stopContainerCmd(containerId).exec(); + dockerRule.getClient().waitContainerCmd(containerId).start().awaitCompletion(15, TimeUnit.SECONDS); + + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withLabelFilter(testLabel) + .withStatusFilter(singletonList("exited")) + .exec(); + + assertThat(filteredContainers, hasSize(1)); + assertThat(filteredContainers.get(0).getId(), is(containerId)); + } + + @Test + public void testVolumeFilter() { + String id; + dockerRule.getClient().createVolumeCmd() + .withName("TestFilterVolume") + .withDriver("local") + .exec(); + + id = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .withHostConfig(newHostConfig() + .withBinds(new Bind("TestFilterVolume", new Volume("/test")))) + .exec() + .getId(); + + dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .exec(); + + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withLabelFilter(testLabel) + .withVolumeFilter(singletonList("TestFilterVolume")) + .exec(); + + assertThat(filteredContainers, hasSize(1)); + assertThat(filteredContainers.get(0).getId(), is(id)); + } + + @Test + public void testNetworkFilter() { + String id; + dockerRule.getClient().createNetworkCmd() + .withName("TestFilterNetwork") + .withDriver("bridge") + .exec(); + + id = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .withHostConfig(newHostConfig() + .withNetworkMode("TestFilterNetwork")) + .exec() + .getId(); + + dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .exec(); + + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withShowAll(true) + .withLabelFilter(testLabel) + .withNetworkFilter(singletonList("TestFilterNetwork")) + .exec(); + + assertThat(filteredContainers.size(), is(1)); + assertThat(filteredContainers.get(0).getId(), is(id)); + } + + @Test + public void testAncestorFilter() throws Exception { + // ancestor filters are broken in swarm + // https://github.com/docker/swarm/issues/1716 + DockerAssume.assumeNotSwarm(dockerRule.getClient()); + + dockerRule.getClient().pullImageCmd("busybox") + .withTag("1.35") + .start() + .awaitCompletion(); + + dockerRule.getClient().createContainerCmd("busybox:1.35") + .withLabels(testLabel) + .exec(); + + String imageId = dockerRule.getClient().inspectImageCmd(DEFAULT_IMAGE).exec().getId(); + + String id = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .exec() + .getId(); + + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withLabelFilter(testLabel) + .withShowAll(true) + .withAncestorFilter(singletonList(imageId)) + .exec(); + + assertThat(filteredContainers.size(), is(1)); + assertThat(filteredContainers.get(0).getId(), is(id)); + } + + @Test + public void testExitedFilter() { + dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .exec(); + + String id = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withLabels(testLabel) + .withCmd("sh", "-c", "exit 42") + .exec() + .getId(); + + dockerRule.getClient().startContainerCmd(id).exec(); + + Integer status = dockerRule.getClient().waitContainerCmd(id).start() + .awaitStatusCode(); + + assertThat(status, is(42)); + + List filteredContainers = dockerRule.getClient().listContainersCmd() + .withLabelFilter(testLabel) + .withShowAll(true) + .withExitedFilter(42) + .exec(); + + assertThat(filteredContainers.size(), is(1)); + assertThat(filteredContainers.get(0).getId(), is(id)); + } + + private void removeContainer(String id) { + if (id != null) { + try { + dockerRule.getClient().removeContainerCmd(id).withForce(true).exec(); + } catch (Exception ignored) {} + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/ListImagesCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/ListImagesCmdIT.java new file mode 100644 index 000000000..67ba85672 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/ListImagesCmdIT.java @@ -0,0 +1,112 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Image; +import com.github.dockerjava.api.model.Info; +import org.apache.commons.lang3.RandomUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Collections; +import java.util.List; + +import static com.github.dockerjava.utils.TestUtils.isNotSwarm; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.emptyArray; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.greaterThan; +import static org.hamcrest.Matchers.hasSize; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.notNullValue; +import static org.junit.Assert.assertTrue; + +public class ListImagesCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(ListImagesCmdIT.class); + + @Test + public void listImages() throws DockerException { + List images = dockerRule.getClient().listImagesCmd().withShowAll(true).exec(); + assertThat(images, notNullValue()); + LOG.info("Images List: {}", images); + Info info = dockerRule.getClient().infoCmd().exec(); + + if (isNotSwarm(dockerRule.getClient())) { + assertThat(images.size(), equalTo(info.getImages())); + } + + Image img = images.get(0); + assertThat(img.getCreated(), is(greaterThan(0L))); + assertThat(img.getSize(), is(greaterThan(0L))); + assertThat(img.getId(), not(is(emptyString()))); + assertThat(img.getRepoTags(), emptyArray()); + } + + @Test + public void listImagesWithDanglingFilter() throws DockerException { + String imageId = createDanglingImage(); + List images = dockerRule.getClient().listImagesCmd().withDanglingFilter(true).withShowAll(true) + .exec(); + assertThat(images, notNullValue()); + LOG.info("Images List: {}", images); + assertThat(images.size(), is(greaterThan(0))); + Boolean imageInFilteredList = isImageInFilteredList(images, imageId); + assertTrue(imageInFilteredList); + } + + @Test + public void listImagesWithReferenceFilter() throws DockerException { + String tag = "" + RandomUtils.nextInt(0, Integer.MAX_VALUE); + + dockerRule.getClient().tagImageCmd("busybox:latest", "docker-java/busybox", tag).exec(); + try { + List images = dockerRule.getClient().listImagesCmd().withReferenceFilter("docker-java/busybox") + .exec(); + assertThat(images, hasSize(1)); + } + finally { + dockerRule.getClient().removeImageCmd("docker-java/busybox:" + tag).exec(); + } + } + + @Test + public void listImagesWithFilter() throws DockerException { + String tag = "" + RandomUtils.nextInt(0, Integer.MAX_VALUE); + + dockerRule.getClient().tagImageCmd("busybox:latest", "docker-java/busybox", tag).exec(); + try { + List images = dockerRule.getClient().listImagesCmd().withFilter("reference", Collections.singletonList("docker-java/busybox")) + .exec(); + assertThat(images, hasSize(1)); + } + finally { + dockerRule.getClient().removeImageCmd("docker-java/busybox:" + tag).exec(); + } + } + + private boolean isImageInFilteredList(List images, String expectedImageId) { + for (Image image : images) { + if (expectedImageId.equals(image.getId())) { + return true; + } + } + return false; + } + + private String createDanglingImage() { + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + LOG.info("Committing container {}", container.toString()); + String imageId = dockerRule.getClient().commitCmd(container.getId()).exec(); + + dockerRule.getClient().stopContainerCmd(container.getId()).exec(); + dockerRule.getClient().removeContainerCmd(container.getId()).exec(); + return imageId; + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/ListNetworksCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/ListNetworksCmdIT.java new file mode 100644 index 000000000..23fb24242 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/ListNetworksCmdIT.java @@ -0,0 +1,30 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Network; +import com.github.dockerjava.utils.TestUtils; +import org.junit.Test; + +import java.util.List; + +import static com.github.dockerjava.junit.DockerAssume.assumeNotSwarm; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; + +public class ListNetworksCmdIT extends CmdIT { + + @Test + public void listNetworks() throws DockerException { + assumeNotSwarm("Swarm has no network", dockerRule); + + List networks = dockerRule.getClient().listNetworksCmd().exec(); + + Network network = TestUtils.findNetwork(networks, "bridge"); + + assertThat(network.getName(), equalTo("bridge")); + assertThat(network.getScope(), equalTo("local")); + assertThat(network.getDriver(), equalTo("bridge")); + assertThat(network.getIpam().getDriver(), equalTo("default")); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/ListVolumesCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/ListVolumesCmdIT.java new file mode 100644 index 000000000..df073e16d --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/ListVolumesCmdIT.java @@ -0,0 +1,32 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateVolumeResponse; +import com.github.dockerjava.api.command.ListVolumesResponse; +import com.github.dockerjava.api.exception.DockerException; +import org.junit.Test; + +import java.util.Collections; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.greaterThanOrEqualTo; + +public class ListVolumesCmdIT extends CmdIT { + + @Test + public void listVolumes() throws DockerException { + + CreateVolumeResponse createVolumeResponse = dockerRule.getClient().createVolumeCmd().withName("volume1") + .withDriver("local").withLabels(Collections.singletonMap("is-timelord", "yes")).exec(); + + assertThat(createVolumeResponse.getName(), equalTo("volume1")); + assertThat(createVolumeResponse.getDriver(), equalTo("local")); + assertThat(createVolumeResponse.getLabels(), equalTo(Collections.singletonMap("is-timelord", "yes"))); + assertThat(createVolumeResponse.getMountpoint(), containsString("/volume1/")); + + ListVolumesResponse listVolumesResponse = dockerRule.getClient().listVolumesCmd().exec(); + + assertThat(listVolumesResponse.getVolumes().size(), greaterThanOrEqualTo(1)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/LoadImageCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/LoadImageCmdIT.java new file mode 100644 index 000000000..36a8d51fc --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/LoadImageCmdIT.java @@ -0,0 +1,78 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.LoadImageCallback; +import com.github.dockerjava.api.model.Image; +import com.github.dockerjava.utils.TestResources; +import net.jcip.annotations.NotThreadSafe; +import org.junit.After; +import org.junit.Before; +import org.junit.Test; + +import java.io.InputStream; +import java.nio.file.Files; +import java.util.List; + +import static java.util.Arrays.asList; +import static java.util.Collections.singletonList; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.core.IsEqual.equalTo; +import static org.hamcrest.core.IsNull.notNullValue; + +@NotThreadSafe +public class LoadImageCmdIT extends CmdIT { + + private String expectedImageId; + + @Before + public void beforeMethod() { + expectedImageId = "sha256:28a8ed28c8b7bd9d7fc00f22ac7df6d385436b93e88ac978943f3dba06d836b4"; + if (findImageWithId(expectedImageId, dockerRule.getClient().listImagesCmd().exec()) != null) { + dockerRule.getClient().removeImageCmd(expectedImageId).exec(); + } + } + + @After + public void afterMethod() { + dockerRule.getClient().removeImageCmd(expectedImageId).exec(); + } + + @Test + public void loadImageFromTar() throws Exception { + try (InputStream uploadStream = Files.newInputStream(TestResources.getApiImagesLoadTestTarball())) { + dockerRule.getClient().loadImageCmd(uploadStream).exec(); + } + + //swarm needs some time to reflect new images + synchronized (this) { + wait(5000); + } + + final Image image = findImageWithId(expectedImageId, dockerRule.getClient().listImagesCmd().exec()); + + assertThat("Can't find expected image after loading from a tar archive!", image, notNullValue()); + assertThat("Image after loading from a tar archive has wrong tags!", + asList(image.getRepoTags()), equalTo(singletonList("docker-java/load:1.0"))); + } + + @Test + public void loadImageFromTarAsync() throws Exception { + try (InputStream uploadStream = Files.newInputStream(TestResources.getApiImagesLoadTestTarball())) { + dockerRule.getClient().loadImageAsyncCmd(uploadStream).exec(new LoadImageCallback()).awaitMessage(); + } + + final Image image = findImageWithId(expectedImageId, dockerRule.getClient().listImagesCmd().exec()); + + assertThat("Can't find expected image after loading from a tar archive!", image, notNullValue()); + assertThat("Image after loading from a tar archive has wrong tags!", + asList(image.getRepoTags()), equalTo(singletonList("docker-java/load:1.0"))); + } + + private Image findImageWithId(final String id, final List images) { + for (Image image : images) { + if (id.equals(image.getId())) { + return image; + } + } + return null; + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/LogContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/LogContainerCmdIT.java new file mode 100644 index 000000000..8593d6ccf --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/LogContainerCmdIT.java @@ -0,0 +1,264 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.api.model.StreamType; +import com.github.dockerjava.utils.LogContainerTestCallback; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; +import java.util.List; +import java.util.concurrent.Callable; +import java.util.concurrent.CopyOnWriteArrayList; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.TimeUnit; +import java.util.concurrent.atomic.AtomicBoolean; +import java.util.stream.Collectors; +import java.util.stream.LongStream; + +import static org.awaitility.Awaitility.await; +import static org.hamcrest.CoreMatchers.everyItem; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.hasSize; +import static org.hamcrest.Matchers.hasToString; +import static org.hamcrest.Matchers.not; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertTrue; + +public class LogContainerCmdIT extends CmdIT { + private static final Logger LOG = LoggerFactory.getLogger(LogContainerCmdIT.class); + + @Test + public void asyncLogContainerWithTtyEnabled() throws Exception { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("/bin/sh", "-c", "while true; do echo hello; sleep 1; done") + .withTty(true) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()) + .exec(); + + LogContainerTestCallback loggingCallback = new LogContainerTestCallback(true); + + // this essentially test the since=0 case + dockerRule.getClient().logContainerCmd(container.getId()) + .withStdErr(true) + .withStdOut(true) + .withFollowStream(true) + .withTailAll() + .exec(loggingCallback); + + loggingCallback.awaitCompletion(3, TimeUnit.SECONDS); + + assertTrue(loggingCallback.toString().contains("hello")); + + assertEquals(StreamType.RAW, loggingCallback.getCollectedFrames().get(0).getStreamType()); + } + + @Test + public void asyncLogContainerWithTtyDisabled() throws Exception { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("/bin/sh", "-c", "while true; do echo hello; sleep 1; done") + .withTty(false) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()) + .exec(); + + LogContainerTestCallback loggingCallback = new LogContainerTestCallback(true); + + // this essentially test the since=0 case + dockerRule.getClient().logContainerCmd(container.getId()) + .withStdErr(true) + .withStdOut(true) + .withFollowStream(true) + .withTailAll() + .exec(loggingCallback); + + loggingCallback.awaitCompletion(3, TimeUnit.SECONDS); + + assertTrue(loggingCallback.toString().contains("hello")); + + assertEquals(StreamType.STDOUT, loggingCallback.getCollectedFrames().get(0).getStreamType()); + } + + @Test + public void asyncLogNonExistingContainer() throws Exception { + + LogContainerTestCallback loggingCallback = new LogContainerTestCallback() { + @Override + public void onError(Throwable throwable) { + + assertEquals(NotFoundException.class.getName(), throwable.getClass().getName()); + + try { + // close the callback to prevent the call to onComplete + close(); + } catch (IOException e) { + throw new RuntimeException(); + } + + super.onError(throwable); + } + + public void onComplete() { + super.onComplete(); + throw new AssertionError("expected NotFoundException"); + }; + }; + + dockerRule.getClient().logContainerCmd("non-existing").withStdErr(true).withStdOut(true).exec(loggingCallback) + .awaitCompletion(); + } + + @Test + public void asyncMultipleLogContainer() throws Exception { + + String snippet = "hello world"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("/bin/echo", snippet) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + int exitCode = dockerRule.getClient().waitContainerCmd(container.getId()) + .start() + .awaitStatusCode(); + + assertThat(exitCode, equalTo(0)); + + LogContainerTestCallback loggingCallback = new LogContainerTestCallback(); + + dockerRule.getClient().logContainerCmd(container.getId()) + .withStdErr(true) + .withStdOut(true) + .exec(loggingCallback); + + loggingCallback.close(); + + loggingCallback = new LogContainerTestCallback(); + + dockerRule.getClient().logContainerCmd(container.getId()) + .withStdErr(true) + .withStdOut(true) + .exec(loggingCallback); + + loggingCallback.close(); + + loggingCallback = new LogContainerTestCallback(); + + dockerRule.getClient().logContainerCmd(container.getId()) + .withStdErr(true) + .withStdOut(true) + .exec(loggingCallback); + + loggingCallback.awaitCompletion(); + + assertTrue(loggingCallback.toString().contains(snippet)); + } + + @Test + public void asyncLogContainerWithSince() throws Exception { + String snippet = "hello world"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("/bin/echo", snippet) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + int timestamp = (int) (System.currentTimeMillis() / 1000); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + int exitCode = dockerRule.getClient().waitContainerCmd(container.getId()) + .start() + .awaitStatusCode(); + + assertThat(exitCode, equalTo(0)); + + LogContainerTestCallback loggingCallback = new LogContainerTestCallback(); + + dockerRule.getClient().logContainerCmd(container.getId()) + .withStdErr(true) + .withStdOut(true) + .withSince(timestamp) + .withUntil(timestamp + 1000) + .exec(loggingCallback); + + loggingCallback.awaitCompletion(); + + assertThat(loggingCallback.toString(), containsString(snippet)); + } + + @Test(timeout = 10_000) + public void simultaneousCommands() throws Exception { + // Create a new client to not affect other tests + DockerClient client = dockerRule.newClient(); + CreateContainerResponse container = client.createContainerCmd("busybox") + .withCmd("/bin/sh", "-c", "echo hello world; sleep infinity") + .exec(); + + client.startContainerCmd(container.getId()).exec(); + + // Simulate 100 simultaneous connections + int connections = 100; + + ExecutorService executor = Executors.newFixedThreadPool(connections); + try { + List firstFrames = new CopyOnWriteArrayList<>(); + executor.invokeAll( + LongStream.range(0, connections).>mapToObj(__ -> { + return () -> { + return client.logContainerCmd(container.getId()) + .withStdOut(true) + .withFollowStream(true) + .exec(new ResultCallback.Adapter() { + + final AtomicBoolean first = new AtomicBoolean(true); + + @Override + public void onNext(Frame object) { + if (first.compareAndSet(true, false)) { + firstFrames.add(object); + } + super.onNext(object); + } + }); + }; + }).collect(Collectors.toList()) + ); + + await().atMost(5, TimeUnit.SECONDS).untilAsserted(() -> { + assertThat(firstFrames, hasSize(connections)); + }); + + assertThat(firstFrames, everyItem(hasToString("STDOUT: hello world"))); + } finally { + executor.shutdownNow(); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/PauseCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/PauseCmdIT.java new file mode 100644 index 000000000..3647e713a --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/PauseCmdIT.java @@ -0,0 +1,74 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.utils.ContainerUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; + +public class PauseCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(PauseCmdIT.class); + + @Test + public void pauseRunningContainer() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + ContainerUtils.startContainer(dockerRule.getClient(), container); + + ContainerUtils.pauseContainer(dockerRule.getClient(), container); + } + + @Test(expected = NotFoundException.class) + public void pauseNonExistingContainer() { + + dockerRule.getClient().pauseContainerCmd("non-existing").exec(); + } + + @Test(expected = DockerException.class) + public void pauseStoppedContainer() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + ContainerUtils.startContainer(dockerRule.getClient(), container); + + ContainerUtils.stopContainer(dockerRule.getClient(), container); + + dockerRule.getClient().pauseContainerCmd(container.getId()).exec(); + } + + @Test(expected = DockerException.class) + public void pausePausedContainer() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + ContainerUtils.startContainer(dockerRule.getClient(), container); + + ContainerUtils.pauseContainer(dockerRule.getClient(), container); + + dockerRule.getClient().pauseContainerCmd(container.getId()).exec(); + } + + @Test(expected = DockerException.class) + public void pauseCreatedContainer() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().pauseContainerCmd(container.getId()).exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/PingCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/PingCmdIT.java new file mode 100644 index 000000000..dde81463c --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/PingCmdIT.java @@ -0,0 +1,13 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.exception.DockerException; +import org.junit.Test; + +public class PingCmdIT extends CmdIT { + + @Test + public void ping() throws DockerException { + dockerRule.getClient().pingCmd().exec(); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/PullImageCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/PullImageCmdIT.java new file mode 100644 index 000000000..3b8dde3ff --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/PullImageCmdIT.java @@ -0,0 +1,165 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.InspectImageResponse; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.Info; +import com.github.dockerjava.core.RemoteApiVersion; +import com.github.dockerjava.junit.PrivateRegistryRule; +import org.junit.ClassRule; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.concurrent.TimeUnit; + +import static com.github.dockerjava.utils.TestUtils.getVersion; +import static com.github.dockerjava.utils.TestUtils.isNotSwarm; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.lessThanOrEqualTo; +import static org.hamcrest.Matchers.notNullValue; + +public class PullImageCmdIT extends CmdIT { + private static final Logger LOG = LoggerFactory.getLogger(PullImageCmdIT.class); + + @ClassRule + public static PrivateRegistryRule REGISTRY = new PrivateRegistryRule(); + + @Rule + public ExpectedException exception = ExpectedException.none(); + + @Test + public void testPullImage() throws Exception { + Info info = dockerRule.getClient().infoCmd().exec(); + LOG.info("Client info: {}", info.toString()); + + int imgCount = info.getImages(); + LOG.info("imgCount1: {}", imgCount); + + // This should be an image that is not used by other repositories + // already + // pulled down, preferably small in size. If tag is not used pull will + // download all images in that repository but tmpImgs will only + // deleted 'latest' image but not images with other tags + String testImage = "alpine:3.17"; + + LOG.info("Removing image: {}", testImage); + + try { + dockerRule.getClient().removeImageCmd(testImage).withForce(true).exec(); + } catch (NotFoundException e) { + // just ignore if not exist + } + + info = dockerRule.getClient().infoCmd().exec(); + LOG.info("Client info: {}", info.toString()); + + imgCount = info.getImages(); + LOG.info("imgCount2: {}", imgCount); + + LOG.info("Pulling image: {}", testImage); + + dockerRule.getClient().pullImageCmd(testImage) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + + info = dockerRule.getClient().infoCmd().exec(); + LOG.info("Client info after pull, {}", info.toString()); + + assertThat(imgCount, lessThanOrEqualTo(info.getImages())); + + InspectImageResponse inspectImageResponse = dockerRule.getClient().inspectImageCmd(testImage).exec(); + LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + assertThat(inspectImageResponse, notNullValue()); + } + + @Test + public void testPullNonExistingImage() throws Exception { + if (isNotSwarm(dockerRule.getClient()) && getVersion(dockerRule.getClient()) + .isGreaterOrEqual(RemoteApiVersion.VERSION_1_26)) { + exception.expect(NotFoundException.class); + } else { + exception.expect(DockerClientException.class); + } + + // stream needs to be fully read in order to close the underlying connection + dockerRule.getClient().pullImageCmd("xvxcv/foo") + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + } + + @Test + public void testPullImageWithValidAuth() throws Exception { + AuthConfig authConfig = REGISTRY.getAuthConfig(); + + String imgName = REGISTRY.createPrivateImage("pull-image-with-valid-auth"); + + // stream needs to be fully read in order to close the underlying connection + dockerRule.getClient().pullImageCmd(imgName) + .withAuthConfig(authConfig) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + } + + @Test + public void testPullImageWithValidAuthAndEmail() throws Exception { + AuthConfig authConfig = REGISTRY.getAuthConfig().withEmail("foo@bar.de"); + + String imgName = REGISTRY.createPrivateImage("pull-image-with-valid-auth"); + + // stream needs to be fully read in order to close the underlying connection + dockerRule.getClient().pullImageCmd(imgName) + .withAuthConfig(authConfig) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + } + + @Test + public void testPullImageWithNoAuth() throws Exception { + AuthConfig authConfig = REGISTRY.getAuthConfig(); + + String imgName = REGISTRY.createPrivateImage("pull-image-with-no-auth"); + + if (isNotSwarm(dockerRule.getClient()) && getVersion(dockerRule.getClient()) + .isGreaterOrEqual(RemoteApiVersion.VERSION_1_30)) { + exception.expect(DockerException.class); + } else { + exception.expect(DockerClientException.class); + } + + // stream needs to be fully read in order to close the underlying connection + dockerRule.getClient().pullImageCmd(imgName) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + } + + + @Test + public void testPullImageWithInvalidAuth() throws Exception { + AuthConfig authConfig = REGISTRY.getAuthConfig(); + AuthConfig invalidAuthConfig = new AuthConfig() + .withUsername("testuser") + .withPassword("testwrongpassword") + .withEmail("foo@bar.de") + .withRegistryAddress(authConfig.getRegistryAddress()); + + String imgName = REGISTRY.createPrivateImage("pull-image-with-invalid-auth"); + + if (isNotSwarm(dockerRule.getClient()) && getVersion(dockerRule.getClient()) + .isGreaterOrEqual(RemoteApiVersion.VERSION_1_30)) { + exception.expect(DockerException.class); + } else { + exception.expect(DockerClientException.class); + } + + // stream needs to be fully read in order to close the underlying connection + dockerRule.getClient().pullImageCmd(imgName) + .withAuthConfig(invalidAuthConfig) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/PushImageCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/PushImageCmdIT.java new file mode 100644 index 000000000..00cd11d51 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/PushImageCmdIT.java @@ -0,0 +1,129 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.core.RemoteApiVersion; +import com.github.dockerjava.junit.PrivateRegistryRule; +import org.junit.Before; +import org.junit.ClassRule; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.UUID; +import java.util.concurrent.TimeUnit; + +import static com.github.dockerjava.utils.TestUtils.getVersion; +import static com.github.dockerjava.utils.TestUtils.isNotSwarm; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; + +public class PushImageCmdIT extends CmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(PushImageCmdIT.class); + + @ClassRule + public static PrivateRegistryRule REGISTRY = new PrivateRegistryRule(); + + @Rule + public ExpectedException exception = ExpectedException.none(); + private AuthConfig authConfig; + + @Before + public void beforeTest() { + authConfig = REGISTRY.getAuthConfig(); + } + + @Test + public void pushLatest() throws Exception { + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("true").exec(); + + LOG.info("Created container {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + LOG.info("Committing container: {}", container.toString()); + String imgName = authConfig.getRegistryAddress() + "/push-latest"; + String imageId = dockerRule.getClient().commitCmd(container.getId()) + .withRepository(imgName) + .exec(); + + // we have to block until image is pushed + dockerRule.getClient().pushImageCmd(imgName) + .withAuthConfig(authConfig) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + + LOG.info("Removing image: {}", imageId); + dockerRule.getClient().removeImageCmd(imageId).exec(); + + dockerRule.getClient().pullImageCmd(imgName) + .withTag("latest") + .withAuthConfig(authConfig) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + } + + @Test + public void pushNonExistentImage() throws Exception { + + if (isNotSwarm(dockerRule.getClient()) && getVersion(dockerRule.getClient()) + .isGreaterOrEqual(RemoteApiVersion.VERSION_1_24)) { + exception.expect(DockerClientException.class); + } else { + exception.expect(NotFoundException.class); + } + + dockerRule.getClient().pushImageCmd(UUID.randomUUID().toString().replace("-", "")) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); // exclude infinite await sleep + + } + + @Test + public void testPushImageWithValidAuth() throws Exception { + String imgName = REGISTRY.createTestImage("push-image-with-valid-auth"); + + // stream needs to be fully read in order to close the underlying connection + dockerRule.getClient().pushImageCmd(imgName) + .withAuthConfig(authConfig) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + } + + @Test + public void testPushImageWithNoAuth() throws Exception { + String imgName = REGISTRY.createTestImage("push-image-with-no-auth"); + + exception.expect(DockerClientException.class); + + // stream needs to be fully read in order to close the underlying connection + dockerRule.getClient().pushImageCmd(imgName) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + } + + @Test + public void testPushImageWithInvalidAuth() throws Exception { + AuthConfig invalidAuthConfig = new AuthConfig() + .withUsername("testuser") + .withPassword("testwrongpassword") + .withEmail("foo@bar.de") + .withRegistryAddress(authConfig.getRegistryAddress()); + + String imgName = REGISTRY.createTestImage("push-image-with-invalid-auth"); + + exception.expect(DockerClientException.class); + + // stream needs to be fully read in order to close the underlying connection + dockerRule.getClient().pushImageCmd(imgName) + .withAuthConfig(invalidAuthConfig) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveContainerCmdImplIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveContainerCmdImplIT.java new file mode 100644 index 000000000..408098148 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveContainerCmdImplIT.java @@ -0,0 +1,49 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Container; +import org.hamcrest.Matcher; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.hasItem; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.startsWith; +import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; + +public class RemoveContainerCmdImplIT extends CmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(RemoveContainerCmdImplIT.class); + + + @Test + public void removeContainer() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("true").exec(); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + dockerRule.getClient().waitContainerCmd(container.getId()).start().awaitStatusCode(); + + LOG.info("Removing container: {}", container.getId()); + dockerRule.getClient().removeContainerCmd(container.getId()).exec(); + + List containers2 = dockerRule.getClient().listContainersCmd().withShowAll(true).exec(); + + Matcher matcher = not(hasItem(hasField("id", startsWith(container.getId())))); + assertThat(containers2, matcher); + + } + + @Test(expected = NotFoundException.class) + public void removeNonExistingContainer() throws DockerException { + + dockerRule.getClient().removeContainerCmd("non-existing").exec(); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveImageCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveImageCmdIT.java new file mode 100644 index 000000000..00d2e6cc1 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveImageCmdIT.java @@ -0,0 +1,55 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Container; +import org.hamcrest.Matcher; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.hasItem; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.startsWith; +import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; + +public class RemoveImageCmdIT extends CmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(RemoveImageCmdIT.class); + + @Test + public void removeImage() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + LOG.info("Committing container {}", container.toString()); + String imageId = dockerRule.getClient().commitCmd(container.getId()).exec(); + + dockerRule.getClient().stopContainerCmd(container.getId()).exec(); + dockerRule.getClient().removeContainerCmd(container.getId()).exec(); + + LOG.info("Removing image: {}", imageId); + dockerRule.getClient().removeImageCmd(imageId).exec(); + + List containers = dockerRule.getClient().listContainersCmd().withShowAll(true).exec(); + + Matcher matcher = not(hasItem(hasField("id", startsWith(imageId)))); + assertThat(containers, matcher); + } + + @Test(expected = NotFoundException.class) + public void removeNonExistingImage() throws DockerException { + + dockerRule.getClient().removeImageCmd("non-existing").exec(); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveNetworkCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveNetworkCmdIT.java new file mode 100644 index 000000000..974022c20 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveNetworkCmdIT.java @@ -0,0 +1,51 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateNetworkResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Network; +import org.hamcrest.Matcher; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +import static com.github.dockerjava.junit.DockerAssume.assumeNotSwarm; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.hasItem; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.startsWith; +import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; + +/** + * @author Kanstantsin Shautsou + */ +public class RemoveNetworkCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(RemoveNetworkCmdIT.class); + + @Test + public void removeNetwork() throws DockerException { + assumeNotSwarm("Swarm has no network", dockerRule); + + CreateNetworkResponse network = dockerRule.getClient().createNetworkCmd() + .withName("test-network") + .exec(); + + LOG.info("Removing network: {}", network.getId()); + dockerRule.getClient().removeNetworkCmd(network.getId()).exec(); + + List networks = dockerRule.getClient().listNetworksCmd().exec(); + + Matcher matcher = not(hasItem(hasField("id", startsWith(network.getId())))); + assertThat(networks, matcher); + + } + + @Test(expected = NotFoundException.class) + public void removeNonExistingContainer() throws DockerException { + assumeNotSwarm("Swarm has no network", dockerRule); + + dockerRule.getClient().removeNetworkCmd("non-existing").exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveVolumeCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveVolumeCmdIT.java new file mode 100644 index 000000000..6d0fdf981 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/RemoveVolumeCmdIT.java @@ -0,0 +1,36 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateVolumeResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import org.junit.Test; + +import java.util.Collections; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.equalTo; + +public class RemoveVolumeCmdIT extends CmdIT { + + + @Test(expected = NotFoundException.class) + public void removeVolume() throws DockerException { + + String volumeName = "volume1"; + + CreateVolumeResponse createVolumeResponse = dockerRule.getClient().createVolumeCmd() + .withName(volumeName) + .withDriver("local") + .withLabels(Collections.singletonMap("is-timelord", "yes")).exec(); + + assertThat(createVolumeResponse.getName(), equalTo(volumeName)); + assertThat(createVolumeResponse.getDriver(), equalTo("local")); + assertThat(createVolumeResponse.getLabels(), equalTo(Collections.singletonMap("is-timelord", "yes"))); + assertThat(createVolumeResponse.getMountpoint(), containsString(volumeName)); + + dockerRule.getClient().removeVolumeCmd(volumeName).exec(); + + dockerRule.getClient().inspectVolumeCmd(volumeName).exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/RenameContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/RenameContainerCmdIT.java new file mode 100644 index 000000000..c50ebcf43 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/RenameContainerCmdIT.java @@ -0,0 +1,53 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.junit.Assert.assertNotEquals; + +public class RenameContainerCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(RenameContainerCmdIT.class); + + @Test + public void renameContainer() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); + + String name1 = inspectContainerResponse.getName(); + + dockerRule.getClient().renameContainerCmd(container.getId()) + .withName("renameContainer") + .exec(); + + InspectContainerResponse inspectContainerResponse2 = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + LOG.info("Container Inspect After Rename: {}", inspectContainerResponse2.toString()); + + String name2 = inspectContainerResponse2.getName(); + + assertNotEquals(name1, name2); + + dockerRule.getClient().killContainerCmd(container.getId()).exec(); + } + + @Test(expected = NotFoundException.class) + public void renameExistingContainer() throws DockerException { + dockerRule.getClient().renameContainerCmd("non-existing") + .withName("renameExistingContainer") + .exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/ResizeContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/ResizeContainerCmdIT.java new file mode 100644 index 000000000..eeb79b2ff --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/ResizeContainerCmdIT.java @@ -0,0 +1,39 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.security.SecureRandom; +import java.util.concurrent.TimeUnit; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; + +public class ResizeContainerCmdIT extends CmdIT { + private static final Logger LOG = LoggerFactory.getLogger(ResizeContainerCmdIT.class); + + private static final int TTY_HEIGHT = 30; + private static final int TTY_WIDTH = 120; + + @Test + public void resizeContainerTtyTest() { + String containerName = "generated_" + new SecureRandom().nextInt(); + + // wait until tty size changed to target size + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withUser("root") + .withCmd("sh", "-c", String.format("until stty size | grep '%d %d'; do : ; done", TTY_HEIGHT, TTY_WIDTH)) + .withName(containerName).withTty(true).withStdinOpen(true).exec(); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + dockerRule.getClient().resizeContainerCmd(container.getId()).withSize(TTY_HEIGHT, TTY_WIDTH).exec(); + + int exitCode = dockerRule.getClient().waitContainerCmd(container.getId()).start() + .awaitStatusCode(10, TimeUnit.SECONDS); + + LOG.info("Container exit code: {}", exitCode); + + assertThat(exitCode, equalTo(0)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/ResizeExecCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/ResizeExecCmdIT.java new file mode 100644 index 000000000..9e5c9b65f --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/ResizeExecCmdIT.java @@ -0,0 +1,48 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.ExecCreateCmdResponse; +import com.github.dockerjava.core.command.ExecStartResultCallback; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.security.SecureRandom; +import java.util.concurrent.TimeUnit; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; + + +public class ResizeExecCmdIT extends CmdIT { + private static final Logger LOG = LoggerFactory.getLogger(ResizeExecCmdIT.class); + + private static final int TTY_HEIGHT = 30; + private static final int TTY_WIDTH = 120; + + @Test + public void resizeExecInstanceTtyTest() throws Exception { + String containerName = "generated_" + new SecureRandom().nextInt(); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withUser("root") + .withCmd("sleep", "9999").withName(containerName).exec(); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + // wait until tty size changed to target size + ExecCreateCmdResponse execCreateCmdResponse = dockerRule.getClient().execCreateCmd(container.getId()).withTty(true) + .withAttachStdout(true).withAttachStderr(true) + .withCmd("sh", "-c", String.format("until stty size | grep '%d %d'; do : ; done", TTY_HEIGHT, TTY_WIDTH)).exec(); + + final ExecStartResultCallback execStartResultCallback = new ExecStartResultCallback(System.out, System.err); + + dockerRule.getClient().execStartCmd(execCreateCmdResponse.getId()).exec(execStartResultCallback).awaitStarted(); + + dockerRule.getClient().resizeExecCmd(execCreateCmdResponse.getId()).withSize(TTY_HEIGHT, TTY_WIDTH).exec(); + + // time out, exec instance resize failed + boolean waitResult = execStartResultCallback.awaitCompletion(10, TimeUnit.SECONDS); + + assertThat(waitResult, equalTo(true)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/RestartContainerCmdImplIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/RestartContainerCmdImplIT.java new file mode 100644 index 000000000..592c9c650 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/RestartContainerCmdImplIT.java @@ -0,0 +1,93 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.core.DefaultDockerClientConfig; +import com.github.dockerjava.core.RemoteApiVersion; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static com.github.dockerjava.junit.DockerMatchers.isGreaterOrEqual; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.not; +import static org.junit.Assume.assumeThat; + +public class RestartContainerCmdImplIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(RestartContainerCmdImplIT.class); + + @Test + public void restartContainer() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); + + String startTime = inspectContainerResponse.getState().getStartedAt(); + + dockerRule.getClient().restartContainerCmd(container.getId()).withtTimeout(2).exec(); + + InspectContainerResponse inspectContainerResponse2 = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + LOG.info("Container Inspect After Restart: {}", inspectContainerResponse2.toString()); + + String startTime2 = inspectContainerResponse2.getState().getStartedAt(); + + assertThat(startTime, not(equalTo(startTime2))); + + assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(true))); + + dockerRule.getClient().killContainerCmd(container.getId()).exec(); + } + + @Test + public void restartContainerWithSignal() throws Exception { + assumeThat("API version should be >= 1.42", dockerRule, isGreaterOrEqual(RemoteApiVersion.VERSION_1_42)); + + DefaultDockerClientConfig dockerClientConfig = DefaultDockerClientConfig.createDefaultConfigBuilder() + .withApiVersion(RemoteApiVersion.VERSION_1_44) + .withRegistryUrl("https://index.docker.io/v1/") + .build(); + try (DockerClient dockerClient = createDockerClient(dockerClientConfig)) { + String expectedUserSignal = "10"; + String initialCommandWithTrap = "trap 'echo \"exit trapped\"' %s; while :; do sleep 1; done"; + final String containerId = dockerClient + .createContainerCmd(DEFAULT_IMAGE) + .withCmd( + "/bin/sh", + "-c", + String.format(initialCommandWithTrap, expectedUserSignal)) + .exec() + .getId(); + assertThat(containerId, not(is(emptyString()))); + dockerClient.startContainerCmd(containerId).exec(); + + // Restart container without signal + dockerClient.restartContainerCmd(containerId).exec(); + String log = dockerRule.containerLog(containerId); + assertThat(log.trim(), emptyString()); + + dockerClient.restartContainerCmd(containerId).withSignal(expectedUserSignal).exec(); + log = dockerRule.containerLog(containerId); + assertThat(log.trim(), is("exit trapped")); + + dockerClient.removeContainerCmd(containerId).withForce(true).withRemoveVolumes(true).exec(); + } + } + + @Test(expected = NotFoundException.class) + public void restartNonExistingContainer() throws DockerException { + + dockerRule.getClient().restartContainerCmd("non-existing").exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/SaveImageCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/SaveImageCmdIT.java new file mode 100644 index 000000000..cb5a4666c --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/SaveImageCmdIT.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.cmd; + +import org.apache.commons.io.IOUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.InputStream; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.not; + +public class SaveImageCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(SaveImageCmdIT.class); + + @Test + public void saveImage() throws Exception { + + try ( + InputStream inputStream = dockerRule.getClient().saveImageCmd("busybox").exec(); + InputStream image = IOUtils.toBufferedInputStream(inputStream) + ) { + assertThat(image.read(), not(-1)); + } + + try ( + InputStream inputStream = dockerRule.getClient().saveImageCmd("busybox").withTag("latest").exec(); + InputStream image2 = IOUtils.toBufferedInputStream(inputStream) + ) { + assertThat(image2.read(), not(-1)); + } + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/SaveImagesCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/SaveImagesCmdIT.java new file mode 100644 index 000000000..86b246029 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/SaveImagesCmdIT.java @@ -0,0 +1,52 @@ +package com.github.dockerjava.cmd; + +import org.apache.commons.io.IOUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.InputStream; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.not; + +public class SaveImagesCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(SaveImagesCmdIT.class); + + @Test + public void saveNoImages() throws Exception { + try ( + InputStream inputStream = dockerRule.getClient().saveImagesCmd().exec(); + InputStream image = IOUtils.toBufferedInputStream(inputStream) + ){ + assertThat(image.read(), not(-1)); + } + + } + + @Test + public void saveImagesWithNameAndTag() throws Exception { + try ( + InputStream inputStream = dockerRule.getClient().saveImagesCmd().withImage("busybox", "latest").exec(); + InputStream image = IOUtils.toBufferedInputStream(inputStream) + ) { + assertThat(image.read(), not(-1)); + } + + } + + @Test + public void saveMultipleImages() throws Exception { + try ( + InputStream inputStream = dockerRule.getClient().saveImagesCmd() + // Not a real life use-case but "busybox" is the only one I dare to assume is really there. + .withImage("busybox", "latest") + .withImage("busybox", "latest") + .exec(); + InputStream image = IOUtils.toBufferedInputStream(inputStream) + ) { + assertThat(image.read(), not(-1)); + } + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/SearchImagesCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/SearchImagesCmdIT.java new file mode 100644 index 000000000..26dcebbeb --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/SearchImagesCmdIT.java @@ -0,0 +1,64 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.SearchItem; +import org.hamcrest.Matcher; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +import static ch.lambdaj.Lambda.filter; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.*; +import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; + +public class SearchImagesCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(SearchImagesCmdIT.class); + + @Test + public void searchImages() throws DockerException { + List dockerSearch = dockerRule.getClient().searchImagesCmd("busybox").exec(); + LOG.info("Search returned {}", dockerSearch.toString()); + + Matcher matcher = hasItem(hasField("name", equalTo("busybox"))); + assertThat(dockerSearch, matcher); + + assertThat(filter(hasField("name", is("busybox")), dockerSearch).size(), equalTo(1)); + } + + @Test(expected = IllegalArgumentException.class) + public void searchImagesWithInvalidMinimumLimit() throws DockerException { + dockerRule.getClient().searchImagesCmd("busybox").withLimit(0).exec(); + } + + @Test(expected = IllegalArgumentException.class) + public void searchImagesWithInvalidMaximumLimit() throws DockerException { + dockerRule.getClient().searchImagesCmd("busybox").withLimit(101).exec(); + } + + @Test + public void searchImagesWithValidMinimumLimit() throws DockerException { + List dockerSearch = dockerRule.getClient().searchImagesCmd("busybox").withLimit(1).exec(); + LOG.info("Search returned {}", dockerSearch.toString()); + + Matcher matcher = hasItem(hasField("name", equalTo("busybox"))); + assertThat(dockerSearch, matcher); + + assertThat(filter(hasField("name", is("busybox")), dockerSearch).size(), equalTo(1)); + + assertThat(dockerSearch.size(), equalTo(1)); + } + + @Test + public void searchImagesWithValidMaximumLimit() throws DockerException { + List dockerSearch = dockerRule.getClient().searchImagesCmd("busybox").withLimit(1).exec(); + LOG.info("Search returned {}", dockerSearch.toString()); + + Matcher matcher = hasItem(hasField("name", equalTo("busybox"))); + assertThat(dockerSearch, matcher); + + assertThat(filter(hasField("name", is("busybox")), dockerSearch).size(), equalTo(1)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/StartContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/StartContainerCmdIT.java new file mode 100644 index 000000000..5d41889ca --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/StartContainerCmdIT.java @@ -0,0 +1,570 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.InternalServerErrorException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Bind; +import com.github.dockerjava.api.model.Device; +import com.github.dockerjava.api.model.ExposedPort; +import com.github.dockerjava.api.model.Link; +import com.github.dockerjava.api.model.Ports; +import com.github.dockerjava.api.model.Ports.Binding; +import com.github.dockerjava.api.model.RestartPolicy; +import com.github.dockerjava.api.model.Volume; +import com.github.dockerjava.api.model.VolumesFrom; +import net.jcip.annotations.NotThreadSafe; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Arrays; +import java.util.List; +import java.util.UUID; + +import static com.github.dockerjava.api.model.AccessMode.ro; +import static com.github.dockerjava.api.model.Capability.MKNOD; +import static com.github.dockerjava.api.model.Capability.NET_ADMIN; +import static com.github.dockerjava.api.model.HostConfig.newHostConfig; +import static com.github.dockerjava.junit.DockerMatchers.mountedVolumes; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.contains; +import static org.hamcrest.Matchers.containsInAnyOrder; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.hasSize; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.Matchers.startsWith; + +@NotThreadSafe +public class StartContainerCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(StartContainerCmdIT.class); + + @Test + public void startContainerWithVolumes() { + + // see http://docs.docker.io/use/working_with_volumes/ + Volume volume1 = new Volume("/opt/webapp1"); + + Volume volume2 = new Volume("/opt/webapp2"); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withVolumes(volume1, volume2) + .withCmd("true") + .withHostConfig(newHostConfig() + .withBinds(new Bind("/tmp/webapp1", volume1, ro), new Bind("/tmp/webapp2", volume2))) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getConfig().getVolumes().keySet(), contains("/opt/webapp1", "/opt/webapp2")); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + dockerRule.getClient().waitContainerCmd(container.getId()).start().awaitStatusCode(); + + inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse, mountedVolumes(containsInAnyOrder(volume1, volume2))); + + final List mounts = inspectContainerResponse.getMounts(); + + assertThat(mounts, hasSize(2)); + + final InspectContainerResponse.Mount mount1 = new InspectContainerResponse.Mount() + .withRw(false).withMode("ro").withDestination(volume1).withSource("/tmp/webapp1"); + final InspectContainerResponse.Mount mount2 = new InspectContainerResponse.Mount() + .withRw(true).withMode("rw").withDestination(volume2).withSource("/tmp/webapp2"); + + assertThat(mounts, containsInAnyOrder(mount1, mount2)); + } + + @Test + public void startContainerWithVolumesFrom() throws DockerException { + + Volume volume1 = new Volume("/opt/webapp1"); + Volume volume2 = new Volume("/opt/webapp2"); + + String container1Name = UUID.randomUUID().toString(); + + CreateContainerResponse container1 = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999") + .withName(container1Name) + .withHostConfig(newHostConfig() + .withBinds(new Bind("/tmp/webapp1", volume1), new Bind("/tmp/webapp2", volume2))) + .exec(); + LOG.info("Created container1 {}", container1.toString()); + + dockerRule.getClient().startContainerCmd(container1.getId()).exec(); + LOG.info("Started container1 {}", container1.toString()); + + InspectContainerResponse inspectContainerResponse1 = dockerRule.getClient().inspectContainerCmd(container1.getId()) + .exec(); + + assertThat(inspectContainerResponse1, mountedVolumes(containsInAnyOrder(volume1, volume2))); + + CreateContainerResponse container2 = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999") + .withHostConfig(newHostConfig() + .withVolumesFrom(new VolumesFrom(container1Name))) + .exec(); + LOG.info("Created container2 {}", container2.toString()); + + dockerRule.getClient().startContainerCmd(container2.getId()).exec(); + LOG.info("Started container2 {}", container2.toString()); + + InspectContainerResponse inspectContainerResponse2 = dockerRule.getClient().inspectContainerCmd(container2.getId()) + .exec(); + + assertThat(inspectContainerResponse2, mountedVolumes(containsInAnyOrder(volume1, volume2))); + } + + @Test + public void startContainerWithDns() throws DockerException { + + String aDnsServer = "8.8.8.8"; + String anotherDnsServer = "8.8.4.4"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("true") + .withHostConfig(newHostConfig() + .withDns(aDnsServer, anotherDnsServer)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDns()), + contains(aDnsServer, anotherDnsServer)); + } + + @Test + public void startContainerWithDnsSearch() throws DockerException { + + String dnsSearch = "example.com"; + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("true") + .withHostConfig(newHostConfig() + .withDnsSearch(dnsSearch)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDnsSearch()), contains(dnsSearch)); + } + + @Test + public void startContainerWithPortBindings() throws DockerException { + int baseport = 20_000; + + ExposedPort tcp22 = ExposedPort.tcp(22); + ExposedPort tcp23 = ExposedPort.tcp(23); + + Ports portBindings = new Ports(); + portBindings.bind(tcp22, Binding.bindPort(baseport + 22)); + portBindings.bind(tcp23, Binding.bindPort(baseport + 23)); + portBindings.bind(tcp23, Binding.bindPort(baseport + 24)); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("true") + .withExposedPorts(tcp22, tcp23) + .withHostConfig(newHostConfig() + .withPortBindings(portBindings)).exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getExposedPorts()), contains(tcp22, tcp23)); + + assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp22)[0], + is(equalTo(Binding.bindPort(baseport + 22)))); + + assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[0], + is(equalTo(Binding.bindPort(baseport + 23)))); + + assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[1], + is(equalTo(Binding.bindPort(baseport + 24)))); + + } + + @Test + public void startContainerWithRandomPortBindings() throws DockerException { + + ExposedPort tcp22 = ExposedPort.tcp(22); + ExposedPort tcp23 = ExposedPort.tcp(23); + + Ports portBindings = new Ports(); + portBindings.bind(tcp22, Binding.empty()); + portBindings.bind(tcp23, Binding.empty()); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999") + .withExposedPorts(tcp22, tcp23).withHostConfig(newHostConfig() + .withPortBindings(portBindings) + .withPublishAllPorts(true)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(Arrays.asList(inspectContainerResponse.getConfig().getExposedPorts()), contains(tcp22, tcp23)); + + assertThat(inspectContainerResponse.getNetworkSettings().getPorts().getBindings().get(tcp22)[0].getHostPortSpec(), + is(not(equalTo(String.valueOf(tcp22.getPort()))))); + + assertThat(inspectContainerResponse.getNetworkSettings().getPorts().getBindings().get(tcp23)[0].getHostPortSpec(), + is(not(equalTo(String.valueOf(tcp23.getPort()))))); + + } + + @Test(expected = InternalServerErrorException.class) + public void startContainerWithConflictingPortBindings() throws DockerException { + + ExposedPort tcp22 = ExposedPort.tcp(22); + ExposedPort tcp23 = ExposedPort.tcp(23); + + Ports portBindings = new Ports(); + portBindings.bind(tcp22, Binding.bindPort(11022)); + portBindings.bind(tcp23, Binding.bindPort(11022)); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("true") + .withExposedPorts(tcp22, tcp23).withHostConfig(newHostConfig() + .withPortBindings(portBindings)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + } + + @Test + public void startContainerWithLinkingDeprecated() throws DockerException { + String container1Name = "containerWithLink1"; + String container2Name = "containerWithLink2"; + dockerRule.ensureContainerRemoved(container1Name); + dockerRule.ensureContainerRemoved(container2Name); + + CreateContainerResponse container1 = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("sleep", "9999") + .withName(container1Name) + .exec(); + + LOG.info("Created container1 {}", container1.toString()); + assertThat(container1.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container1.getId()).exec(); + + InspectContainerResponse inspectContainerResponse1 = dockerRule.getClient().inspectContainerCmd(container1.getId()) + .exec(); + LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); + + assertThat(inspectContainerResponse1.getConfig(), is(notNullValue())); + assertThat(inspectContainerResponse1.getId(), not(is(emptyString()))); + assertThat(inspectContainerResponse1.getId(), startsWith(container1.getId())); + assertThat(inspectContainerResponse1.getName(), equalTo("/" + container1Name)); + assertThat(inspectContainerResponse1.getImageId(), not(is(emptyString()))); + assertThat(inspectContainerResponse1.getState(), is(notNullValue())); + assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); + + if (!inspectContainerResponse1.getState().getRunning()) { + assertThat(inspectContainerResponse1.getState().getExitCode(), is(equalTo(0))); + } + + CreateContainerResponse container2 = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999") + .withName(container2Name).withHostConfig(newHostConfig() + .withLinks(new Link(container1Name, container1Name + "Link"))) + .exec(); + + LOG.info("Created container2 {}", container2.toString()); + assertThat(container2.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container2.getId()).exec(); + + InspectContainerResponse inspectContainerResponse2 = dockerRule.getClient().inspectContainerCmd(container2.getId()) + .exec(); + LOG.info("Container2 Inspect: {}", inspectContainerResponse2.toString()); + + assertThat(inspectContainerResponse2.getConfig(), is(notNullValue())); + assertThat(inspectContainerResponse2.getId(), not(is(emptyString()))); + assertThat(inspectContainerResponse2.getHostConfig(), is(notNullValue())); + assertThat(inspectContainerResponse2.getHostConfig().getLinks(), is(notNullValue())); + assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[]{new Link(container1Name, + container1Name + "Link")})); + assertThat(inspectContainerResponse2.getId(), startsWith(container2.getId())); + assertThat(inspectContainerResponse2.getName(), equalTo("/" + container2Name)); + assertThat(inspectContainerResponse2.getImageId(), not(is(emptyString()))); + assertThat(inspectContainerResponse2.getState(), is(notNullValue())); + assertThat(inspectContainerResponse2.getState().getRunning(), is(true)); + + } + + @Test + public void startContainerWithLinking() throws DockerException { + String container1Name = "containerWithLinking1"; + String container2Name = "containerWithLinking2"; + dockerRule.ensureContainerRemoved(container1Name); + dockerRule.ensureContainerRemoved(container2Name); + + CreateContainerResponse container1 = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("sleep", "9999") + .withName(container1Name) + .exec(); + + LOG.info("Created container1 {}", container1.toString()); + assertThat(container1.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container1.getId()).exec(); + + InspectContainerResponse inspectContainerResponse1 = dockerRule.getClient().inspectContainerCmd(container1.getId()) + .exec(); + LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); + + assertThat(inspectContainerResponse1.getConfig(), is(notNullValue())); + assertThat(inspectContainerResponse1.getId(), not(is(emptyString()))); + assertThat(inspectContainerResponse1.getId(), startsWith(container1.getId())); + assertThat(inspectContainerResponse1.getName(), equalTo("/" + container1Name)); + assertThat(inspectContainerResponse1.getImageId(), not(is(emptyString()))); + assertThat(inspectContainerResponse1.getState(), is(notNullValue())); + assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); + + if (!inspectContainerResponse1.getState().getRunning()) { + assertThat(inspectContainerResponse1.getState().getExitCode(), is(equalTo(0))); + } + + CreateContainerResponse container2 = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999") + .withName(container2Name).withHostConfig(newHostConfig() + .withLinks(new Link(container1Name, container1Name + "Link"))) + .exec(); + + LOG.info("Created container2 {}", container2.toString()); + assertThat(container2.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container2.getId()).exec(); + + InspectContainerResponse inspectContainerResponse2 = dockerRule.getClient().inspectContainerCmd(container2.getId()) + .exec(); + LOG.info("Container2 Inspect: {}", inspectContainerResponse2.toString()); + + assertThat(inspectContainerResponse2.getConfig(), is(notNullValue())); + assertThat(inspectContainerResponse2.getId(), not(is(emptyString()))); + assertThat(inspectContainerResponse2.getHostConfig(), is(notNullValue())); + assertThat(inspectContainerResponse2.getHostConfig().getLinks(), is(notNullValue())); + assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[]{new Link(container1Name, + container1Name + "Link")})); + assertThat(inspectContainerResponse2.getId(), startsWith(container2.getId())); + assertThat(inspectContainerResponse2.getName(), equalTo("/" + container2Name)); + assertThat(inspectContainerResponse2.getImageId(), not(is(emptyString()))); + assertThat(inspectContainerResponse2.getState(), is(notNullValue())); + assertThat(inspectContainerResponse2.getState().getRunning(), is(true)); + + } + + @Test + public void startContainer() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd(new String[]{"top"}) + .exec(); + + LOG.info("Created container {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); + + assertThat(inspectContainerResponse.getConfig(), is(notNullValue())); + assertThat(inspectContainerResponse.getId(), not(is(emptyString()))); + + assertThat(inspectContainerResponse.getId(), startsWith(container.getId())); + + assertThat(inspectContainerResponse.getImageId(), not(is(emptyString()))); + assertThat(inspectContainerResponse.getState(), is(notNullValue())); + + assertThat(inspectContainerResponse.getState().getRunning(), is(true)); + + if (!inspectContainerResponse.getState().getRunning()) { + assertThat(inspectContainerResponse.getState().getExitCode(), is(equalTo(0))); + } + } + + @Test(expected = NotFoundException.class) + public void testStartNonExistingContainer() throws DockerException { + + dockerRule.getClient().startContainerCmd("non-existing").exec(); + } + + /** + * This tests support for --net option for the docker run command: --net="bridge" Set the Network mode for the container 'bridge': + * creates a new network stack for the container on the docker bridge 'none': no networking for this container 'container:': reuses + * another container network stack 'host': use the host network stack inside the container. Note: the host mode gives the container full + * access to local system services such as D-bus and is therefore considered insecure. + */ + @Test + public void startContainerWithNetworkMode() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("true") + .withHostConfig(newHostConfig() + .withNetworkMode("host")) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getHostConfig().getNetworkMode(), is(equalTo("host"))); + } + + @Test + public void startContainerWithCapAddAndCapDrop() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("sleep", "9999") + .withHostConfig(newHostConfig() + .withCapAdd(NET_ADMIN) + .withCapDrop(MKNOD)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getState().getRunning(), is(true)); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapAdd()), contains(NET_ADMIN)); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapDrop()), contains(MKNOD)); + } + + @Test + public void startContainerWithDevices() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("sleep", "9999") + .withHostConfig(newHostConfig() + .withDevices(new Device("rwm", "/dev/nulo", "/dev/zero"))) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getState().getRunning(), is(true)); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDevices()), contains(new Device("rwm", + "/dev/nulo", "/dev/zero"))); + } + + @Test + public void startContainerWithExtraHosts() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999") + .withHostConfig(newHostConfig() + .withExtraHosts("dockerhost:127.0.0.1")) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getState().getRunning(), is(true)); + + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getExtraHosts()), + contains("dockerhost:127.0.0.1")); + } + + @Test + public void startContainerWithRestartPolicy() throws DockerException { + + RestartPolicy restartPolicy = RestartPolicy.onFailureRestart(5); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("sleep", "9999") + .withHostConfig(newHostConfig() + .withRestartPolicy(restartPolicy)) + .exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + assertThat(inspectContainerResponse.getState().getRunning(), is(true)); + + assertThat(inspectContainerResponse.getHostConfig().getRestartPolicy(), is(equalTo(restartPolicy))); + } + + @Test + public void existingHostConfigIsPreservedByBlankStartCmd() throws DockerException { + + String dnsServer = "8.8.8.8"; + + // prepare a container with custom DNS + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withHostConfig(newHostConfig() + .withDns(dnsServer)) + .withCmd("true").exec(); + + LOG.info("Created container {}", container.toString()); + + assertThat(container.getId(), not(is(emptyString()))); + + // start container _without_any_customization_ (important!) + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + + // The DNS setting survived. + assertThat(inspectContainerResponse.getHostConfig().getDns(), is(notNullValue())); + assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDns()), contains(dnsServer)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/StatsCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/StatsCmdIT.java new file mode 100644 index 000000000..c4f9fef57 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/StatsCmdIT.java @@ -0,0 +1,95 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.async.ResultCallbackTemplate; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.model.Statistics; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.TimeUnit; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertTrue; + +public class StatsCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(StatsCmdIT.class); + + private static int NUM_STATS = 3; + + @Test + public void testStatsStreaming() throws InterruptedException, IOException { + CountDownLatch countDownLatch = new CountDownLatch(NUM_STATS); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("top").exec(); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + boolean gotStats = false; + try (StatsCallbackTest statsCallback = dockerRule.getClient() + .statsCmd(container.getId()) + .exec(new StatsCallbackTest(countDownLatch))) { + + assertTrue(countDownLatch.await(10, TimeUnit.SECONDS)); + gotStats = statsCallback.gotStats(); + + LOG.info("Stop stats collection"); + } + + LOG.info("Stopping container"); + dockerRule.getClient().stopContainerCmd(container.getId()).exec(); + dockerRule.getClient().removeContainerCmd(container.getId()).exec(); + + LOG.info("Completed test"); + assertTrue("Expected true", gotStats); + } + + @Test + public void testStatsNoStreaming() throws InterruptedException, IOException { + CountDownLatch countDownLatch = new CountDownLatch(NUM_STATS); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("top").exec(); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + try (StatsCallbackTest statsCallback = dockerRule.getClient().statsCmd(container.getId()) + .withNoStream(true) + .exec(new StatsCallbackTest(countDownLatch))) { + countDownLatch.await(5, TimeUnit.SECONDS); + + LOG.info("Stop stats collection"); + } + + LOG.info("Stopping container"); + dockerRule.getClient().stopContainerCmd(container.getId()).exec(); + dockerRule.getClient().removeContainerCmd(container.getId()).exec(); + + LOG.info("Completed test"); + assertEquals("Expected stats called only once", countDownLatch.getCount(), NUM_STATS - 1); + } + + private static class StatsCallbackTest extends ResultCallbackTemplate { + private final CountDownLatch countDownLatch; + + private Boolean gotStats = false; + + public StatsCallbackTest(CountDownLatch countDownLatch) { + this.countDownLatch = countDownLatch; + } + + @Override + public void onNext(Statistics stats) { + LOG.info("Received stats #{}: {}", countDownLatch.getCount(), stats); + if (stats != null) { + gotStats = true; + } + countDownLatch.countDown(); + } + + public Boolean gotStats() { + return gotStats; + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/StopContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/StopContainerCmdIT.java new file mode 100644 index 000000000..7e88cf088 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/StopContainerCmdIT.java @@ -0,0 +1,49 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; + +public class StopContainerCmdIT extends CmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(StopContainerCmdIT.class); + + @Test + public void testStopContainer() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + LOG.info("Stopping container: {}", container.getId()); + dockerRule.getClient().stopContainerCmd(container.getId()).withTimeout(2).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); + + assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); + + final Integer exitCode = inspectContainerResponse.getState().getExitCode(); + + assertThat(exitCode, is(137)); + + } + + @Test(expected = NotFoundException.class) + public void testStopNonExistingContainer() throws DockerException { + + dockerRule.getClient().stopContainerCmd("non-existing").withTimeout(2).exec(); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/TagImageCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/TagImageCmdIT.java new file mode 100644 index 000000000..fc5894455 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/TagImageCmdIT.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.exception.NotFoundException; +import org.apache.commons.lang3.RandomUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class TagImageCmdIT extends CmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(TagImageCmdIT.class); + + @Test + public void tagImage() { + String tag = "" + RandomUtils.nextInt(0, Integer.MAX_VALUE); + + dockerRule.getClient().tagImageCmd("busybox:latest", "docker-java/busybox", tag).exec(); + + dockerRule.getClient().removeImageCmd("docker-java/busybox:" + tag).exec(); + } + + @Test(expected = NotFoundException.class) + public void tagNonExistingImage() { + + String tag = "" + RandomUtils.nextInt(0, Integer.MAX_VALUE); + dockerRule.getClient().tagImageCmd("non-existing", "docker-java/busybox", tag).exec(); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/TrackingDockerHttpClient.java b/docker-java/src/test/java/com/github/dockerjava/cmd/TrackingDockerHttpClient.java new file mode 100644 index 000000000..3c991a8f1 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/TrackingDockerHttpClient.java @@ -0,0 +1,91 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.transport.DockerHttpClient; + +import javax.annotation.Nonnull; +import javax.annotation.Nullable; +import java.io.IOException; +import java.io.InputStream; +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.concurrent.ConcurrentHashMap; + +class TrackingDockerHttpClient implements DockerHttpClient { + + static final Set ACTIVE_RESPONSES = Collections.newSetFromMap(new ConcurrentHashMap<>()); + + private final DockerHttpClient delegate; + + TrackingDockerHttpClient(DockerHttpClient delegate) { + this.delegate = delegate; + } + + @Override + public Response execute(Request request) { + return new TrackedResponse(delegate.execute(request)) { + { + synchronized (ACTIVE_RESPONSES) { + ACTIVE_RESPONSES.add(this); + } + } + + @Override + public void close() { + synchronized (ACTIVE_RESPONSES) { + ACTIVE_RESPONSES.remove(this); + } + super.close(); + } + }; + } + + @Override + public void close() throws IOException { + delegate.close(); + } + + static class TrackedResponse implements Response { + + private static class AllocatedAt extends Exception { + public AllocatedAt(String message) { + super(message); + } + } + + final Exception allocatedAt = new AllocatedAt(this.toString()); + + private final Response delegate; + + TrackedResponse(Response delegate) { + this.delegate = delegate; + } + + @Override + public int getStatusCode() { + return delegate.getStatusCode(); + } + + @Override + public Map> getHeaders() { + return delegate.getHeaders(); + } + + @Override + public InputStream getBody() { + return delegate.getBody(); + } + + @Override + public void close() { + delegate.close(); + } + + @Override + @Nullable + public String getHeader(@Nonnull String name) { + return delegate.getHeader(name); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/UnpauseCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/UnpauseCmdIT.java new file mode 100644 index 000000000..2c970aee1 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/UnpauseCmdIT.java @@ -0,0 +1,90 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.exception.InternalServerErrorException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.utils.ContainerUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; + +public class UnpauseCmdIT extends CmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(UnpauseCmdIT.class); + + @Test + public void unpausePausedContainer() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + ContainerUtils.startContainer(dockerRule.getClient(), container); + + ContainerUtils.pauseContainer(dockerRule.getClient(), container); + + ContainerUtils.unpauseContainer(dockerRule.getClient(), container); + } + + @Test(expected = InternalServerErrorException.class) + public void unpauseRunningContainer() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + ContainerUtils.startContainer(dockerRule.getClient(), container); + + dockerRule.getClient().unpauseContainerCmd(container.getId()).exec(); + } + + @Test(expected = InternalServerErrorException.class) + public void unpauseStoppedContainer() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + ContainerUtils.startContainer(dockerRule.getClient(), container); + + ContainerUtils.stopContainer(dockerRule.getClient(), container); + + dockerRule.getClient().unpauseContainerCmd(container.getId()).exec(); + } + + @Test(expected = NotFoundException.class) + public void unpauseNonExistingContainer() { + + dockerRule.getClient().unpauseContainerCmd("non-existing").exec(); + } + + @Test(expected = InternalServerErrorException.class) + public void unpauseCreatedContainer() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().unpauseContainerCmd(container.getId()).exec(); + } + + @Test(expected = InternalServerErrorException.class) + public void unpauseUnpausedContainer() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + ContainerUtils.startContainer(dockerRule.getClient(), container); + + ContainerUtils.pauseContainer(dockerRule.getClient(), container); + + dockerRule.getClient().unpauseContainerCmd(container.getId()).exec(); + dockerRule.getClient().unpauseContainerCmd(container.getId()).exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/UpdateContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/UpdateContainerCmdIT.java new file mode 100644 index 000000000..e1e637809 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/UpdateContainerCmdIT.java @@ -0,0 +1,94 @@ +package com.github.dockerjava.cmd; + +import com.fasterxml.jackson.databind.JavaType; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.HostConfig; +import com.github.dockerjava.core.command.UpdateContainerCmdImpl; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Ignore; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; + +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_22; +import static com.github.dockerjava.junit.DockerMatchers.isGreaterOrEqual; +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.core.IsNull.notNullValue; +import static org.junit.Assume.assumeThat; + +/** + * @author Kanstantsin Shautsou + */ +public class UpdateContainerCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(UpdateContainerCmdIT.class); + + + @Test + public void updateContainer() throws DockerException { + assumeThat("API version should be >= 1.22", dockerRule, isGreaterOrEqual(VERSION_1_22)); + + CreateContainerResponse response = dockerRule.getClient().createContainerCmd(DEFAULT_IMAGE) + .withCmd("sleep", "9999") + .exec(); + + String containerId = response.getId(); + dockerRule.getClient().startContainerCmd(containerId).exec(); + + InspectContainerResponse inspectBefore = dockerRule.getClient().inspectContainerCmd(containerId).exec(); + LOG.debug("Inspect: {}", inspectBefore); + final Long memory = inspectBefore.getHostConfig().getMemory(); + + dockerRule.getClient().updateContainerCmd(containerId) + .withBlkioWeight(300) + .withCpuShares(512) + .withCpuPeriod(100000L) + .withCpuQuota(50000L) +// .withCpusetCpus("0") // depends on env + .withCpusetMems("0") +// .withMemory(209715200L + 2L) +// .withMemorySwap(514288000L) Your kernel does not support swap limit capabilities, memory limited without swap. +// .withMemoryReservation(209715200L) +// .withKernelMemory(52428800) Can not update kernel memory to a running container, please stop it first. + .exec(); + + // true only on docker toolbox (1.10.1) +// assertThat(updateResponse.getWarnings(), hasSize(1)); +// assertThat(updateResponse.getWarnings().get(0), +// is("Your kernel does not support Block I/O weight. Weight discarded.")); + + InspectContainerResponse inspectAfter = dockerRule.getClient().inspectContainerCmd(containerId).exec(); + final HostConfig afterHostConfig = inspectAfter.getHostConfig(); + +// assertThat(afterHostConfig.getMemory(), is(209715200L + 2L)); + +// assertThat(afterHostConfig.getBlkioWeight(), is(300)); + assertThat(afterHostConfig.getCpuShares(), is(512)); + assertThat(afterHostConfig.getCpuPeriod(), is(100000L)); + assertThat(afterHostConfig.getCpuQuota(), is(50000L)); + assertThat(afterHostConfig.getCpusetMems(), is("0")); + +// assertThat(afterHostConfig.getMemoryReservation(), is(209715200L)); +// assertThat(afterHostConfig.getMemorySwap(), is(514288000L)); + + } + + @Ignore("impossible to serder because model bundled in cmd") + @Test + public void serDerDocs1() throws IOException { + final JavaType type = JSONTestHelper.getMapper().getTypeFactory().constructType(UpdateContainerCmdImpl.class); + + final UpdateContainerCmdImpl upd = testRoundTrip(VERSION_1_22, + "/containers/container/update/docs.json", + type + ); + + assertThat(upd, notNullValue()); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/VersionCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/VersionCmdIT.java new file mode 100644 index 000000000..90c7e534e --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/VersionCmdIT.java @@ -0,0 +1,29 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Version; +import org.apache.commons.lang3.StringUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertTrue; + +public class VersionCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(VersionCmdIT.class); + + + @Test + public void version() throws DockerException { + Version version = dockerRule.getClient().versionCmd().exec(); + LOG.info(version.toString()); + + assertTrue(version.getGoVersion().length() > 0); + assertTrue(version.getVersion().length() > 0); + + assertEquals(3, StringUtils.split(version.getVersion(), ".").length); + + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/WaitContainerCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/WaitContainerCmdIT.java new file mode 100644 index 000000000..3a39b3eea --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/WaitContainerCmdIT.java @@ -0,0 +1,185 @@ +package com.github.dockerjava.cmd; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.BuildImageCmd; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.command.WaitContainerResultCallback; +import com.github.dockerjava.api.exception.DockerClientException; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.WaitContainerCondition; +import com.github.dockerjava.api.model.WaitResponse; +import org.junit.Assert; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.concurrent.TimeUnit; + +import static com.github.dockerjava.api.model.HostConfig.newHostConfig; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_25; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_30; +import static com.github.dockerjava.junit.DockerMatchers.isGreaterOrEqual; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.emptyString; +import static org.hamcrest.Matchers.not; +import static org.junit.Assume.assumeThat; + +public class WaitContainerCmdIT extends CmdIT { + public static final Logger LOG = LoggerFactory.getLogger(BuildImageCmd.class); + + @Test + public void testWaitContainer() throws DockerException { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("true").exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + int exitCode = dockerRule.getClient().waitContainerCmd(container.getId()).start() + .awaitStatusCode(); + LOG.info("Container exit code: {}", exitCode); + + assertThat(exitCode, equalTo(0)); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); + + assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); + assertThat(inspectContainerResponse.getState().getExitCode(), is(equalTo(exitCode))); + } + + @Test(expected = NotFoundException.class) + public void testWaitNonExistingContainer() throws Exception { + + ResultCallback.Adapter callback = new ResultCallback.Adapter() { + public void onNext(WaitResponse waitResponse) { + throw new AssertionError("expected NotFoundException"); + } + }; + + dockerRule.getClient().waitContainerCmd("non-existing").exec(callback).awaitCompletion(); + } + + @Test + public void testWaitContainerAbort() throws Exception { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "9999").exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + WaitContainerResultCallback callback = dockerRule.getClient().waitContainerCmd(container.getId()).start(); + + Thread.sleep(5000); + + callback.close(); + + dockerRule.getClient().killContainerCmd(container.getId()).exec(); + + InspectContainerResponse inspectContainerResponse = dockerRule.getClient().inspectContainerCmd(container.getId()).exec(); + LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); + + assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); + } + + @Test + public void testWaitContainerTimeout() { + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox").withCmd("sleep", "10").exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + WaitContainerResultCallback callback = dockerRule.getClient().waitContainerCmd(container.getId()).exec( + new WaitContainerResultCallback()); + try { + callback.awaitStatusCode(100, TimeUnit.MILLISECONDS); + throw new AssertionError("Should throw exception on timeout."); + } catch (DockerClientException e) { + LOG.info(e.getMessage()); + } + } + + @Test + public void testWaitNotStartedContainer() { + assumeThat("API version should be > 1.25", dockerRule, isGreaterOrEqual(VERSION_1_25)); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withHostConfig(newHostConfig().withAutoRemove(true)) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + WaitContainerResultCallback callback = dockerRule.getClient().waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()); + + Integer statusCode = callback.awaitStatusCode(100, TimeUnit.MILLISECONDS); + Assert.assertEquals(0, statusCode.intValue()); + } + + @Test + public void testWaitContainerWithAutoRemoval() { + assumeThat("API version should be > 1.30", dockerRule, isGreaterOrEqual(VERSION_1_30)); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("false") + .withHostConfig(newHostConfig().withAutoRemove(true)) + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + WaitContainerResultCallback removedCondition = dockerRule.getClient().waitContainerCmd(container.getId()) + .withCondition(WaitContainerCondition.REMOVED) + .exec(new WaitContainerResultCallback()); + + WaitContainerResultCallback nextExitCondition = dockerRule.getClient().waitContainerCmd(container.getId()) + .withCondition(WaitContainerCondition.NEXT_EXIT) + .exec(new WaitContainerResultCallback()); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + Assert.assertEquals(1, removedCondition.awaitStatusCode(100, TimeUnit.MILLISECONDS).intValue()); + Assert.assertEquals(1, nextExitCondition.awaitStatusCode(100, TimeUnit.MILLISECONDS).intValue()); + } + + @Test + public void testWaitRestartedContainer() { + assumeThat("API version should be > 1.30", dockerRule, isGreaterOrEqual(VERSION_1_30)); + + CreateContainerResponse container = dockerRule.getClient().createContainerCmd("busybox") + .withCmd("sh", "-c", "[ -f \"$HOME/.first_run_marker\" ] && exit 2 || { touch \"$HOME/.first_run_marker\"; exit 1; }") + .exec(); + + LOG.info("Created container: {}", container.toString()); + assertThat(container.getId(), not(is(emptyString()))); + + WaitContainerResultCallback firstExitCallback = dockerRule.getClient().waitContainerCmd(container.getId()) + .withCondition(WaitContainerCondition.NEXT_EXIT) + .exec(new WaitContainerResultCallback()); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + Integer firstStatusCode = firstExitCallback.awaitStatusCode(100, TimeUnit.MILLISECONDS); + Assert.assertEquals(1, firstStatusCode.intValue()); + + WaitContainerResultCallback callback = dockerRule.getClient().waitContainerCmd(container.getId()) + .withCondition(WaitContainerCondition.NEXT_EXIT) + .exec(new WaitContainerResultCallback()); + + dockerRule.getClient().startContainerCmd(container.getId()).exec(); + + Integer statusCode = callback.awaitStatusCode(100, TimeUnit.MILLISECONDS); + Assert.assertEquals(2, statusCode.intValue()); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/CreateConfigCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/CreateConfigCmdExecIT.java new file mode 100644 index 000000000..8ebb610b9 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/CreateConfigCmdExecIT.java @@ -0,0 +1,31 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.CreateConfigResponse; +import org.apache.commons.lang3.RandomStringUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.notNullValue; + +public class CreateConfigCmdExecIT extends SwarmCmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(CreateConfigCmdExecIT.class); + + @Test + public void testCreateConfig() { + DockerClient dockerClient = startSwarm(); + String configName = RandomStringUtils.random(10, true, false); + CreateConfigResponse response = dockerClient.createConfigCmd() + .withName(configName) + .withData("configuration data".getBytes()).exec(); + assertThat(response, notNullValue()); + String configId = response.getId(); + assertThat(configId, notNullValue()); + + dockerClient.removeConfigCmd(configId).exec(); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/CreateSecretCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/CreateSecretCmdExecIT.java new file mode 100644 index 000000000..4644a8330 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/CreateSecretCmdExecIT.java @@ -0,0 +1,52 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.CreateSecretResponse; +import com.github.dockerjava.api.model.Secret; +import com.github.dockerjava.api.model.SecretSpec; +import com.google.common.collect.Lists; +import org.apache.commons.lang3.RandomStringUtils; +import org.hamcrest.collection.IsCollectionWithSize; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.notNullValue; + +public class CreateSecretCmdExecIT extends SwarmCmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(CreateSecretCmdExecIT.class); + + @Test + public void testCreateSecret() { + DockerClient dockerClient = startSwarm(); + int length = 10; + boolean useLetters = true; + boolean useNumbers = false; + String secretName = RandomStringUtils.random(length, useLetters, useNumbers); + CreateSecretResponse exec = dockerClient.createSecretCmd(new SecretSpec().withName(secretName).withData("mon secret en clair")).exec(); + assertThat(exec, notNullValue()); + assertThat(exec.getId(), notNullValue()); + LOG.info("Secret created with ID {}", exec.getId()); + + + List secrets = dockerClient.listSecretsCmd() + .withNameFilter(Lists.newArrayList(secretName)) + .exec(); + + assertThat(secrets, IsCollectionWithSize.hasSize(1)); + + dockerClient.removeSecretCmd(secrets.get(0).getId()) + .exec(); + LOG.info("Secret removed with ID {}", exec.getId()); + List secretsAfterRemoved = dockerClient.listSecretsCmd() + .withNameFilter(Lists.newArrayList(secretName)) + .exec(); + + assertThat(secretsAfterRemoved, IsCollectionWithSize.hasSize(0)); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/CreateServiceCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/CreateServiceCmdExecIT.java new file mode 100644 index 000000000..e221d9cd3 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/CreateServiceCmdExecIT.java @@ -0,0 +1,184 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.exception.ConflictException; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.ContainerSpec; +import com.github.dockerjava.api.model.EndpointResolutionMode; +import com.github.dockerjava.api.model.EndpointSpec; +import com.github.dockerjava.api.model.Mount; +import com.github.dockerjava.api.model.Network; +import com.github.dockerjava.api.model.NetworkAttachmentConfig; +import com.github.dockerjava.api.model.PortConfig; +import com.github.dockerjava.api.model.PortConfigProtocol; +import com.github.dockerjava.api.model.Service; +import com.github.dockerjava.api.model.ServiceModeConfig; +import com.github.dockerjava.api.model.ServiceReplicatedModeOptions; +import com.github.dockerjava.api.model.ServiceSpec; +import com.github.dockerjava.api.model.TaskSpec; +import com.github.dockerjava.api.model.TmpfsOptions; +import com.github.dockerjava.junit.PrivateRegistryRule; +import com.google.common.collect.ImmutableMap; +import com.google.common.collect.Lists; +import org.junit.Before; +import org.junit.ClassRule; +import org.junit.Ignore; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Collections; +import java.util.List; + +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.hasSize; +import static org.hamcrest.Matchers.is; + +public class CreateServiceCmdExecIT extends SwarmCmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(CreateServiceCmdExecIT.class); + private static final String SERVICE_NAME = "theservice"; + + @ClassRule + public static PrivateRegistryRule REGISTRY = new PrivateRegistryRule(); + + @Rule + public ExpectedException exception = ExpectedException.none(); + private AuthConfig authConfig; + + private DockerClient dockerClient; + + @Before + public final void setUpCreateServiceCmdExecIT() { + authConfig = REGISTRY.getAuthConfig(); + dockerClient = startSwarm(); + } + + @Test + public void testCreateService() throws DockerException { + dockerClient.createServiceCmd(new ServiceSpec() + .withName(SERVICE_NAME) + .withTaskTemplate(new TaskSpec() + .withContainerSpec(new ContainerSpec() + .withImage(DEFAULT_IMAGE)))) + .exec(); + + List services = dockerClient.listServicesCmd() + .withNameFilter(Lists.newArrayList(SERVICE_NAME)) + .exec(); + + assertThat(services, hasSize(1)); + + dockerClient.removeServiceCmd(SERVICE_NAME).exec(); + } + + @Test + public void testCreateServiceWithNetworks() { + String networkId = dockerClient.createNetworkCmd().withName("networkname") + .withDriver("overlay") + .withIpam(new Network.Ipam() + .withDriver("default")) + .exec().getId(); + ServiceSpec spec = new ServiceSpec() + .withName(SERVICE_NAME) + .withTaskTemplate(new TaskSpec() + .withForceUpdate(0) + .withRuntime("container") + .withContainerSpec(new ContainerSpec() + .withImage("busybox")) + ) + .withNetworks(Lists.newArrayList( + new NetworkAttachmentConfig() + .withTarget(networkId) + .withAliases(Lists.newArrayList("alias1", "alias2")) + )) + .withLabels(ImmutableMap.of("com.docker.java.usage", "SwarmServiceIT")) + .withMode(new ServiceModeConfig().withReplicated( + new ServiceReplicatedModeOptions() + .withReplicas(1) + )).withEndpointSpec(new EndpointSpec() + .withMode(EndpointResolutionMode.VIP) + .withPorts(Lists.newArrayList(new PortConfig() + .withPublishMode(PortConfig.PublishMode.host) + .withPublishedPort(22) + .withProtocol(PortConfigProtocol.TCP) + ))); + + dockerClient.createServiceCmd(spec).exec(); + + List services = dockerClient.listServicesCmd() + .withNameFilter(Lists.newArrayList(SERVICE_NAME)) + .exec(); + + assertThat(services, hasSize(1)); + + assertThat(services.get(0).getSpec(), is(spec)); + + dockerClient.removeServiceCmd(SERVICE_NAME).exec(); + } + + @Test + public void testCreateServiceWithTmpfs() { + Mount tmpMount = new Mount().withTmpfsOptions(new TmpfsOptions().withSizeBytes(600L)).withTarget("/tmp/foo"); + + dockerClient.createServiceCmd(new ServiceSpec() + .withName(SERVICE_NAME) + .withTaskTemplate(new TaskSpec() + .withContainerSpec(new ContainerSpec().withImage(DEFAULT_IMAGE).withMounts(Collections.singletonList(tmpMount))))) + .exec(); + + List services = dockerClient.listServicesCmd() + .withNameFilter(Lists.newArrayList(SERVICE_NAME)) + .exec(); + + assertThat(services, hasSize(1)); + List mounts = dockerClient.inspectServiceCmd(SERVICE_NAME).exec().getSpec().getTaskTemplate() + .getContainerSpec().getMounts(); + assertThat(mounts, hasSize(1)); + assertThat(mounts.get(0), is(tmpMount)); + dockerClient.removeServiceCmd(SERVICE_NAME).exec(); + } + + @Test + public void testCreateServiceWithValidAuth() throws DockerException { + dockerClient.createServiceCmd(new ServiceSpec() + .withName(SERVICE_NAME) + .withTaskTemplate(new TaskSpec() + .withContainerSpec(new ContainerSpec() + .withImage(DEFAULT_IMAGE)))) + .withAuthConfig(authConfig) + .exec(); + + List services = dockerClient.listServicesCmd() + .withNameFilter(Lists.newArrayList(SERVICE_NAME)) + .exec(); + + assertThat(services, hasSize(1)); + + dockerClient.removeServiceCmd(SERVICE_NAME).exec(); + } + + @Test + @Ignore // TODO rework test (does not throw as expected atm) + public void testCreateServiceWithInvalidAuth() throws DockerException { + AuthConfig invalidAuthConfig = new AuthConfig() + .withUsername("testuser") + .withPassword("testwrongpassword") + .withEmail("foo@bar.de") + .withRegistryAddress(authConfig.getRegistryAddress()); + + exception.expect(ConflictException.class); + + dockerClient.createServiceCmd(new ServiceSpec() + .withName(SERVICE_NAME) + .withTaskTemplate(new TaskSpec() + .withContainerSpec(new ContainerSpec() + .withImage(DEFAULT_IMAGE)))) + .withAuthConfig(invalidAuthConfig) + .exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/InitializeSwarmCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/InitializeSwarmCmdExecIT.java new file mode 100644 index 000000000..6341f4b30 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/InitializeSwarmCmdExecIT.java @@ -0,0 +1,63 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Swarm; +import com.github.dockerjava.api.model.SwarmCAConfig; +import com.github.dockerjava.api.model.SwarmDispatcherConfig; +import com.github.dockerjava.api.model.SwarmOrchestration; +import com.github.dockerjava.api.model.SwarmRaftConfig; +import com.github.dockerjava.api.model.SwarmSpec; +import com.github.dockerjava.api.model.TaskDefaults; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.is; + +public class InitializeSwarmCmdExecIT extends SwarmCmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(InitializeSwarmCmdExecIT.class); + + @Test + public void initializeSwarm() throws Exception { + DockerClient dockerClient = startDockerInDocker(); + SwarmSpec swarmSpec = new SwarmSpec() + .withName("default") + .withDispatcher(new SwarmDispatcherConfig() + .withHeartbeatPeriod(10000000L) + ).withOrchestration(new SwarmOrchestration() + .withTaskHistoryRententionLimit(100) + ).withCaConfig(new SwarmCAConfig() + .withNodeCertExpiry(60 * 60 * 1000000000L /*ns */)) + .withRaft(new SwarmRaftConfig() + .withElectionTick(8) + .withSnapshotInterval(20000) + .withHeartbeatTick(5) + .withLogEntriesForSlowFollowers(200) + ).withTaskDefaults(new TaskDefaults()); + + dockerClient.initializeSwarmCmd(swarmSpec) + .withListenAddr("127.0.0.1") + .withAdvertiseAddr("127.0.0.1") + .exec(); + LOG.info("Initialized swarm: {}", swarmSpec.toString()); + + Swarm swarm = dockerClient.inspectSwarmCmd().exec(); + LOG.info("Inspected swarm: {}", swarm.toString()); + assertThat(swarm.getSpec(), is(equalTo(swarmSpec))); + } + + @Test(expected = DockerException.class) + public void initializingSwarmThrowsWhenAlreadyInSwarm() throws DockerException { + DockerClient dockerClient = startSwarm(); + + // Initializing a swarm if already in swarm mode should fail + dockerClient.initializeSwarmCmd(new SwarmSpec()) + .withListenAddr("127.0.0.1") + .exec(); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/InspectConfigCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/InspectConfigCmdIT.java new file mode 100644 index 000000000..12c69c996 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/InspectConfigCmdIT.java @@ -0,0 +1,33 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.CreateConfigResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Config; +import org.apache.commons.lang3.RandomStringUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.junit.Assert.assertEquals; + +public class InspectConfigCmdIT extends SwarmCmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(InspectConfigCmdIT.class); + + @Test + public void inspectConfig() throws DockerException { + DockerClient dockerClient = startSwarm(); + + String configName = RandomStringUtils.random(10, true, false); + + CreateConfigResponse configResponse = dockerClient.createConfigCmd() + .withName(configName) + .withData("configuration data".getBytes()).exec(); + LOG.info("Config created with ID {}", configResponse.getId()); + + Config config = dockerClient.inspectConfigCmd(configResponse.getId()).exec(); + assertEquals(configResponse.getId(), config.getId()); + assertEquals(configName, config.getSpec().getName()); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/JoinSwarmCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/JoinSwarmCmdExecIT.java new file mode 100644 index 000000000..16f1b0911 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/JoinSwarmCmdExecIT.java @@ -0,0 +1,86 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Info; +import com.github.dockerjava.api.model.LocalNodeState; +import com.github.dockerjava.api.model.Swarm; +import com.github.dockerjava.api.model.SwarmJoinTokens; +import com.github.dockerjava.api.model.SwarmSpec; +import com.google.common.collect.Lists; +import org.junit.Before; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.is; + +public class JoinSwarmCmdExecIT extends SwarmCmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(JoinSwarmCmdExecIT.class); + + private DockerClient docker1; + + private DockerClient docker2; + + @Before + public void setUp() throws Exception { + docker1 = startDockerInDocker(); + docker2 = startDockerInDocker(); + } + + private SwarmJoinTokens initSwarmOnDocker(DockerClient docker) { + SwarmSpec swarmSpec = new SwarmSpec(); + docker.initializeSwarmCmd(swarmSpec) + .exec(); + LOG.info("Initialized swarm docker1: {}", swarmSpec.toString()); + + Swarm swarm = docker.inspectSwarmCmd().exec(); + LOG.info("Inspected swarm on docker1: {}", swarm.toString()); + return swarm.getJoinTokens(); + } + + @Test + public void joinSwarmAsWorker() { + SwarmJoinTokens tokens = initSwarmOnDocker(docker1); + + docker2.joinSwarmCmd() + .withRemoteAddrs(Lists.newArrayList("docker1")) + .withJoinToken(tokens.getWorker()) + .exec(); + LOG.info("docker2 joined docker1's swarm"); + + Info info = docker2.infoCmd().exec(); + LOG.info("Inspected docker2: {}", info.toString()); + assertThat(info.getSwarm().getLocalNodeState(), is(equalTo(LocalNodeState.ACTIVE))); + } + + @Test + public void joinSwarmAsManager() throws DockerException { + SwarmJoinTokens tokens = initSwarmOnDocker(docker1); + + docker2.joinSwarmCmd() + .withRemoteAddrs(Lists.newArrayList("docker1")) + .withJoinToken(tokens.getManager()) + .exec(); + LOG.info("docker2 joined docker1's swarm"); + + Info info = docker2.infoCmd().exec(); + LOG.info("Inspected docker2: {}", info.toString()); + assertThat(info.getSwarm().getLocalNodeState(), is(equalTo(LocalNodeState.ACTIVE))); + } + + @Test(expected = DockerException.class) + public void joinSwarmIfAlreadyInSwarm() { + SwarmJoinTokens tokens = initSwarmOnDocker(docker1); + + initSwarmOnDocker(docker2); + + docker2.joinSwarmCmd() + .withRemoteAddrs(Lists.newArrayList("docker1")) + .withJoinToken(tokens.getWorker()) + .exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/LeaveSwarmCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/LeaveSwarmCmdExecIT.java new file mode 100644 index 000000000..e6d652d43 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/LeaveSwarmCmdExecIT.java @@ -0,0 +1,45 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Info; +import com.github.dockerjava.api.model.LocalNodeState; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.is; + +public class LeaveSwarmCmdExecIT extends SwarmCmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(LeaveSwarmCmdExecIT.class); + + @Test + public void leaveSwarmAsMaster() throws DockerException { + DockerClient dockerClient = startSwarm(); + + Info info = dockerClient.infoCmd().exec(); + LOG.info("Inspected docker: {}", info.toString()); + + assertThat(info.getSwarm().getLocalNodeState(), is(LocalNodeState.ACTIVE)); + + dockerClient.leaveSwarmCmd() + .withForceEnabled(true) + .exec(); + LOG.info("Left swarm"); + + info = dockerClient.infoCmd().exec(); + LOG.info("Inspected docker: {}", info.toString()); + + assertThat(info.getSwarm().getLocalNodeState(), is(LocalNodeState.INACTIVE)); + + } + + @Test(expected = DockerException.class) + public void leavingSwarmThrowsWhenNotInSwarm() throws Exception { + DockerClient dockerClient = startDockerInDocker(); + dockerClient.leaveSwarmCmd().exec(); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListConfigCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListConfigCmdExecIT.java new file mode 100644 index 000000000..2c19a7f0e --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListConfigCmdExecIT.java @@ -0,0 +1,54 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.CreateConfigResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Config; +import org.apache.commons.lang3.RandomStringUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Arrays; +import java.util.Collections; +import java.util.List; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.collection.IsCollectionWithSize.hasSize; + +public class ListConfigCmdExecIT extends SwarmCmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(ListConfigCmdExecIT.class); + + @Test + public void tesListConfig() throws DockerException { + DockerClient dockerClient = startSwarm(); + String configName = RandomStringUtils.random(10, true, false); + CreateConfigResponse response = dockerClient.createConfigCmd() + .withName(configName) + .withData("configuration data".getBytes()) + .exec(); + String configId = response.getId(); + + try { + LOG.info("Config created with ID {}", configId); + + List configs = dockerClient.listConfigsCmd() + .withFilters(Collections.singletonMap("name", Arrays.asList(configName))) + .exec(); + + assertThat(configs, hasSize(1)); + } finally { + dockerClient.removeConfigCmd(configId).exec(); + LOG.info("Config removed with ID {}", configId); + } + + List configsAfterRemoved = dockerClient.listConfigsCmd() + .withFilters(Collections.singletonMap("name", Arrays.asList(configName))) + .exec(); + + assertThat(configsAfterRemoved, hasSize(0)); + + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListSecretCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListSecretCmdExecIT.java new file mode 100644 index 000000000..ce90f23a0 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListSecretCmdExecIT.java @@ -0,0 +1,54 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.CreateSecretResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Secret; +import com.github.dockerjava.api.model.SecretSpec; +import com.google.common.collect.Lists; +import org.apache.commons.lang3.RandomStringUtils; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.collection.IsCollectionWithSize.hasSize; + +public class ListSecretCmdExecIT extends SwarmCmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(ListSecretCmdExecIT.class); + + @Test + public void tesListSecret() throws DockerException { + DockerClient dockerClient = startSwarm(); + int length = 10; + boolean useLetters = true; + boolean useNumbers = false; + String secretName = RandomStringUtils.random(length, useLetters, useNumbers); + CreateSecretResponse exec = dockerClient.createSecretCmd(new SecretSpec().withName(secretName).withData("mon secret en clair")).exec(); + assertThat(exec, notNullValue()); + assertThat(exec.getId(), notNullValue()); + LOG.info("Secret created with ID {}", exec.getId()); + + + List secrets = dockerClient.listSecretsCmd() + .withNameFilter(Lists.newArrayList(secretName)) + .exec(); + + assertThat(secrets, hasSize(1)); + + dockerClient.removeSecretCmd(secrets.get(0).getId()) + .exec(); + LOG.info("Secret removed with ID {}", exec.getId()); + List secretsAfterRemoved = dockerClient.listSecretsCmd() + .withNameFilter(Lists.newArrayList(secretName)) + .exec(); + + assertThat(secretsAfterRemoved, hasSize(0)); + + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListServicesCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListServicesCmdExecIT.java new file mode 100644 index 000000000..715ba60c8 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListServicesCmdExecIT.java @@ -0,0 +1,56 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.CreateServiceResponse; +import com.github.dockerjava.api.model.ContainerSpec; +import com.github.dockerjava.api.model.Service; +import com.github.dockerjava.api.model.ServiceModeConfig; +import com.github.dockerjava.api.model.ServiceReplicatedModeOptions; +import com.github.dockerjava.api.model.ServiceSpec; +import com.github.dockerjava.api.model.TaskSpec; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.hasSize; + +public class ListServicesCmdExecIT extends SwarmCmdIT { + public static final Logger LOG = LoggerFactory.getLogger(CreateServiceCmdExecIT.class); + private static final String SERVICE_NAME = "inspect_service"; + private static final String LABEL_KEY = "com.github.dockerjava.usage"; + private static final String LABEL_VALUE = "test"; + + @Test + public void testListServices() { + DockerClient dockerClient = startSwarm(); + Map serviceLabels = Collections.singletonMap(LABEL_KEY, LABEL_VALUE); + CreateServiceResponse response = dockerClient.createServiceCmd(new ServiceSpec() + .withLabels(serviceLabels) + .withName(SERVICE_NAME) + .withMode(new ServiceModeConfig().withReplicated( + new ServiceReplicatedModeOptions() + .withReplicas(1) + )) + .withTaskTemplate(new TaskSpec() + .withContainerSpec(new ContainerSpec() + .withImage(DEFAULT_IMAGE)))) + .exec(); + String serviceId = response.getId(); + //filtering with service id + List services = dockerClient.listServicesCmd().withIdFilter(Collections.singletonList(serviceId)).exec(); + assertThat(services, hasSize(1)); + //filtering with service name + services = dockerClient.listServicesCmd().withNameFilter(Collections.singletonList(SERVICE_NAME)).exec(); + assertThat(services, hasSize(1)); + //filter labels + services = dockerClient.listServicesCmd().withLabelFilter(serviceLabels).exec(); + assertThat(services, hasSize(1)); + dockerClient.removeServiceCmd(SERVICE_NAME).exec(); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListSwarmNodesCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListSwarmNodesCmdExecIT.java new file mode 100644 index 000000000..853dc6c03 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListSwarmNodesCmdExecIT.java @@ -0,0 +1,89 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.model.SwarmNode; +import org.junit.Test; + +import java.util.Collections; +import java.util.List; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.is; + +public class ListSwarmNodesCmdExecIT extends SwarmCmdIT { + @Test + public void testListSwarmNodes() { + DockerClient dockerClient = startSwarm(); + + List nodes = dockerClient.listSwarmNodesCmd().exec(); + assertThat(nodes.size(), is(1)); + } + + @Test + public void testListSwarmNodesWithIdFilter() { + DockerClient dockerClient = startSwarm(); + + List nodes = dockerClient.listSwarmNodesCmd().exec(); + assertThat(nodes.size(), is(1)); + + String nodeId = nodes.get(0).getId(); + List nodesWithId = dockerClient.listSwarmNodesCmd() + .withIdFilter(Collections.singletonList(nodeId)) + .exec(); + assertThat(nodesWithId.size(), is(1)); + + List nodesWithNonexistentId = dockerClient.listSwarmNodesCmd() + .withIdFilter(Collections.singletonList("__nonexistent__")) + .exec(); + assertThat(nodesWithNonexistentId.size(), is(0)); + } + + @Test + public void testListSwarmNodesWithNameFilter() { + DockerClient dockerClient = startSwarm(); + + List nodes = dockerClient.listSwarmNodesCmd().exec(); + assertThat(nodes.size(), is(1)); + + String nodeName = nodes.get(0).getSpec().getName(); + List nodesWithFirstNodesName = dockerClient.listSwarmNodesCmd() + .withNameFilter(Collections.singletonList(nodeName)) + .exec(); + assertThat(nodesWithFirstNodesName.size(), is(1)); + + List nodesWithNonexistentName = dockerClient.listSwarmNodesCmd() + .withNameFilter(Collections.singletonList("__nonexistent__")) + .exec(); + assertThat(nodesWithNonexistentName.size(), is(0)); + } + + @Test + public void testListSwarmNodesWithMembershipFilter() { + DockerClient dockerClient = startSwarm(); + + List nodesWithAcceptedMembership = dockerClient.listSwarmNodesCmd() + .withMembershipFilter(Collections.singletonList("accepted")) + .exec(); + assertThat(nodesWithAcceptedMembership.size(), is(1)); + + List nodesWithPendingMembership = dockerClient.listSwarmNodesCmd() + .withMembershipFilter(Collections.singletonList("pending")) + .exec(); + assertThat(nodesWithPendingMembership.size(), is(0)); + } + + @Test + public void testListSwarmNodesWithRoleFilter() { + DockerClient dockerClient = startSwarm(); + + List nodesWithManagerRole = dockerClient.listSwarmNodesCmd() + .withRoleFilter(Collections.singletonList("manager")) + .exec(); + assertThat(nodesWithManagerRole.size(), is(1)); + + List nodesWithWorkerRole = dockerClient.listSwarmNodesCmd() + .withRoleFilter(Collections.singletonList("worker")) + .exec(); + assertThat(nodesWithWorkerRole.size(), is(0)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListTasksCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListTasksCmdExecIT.java new file mode 100644 index 000000000..8ce672b1b --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/ListTasksCmdExecIT.java @@ -0,0 +1,81 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.CreateServiceResponse; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.ContainerSpec; +import com.github.dockerjava.api.model.ServiceModeConfig; +import com.github.dockerjava.api.model.ServiceReplicatedModeOptions; +import com.github.dockerjava.api.model.ServiceSpec; +import com.github.dockerjava.api.model.Task; +import com.github.dockerjava.api.model.TaskSpec; +import com.github.dockerjava.api.model.TaskState; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static org.awaitility.Awaitility.await; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.hasSize; +import static org.hamcrest.Matchers.is; + +public class ListTasksCmdExecIT extends SwarmCmdIT { + public static final Logger LOG = LoggerFactory.getLogger(CreateServiceCmdExecIT.class); + private static final String SERVICE_NAME = "inspect_task"; + private static final String TASK_LABEL_KEY = "com.github.dockerjava.usage"; + private static final String TASK_LABEL_VALUE = "test"; + + @Test + public void testListTasks() throws DockerException { + DockerClient dockerClient = startSwarm(); + Map taskLabels = Collections.singletonMap(TASK_LABEL_KEY, TASK_LABEL_VALUE); + CreateServiceResponse response = dockerClient.createServiceCmd(new ServiceSpec() + .withName(SERVICE_NAME) + .withMode(new ServiceModeConfig().withReplicated( + new ServiceReplicatedModeOptions() + .withReplicas(2) + )) + .withTaskTemplate(new TaskSpec() + .withContainerSpec(new ContainerSpec() + .withImage(DEFAULT_IMAGE))).withLabels(taskLabels)) + .exec(); + String serviceId = response.getId(); + //filtering with service id + List tasks = await().until( + () -> dockerClient.listTasksCmd().withServiceFilter(serviceId).exec(), + hasSize(2) + ); + String taskId = tasks.get(0).getId(); + String secondTaskId = tasks.get(1).getId(); + //filtering with unique id + tasks = dockerClient.listTasksCmd().withIdFilter(taskId).exec(); + assertThat(tasks, hasSize(1)); + assertThat(tasks.get(0).getId(), is(taskId)); + //filtering with multiple id + tasks = dockerClient.listTasksCmd().withIdFilter(secondTaskId, taskId).exec(); + assertThat(tasks, hasSize(2)); + //filtering node id + // Wait for node assignment + String nodeId = await().until(() -> { + return dockerClient.listTasksCmd().withIdFilter(secondTaskId).exec() + .get(0) + .getNodeId(); + }, Objects::nonNull); + tasks = dockerClient.listTasksCmd().withNodeFilter(nodeId).exec(); + assertThat(tasks.get(0).getNodeId(), is(nodeId)); + //filtering with state + tasks = dockerClient.listTasksCmd().withStateFilter(TaskState.RUNNING).exec(); + assertThat(tasks, hasSize(2)); + //filter labels + tasks = dockerClient.listTasksCmd().withLabelFilter(taskLabels).exec(); + assertThat(tasks, hasSize(2)); + tasks = dockerClient.listTasksCmd().withLabelFilter(TASK_LABEL_KEY + "=" + TASK_LABEL_VALUE).exec(); + assertThat(tasks, hasSize(2)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/LogSwarmObjectIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/LogSwarmObjectIT.java new file mode 100644 index 000000000..11606dce0 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/LogSwarmObjectIT.java @@ -0,0 +1,76 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.LogSwarmObjectCmd; +import com.github.dockerjava.api.model.ContainerSpec; +import com.github.dockerjava.api.model.ServiceModeConfig; +import com.github.dockerjava.api.model.ServiceReplicatedModeOptions; +import com.github.dockerjava.api.model.ServiceRestartCondition; +import com.github.dockerjava.api.model.ServiceRestartPolicy; +import com.github.dockerjava.api.model.ServiceSpec; +import com.github.dockerjava.api.model.Task; +import com.github.dockerjava.api.model.TaskSpec; +import com.github.dockerjava.api.model.TaskState; +import com.github.dockerjava.utils.LogContainerTestCallback; +import org.junit.Ignore; +import org.junit.Test; + +import java.io.IOException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.concurrent.TimeUnit; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.core.Is.is; + +public class LogSwarmObjectIT extends SwarmCmdIT { + + @Ignore + @Test + public void testLogsCmd() throws InterruptedException, IOException { + DockerClient dockerClient = startSwarm(); + String snippet = "hello world"; + TaskSpec taskSpec = new TaskSpec().withContainerSpec( + new ContainerSpec().withImage("busybox").withCommand(Arrays.asList("echo", snippet))) + .withRestartPolicy(new ServiceRestartPolicy().withCondition(ServiceRestartCondition.NONE)); + ServiceSpec serviceSpec = new ServiceSpec() + .withMode(new ServiceModeConfig().withReplicated(new ServiceReplicatedModeOptions().withReplicas(1))) + .withTaskTemplate(taskSpec) + .withName("log-worker"); + String serviceId = dockerClient.createServiceCmd(serviceSpec).exec().getId(); + int since = (int) System.currentTimeMillis() / 1000; + //wait the service to end + List tasks = new ArrayList<>(); + for (int i = 0; i < 10; i++) { + tasks = dockerClient.listTasksCmd().withServiceFilter(serviceId).withStateFilter(TaskState.SHUTDOWN).exec(); + if (tasks.size() == 1) { + break; + } else { + TimeUnit.SECONDS.sleep(3); + } + } + assertThat(tasks.size(), is(1)); + String taskId = tasks.get(0).getId(); + //check service log + validateLog(dockerClient.logServiceCmd(serviceId).withStdout(true), snippet); + //check task log + validateLog(dockerClient.logTaskCmd(taskId).withStdout(true), snippet); + //check details/context + // FIXME + // validateLog(docker1.logServiceCmd(serviceId).withStdout(true).withDetails(true), "com.docker.swarm.service.id=" + serviceId); + // validateLog(docker1.logTaskCmd(taskId).withStdout(true).withDetails(true), "com.docker.swarm.service.id=" + serviceId); + //check since + validateLog(dockerClient.logServiceCmd(serviceId).withStdout(true).withSince(since), snippet); + validateLog(dockerClient.logTaskCmd(taskId).withStdout(true).withSince(since), snippet); + dockerClient.removeServiceCmd(serviceId).exec(); + } + + private void validateLog(LogSwarmObjectCmd logCmd, String messsage) throws InterruptedException, IOException { + try (LogContainerTestCallback loggingCallback = logCmd.exec(new LogContainerTestCallback(true))) { + loggingCallback.awaitCompletion(5, TimeUnit.SECONDS); + assertThat(loggingCallback.toString(), containsString(messsage)); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/RemoveSwarmNodeCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/RemoveSwarmNodeCmdExecIT.java new file mode 100644 index 000000000..8bdee6947 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/RemoveSwarmNodeCmdExecIT.java @@ -0,0 +1,44 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.model.Swarm; +import com.github.dockerjava.api.model.SwarmNode; +import com.github.dockerjava.api.model.SwarmNodeRole; +import com.google.common.collect.Lists; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; +import java.util.Optional; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.is; + +public class RemoveSwarmNodeCmdExecIT extends SwarmCmdIT { + + private static final Logger LOGGER = LoggerFactory.getLogger(RemoveSwarmNodeCmdExecIT.class); + + @Test + public void testRemoveSwarmNode() throws Exception { + DockerClient dockerClient = startSwarm(); + Swarm swarm = dockerClient.inspectSwarmCmd().exec(); + + DockerClient docker2 = startDockerInDocker(); + docker2.joinSwarmCmd() + .withRemoteAddrs(Lists.newArrayList("docker1")) + .withJoinToken(swarm.getJoinTokens().getWorker()) + .exec(); + LOGGER.info("docker2 joined docker's swarm"); + + List nodes = dockerClient.listSwarmNodesCmd().exec(); + assertThat(2, is(nodes.size())); + Optional firstWorkNode = nodes.stream().filter(node -> node.getSpec().getRole() == SwarmNodeRole.WORKER) + .findFirst(); + dockerClient.removeSwarmNodeCmd(firstWorkNode.get().getId()) + .withForce(true) + .exec(); + nodes = dockerClient.listSwarmNodesCmd().exec(); + assertThat(nodes.size(), is(1)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/SwarmCmdIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/SwarmCmdIT.java new file mode 100644 index 000000000..36bcab840 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/SwarmCmdIT.java @@ -0,0 +1,145 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.exception.ConflictException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.ExposedPort; +import com.github.dockerjava.api.model.PortBinding; +import com.github.dockerjava.api.model.Ports; +import com.github.dockerjava.api.model.PullResponseItem; +import com.github.dockerjava.api.model.SwarmSpec; +import com.github.dockerjava.cmd.CmdIT; +import com.github.dockerjava.core.DefaultDockerClientConfig; +import com.github.dockerjava.junit.category.Integration; +import com.github.dockerjava.junit.category.SwarmModeIntegration; +import org.junit.After; +import org.junit.Before; +import org.junit.experimental.categories.Category; + +import java.io.IOException; +import java.time.Duration; +import java.util.Arrays; +import java.util.HashSet; +import java.util.Set; +import java.util.concurrent.TimeUnit; +import java.util.concurrent.atomic.AtomicInteger; + +import static com.github.dockerjava.api.model.HostConfig.newHostConfig; +import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_24; +import static com.github.dockerjava.junit.DockerMatchers.isGreaterOrEqual; +import static org.awaitility.Awaitility.await; +import static org.junit.Assume.assumeThat; + +@Category({SwarmModeIntegration.class, Integration.class}) +public abstract class SwarmCmdIT extends CmdIT { + + private static final String DOCKER_IN_DOCKER_IMAGE_REPOSITORY = "docker"; + + private static final String DOCKER_IN_DOCKER_IMAGE_TAG = "26.1.3-dind"; + + private static final String DOCKER_IN_DOCKER_CONTAINER_PREFIX = "docker"; + + private static final String NETWORK_NAME = "dind-network"; + + private final AtomicInteger numberOfDockersInDocker = new AtomicInteger(); + + private final Set startedContainerIds = new HashSet<>(); + + @Before + public final void setUpMultiNodeSwarmCmdIT() { + assumeThat(dockerRule, isGreaterOrEqual(VERSION_1_24)); + } + + protected DockerClient startSwarm() { + DockerClient dockerClient; + try { + dockerClient = startDockerInDocker(); + } catch (InterruptedException e) { + throw new RuntimeException(e); + } + dockerClient.initializeSwarmCmd(new SwarmSpec()).exec(); + return dockerClient; + } + + @After + public final void tearDownMultiNodeSwarmCmdIT() { + for (String containerId : startedContainerIds) { + try { + dockerRule.getClient().removeContainerCmd(containerId).withForce(true).exec(); + } catch (NotFoundException e) { + // container does not exist + } + } + + try { + dockerRule.getClient().removeNetworkCmd(NETWORK_NAME).exec(); + } catch (NotFoundException e) { + // network does not exist + } + } + + protected DockerClient startDockerInDocker() throws InterruptedException { + // Create network if not already exists + DockerClient hostDockerClient = dockerRule.getClient(); + try { + hostDockerClient.inspectNetworkCmd().withNetworkId(NETWORK_NAME).exec(); + } catch (NotFoundException e) { + try { + hostDockerClient.createNetworkCmd().withName(NETWORK_NAME).exec(); + } catch (ConflictException e2) { + // already exists + } + } + + try ( + ResultCallback.Adapter callback = hostDockerClient.pullImageCmd(DOCKER_IN_DOCKER_IMAGE_REPOSITORY) + .withTag(DOCKER_IN_DOCKER_IMAGE_TAG) + .start() + ) { + callback.awaitCompletion(); + } catch (IOException e) { + throw new RuntimeException(e); + } + + ExposedPort exposedPort = ExposedPort.tcp(2375); + CreateContainerResponse response = hostDockerClient + .createContainerCmd(DOCKER_IN_DOCKER_IMAGE_REPOSITORY + ":" + DOCKER_IN_DOCKER_IMAGE_TAG) + .withEntrypoint("dockerd") + .withCmd(Arrays.asList("--host=tcp://0.0.0.0:2375", "--host=unix:///var/run/docker.sock", "--tls=false")) + .withHostConfig(newHostConfig() + .withNetworkMode(NETWORK_NAME) + .withPortBindings(new PortBinding( + Ports.Binding.bindIp("127.0.0.1"), + exposedPort)) + .withPrivileged(true)) + .withAliases(DOCKER_IN_DOCKER_CONTAINER_PREFIX + numberOfDockersInDocker.incrementAndGet()) + .exec(); + + String containerId = response.getId(); + startedContainerIds.add(containerId); + + hostDockerClient.startContainerCmd(containerId).exec(); + + InspectContainerResponse inspectContainerResponse = hostDockerClient.inspectContainerCmd(containerId).exec(); + + Ports.Binding binding = inspectContainerResponse.getNetworkSettings().getPorts().getBindings().get(exposedPort)[0]; + + DockerClient dockerClient = initializeDockerClient(binding); + + await().pollDelay(Duration.ofSeconds(5)).atMost(10, TimeUnit.SECONDS).untilAsserted(() -> { + dockerClient.pingCmd().exec(); + }); + + return dockerClient; + } + + private DockerClient initializeDockerClient(Ports.Binding binding) { + DefaultDockerClientConfig config = DefaultDockerClientConfig.createDefaultConfigBuilder() + .withRegistryUrl("https://index.docker.io/v1/") + .withDockerHost("tcp://" + binding).build(); + return createDockerClient(config); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/UpdateSwarmCmdExecIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/UpdateSwarmCmdExecIT.java new file mode 100644 index 000000000..ea3818836 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/UpdateSwarmCmdExecIT.java @@ -0,0 +1,66 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.model.Swarm; +import com.github.dockerjava.api.model.SwarmCAConfig; +import com.github.dockerjava.api.model.SwarmDispatcherConfig; +import com.github.dockerjava.api.model.SwarmOrchestration; +import com.github.dockerjava.api.model.SwarmRaftConfig; +import com.github.dockerjava.api.model.SwarmSpec; +import com.github.dockerjava.api.model.TaskDefaults; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.not; + +public class UpdateSwarmCmdExecIT extends SwarmCmdIT { + + public static final Logger LOG = LoggerFactory.getLogger(UpdateSwarmCmdExecIT.class); + + @Test + public void updateSwarm() throws DockerException { + DockerClient dockerClient = startSwarm(); + + SwarmSpec newSpec = new SwarmSpec() + .withName("default") + .withDispatcher(new SwarmDispatcherConfig() + .withHeartbeatPeriod(10000000L) + ).withOrchestration(new SwarmOrchestration() + .withTaskHistoryRententionLimit(100) + ).withCaConfig(new SwarmCAConfig() + .withNodeCertExpiry(60 * 60 * 1000000000L /*ns */)) + .withRaft(new SwarmRaftConfig() + .withElectionTick(8) + .withSnapshotInterval(20000) + .withHeartbeatTick(5) + .withLogEntriesForSlowFollowers(200) + ).withTaskDefaults(new TaskDefaults()); + + Swarm swarm = dockerClient.inspectSwarmCmd().exec(); + LOG.info("Inspected swarm: {}", swarm.toString()); + assertThat(swarm.getSpec(), is(not(equalTo(newSpec)))); + + dockerClient.updateSwarmCmd(newSpec) + .withVersion(swarm.getVersion().getIndex()) + .exec(); + LOG.info("Updated swarm: {}", newSpec.toString()); + + swarm = dockerClient.inspectSwarmCmd().exec(); + LOG.info("Inspected swarm: {}", swarm.toString()); + assertThat(swarm.getSpec(), is(equalTo(newSpec))); + } + + @Test(expected = DockerException.class) + public void updatingSwarmThrowsWhenNotInSwarm() throws Exception { + DockerClient dockerClient = startDockerInDocker(); + dockerClient.updateSwarmCmd(new SwarmSpec()) + .withVersion(1L) + .exec(); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/UpdateSwarmNodeIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/UpdateSwarmNodeIT.java new file mode 100644 index 000000000..d26e051b1 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/UpdateSwarmNodeIT.java @@ -0,0 +1,31 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.model.SwarmNode; +import com.github.dockerjava.api.model.SwarmNodeAvailability; +import com.github.dockerjava.api.model.SwarmNodeSpec; +import com.github.dockerjava.api.model.SwarmNodeState; +import org.junit.Test; + +import java.util.List; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.is; + +public class UpdateSwarmNodeIT extends SwarmCmdIT { + @Test + public void testUpdateSwarmNode() { + DockerClient dockerClient = startSwarm(); + List nodes = dockerClient.listSwarmNodesCmd().exec(); + assertThat(1, is(nodes.size())); + SwarmNode node = nodes.get(0); + assertThat(SwarmNodeState.READY, is(node.getStatus().getState())); + //update the node availability + SwarmNodeSpec nodeSpec = node.getSpec().withAvailability(SwarmNodeAvailability.PAUSE); + dockerClient.updateSwarmNodeCmd().withSwarmNodeId(node.getId()).withVersion(node.getVersion().getIndex()) + .withSwarmNodeSpec(nodeSpec).exec(); + nodes = dockerClient.listSwarmNodesCmd().exec(); + assertThat(nodes.size(), is(1)); + assertThat(nodes.get(0).getSpec().getAvailability(), is(SwarmNodeAvailability.PAUSE)); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/UpdateSwarmServiceIT.java b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/UpdateSwarmServiceIT.java new file mode 100644 index 000000000..c477320bf --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/cmd/swarm/UpdateSwarmServiceIT.java @@ -0,0 +1,49 @@ +package com.github.dockerjava.cmd.swarm; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.model.ContainerSpec; +import com.github.dockerjava.api.model.Network; +import com.github.dockerjava.api.model.NetworkAttachmentConfig; +import com.github.dockerjava.api.model.Service; +import com.github.dockerjava.api.model.ServiceModeConfig; +import com.github.dockerjava.api.model.ServiceReplicatedModeOptions; +import com.github.dockerjava.api.model.ServiceSpec; +import com.github.dockerjava.api.model.TaskSpec; +import com.google.common.collect.Lists; +import org.junit.Test; + +import java.util.Arrays; +import java.util.List; + +import static org.awaitility.Awaitility.await; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.is; + +public class UpdateSwarmServiceIT extends SwarmCmdIT { + @Test + public void testUpdateServiceReplicate() { + DockerClient dockerClient = startSwarm(); + //create network + String networkId = dockerClient.createNetworkCmd().withName("networkname").withDriver("overlay") + .withIpam(new Network.Ipam().withDriver("default")).exec().getId(); + TaskSpec taskSpec = new TaskSpec().withContainerSpec( + new ContainerSpec().withImage("busybox").withArgs(Arrays.asList("sleep", "3600")).withInit(true)); + ServiceSpec serviceSpec = new ServiceSpec() + .withMode(new ServiceModeConfig().withReplicated(new ServiceReplicatedModeOptions().withReplicas(1))) + .withTaskTemplate(taskSpec) + .withNetworks(Lists.newArrayList(new NetworkAttachmentConfig().withTarget(networkId))) + .withName("worker"); + String serviceId = dockerClient.createServiceCmd(serviceSpec).exec().getId(); + await().untilAsserted(() -> { + List services = dockerClient.listServicesCmd().withIdFilter(Arrays.asList(serviceId)).exec(); + assertThat(services.size(), is(1)); + Service service = services.get(0); + ServiceSpec updateServiceSpec = service.getSpec() + .withMode(new ServiceModeConfig().withReplicated(new ServiceReplicatedModeOptions().withReplicas(2))); + dockerClient.updateServiceCmd(service.getId(), updateServiceSpec).withVersion(service.getVersion().getIndex()).exec(); + //verify the replicate + Service updateService = dockerClient.listServicesCmd().withIdFilter(Arrays.asList(serviceId)).exec().get(0); + assertThat(updateService.getSpec().getMode().getReplicated().getReplicas(), is(2L)); + }); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/core/DefaultDockerClientConfigTest.java b/docker-java/src/test/java/com/github/dockerjava/core/DefaultDockerClientConfigTest.java new file mode 100644 index 000000000..6c7787caf --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/DefaultDockerClientConfigTest.java @@ -0,0 +1,350 @@ +package com.github.dockerjava.core; + +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.AuthConfigurations; +import com.google.common.io.Resources; +import java.io.IOException; +import org.apache.commons.lang3.SerializationUtils; +import org.junit.Test; + +import java.io.File; +import java.lang.reflect.Field; +import java.net.URI; +import java.net.URISyntaxException; +import java.util.Collections; +import java.util.HashMap; +import java.util.Map; +import java.util.Properties; +import java.util.UUID; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.notNullValue; +import static org.hamcrest.core.Is.is; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertTrue; + +public class DefaultDockerClientConfigTest { + + public static final DefaultDockerClientConfig EXAMPLE_CONFIG = newExampleConfig(); + public static final DefaultDockerClientConfig EXAMPLE_CONFIG_FULLY_LOADED = newExampleConfigFullyLoaded(); + + private static DefaultDockerClientConfig newExampleConfig() { + String dockerCertPath = dockerCertPath(); + return new DefaultDockerClientConfig(URI.create("tcp://foo"), null, "dockerConfig", "apiVersion", "registryUrl", + "registryUsername", "registryPassword", "registryEmail", + new LocalDirectorySSLConfig(dockerCertPath)); + } + + private static DefaultDockerClientConfig newExampleConfigFullyLoaded() { + try { + String dockerCertPath = dockerCertPath(); + String dockerConfig = "dockerConfig"; + DockerConfigFile loadedConfigFile = DockerConfigFile.loadConfig(DockerClientConfig.getDefaultObjectMapper(), dockerConfig); + return new DefaultDockerClientConfig(URI.create("tcp://foo"), loadedConfigFile, dockerConfig, "apiVersion", "registryUrl", + "registryUsername", "registryPassword", "registryEmail", + new LocalDirectorySSLConfig(dockerCertPath)); + } catch (IOException exception) { + throw new RuntimeException(exception); + } + } + + private static String homeDir() { + return "target/test-classes/someHomeDir"; + } + + private static String dockerCertPath() { + return homeDir() + "/.docker"; + } + + @Test + public void equals() { + assertEquals(EXAMPLE_CONFIG, newExampleConfig()); + } + + @Test + public void environmentDockerHost() { + + // given docker host in env + Map env = new HashMap<>(); + env.put(DefaultDockerClientConfig.DOCKER_HOST, "tcp://baz:8768"); + // and it looks to be SSL disabled + env.remove("DOCKER_CERT_PATH"); + + // given default cert path + Properties systemProperties = new Properties(); + systemProperties.setProperty("user.name", "someUserName"); + systemProperties.setProperty("user.home", homeDir()); + + // when you build a config + DefaultDockerClientConfig config = buildConfig(env, systemProperties); + + assertEquals(URI.create("tcp://baz:8768"), config.getDockerHost()); + } + + @Test + public void dockerContextFromConfig() { + // given home directory with docker contexts configured + Properties systemProperties = new Properties(); + systemProperties.setProperty("user.home", "target/test-classes/dockerContextHomeDir"); + + // and an empty environment + Map env = new HashMap<>(); + + // when you build a config + DefaultDockerClientConfig config = buildConfig(env, systemProperties); + + assertEquals(URI.create("unix:///configcontext.sock"), config.getDockerHost()); + } + + @Test + public void dockerContextFromEnvironmentVariable() { + // given home directory with docker contexts + Properties systemProperties = new Properties(); + systemProperties.setProperty("user.home", "target/test-classes/dockerContextHomeDir"); + + // and an environment variable that overrides docker context + Map env = new HashMap<>(); + env.put(DefaultDockerClientConfig.DOCKER_CONTEXT, "envvarcontext"); + + // when you build a config + DefaultDockerClientConfig config = buildConfig(env, systemProperties); + + assertEquals(URI.create("unix:///envvarcontext.sock"), config.getDockerHost()); + } + + @Test + public void dockerContextWithDockerHostAndTLS() { + // given home directory with docker contexts + Properties systemProperties = new Properties(); + systemProperties.setProperty("user.home", "target/test-classes/dockerContextHomeDir"); + + // and an environment variable that overrides docker context + Map env = new HashMap<>(); + env.put(DefaultDockerClientConfig.DOCKER_CONTEXT, "remote"); + + // when you build a config + DefaultDockerClientConfig config = buildConfig(env, systemProperties); + + assertEquals(URI.create("tcp://remote:2376"), config.getDockerHost()); + assertTrue("SSL config is set", config.getSSLConfig() instanceof LocalDirectorySSLConfig); + assertTrue("SSL directory is set", ((LocalDirectorySSLConfig)config.getSSLConfig()).getDockerCertPath().endsWith("dockerContextHomeDir/.docker/contexts/tls/b71199ebd070b36beab7317920c2c2f1d777df8d05e5527d8458fda57cb17a7a/docker")); + } + + @Test + public void environment() { + + // given a default config in env properties + Map env = new HashMap<>(); + env.put(DefaultDockerClientConfig.DOCKER_HOST, "tcp://foo"); + env.put(DefaultDockerClientConfig.API_VERSION, "apiVersion"); + env.put(DefaultDockerClientConfig.REGISTRY_USERNAME, "registryUsername"); + env.put(DefaultDockerClientConfig.REGISTRY_PASSWORD, "registryPassword"); + env.put(DefaultDockerClientConfig.REGISTRY_EMAIL, "registryEmail"); + env.put(DefaultDockerClientConfig.REGISTRY_URL, "registryUrl"); + env.put(DefaultDockerClientConfig.DOCKER_CONFIG, "dockerConfig"); + env.put(DefaultDockerClientConfig.DOCKER_CERT_PATH, dockerCertPath()); + env.put(DefaultDockerClientConfig.DOCKER_TLS_VERIFY, "1"); + + // when you build a config + DefaultDockerClientConfig config = buildConfig(env, new Properties()); + + // then we get the example object + assertEquals(EXAMPLE_CONFIG_FULLY_LOADED, config); + } + + @Test + public void emptyHost() { + Map env = new HashMap<>(); + env.put(DefaultDockerClientConfig.DOCKER_HOST, " "); + + DefaultDockerClientConfig config = buildConfig(env, new Properties()); + + assertEquals( + DefaultDockerClientConfig.DEFAULT_DOCKER_HOST, + config.getDockerHost().toString() + ); + } + + private DefaultDockerClientConfig buildConfig(Map env, Properties systemProperties) { + return DefaultDockerClientConfig.createDefaultConfigBuilder(env, systemProperties).build(); + } + + @Test + public void defaults() { + + // given default cert path + Properties systemProperties = new Properties(); + systemProperties.setProperty("user.name", "someUserName"); + systemProperties.setProperty("user.home", homeDir()); + + // when you build config + DefaultDockerClientConfig config = buildConfig(Collections. emptyMap(), systemProperties); + + // then the cert path is as expected + assertEquals(URI.create("unix:///var/run/docker.sock"), config.getDockerHost()); + assertEquals("someUserName", config.getRegistryUsername()); + assertEquals(AuthConfig.DEFAULT_SERVER_ADDRESS, config.getRegistryUrl()); + assertEquals(RemoteApiVersion.unknown(), config.getApiVersion()); + assertEquals(homeDir() + "/.docker", config.getDockerConfigPath()); + assertNull(config.getSSLConfig()); + } + + @Test + public void systemProperties() { + + // given system properties based on the example + Properties systemProperties = new Properties(); + systemProperties.put(DefaultDockerClientConfig.DOCKER_HOST, "tcp://foo"); + systemProperties.put(DefaultDockerClientConfig.API_VERSION, "apiVersion"); + systemProperties.put(DefaultDockerClientConfig.REGISTRY_USERNAME, "registryUsername"); + systemProperties.put(DefaultDockerClientConfig.REGISTRY_PASSWORD, "registryPassword"); + systemProperties.put(DefaultDockerClientConfig.REGISTRY_EMAIL, "registryEmail"); + systemProperties.put(DefaultDockerClientConfig.REGISTRY_URL, "registryUrl"); + systemProperties.put(DefaultDockerClientConfig.DOCKER_CONFIG, "dockerConfig"); + systemProperties.put(DefaultDockerClientConfig.DOCKER_CERT_PATH, dockerCertPath()); + systemProperties.put(DefaultDockerClientConfig.DOCKER_TLS_VERIFY, "1"); + + // when you build new config + DefaultDockerClientConfig config = buildConfig(Collections. emptyMap(), systemProperties); + + // then it is the same as the example + assertEquals(EXAMPLE_CONFIG_FULLY_LOADED, config); + + } + + @Test + public void serializableTest() { + final byte[] serialized = SerializationUtils.serialize(EXAMPLE_CONFIG); + final DefaultDockerClientConfig deserialized = (DefaultDockerClientConfig) SerializationUtils.deserialize(serialized); + + assertThat("Deserialized object mush match source object", deserialized, equalTo(EXAMPLE_CONFIG)); + } + + @Test() + public void testSslContextEmpty() { + new DefaultDockerClientConfig(URI.create("tcp://foo"), new DockerConfigFile(), "dockerConfig", "apiVersion", "registryUrl", "registryUsername", "registryPassword", "registryEmail", + null); + } + + + + @Test() + public void testTlsVerifyAndCertPath() { + new DefaultDockerClientConfig(URI.create("tcp://foo"), new DockerConfigFile(), "dockerConfig", "apiVersion", "registryUrl", "registryUsername", "registryPassword", "registryEmail", + new LocalDirectorySSLConfig(dockerCertPath())); + } + + @Test() + public void testAnyHostScheme() { + URI dockerHost = URI.create("a" + UUID.randomUUID().toString().replace("-", "") + "://foo"); + new DefaultDockerClientConfig(dockerHost, new DockerConfigFile(), "dockerConfig", "apiVersion", "registryUrl", "registryUsername", "registryPassword", "registryEmail", + null); + } + + @Test + public void withDockerTlsVerify() throws Exception { + DefaultDockerClientConfig.Builder builder = new DefaultDockerClientConfig.Builder(); + Field field = builder.getClass().getDeclaredField("dockerTlsVerify"); + field.setAccessible(true); + + builder.withDockerTlsVerify(""); + assertThat((Boolean) field.get(builder), is(false)); + + builder.withDockerTlsVerify("false"); + assertThat((Boolean) field.get(builder), is(false)); + + builder.withDockerTlsVerify("FALSE"); + assertThat((Boolean) field.get(builder), is(false)); + + builder.withDockerTlsVerify("true"); + assertThat((Boolean) field.get(builder), is(true)); + + builder.withDockerTlsVerify("TRUE"); + assertThat((Boolean) field.get(builder), is(true)); + + builder.withDockerTlsVerify("0"); + assertThat((Boolean) field.get(builder), is(false)); + + builder.withDockerTlsVerify("1"); + assertThat((Boolean) field.get(builder), is(true)); + } + + @Test + public void dockerHostSetExplicitlyOnSetter() { + DefaultDockerClientConfig.Builder builder = DefaultDockerClientConfig.createDefaultConfigBuilder(Collections.emptyMap(), new Properties()); + assertThat(builder.isDockerHostSetExplicitly(), is(false)); + + builder.withDockerHost("tcp://foo"); + assertThat(builder.isDockerHostSetExplicitly(), is(true)); + } + + @Test + public void dockerHostSetExplicitlyOnSystemProperty() { + Properties systemProperties = new Properties(); + systemProperties.put(DefaultDockerClientConfig.DOCKER_HOST, "tcp://foo"); + + DefaultDockerClientConfig.Builder builder = DefaultDockerClientConfig.createDefaultConfigBuilder(Collections.emptyMap(), systemProperties); + + assertThat(builder.isDockerHostSetExplicitly(), is(true)); + } + + @Test + public void dockerHostSetExplicitlyOnEnv() { + Map env = new HashMap<>(); + env.put(DefaultDockerClientConfig.DOCKER_HOST, "tcp://foo"); + + DefaultDockerClientConfig.Builder builder = DefaultDockerClientConfig.createDefaultConfigBuilder(env, new Properties()); + + assertThat(builder.isDockerHostSetExplicitly(), is(true)); + } + + @Test + public void dockerHostSetExplicitlyIfSetToDefaultByUser() { + Map env = new HashMap<>(); + env.put(DefaultDockerClientConfig.DOCKER_HOST, DefaultDockerClientConfig.DEFAULT_DOCKER_HOST); + + DefaultDockerClientConfig.Builder builder = DefaultDockerClientConfig.createDefaultConfigBuilder(env, new Properties()); + + assertThat(builder.isDockerHostSetExplicitly(), is(true)); + } + + + @Test + public void testGetAuthConfigurationsFromDockerCfg() throws URISyntaxException, IOException { + File cfgFile = new File(Resources.getResource("com.github.dockerjava.core/registry.v1").toURI()); + DockerConfigFile dockerConfigFile = + DockerConfigFile.loadConfig(DockerClientConfig.getDefaultObjectMapper(), cfgFile.getAbsolutePath()); + DefaultDockerClientConfig clientConfig = new DefaultDockerClientConfig(URI.create( + "unix://foo"), dockerConfigFile, cfgFile.getAbsolutePath(), "apiVersion", "registryUrl", "registryUsername", "registryPassword", + "registryEmail", null); + + AuthConfigurations authConfigurations = clientConfig.getAuthConfigurations(); + assertThat(authConfigurations, notNullValue()); + assertThat(authConfigurations.getConfigs().get("https://test.docker.io/v1/"), notNullValue()); + + AuthConfig authConfig = authConfigurations.getConfigs().get("https://test.docker.io/v1/"); + assertThat(authConfig.getUsername(), equalTo("user")); + assertThat(authConfig.getPassword(), equalTo("password")); + } + + @Test + public void testGetAuthConfigurationsFromConfigJson() throws URISyntaxException, IOException { + File cfgFile = new File(Resources.getResource("com.github.dockerjava.core/registry.v2").toURI()); + DockerConfigFile dockerConfigFile = + DockerConfigFile.loadConfig(DockerClientConfig.getDefaultObjectMapper(), cfgFile.getAbsolutePath()); + DefaultDockerClientConfig clientConfig = new DefaultDockerClientConfig(URI.create( + "unix://foo"), dockerConfigFile, cfgFile.getAbsolutePath(), "apiVersion", "registryUrl", "registryUsername", "registryPassword", + "registryEmail", null); + + AuthConfigurations authConfigurations = clientConfig.getAuthConfigurations(); + assertThat(authConfigurations, notNullValue()); + assertThat(authConfigurations.getConfigs().get("https://test.docker.io/v2/"), notNullValue()); + + AuthConfig authConfig = authConfigurations.getConfigs().get("https://test.docker.io/v2/"); + assertThat(authConfig.getUsername(), equalTo("user")); + assertThat(authConfig.getPassword(), equalTo("password")); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/core/DockerClientBuilderTest.java b/docker-java/src/test/java/com/github/dockerjava/core/DockerClientBuilderTest.java new file mode 100644 index 000000000..d826a98ce --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/DockerClientBuilderTest.java @@ -0,0 +1,76 @@ +package com.github.dockerjava.core; + + +import com.github.dockerjava.api.command.DockerCmdExecFactory; +import org.junit.Test; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashSet; +import java.util.List; +import java.util.Set; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertNotNull; + +public class DockerClientBuilderTest { + // Amount of instances created in test + private static final int AMOUNT = 100; + + @Test + public void testConcurrentClientBuilding() throws Exception { + // we use it to check instance uniqueness + final Set instances = Collections.synchronizedSet(new HashSet<>()); + + Runnable runnable = () -> { + DockerCmdExecFactory factory = DockerClientBuilder.getDefaultDockerCmdExecFactory(); + // factory created + assertNotNull(factory); + // and is unique + assertFalse(instances.contains(factory)); + instances.add(factory); + }; + + parallel(AMOUNT, runnable); + // set contains all required unique instances + assertEquals(AMOUNT, instances.size()); + } + + public static void parallel(int threads, final Runnable task) throws Exception { + final ExceptionListener exceptionListener = new ExceptionListener(); + Runnable runnable = () -> { + try { + task.run(); + } catch (Throwable e) { + exceptionListener.onException(e); + } + }; + + List threadList = new ArrayList<>(threads); + for (int i = 0; i < threads; i++) { + Thread thread = new Thread(runnable); + thread.start(); + threadList.add(thread); + } + for (Thread thread : threadList) { + thread.join(); + } + Throwable exception = exceptionListener.getException(); + if (exception != null) { + throw new RuntimeException(exception); + } + } + + private static class ExceptionListener { + private Throwable exception; + + private synchronized void onException(Throwable e) { + exception = e; + } + + private synchronized Throwable getException() { + return exception; + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/core/DockerClientImplTest.java b/docker-java/src/test/java/com/github/dockerjava/core/DockerClientImplTest.java new file mode 100644 index 000000000..6ae88ffd1 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/DockerClientImplTest.java @@ -0,0 +1,42 @@ +package com.github.dockerjava.core; + +import org.junit.Test; + +import java.net.URI; + +import static org.junit.Assert.assertEquals; + + +public class DockerClientImplTest { + + @Test + public void configuredInstanceAuthConfig() { + // given a config with null serverAddress + DefaultDockerClientConfig dockerClientConfig = new DefaultDockerClientConfig(URI.create("tcp://foo"), + new DockerConfigFile(), null, null, null, "", "", "", null); + DockerClientImpl dockerClient = DockerClientImpl.getInstance(dockerClientConfig); + + // when we get the auth config + try { + dockerClient.authConfig(); + throw new AssertionError(); + } catch (NullPointerException e) { + // then we get a NPE with expected message + assertEquals("Configured serverAddress is null.", e.getMessage()); + } + } + + @Test + public void defaultInstanceAuthConfig() { + + System.setProperty("user.home", "target/test-classes/someHomeDir"); + + // given a default client + DockerClientImpl dockerClient = DockerClientImpl.getInstance(); + + // when we get the auth config + dockerClient.authConfig(); + + // then we do not get an exception + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/core/DockerCmdExecFactoryDelegate.java b/docker-java/src/test/java/com/github/dockerjava/core/DockerCmdExecFactoryDelegate.java new file mode 100644 index 000000000..463c63ffe --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/DockerCmdExecFactoryDelegate.java @@ -0,0 +1,25 @@ +package com.github.dockerjava.core; + +import com.github.dockerjava.api.command.DelegatingDockerCmdExecFactory; +import com.github.dockerjava.api.command.DockerCmdExecFactory; + +class DockerCmdExecFactoryDelegate extends DelegatingDockerCmdExecFactory implements DockerClientConfigAware { + + final DockerCmdExecFactory delegate; + + DockerCmdExecFactoryDelegate(DockerCmdExecFactory delegate) { + this.delegate = delegate; + } + + @Override + public final DockerCmdExecFactory getDockerCmdExecFactory() { + return delegate; + } + + @Override + public void init(DockerClientConfig dockerClientConfig) { + if (delegate instanceof DockerClientConfigAware) { + ((DockerClientConfigAware) delegate).init(dockerClientConfig); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/core/DockerConfigFileTest.java b/docker-java/src/test/java/com/github/dockerjava/core/DockerConfigFileTest.java new file mode 100644 index 000000000..76211fc55 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/DockerConfigFileTest.java @@ -0,0 +1,177 @@ +/** + * Copyright (C) 2014 SignalFuse, Inc. + */ +package com.github.dockerjava.core; + +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.ExpectedException; + +import java.io.File; +import java.io.IOException; + +import static org.hamcrest.core.Is.is; +import static org.hamcrest.MatcherAssert.assertThat; + +public class DockerConfigFileTest { + @Rule + public ExpectedException expectedEx = ExpectedException.none(); + + private final File FILESROOT = new File(DockerConfigFileTest.class.getResource("/testAuthConfigFile").getFile()); + + @Test + public void emptyFile() throws IOException { + expectedEx.expect(IOException.class); + expectedEx.expectMessage("The Auth Config file is empty"); + + runTest("emptyFile"); + } + + @Test + public void tooSmallFile() throws IOException { + expectedEx.expect(IOException.class); + expectedEx.expectMessage("The Auth Config file is empty"); + + runTest("tooSmallFile"); + } + + @Test + public void invalidJsonInvalidAuth() throws IOException { + expectedEx.expect(IOException.class); + expectedEx.expectMessage("Invalid auth configuration file"); + + runTest("invalidJsonInvalidAuth"); + } + + @Test + public void invalidLegacyAuthLine() throws IOException { + expectedEx.expect(IOException.class); + expectedEx.expectMessage("Invalid Auth config file"); + + runTest("invalidLegacyAuthLine"); + } + + @Test + public void invalidLegacyInvalidAuth() throws IOException { + expectedEx.expect(IOException.class); + expectedEx.expectMessage("Invalid auth configuration file"); + + runTest("invalidLegacyInvalidAuth"); + } + + @Test + public void invalidLegacyEmailLine() throws IOException { + expectedEx.expect(IOException.class); + expectedEx.expectMessage("Invalid Auth config file"); + + runTest("invalidLegacyEmailLine"); + } + + @Test + public void validLegacyJson() throws IOException { + AuthConfig authConfig1 = new AuthConfig() + .withEmail("foo@example.com") + .withUsername("foo") + .withPassword("bar") + .withRegistryAddress("quay.io"); + + AuthConfig authConfig2 = new AuthConfig() + .withEmail("moo@example.com") + .withUsername("foo1") + .withPassword("bar1") + .withRegistryAddress(AuthConfig.DEFAULT_SERVER_ADDRESS); + + DockerConfigFile expected = new DockerConfigFile(); + expected.addAuthConfig(authConfig1); + expected.addAuthConfig(authConfig2); + + assertThat(runTest("validLegacyJson"), is(expected)); + } + + @Test + public void validJsonWithUnknown() throws IOException { + AuthConfig authConfig1 = new AuthConfig() + .withRegistryAddress("192.168.99.100:32768"); + + AuthConfig authConfig2 = new AuthConfig() + .withEmail("foo@example.com") + .withUsername("foo") + .withPassword("bar") + .withRegistryAddress("https://index.docker.io/v1/"); + + DockerConfigFile expected = new DockerConfigFile(); + expected.addAuthConfig(authConfig1); + expected.addAuthConfig(authConfig2); + DockerConfigFile actual = runTest("validJsonWithUnknown"); + assertThat(actual, is(expected)); + } + + @Test + public void validJsonWithOnlyUnknown() throws IOException { + DockerConfigFile expected = new DockerConfigFile(); + DockerConfigFile actual = runTest("validJsonWithOnlyUnknown"); + assertThat(actual, is(expected)); + } + + @Test + public void validLegacy() throws IOException { + AuthConfig authConfig = new AuthConfig() + .withEmail("foo@example.com") + .withUsername("foo") + .withPassword("bar") + .withRegistryAddress(AuthConfig.DEFAULT_SERVER_ADDRESS); + + DockerConfigFile expected = new DockerConfigFile(); + expected.addAuthConfig(authConfig); + + assertThat(runTest("validLegacy"), is(expected)); + } + + @Test + public void validDockerConfig() throws IOException { + AuthConfig authConfig1 = new AuthConfig() + .withEmail("foo@example.com") + .withUsername("foo") + .withPassword("bar") + .withRegistryAddress("quay.io"); + + AuthConfig authConfig2 = new AuthConfig() + .withEmail("moo@example.com") + .withUsername("foo1") + .withPassword("bar1") + .withRegistryAddress(AuthConfig.DEFAULT_SERVER_ADDRESS); + + DockerConfigFile expected = new DockerConfigFile(); + expected.addAuthConfig(authConfig1); + expected.addAuthConfig(authConfig2); + + assertThat(runTest("validDockerConfig"), is(expected)); + } + + @Test + public void validDockerConfigWithCurrentContext() throws IOException { + DockerConfigFile expected = new DockerConfigFile(); + expected.setCurrentContext("expectedContext"); + + assertThat(runTest("validDockerConfigWithCurrentContext"), is(expected)); + } + + @Test + public void nonExistent() throws IOException { + DockerConfigFile expected = new DockerConfigFile(); + assertThat(runTest("idontexist"), is(expected)); + } + + @Test + public void validJsonAuthsNull() throws IOException { + DockerConfigFile expected = new DockerConfigFile(); + assertThat(runTest("validJsonAuthsNull"), is(expected)); + } + + private DockerConfigFile runTest(String testFileName) throws IOException { + return DockerConfigFile.loadConfig(JSONTestHelper.getMapper(), new File(FILESROOT, testFileName).getAbsolutePath()); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/core/DockerRule.java b/docker-java/src/test/java/com/github/dockerjava/core/DockerRule.java new file mode 100644 index 000000000..af606a5b1 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/DockerRule.java @@ -0,0 +1,212 @@ +package com.github.dockerjava.core; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.DockerClientDelegate; +import com.github.dockerjava.api.command.CreateContainerCmd; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.CreateNetworkCmd; +import com.github.dockerjava.api.command.CreateNetworkResponse; +import com.github.dockerjava.api.command.CreateVolumeCmd; +import com.github.dockerjava.api.command.CreateVolumeResponse; +import com.github.dockerjava.api.exception.ConflictException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.cmd.CmdIT; +import com.github.dockerjava.utils.LogContainerTestCallback; +import org.junit.rules.ExternalResource; +import org.junit.runner.Description; +import org.junit.runners.model.Statement; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.File; +import java.util.HashSet; +import java.util.Set; + +/** + * @author Kanstantsin Shautsou + */ +public class DockerRule extends ExternalResource { + public static final Logger LOG = LoggerFactory.getLogger(DockerRule.class); + public static final String DEFAULT_IMAGE = "busybox:latest"; + + private DockerClient dockerClient; + + private final Set createdContainerIds = new HashSet<>(); + + private final Set createdNetworkIds = new HashSet<>(); + + private final Set createdVolumeNames = new HashSet<>(); + + private final DefaultDockerClientConfig config = config(); + + public DockerClient newClient() { + DockerClientImpl dockerClient = CmdIT.createDockerClient(config); + + dockerClient.withDockerCmdExecFactory( + new DockerCmdExecFactoryDelegate(dockerClient.dockerCmdExecFactory) { + @Override + public CreateContainerCmd.Exec createCreateContainerCmdExec() { + CreateContainerCmd.Exec exec = super.createCreateContainerCmdExec(); + return command -> { + CreateContainerResponse response = exec.exec(command); + createdContainerIds.add(response.getId()); + return response; + }; + } + + @Override + public CreateNetworkCmd.Exec createCreateNetworkCmdExec() { + CreateNetworkCmd.Exec exec = super.createCreateNetworkCmdExec(); + return command -> { + CreateNetworkResponse response = exec.exec(command); + createdNetworkIds.add(response.getId()); + return response; + }; + } + + @Override + public CreateVolumeCmd.Exec createCreateVolumeCmdExec() { + CreateVolumeCmd.Exec exec = super.createCreateVolumeCmdExec(); + return command -> { + CreateVolumeResponse response = exec.exec(command); + createdVolumeNames.add(response.getName()); + return response; + }; + } + } + ); + + return new DockerClientDelegate() { + @Override + protected DockerClient getDockerClient() { + return dockerClient; + } + }; + } + + public DefaultDockerClientConfig getConfig() { + return config; + } + + public DockerClient getClient() { + if (dockerClient != null) { + return dockerClient; + } + return this.dockerClient = newClient(); + } + + @Override + public Statement apply(Statement base, Description description) { + return super.apply(base, description); + } + + @Override + protected void before() throws Throwable { +// LOG.info("======================= BEFORETEST ======================="); + LOG.debug("Connecting to Docker server"); + + + try { + getClient().inspectImageCmd(DEFAULT_IMAGE).exec(); + } catch (NotFoundException e) { + LOG.info("Pulling image 'busybox'"); + // need to block until image is pulled completely + getClient().pullImageCmd("busybox") + .withTag("latest") + .start() + .awaitCompletion(); + } + +// assertThat(getClient(), notNullValue()); +// LOG.info("======================= END OF BEFORETEST =======================\n\n"); + } + + @Override + protected void after() { +// LOG.debug("======================= END OF AFTERTEST ======================="); + createdContainerIds.parallelStream().forEach(containerId -> { + try { + dockerClient.removeContainerCmd(containerId) + .withForce(true) + .withRemoveVolumes(true) + .exec(); + } catch (ConflictException | NotFoundException ignored) { + } catch (Throwable e) { + if (e instanceof InterruptedException) { + Thread.currentThread().interrupt(); + } + LOG.debug("Failed to remove container {}", containerId, e); + } + }); + createdNetworkIds.parallelStream().forEach(networkId -> { + try { + dockerClient.removeNetworkCmd(networkId).exec(); + } catch (ConflictException | NotFoundException ignored) { + } catch (Throwable e) { + if (e instanceof InterruptedException) { + Thread.currentThread().interrupt(); + } + LOG.debug("Failed to remove network {}", networkId, e); + } + }); + createdVolumeNames.parallelStream().forEach(volumeName -> { + try { + dockerClient.removeVolumeCmd(volumeName).exec(); + } catch (ConflictException | NotFoundException ignored) { + } catch (Throwable e) { + if (e instanceof InterruptedException) { + Thread.currentThread().interrupt(); + } + LOG.debug("Failed to remove volume {}", volumeName, e); + } + }); + + try { + dockerClient.close(); + } catch (Exception e) { + LOG.warn("Failed to close the DockerClient", e); + } + } + + private static DefaultDockerClientConfig config() { + return config(null); + } + + public static DefaultDockerClientConfig config(String password) { + DefaultDockerClientConfig.Builder builder = DefaultDockerClientConfig.createDefaultConfigBuilder() + .withApiVersion(RemoteApiVersion.VERSION_1_44) + .withRegistryUrl("https://index.docker.io/v1/"); + if (password != null) { + builder = builder.withRegistryPassword(password); + } + + return builder.build(); + } + + public String buildImage(File baseDir) throws Exception { + return getClient().buildImageCmd(baseDir) + .withNoCache(true) + .start() + .awaitImageId(); + } + + public String containerLog(String containerId) throws Exception { + return getClient().logContainerCmd(containerId) + .withStdOut(true) + .exec(new LogContainerTestCallback()) + .awaitCompletion() + .toString(); + } + + public void ensureContainerRemoved(String container1Name) { + try { + getClient().removeContainerCmd(container1Name) + .withForce(true) + .withRemoveVolumes(true) + .exec(); + } catch (NotFoundException ex) { + // ignore + } + } + +} diff --git a/src/test/java/com/github/dockerjava/core/GoLangFileMatchTest.java b/docker-java/src/test/java/com/github/dockerjava/core/GoLangFileMatchTest.java similarity index 91% rename from src/test/java/com/github/dockerjava/core/GoLangFileMatchTest.java rename to docker-java/src/test/java/com/github/dockerjava/core/GoLangFileMatchTest.java index 62b342595..11ea90e57 100644 --- a/src/test/java/com/github/dockerjava/core/GoLangFileMatchTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/core/GoLangFileMatchTest.java @@ -3,45 +3,23 @@ */ package com.github.dockerjava.core; -import java.io.IOException; - +import com.github.dockerjava.core.exception.GoLangFileMatchException; import org.apache.commons.io.FilenameUtils; -import org.testng.annotations.DataProvider; -import org.testng.annotations.Test; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.junit.runners.Parameterized; -import com.github.dockerjava.core.exception.GoLangFileMatchException; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.fail; -import junit.framework.Assert; +import java.io.IOException; +@RunWith(Parameterized.class) public class GoLangFileMatchTest { - @Test(dataProvider = "getTestData") - public void testMatch(MatchTestCase testCase) throws IOException { - String pattern = testCase.pattern; - String s = testCase.s; - if (GoLangFileMatch.IS_WINDOWS) { - if (pattern.indexOf('\\') >= 0) { - // no escape allowed on windows. - return; - } - pattern = FilenameUtils.normalize(pattern); - s = FilenameUtils.normalize(s); - } - try { - Boolean matched = GoLangFileMatch.match(pattern, s); - if (testCase.expectException) { - Assert.fail("Expected GoFileMatchException"); - } - Assert.assertEquals(testCase.toString(), testCase.matches, matched); - } catch (GoLangFileMatchException e) { - if (!testCase.expectException) { - throw e; - } - } - } - @DataProvider - public Object[][] getTestData() { + @Parameterized.Parameters + public static Object[][] getTestData() { return new Object[][] {new Object[] {new MatchTestCase("", "abc", false, false)}, new Object[] {new MatchTestCase("abc", "abc", true, false)}, new Object[] {new MatchTestCase("*", "abc", true, false)}, @@ -103,7 +81,35 @@ public Object[][] getTestData() { new Object[] {new MatchTestCase("**/c", "a/b/c", true, false)}}; } - private final class MatchTestCase { + @Parameterized.Parameter + public MatchTestCase testCase; + + @Test + public void testMatch() { + String pattern = testCase.pattern; + String s = testCase.s; + if (GoLangFileMatch.IS_WINDOWS) { + if (pattern.indexOf('\\') >= 0) { + // no escape allowed on windows. + return; + } + pattern = FilenameUtils.normalize(pattern); + s = FilenameUtils.normalize(s); + } + try { + Boolean matched = GoLangFileMatch.match(pattern, s); + if (testCase.expectException) { + fail("Expected GoFileMatchException"); + } + assertEquals(testCase.toString(), testCase.matches, matched); + } catch (GoLangFileMatchException e) { + if (!testCase.expectException) { + throw e; + } + } + } + + private final static class MatchTestCase { private final String pattern; private final String s; diff --git a/docker-java/src/test/java/com/github/dockerjava/core/NameParserTest.java b/docker-java/src/test/java/com/github/dockerjava/core/NameParserTest.java new file mode 100644 index 000000000..89ad131f6 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/NameParserTest.java @@ -0,0 +1,191 @@ +/* + * Created on 17.08.2015 + */ +package com.github.dockerjava.core; + +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.core.NameParser.HostnameReposName; +import com.github.dockerjava.core.NameParser.ReposTag; +import com.github.dockerjava.core.exception.InvalidRepositoryNameException; +import org.apache.commons.lang3.StringUtils; +import org.junit.Test; + +import static org.junit.Assert.assertEquals; + +/** + * + * + * @author Marcus Linke + * + */ +public class NameParserTest { + + @Test + public void testValidateRepoName() { + NameParser.validateRepoName("repository"); + NameParser.validateRepoName("namespace/repository"); + NameParser.validateRepoName("namespace-with-dashes/repository"); + NameParser.validateRepoName("namespace/repository-with-dashes"); + NameParser.validateRepoName("namespace.with.dots/repository"); + NameParser.validateRepoName("namespace/repository.with.dots"); + NameParser.validateRepoName("namespace_with_underscores/repository"); + NameParser.validateRepoName("namespace/repository_with_underscores"); + } + + @Test(expected = InvalidRepositoryNameException.class) + public void testValidateRepoNameEmpty() { + NameParser.validateRepoName(""); + } + + @Test(expected = InvalidRepositoryNameException.class) + public void testValidateRepoNameExceedsMaxLength() { + NameParser.validateRepoName(StringUtils.repeat("repository", 255)); + } + + @Test(expected = InvalidRepositoryNameException.class) + public void testValidateRepoNameEndWithDash() { + NameParser.validateRepoName("repository-"); + } + + @Test(expected = InvalidRepositoryNameException.class) + public void testValidateRepoNameStartWithDash() { + NameParser.validateRepoName("-repository"); + } + + @Test(expected = InvalidRepositoryNameException.class) + public void testValidateRepoNameEndWithDot() { + NameParser.validateRepoName("repository."); + } + + @Test(expected = InvalidRepositoryNameException.class) + public void testValidateRepoNameStartWithDot() { + NameParser.validateRepoName(".repository"); + } + + @Test(expected = InvalidRepositoryNameException.class) + public void testValidateRepoNameEndWithUnderscore() { + NameParser.validateRepoName("repository_"); + } + + @Test(expected = InvalidRepositoryNameException.class) + public void testValidateRepoNameStartWithUnderscore() { + NameParser.validateRepoName("_repository"); + } + + @Test(expected = InvalidRepositoryNameException.class) + public void testValidateRepoNameWithColon() { + NameParser.validateRepoName("repository:with:colon"); + } + + @Test + public void testResolveSimpleRepositoryName() { + HostnameReposName resolved = NameParser.resolveRepositoryName("repository"); + assertEquals(new HostnameReposName(AuthConfig.DEFAULT_SERVER_ADDRESS, "repository"), resolved); + } + + @Test + public void testResolveRepositoryNameWithTag() { + HostnameReposName resolved = NameParser.resolveRepositoryName("repository:tag"); + assertEquals(new HostnameReposName(AuthConfig.DEFAULT_SERVER_ADDRESS, "repository"), resolved); + } + + @Test + public void testResolveRepositoryNameWithSHA256() { + HostnameReposName resolved = NameParser.resolveRepositoryName("repository@sha256:sha256"); + assertEquals(new HostnameReposName(AuthConfig.DEFAULT_SERVER_ADDRESS, "repository"), resolved); + } + + @Test + public void testResolveRepositoryNameWithTagAndSHA256() { + HostnameReposName resolved = NameParser.resolveRepositoryName("repository:tag@sha256:sha256"); + assertEquals(new HostnameReposName(AuthConfig.DEFAULT_SERVER_ADDRESS, "repository"), resolved); + } + + @Test + public void testResolveRepositoryNameWithNamespace() { + HostnameReposName resolved = NameParser.resolveRepositoryName("namespace/repository"); + assertEquals(new HostnameReposName(AuthConfig.DEFAULT_SERVER_ADDRESS, "namespace/repository"), resolved); + } + + @Test + public void testResolveRepositoryNameWithNamespaceAndSHA256() { + HostnameReposName resolved = NameParser.resolveRepositoryName("namespace/repository@sha256:sha256"); + assertEquals(new HostnameReposName(AuthConfig.DEFAULT_SERVER_ADDRESS, "namespace/repository"), resolved); + } + + @Test + public void testResolveRepositoryNameWithNamespaceAndHostname() { + HostnameReposName resolved = NameParser.resolveRepositoryName("localhost:5000/namespace/repository"); + assertEquals(new HostnameReposName("localhost:5000", "namespace/repository"), resolved); + } + + @Test + public void testResolveRepositoryNameWithNamespaceAndHostnameAndSHA256() { + HostnameReposName resolved = NameParser.resolveRepositoryName("localhost:5000/namespace/repository@sha256:sha256"); + assertEquals(new HostnameReposName("localhost:5000", "namespace/repository"), resolved); + } + + @Test + public void testResolveRepositoryNameWithNamespaceAndHostnameAndTag() { + HostnameReposName resolved = NameParser.resolveRepositoryName("localhost:5000/namespace/repository:tag"); + assertEquals(new HostnameReposName("localhost:5000", "namespace/repository"), resolved); + } + @Test + public void testResolveRepositoryNameWithNamespaceAndHostnameAndTagAndSHA256() { + HostnameReposName resolved = NameParser.resolveRepositoryName("localhost:5000/namespace/repository:tag@sha256:sha256"); + assertEquals(new HostnameReposName("localhost:5000", "namespace/repository"), resolved); + } + + @Test(expected = InvalidRepositoryNameException.class) + public void testResolveRepositoryNameWithIndex() { + NameParser.resolveRepositoryName("index.docker.io/repository"); + } + + @Test + public void testResolveReposTagWithoutTagSimple() { + ReposTag resolved = NameParser.parseRepositoryTag("repository"); + assertEquals(new ReposTag("repository", ""), resolved); + + resolved = NameParser.parseRepositoryTag("namespace/repository"); + assertEquals(new ReposTag("namespace/repository", ""), resolved); + + resolved = NameParser.parseRepositoryTag("localhost:5000/namespace/repository"); + assertEquals(new ReposTag("localhost:5000/namespace/repository", ""), resolved); + } + + @Test + public void testResolveReposTagWithTag() { + ReposTag resolved = NameParser.parseRepositoryTag("repository:tag"); + assertEquals(new ReposTag("repository", "tag"), resolved); + + resolved = NameParser.parseRepositoryTag("namespace/repository:tag"); + assertEquals(new ReposTag("namespace/repository", "tag"), resolved); + + resolved = NameParser.parseRepositoryTag("localhost:5000/namespace/repository:tag"); + assertEquals(new ReposTag("localhost:5000/namespace/repository", "tag"), resolved); + } + + @Test + public void testResolveReposTagWithSHA256() { + ReposTag resolved = NameParser.parseRepositoryTag("repository@sha256:sha256"); + assertEquals(new ReposTag("repository@sha256:sha256", ""), resolved); + + resolved = NameParser.parseRepositoryTag("namespace/repository@sha256:sha256"); + assertEquals(new ReposTag("namespace/repository@sha256:sha256", ""), resolved); + + resolved = NameParser.parseRepositoryTag("localhost:5000/namespace/repository@sha256:sha256"); + assertEquals(new ReposTag("localhost:5000/namespace/repository@sha256:sha256", ""), resolved); + } + + @Test + public void testResolveReposTagWithTagAndSHA256() { + ReposTag resolved = NameParser.parseRepositoryTag("repository:tag@sha256:sha256"); + assertEquals(new ReposTag("repository:tag@sha256:sha256", ""), resolved); + + resolved = NameParser.parseRepositoryTag("namespace/repository:tag@sha256:sha256"); + assertEquals(new ReposTag("namespace/repository:tag@sha256:sha256", ""), resolved); + + resolved = NameParser.parseRepositoryTag("localhost:5000/namespace/repository:tag@sha256:sha256"); + assertEquals(new ReposTag("localhost:5000/namespace/repository:tag@sha256:sha256", ""), resolved); + } +} diff --git a/src/test/java/com/github/dockerjava/core/RemoteApiVersionTest.java b/docker-java/src/test/java/com/github/dockerjava/core/RemoteApiVersionTest.java similarity index 89% rename from src/test/java/com/github/dockerjava/core/RemoteApiVersionTest.java rename to docker-java/src/test/java/com/github/dockerjava/core/RemoteApiVersionTest.java index 011f909ad..06b825868 100644 --- a/src/test/java/com/github/dockerjava/core/RemoteApiVersionTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/core/RemoteApiVersionTest.java @@ -1,7 +1,7 @@ package com.github.dockerjava.core; -import org.apache.commons.lang.SerializationUtils; -import org.testng.annotations.Test; +import org.apache.commons.lang3.SerializationUtils; +import org.junit.Test; import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.Matchers.equalTo; diff --git a/docker-java/src/test/java/com/github/dockerjava/core/command/DockerfileFixture.java b/docker-java/src/test/java/com/github/dockerjava/core/command/DockerfileFixture.java new file mode 100644 index 000000000..913d1758b --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/command/DockerfileFixture.java @@ -0,0 +1,81 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.exception.DockerException; +import com.github.dockerjava.api.exception.InternalServerErrorException; +import com.github.dockerjava.api.exception.NotFoundException; +import com.github.dockerjava.api.model.Image; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.File; + +/** + * Start and stop a single container for testing. + */ +public class DockerfileFixture implements AutoCloseable { + + private static final Logger LOGGER = LoggerFactory.getLogger(DockerfileFixture.class); + + private final DockerClient client; + + private String directory; + + private String repository; + + private String containerId; + + public DockerfileFixture(DockerClient client, String directory) { + this.client = client; + this.directory = directory; + } + + public void open() throws Exception { + + LOGGER.info("building {}", directory); + + client.buildImageCmd(new File("src/test/resources", directory)).withNoCache(true) + .start().awaitImageId(); + + Image lastCreatedImage = client.listImagesCmd().exec().get(0); + + repository = lastCreatedImage.getRepoTags()[0]; + + LOGGER.info("created {} {}", lastCreatedImage.getId(), repository); + + containerId = client.createContainerCmd(lastCreatedImage.getId()).exec().getId(); + + LOGGER.info("starting {}", containerId); + + client.startContainerCmd(containerId).exec(); + } + + @Override + public void close() throws Exception { + + if (containerId != null) { + LOGGER.info("removing container {}", containerId); + try { + client.removeContainerCmd(containerId).withForce(true) // stop too + .exec(); + } catch (NotFoundException | InternalServerErrorException ignored) { + LOGGER.info("ignoring {}", ignored.getMessage()); + } + containerId = null; + } + + if (repository != null) { + LOGGER.info("removing repository {}", repository); + try { + client.removeImageCmd(repository).withForce(true).exec(); + } catch (DockerException e) { + LOGGER.info("ignoring {}", e.getMessage()); + } + repository = null; + } + } + + public String getContainerId() { + return containerId; + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/core/command/FrameReaderITest.java b/docker-java/src/test/java/com/github/dockerjava/core/command/FrameReaderITest.java new file mode 100644 index 000000000..16c456164 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/command/FrameReaderITest.java @@ -0,0 +1,111 @@ +package com.github.dockerjava.core.command; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.api.model.StreamType; +import com.github.dockerjava.core.DockerClientBuilder; +import com.github.dockerjava.junit.category.Integration; +import org.junit.After; +import org.junit.Before; +import org.junit.Test; +import org.junit.experimental.categories.Category; + +import java.util.ArrayList; +import java.util.Iterator; +import java.util.List; + +import static com.github.dockerjava.junit.DockerAssume.assumeNotSwarm; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsInAnyOrder; +import static org.hamcrest.Matchers.hasSize; +import static org.junit.Assert.assertEquals; + +@Category(Integration.class) +public class FrameReaderITest { + + private DockerClient dockerClient; + + private DockerfileFixture dockerfileFixture; + + @Before + public void beforeTest() throws Exception { + dockerClient = DockerClientBuilder.getInstance().build(); + dockerfileFixture = new DockerfileFixture(dockerClient, "frameReaderDockerfile"); + dockerfileFixture.open(); + } + + @After + public void deleteDockerContainerImage() throws Exception { + dockerfileFixture.close(); + dockerClient.close(); + } + + @Test + public void canCloseFrameReaderAndReadExpectedLines() throws Exception { + assumeNotSwarm("", dockerClient); + + // wait for the container to be successfully executed + int exitCode = dockerClient.waitContainerCmd(dockerfileFixture.getContainerId()) + .start().awaitStatusCode(); + assertEquals(0, exitCode); + + final List loggingFrames = getLoggingFrames(); + final Frame outFrame = new Frame(StreamType.STDOUT, "to stdout\n".getBytes()); + final Frame errFrame = new Frame(StreamType.STDERR, "to stderr\n".getBytes()); + + assertThat(loggingFrames, containsInAnyOrder(outFrame, errFrame)); + assertThat(loggingFrames, hasSize(2)); + } + + private List getLoggingFrames() throws Exception { + + FrameReaderITestCallback collectFramesCallback = new FrameReaderITestCallback(); + + dockerClient.logContainerCmd(dockerfileFixture.getContainerId()).withStdOut(true).withStdErr(true) + .withTailAll() + // we can't follow stream here as it blocks reading from resulting InputStream infinitely + // .withFollowStream() + .exec(collectFramesCallback).awaitCompletion(); + + return collectFramesCallback.frames; + } + + @Test + public void canLogInOneThreadAndExecuteCommandsInAnother() throws Exception { + + Thread thread = new Thread(() -> { + try { + Iterator frames = getLoggingFrames().iterator(); + + while (frames.hasNext()) { + frames.next(); + } + + } catch (Exception e) { + throw new RuntimeException(e); + } + }); + + thread.start(); + + try (DockerfileFixture busyboxDockerfile = new DockerfileFixture(dockerClient, "busyboxDockerfile")) { + busyboxDockerfile.open(); + } + + thread.join(); + + } + + public static class FrameReaderITestCallback extends ResultCallback.Adapter { + + public List frames = new ArrayList<>(); + + @Override + public void onNext(Frame item) { + frames.add(item); + super.onNext(item); + } + + } +} diff --git a/src/test/java/com/github/dockerjava/core/command/FrameReaderTest.java b/docker-java/src/test/java/com/github/dockerjava/core/command/FrameReaderTest.java similarity index 75% rename from src/test/java/com/github/dockerjava/core/command/FrameReaderTest.java rename to docker-java/src/test/java/com/github/dockerjava/core/command/FrameReaderTest.java index 38ddf1402..580e278f4 100644 --- a/src/test/java/com/github/dockerjava/core/command/FrameReaderTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/core/command/FrameReaderTest.java @@ -1,17 +1,17 @@ package com.github.dockerjava.core.command; -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertNull; +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.api.model.StreamType; +import org.junit.Test; import java.io.IOException; import java.io.InputStream; import java.util.ArrayList; import java.util.List; -import org.testng.annotations.Test; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNull; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.api.model.StreamType; public class FrameReaderTest { public static final int HEADER_SIZE = 8; @@ -34,7 +34,7 @@ public void endOfStreamReturnsNull() throws Exception { @Test public void stdInBytesFrameReturnsFrame() throws Exception { - assertEquals(nextFrame(0, 0, 0, 0, 0, 0, 0, 0), new Frame(StreamType.STDIN, new byte[0])); + assertEquals(new Frame(StreamType.STDIN, new byte[0]), nextFrame(0, 0, 0, 0, 0, 0, 0, 0)); } private Frame nextFrame(int... bytes) throws IOException { @@ -44,12 +44,12 @@ private Frame nextFrame(int... bytes) throws IOException { @Test public void stdOutBytesFrameReturnsFrame() throws Exception { - assertEquals(nextFrame(1, 0, 0, 0, 0, 0, 0, 0), new Frame(StreamType.STDOUT, new byte[0])); + assertEquals(new Frame(StreamType.STDOUT, new byte[0]), nextFrame(1, 0, 0, 0, 0, 0, 0, 0)); } @Test public void stdErrBytesFrameReturnsFrame() throws Exception { - assertEquals(nextFrame(2, 0, 0, 0, 0, 0, 0, 0), new Frame(StreamType.STDERR, new byte[0])); + assertEquals(new Frame(StreamType.STDERR, new byte[0]), nextFrame(2, 0, 0, 0, 0, 0, 0, 0)); } private void setBytes(int... bytes) { diff --git a/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileAddMultipleFilesTest.java b/docker-java/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileAddMultipleFilesTest.java similarity index 79% rename from src/test/java/com/github/dockerjava/core/dockerfile/DockerfileAddMultipleFilesTest.java rename to docker-java/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileAddMultipleFilesTest.java index b37754a6a..6406647cf 100644 --- a/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileAddMultipleFilesTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileAddMultipleFilesTest.java @@ -1,29 +1,30 @@ package com.github.dockerjava.core.dockerfile; -import static com.google.common.collect.Collections2.transform; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsInAnyOrder; +import com.google.common.base.Function; +import org.junit.Test; import java.io.File; import java.io.IOException; import java.util.Collection; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.annotations.Test; - -import com.google.common.base.Function; +import static com.google.common.collect.Collections2.transform; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsInAnyOrder; public class DockerfileAddMultipleFilesTest { - private static final Logger log = LoggerFactory.getLogger(DockerfileAddMultipleFilesTest.class); + private static final Function TO_FILE_NAMES = File::getName; + + @Test + public void ignoreAllBut() throws Exception { + File baseDir = fileFromBuildTestResource("dockerignore/IgnoreAllBut"); + Dockerfile dockerfile = new Dockerfile(new File(baseDir, "Dockerfile"), baseDir); + Dockerfile.ScannedResult result = dockerfile.parse(); + Collection filesToAdd = transform(result.filesToAdd, TO_FILE_NAMES); - private static final Function TO_FILE_NAMES = new Function() { - @Override - public String apply(File file) { - return file.getName(); - } - }; + assertThat(filesToAdd, + containsInAnyOrder("Dockerfile", "foo.jar")); + } @Test public void nestedDirsPatterns() throws Exception { @@ -59,11 +60,12 @@ public void ineffectiveIgnorePattern() throws Exception { @Test public void addFiles() throws IOException { File baseDir = fileFromBuildTestResource("ADD/files"); + new File(baseDir, "emptydir").mkdir(); Dockerfile dockerfile = new Dockerfile(new File(baseDir, "Dockerfile"), baseDir); Dockerfile.ScannedResult result = dockerfile.parse(); Collection filesToAdd = transform(result.filesToAdd, TO_FILE_NAMES); - assertThat(filesToAdd, containsInAnyOrder("Dockerfile", "src1", "src2")); + assertThat(filesToAdd, containsInAnyOrder("emptydir", "Dockerfile", "src1", "src2")); } private File fileFromBuildTestResource(String resource) { diff --git a/docker-java/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileStatementAddTest.java b/docker-java/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileStatementAddTest.java new file mode 100644 index 000000000..d36aa4f4e --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileStatementAddTest.java @@ -0,0 +1,58 @@ +package com.github.dockerjava.core.dockerfile; + +import com.github.dockerjava.api.exception.DockerClientException; +import com.google.common.base.Optional; +import org.hamcrest.Matcher; +import org.junit.Test; +import org.junit.experimental.runners.Enclosed; +import org.junit.runner.RunWith; +import org.junit.runners.Parameterized; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.contains; +import static org.hamcrest.Matchers.containsInAnyOrder; +import static org.hamcrest.Matchers.is; + +@RunWith(Enclosed.class) +public class DockerfileStatementAddTest { + + @RunWith(Parameterized.class) + public static final class ParamTests { + + @Parameterized.Parameters(name = "{0} {1} {2}") + public static Object[][] data() { + return new Object[][]{{"ADD src dest", contains("src"), "dest"}, + {"ADD \"src file\" \"dest\"", contains("src file"), "dest"}, + {"ADD src\"file dest", contains("src\"file"), "dest"}, + {"ADD src1 src2 dest", containsInAnyOrder("src1", "src2"), "dest"}, + {"COPY src dest", contains("src"), "dest"}, + {"COPY \"src file\" \"dest\"", contains("src file"), "dest"}, + {"COPY src\"file dest", contains("src\"file"), "dest"}, + {"COPY src1 src2 dest", containsInAnyOrder("src1", "src2"), "dest"}}; + } + + @Parameterized.Parameter + public String command; + + @Parameterized.Parameter(1) + public Matcher matchesExpectation; + + @Parameterized.Parameter(2) + public String expectedDest; + + @Test + public void testAddOrCopyPattern() { + Optional optionalAdd = DockerfileStatement.Add.create(command); + assertThat(optionalAdd.isPresent(), is(true)); + assertThat(optionalAdd.get().sources, matchesExpectation); + assertThat(optionalAdd.get().destination, is(expectedDest)); + } + } + + public static final class Tests { + @Test(expected = DockerClientException.class) + public void shouldThrowExceptionIfDestNotSpecified() { + DockerfileStatement.Add.create("ADD src"); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/core/util/CertificateUtilsTest.java b/docker-java/src/test/java/com/github/dockerjava/core/util/CertificateUtilsTest.java new file mode 100644 index 000000000..c29cedcf9 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/util/CertificateUtilsTest.java @@ -0,0 +1,108 @@ +package com.github.dockerjava.core.util; + +import org.bouncycastle.jce.provider.BouncyCastleProvider; +import org.junit.AfterClass; +import org.junit.BeforeClass; +import org.junit.Test; + +import java.io.File; +import java.io.IOException; +import java.nio.file.Files; +import java.nio.file.Paths; +import java.security.KeyStore; +import java.security.Security; +import java.security.cert.Certificate; +import java.util.List; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.core.Is.is; + +public class CertificateUtilsTest { + private static final String baseDir = CertificateUtilsTest.class.getResource( + CertificateUtilsTest.class.getSimpleName() + "/").getFile(); + + @BeforeClass + public static void init() { + Security.addProvider(new BouncyCastleProvider()); + } + + @AfterClass + public static void tearDown() { + Security.removeProvider(BouncyCastleProvider.PROVIDER_NAME); + } + + @Test + public void allFilesExist() { + assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "allFilesExist"), is(true)); + } + + @Test + public void caAndCertAndKeyMissing() { + assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "caAndCertAndKeyMissing"), is(false)); + } + + @Test + public void caAndCertMissing() { + assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "caAndCertMissing"), is(false)); + } + + @Test + public void caAndKeyMissing() { + assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "caAndKeyMissing"), is(false)); + } + + @Test + public void caMissing() { + assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "caMissing"), is(false)); + } + + @Test + public void certAndKeyMissing() { + assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "certAndKeyMissing"), is(false)); + } + + @Test + public void certMissing() { + assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "certMissing"), is(false)); + } + + @Test + public void keyMissing() { + assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "keyMissing"), is(false)); + } + + @Test + public void readCaCert() throws Exception { + String capem = readFileAsString("caTest/single_ca.pem"); + KeyStore keyStore = CertificateUtils.createTrustStore(capem); + assertThat(keyStore.size(), is(1)); + assertThat(keyStore.isCertificateEntry("ca-1"), is(true)); + } + + @Test + public void readMultipleCaCerts() throws Exception { + String capem = readFileAsString("caTest/multiple_ca.pem"); + KeyStore keyStore = CertificateUtils.createTrustStore(capem); + assertThat(keyStore.size(), is(2)); + assertThat(keyStore.isCertificateEntry("ca-1"), is(true)); + assertThat(keyStore.isCertificateEntry("ca-2"), is(true)); + } + + @Test + public void readCert() throws Exception { + String certpem = readFileAsString("caTest/single_ca.pem"); + List certs = CertificateUtils.loadCertificates(certpem); + assertThat(certs.size(), is(1)); + } + + @Test + public void readMultipleCerts() throws Exception { + String certpem = readFileAsString("caTest/multiple_ca.pem"); + List certs = CertificateUtils.loadCertificates(certpem); + assertThat(certs.size(), is(2)); + } + + private String readFileAsString(String path) throws IOException { + return new String(Files.readAllBytes(Paths.get(new File(baseDir + path).getPath()))); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/core/util/CompressArchiveUtilTest.java b/docker-java/src/test/java/com/github/dockerjava/core/util/CompressArchiveUtilTest.java new file mode 100644 index 000000000..f15085d1c --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/core/util/CompressArchiveUtilTest.java @@ -0,0 +1,320 @@ +package com.github.dockerjava.core.util; + +import org.apache.commons.compress.archivers.tar.TarArchiveEntry; +import org.apache.commons.compress.archivers.tar.TarArchiveInputStream; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.TemporaryFolder; + +import java.io.BufferedInputStream; +import java.io.ByteArrayInputStream; +import java.io.File; +import java.io.FileInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.util.ArrayList; +import java.util.List; +import java.util.zip.GZIPInputStream; + +import static java.util.Arrays.asList; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertTrue; + +public class CompressArchiveUtilTest { + + @Rule + public TemporaryFolder tempFolder = new TemporaryFolder(); + + @Test + public void tarWithRegularFileAsInput() throws Exception { + Path archiveSourceFile = tempFolder.getRoot().toPath().resolve("sourceFile"); + createFileWithContent(archiveSourceFile); + + // ChildrenOnly = false + Path tarGzFile = tempFolder.newFile("archive.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceFile, tarGzFile, true, false); + assertEquals(1, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsNonEmptyFile(tarGzFile.toFile(), "sourceFile"); + + // ChildrenOnly = true, this option make no sense when input is a file but still, let's test it + // to make sure it behaves as expected + tarGzFile = tempFolder.newFile("archiveChildrenOnly.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceFile, tarGzFile, true, false); + assertEquals(1, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsNonEmptyFile(tarGzFile.toFile(), "sourceFile"); + } + + @Test + public void tarWithExecutableFileAsInput() throws Exception { + Path archiveSourceFile = tempFolder.getRoot().toPath().resolve("executableFile.sh"); + createFileWithContent(archiveSourceFile); + archiveSourceFile.toFile().setExecutable(true); + + // ChildrenOnly = false + Path tarGzFile = tempFolder.newFile("archive.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceFile, tarGzFile, true, false); + assertEquals(1, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsExecutableFile(tarGzFile.toFile(), "executableFile.sh"); + + // ChildrenOnly = true, this option make no sense when input is a file but still, let's test it + // to make sure it behaves as expected + tarGzFile = tempFolder.newFile("archiveChildrenOnly.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceFile, tarGzFile, true, false); + assertEquals(1, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsExecutableFile(tarGzFile.toFile(), "executableFile.sh"); + } + + @Test + public void tarWithSymbolicLinkFileAsInput() throws IOException { + Path archiveSourceFile = tempFolder.getRoot().toPath().resolve("symlinkFile"); + Path linkTargetFile = tempFolder.newFile("link-target").toPath(); + Files.createSymbolicLink(archiveSourceFile, linkTargetFile); + + // ChildrenOnly = false + Path tarGzFile = tempFolder.newFile("archive.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceFile, tarGzFile, true, false); + assertEquals(1, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsSymlink(tarGzFile.toFile(), "symlinkFile", linkTargetFile.toString()); + + // ChildrenOnly = true, this option make no sense when input is a file but still, let's test it + // to make sure it behaves as expected + tarGzFile = tempFolder.newFile("archiveChildrenOnly.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceFile, tarGzFile, true, false); + assertEquals(1, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsSymlink(tarGzFile.toFile(), "symlinkFile", linkTargetFile.toString()); + } + + @Test + public void tarWithfolderAsInput() throws Exception { + Path archiveSourceDir = tempFolder.newFolder("archive-source").toPath(); + createFoldersAndSubFolderWithFiles(archiveSourceDir); + + // ChildrenOnly = false + Path tarGzFile = tempFolder.newFile("archive.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceDir, tarGzFile, true, false); + assertEquals(7, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsDirectory(tarGzFile.toFile(), "archive-source"); + assertTarArchiveEntryIsDirectory(tarGzFile.toFile(), "folderA"); + assertTarArchiveEntryIsDirectory(tarGzFile.toFile(), "folderB"); + assertTarArchiveEntryIsDirectory(tarGzFile.toFile(), "subFolderB"); + assertTarArchiveEntryIsNonEmptyFile(tarGzFile.toFile(), "fileA"); + assertTarArchiveEntryIsNonEmptyFile(tarGzFile.toFile(), "fileB"); + assertTarArchiveEntryIsNonEmptyFile(tarGzFile.toFile(), "subFileB"); + + // ChildrenOnly = true + tarGzFile = tempFolder.newFile("archiveChildrenOnly.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceDir, tarGzFile, true, true); + assertEquals(6, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsDirectory(tarGzFile.toFile(), "folderA"); + assertTarArchiveEntryIsDirectory(tarGzFile.toFile(), "folderB"); + assertTarArchiveEntryIsDirectory(tarGzFile.toFile(), "subFolderB"); + assertTarArchiveEntryIsNonEmptyFile(tarGzFile.toFile(), "fileA"); + assertTarArchiveEntryIsNonEmptyFile(tarGzFile.toFile(), "fileB"); + assertTarArchiveEntryIsNonEmptyFile(tarGzFile.toFile(), "subFileB"); + } + + @Test + public void tarWithfolderAsInputAndNestedExecutableFile() throws Exception { + Path archiveSourceDir = tempFolder.newFolder("archive-source").toPath(); + Path executableFile = archiveSourceDir.resolve("executableFile.sh"); + createFileWithContent(executableFile); + executableFile.toFile().setExecutable(true); + + // ChildrenOnly = false + Path tarGzFile = tempFolder.newFile("archive.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceDir, tarGzFile, true, false); + assertEquals(2, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsDirectory(tarGzFile.toFile(), "archive-source"); + assertTarArchiveEntryIsExecutableFile(tarGzFile.toFile(), "executableFile.sh"); + + // ChildrenOnly = true + tarGzFile = tempFolder.newFile("archiveChildrenOnly.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceDir, tarGzFile, true, true); + assertEquals(1, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsExecutableFile(tarGzFile.toFile(), "executableFile.sh"); + } + + @Test + public void tarWithfolderAsInputAndNestedSymbolicLinkFile() throws Exception { + Path archiveSourceDir = tempFolder.newFolder("archive-source").toPath(); + Path linkTargetFile = tempFolder.newFile("link-target").toPath(); + Path symlinkFile = archiveSourceDir.resolve("symlinkFile"); + Files.createSymbolicLink(symlinkFile, linkTargetFile); + + // ChildrenOnly = false + Path tarGzFile = tempFolder.newFile("archive.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceDir, tarGzFile, true, false); + assertEquals(2, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsDirectory(tarGzFile.toFile(), "archive-source"); + assertTarArchiveEntryIsSymlink(tarGzFile.toFile(), "symlinkFile", linkTargetFile.toString()); + + // ChildrenOnly = true + tarGzFile = tempFolder.newFile("archiveChildrenOnly.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceDir, tarGzFile, true, true); + assertEquals(1, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsSymlink(tarGzFile.toFile(), "symlinkFile", linkTargetFile.toString()); + } + + @Test + public void tarWithfolderAsInputAndNestedSymbolicLinkDir() throws Exception { + Path archiveSourceDir = tempFolder.newFolder("archive-source").toPath(); + Path linkTargetDir = tempFolder.newFolder("link-target").toPath(); + Path symlinkFile = archiveSourceDir.resolve("symlinkFile"); + Files.createSymbolicLink(symlinkFile, linkTargetDir); + + // ChildrenOnly = false + Path tarGzFile = tempFolder.newFile("archive.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceDir, tarGzFile, true, false); + assertEquals(2, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsDirectory(tarGzFile.toFile(), "archive-source"); + assertTarArchiveEntryIsSymlink(tarGzFile.toFile(), "symlinkFile", linkTargetDir.toString()); + + // ChildrenOnly = true + tarGzFile = tempFolder.newFile("archiveChildrenOnly.tar.gz").toPath(); + CompressArchiveUtil.tar(archiveSourceDir, tarGzFile, true, true); + assertEquals(1, getNumberOfEntryInArchive(tarGzFile.toFile())); + assertTarArchiveEntryIsSymlink(tarGzFile.toFile(), "symlinkFile", linkTargetDir.toString()); + } + + @Test + public void archiveTARFilesWithFolderAndFiles() throws Exception { + File archive = CompressArchiveUtil.archiveTARFiles(tempFolder.getRoot(), + createFoldersAndSubFolderWithFiles(tempFolder.getRoot().toPath()), "archive"); + assertEquals(6, getNumberOfEntryInArchive(archive)); + assertTarArchiveEntryIsDirectory(archive, "folderA"); + assertTarArchiveEntryIsDirectory(archive, "folderB"); + assertTarArchiveEntryIsDirectory(archive, "subFolderB"); + assertTarArchiveEntryIsNonEmptyFile(archive, "fileA"); + assertTarArchiveEntryIsNonEmptyFile(archive, "fileB"); + assertTarArchiveEntryIsNonEmptyFile(archive, "subFileB"); + } + + @Test + public void archiveTARFilesWithExecutableFile() throws Exception { + File executableFile = tempFolder.newFile("executableFile.sh"); + executableFile.setExecutable(true); + + File archive = CompressArchiveUtil.archiveTARFiles(tempFolder.getRoot(), asList(executableFile), "archive"); + assertEquals(1, getNumberOfEntryInArchive(archive)); + assertTarArchiveEntryIsExecutableFile(archive, "executableFile.sh"); + } + + @Test + public void archiveTARFilesWithSymbolicLinkFile() throws Exception { + Path linkTargetFile = tempFolder.newFile("link-target").toPath(); + Path symlinkFile = tempFolder.getRoot().toPath().resolve("symlinkFile"); + Files.createSymbolicLink(symlinkFile, linkTargetFile); + + File archive = CompressArchiveUtil.archiveTARFiles(tempFolder.getRoot(), asList(symlinkFile.toFile()), "archive"); + assertEquals(1, getNumberOfEntryInArchive(archive)); + assertTarArchiveEntryIsSymlink(archive, "symlinkFile", linkTargetFile.toString()); + } + + @Test + public void archiveTARFilesWithSymbolicLinkDir() throws Exception { + Path linkTargetDir = tempFolder.newFolder("link-target").toPath(); + Path symlinkFile = tempFolder.getRoot().toPath().resolve("symlinkFile"); + Files.createSymbolicLink(symlinkFile, linkTargetDir); + + File archive = CompressArchiveUtil.archiveTARFiles(tempFolder.getRoot(), asList(symlinkFile.toFile()), "archive"); + assertEquals(1, getNumberOfEntryInArchive(archive)); + assertTarArchiveEntryIsSymlink(archive, "symlinkFile", linkTargetDir.toString()); + } + + private static void assertTarArchiveEntryIsDirectory(File archive, String directoryName) throws IOException { + TarArchiveEntry tarArchiveEntry = getTarArchiveEntry(archive, directoryName); + assertNotNull(tarArchiveEntry); + assertTrue(tarArchiveEntry.isDirectory()); + } + + private static void assertTarArchiveEntryIsNonEmptyFile(File archive, String fileName) throws IOException { + TarArchiveEntry tarArchiveEntry = getTarArchiveEntry(archive, fileName); + assertNotNull(tarArchiveEntry); + assertTrue(tarArchiveEntry.isFile()); + assertTrue(tarArchiveEntry.getSize()>0); + } + + private static void assertTarArchiveEntryIsExecutableFile(File archive, String fileName) throws IOException { + TarArchiveEntry tarArchiveEntry = getTarArchiveEntry(archive, fileName); + assertNotNull(tarArchiveEntry); + assertTrue(tarArchiveEntry.isFile()); + assertEquals("should be executable", 0755, (tarArchiveEntry.getMode() & 0755)); + } + + private static void assertTarArchiveEntryIsSymlink(File archive, String fileName, String expectedTarget) throws IOException { + TarArchiveEntry tarArchiveEntry = getTarArchiveEntry(archive, fileName); + assertNotNull(tarArchiveEntry); + assertTrue("should be a symbolic link", tarArchiveEntry.isSymbolicLink()); + assertEquals(expectedTarget, tarArchiveEntry.getLinkName()); + } + + /** + * Creates the following directory structure with files in the specified + * destination folder + * + * destinationFolder + * |__folderA + * | |__fileA + * |__folderB + * |__fileB + * |__subFolderB + * |__subFileB + * + * + * @param destinationFolder where to create the folder/files. + * @return the list of created files. + * @throws IOException if an error occurs while creating the folders/files. + */ + private static List createFoldersAndSubFolderWithFiles(Path destinationFolder) throws IOException { + List createdFiles = new ArrayList<>(); + Path folderA = destinationFolder.resolve("folderA"); + createdFiles.add(Files.createDirectories(folderA).toFile()); + createdFiles.add(createFileWithContent(folderA.resolve("fileA"))); + + Path folderB = destinationFolder.resolve("folderB"); + createdFiles.add(Files.createDirectories(folderB).toFile()); + createdFiles.add(createFileWithContent(folderB.resolve("fileB"))); + + Path subFolderB = folderB.resolve("subFolderB"); + createdFiles.add(Files.createDirectories(subFolderB).toFile()); + createdFiles.add(createFileWithContent(folderA.resolve("subFileB"))); + return createdFiles; + } + + private static File createFileWithContent(Path fileToCreate) throws IOException { + try (InputStream in = new ByteArrayInputStream("some content".getBytes())) { + Files.copy(in, fileToCreate); + } + return fileToCreate.toFile(); + } + + private static TarArchiveEntry getTarArchiveEntry(File tarArchive, String filename) throws IOException { + try (TarArchiveInputStream tarArchiveInputStream = new TarArchiveInputStream( + new GZIPInputStream(new BufferedInputStream(new FileInputStream(tarArchive))))) { + TarArchiveEntry entry; + while ((entry = tarArchiveInputStream.getNextTarEntry()) != null) { + if (entry.getName().equals(filename) + || entry.getName().endsWith("/" + filename) + || entry.getName().equals(filename + "/") + || entry.getName().endsWith("/" + filename + "/")) { + return entry; + } + } + } + return null; + } + + private static int getNumberOfEntryInArchive(File tarArchive) throws IOException { + int numberOfEntries = 0; + try (TarArchiveInputStream tarArchiveInputStream = new TarArchiveInputStream( + new GZIPInputStream(new BufferedInputStream(new FileInputStream(tarArchive))))) { + while ((tarArchiveInputStream.getNextTarEntry()) != null) { + numberOfEntries++; + } + } + return numberOfEntries; + } +} diff --git a/src/test/java/com/github/dockerjava/core/util/FiltersBuilderTest.java b/docker-java/src/test/java/com/github/dockerjava/core/util/FiltersBuilderTest.java similarity index 93% rename from src/test/java/com/github/dockerjava/core/util/FiltersBuilderTest.java rename to docker-java/src/test/java/com/github/dockerjava/core/util/FiltersBuilderTest.java index e70a3b057..08bf62fd3 100644 --- a/src/test/java/com/github/dockerjava/core/util/FiltersBuilderTest.java +++ b/docker-java/src/test/java/com/github/dockerjava/core/util/FiltersBuilderTest.java @@ -1,12 +1,13 @@ package com.github.dockerjava.core.util; import com.google.common.collect.Maps; -import org.testng.annotations.Test; +import org.junit.Test; import java.util.Map; -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertNotEquals; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotEquals; + /** * @author Vincent Latombe diff --git a/docker-java/src/test/java/com/github/dockerjava/junit/DockerAssume.java b/docker-java/src/test/java/com/github/dockerjava/junit/DockerAssume.java new file mode 100644 index 000000000..2971d7bf3 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/junit/DockerAssume.java @@ -0,0 +1,34 @@ +package com.github.dockerjava.junit; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.core.DockerRule; + +import static com.github.dockerjava.utils.TestUtils.isSwarm; +import static org.junit.Assume.assumeFalse; +import static org.junit.Assume.assumeTrue; + +/** + * @author Kanstantsin Shautsou + */ +public class DockerAssume { + public static void assumeSwarm(DockerClient client) { + assumeTrue(isSwarm(client)); + } + + public static void assumeSwarm(String message, DockerClient client) { + assumeTrue(message, isSwarm(client)); + } + + public static void assumeNotSwarm(DockerClient client) { + assumeFalse(isSwarm(client)); + } + + public static void assumeNotSwarm(String message, DockerRule dockerRule) { + assumeNotSwarm(message, dockerRule.getClient()); + } + + public static void assumeNotSwarm(String message, DockerClient client) { + assumeFalse(message, isSwarm(client)); + } + +} diff --git a/docker-java/src/test/java/com/github/dockerjava/junit/DockerMatchers.java b/docker-java/src/test/java/com/github/dockerjava/junit/DockerMatchers.java new file mode 100644 index 000000000..d0c2a22e6 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/junit/DockerMatchers.java @@ -0,0 +1,66 @@ +package com.github.dockerjava.junit; + +import com.github.dockerjava.api.command.InspectContainerResponse; +import com.github.dockerjava.api.model.Volume; +import com.github.dockerjava.core.DockerRule; +import com.github.dockerjava.core.RemoteApiVersion; +import org.hamcrest.CustomTypeSafeMatcher; +import org.hamcrest.Description; +import org.hamcrest.FeatureMatcher; +import org.hamcrest.Matcher; + +import java.util.ArrayList; +import java.util.List; + +import static com.github.dockerjava.utils.TestUtils.getVersion; + +/** + * @author Kanstantsin Shautsou + */ +public class DockerMatchers { + private DockerMatchers() { + } + + public static MountedVolumes mountedVolumes(Matcher> subMatcher) { + return new MountedVolumes(subMatcher, "Mounted volumes", "mountedVolumes"); + } + + public static class MountedVolumes extends FeatureMatcher> { + public MountedVolumes(Matcher> subMatcher, String featureDescription, String featureName) { + super(subMatcher, featureDescription, featureName); + } + + @Override + public List featureValueOf(InspectContainerResponse item) { + List volumes = new ArrayList<>(); + for (InspectContainerResponse.Mount mount : item.getMounts()) { + volumes.add(mount.getDestination()); + } + return volumes; + } + } + + + public static Matcher apiVersionGreater(final RemoteApiVersion version) { + return new CustomTypeSafeMatcher("is greater") { + public boolean matchesSafely(DockerRule dockerRule) { + return getVersion(dockerRule.getClient()).isGreater(version); + } + }; + } + + public static Matcher isGreaterOrEqual(final RemoteApiVersion version) { + return new CustomTypeSafeMatcher("is greater or equal") { + public boolean matchesSafely(DockerRule dockerRule) { + return getVersion(dockerRule.getClient()).isGreaterOrEqual(version); + } + + @Override + protected void describeMismatchSafely(DockerRule rule, Description mismatchDescription) { + mismatchDescription + .appendText(" was ") + .appendText(getVersion(rule.getClient()).toString()); + } + }; + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/junit/PrivateRegistryRule.java b/docker-java/src/test/java/com/github/dockerjava/junit/PrivateRegistryRule.java new file mode 100644 index 000000000..327bfc941 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/junit/PrivateRegistryRule.java @@ -0,0 +1,119 @@ +package com.github.dockerjava.junit; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.command.CreateContainerResponse; +import com.github.dockerjava.api.command.InspectImageResponse; +import com.github.dockerjava.api.model.AuthConfig; +import com.github.dockerjava.api.model.ExposedPort; +import com.github.dockerjava.api.model.PortBinding; +import com.github.dockerjava.api.model.Ports; +import com.github.dockerjava.cmd.CmdIT; +import com.github.dockerjava.core.DockerRule; +import org.junit.rules.ExternalResource; + +import java.io.File; +import java.util.concurrent.TimeUnit; + +import static com.github.dockerjava.api.model.HostConfig.newHostConfig; +import static com.github.dockerjava.core.DockerRule.DEFAULT_IMAGE; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.nullValue; + +public class PrivateRegistryRule extends ExternalResource { + + private final DockerClient dockerClient; + + private AuthConfig authConfig; + + private String containerId; + + public PrivateRegistryRule() { + this.dockerClient = CmdIT.createDockerClient(DockerRule.config(null)); + } + + public AuthConfig getAuthConfig() { + return authConfig; + } + + public String createPrivateImage(String tagName) throws InterruptedException { + String imgNameWithTag = createTestImage(tagName); + + dockerClient.pushImageCmd(imgNameWithTag) + .withAuthConfig(authConfig) + .start() + .awaitCompletion(30, TimeUnit.SECONDS); + + dockerClient.removeImageCmd(imgNameWithTag).exec(); + + //ensures that the image is available, the private registry needs some time to reflect a tag push + Thread.sleep(5000); + + return imgNameWithTag; + } + + public String createTestImage(String tagName) { + String imgName = authConfig.getRegistryAddress() + "/busybox"; + + dockerClient.tagImageCmd(DEFAULT_IMAGE, imgName, tagName).exec(); + return imgName + ":" + tagName; + } + + /** + * Starts a local test registry when it is not already started and returns the auth configuration for it + * This method is synchronized so that only the first invocation starts the registry + */ + @Override + protected void before() throws Throwable { + + int port = 5050; + + String imageName = "private-registry-image"; + + File baseDir = new File(DockerRule.class.getResource("/privateRegistry").getFile()); + + String registryImageId = dockerClient.buildImageCmd(baseDir) + .withNoCache(true) + .start() + .awaitImageId(); + + InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(registryImageId).exec(); + assertThat(inspectImageResponse, not(nullValue())); + DockerRule.LOG.info("Image Inspect: {}", inspectImageResponse.toString()); + + dockerClient.tagImageCmd(registryImageId, imageName, "2") + .withForce().exec(); + + // see https://github.com/docker/distribution/blob/master/docs/deploying.md#native-basic-auth + CreateContainerResponse testregistry = dockerClient + .createContainerCmd(imageName + ":2") + .withHostConfig(newHostConfig() + .withPortBindings(new PortBinding(Ports.Binding.bindPort(port), ExposedPort.tcp(5000)))) + .withEnv("REGISTRY_AUTH=htpasswd", "REGISTRY_AUTH_HTPASSWD_REALM=Registry Realm", + "REGISTRY_AUTH_HTPASSWD_PATH=/auth/htpasswd", "REGISTRY_LOG_LEVEL=debug", + "REGISTRY_HTTP_TLS_CERTIFICATE=/certs/domain.crt", "REGISTRY_HTTP_TLS_KEY=/certs/domain.key") + .exec(); + + containerId = testregistry.getId(); + dockerClient.startContainerCmd(containerId).exec(); + + // wait for registry to boot + Thread.sleep(3000); + + // credentials as configured in /auth/htpasswd + authConfig = new AuthConfig() + .withUsername("testuser") + .withPassword("testpassword") + .withRegistryAddress("localhost:" + port); + } + + @Override + protected void after() { + if (containerId != null) { + dockerClient.removeContainerCmd(containerId) + .withForce(true) + .withRemoveVolumes(true) + .exec(); + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/junit/category/AuthIntegration.java b/docker-java/src/test/java/com/github/dockerjava/junit/category/AuthIntegration.java new file mode 100644 index 000000000..21dd8618c --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/junit/category/AuthIntegration.java @@ -0,0 +1,4 @@ +package com.github.dockerjava.junit.category; + +public interface AuthIntegration { +} diff --git a/docker-java/src/test/java/com/github/dockerjava/junit/category/Integration.java b/docker-java/src/test/java/com/github/dockerjava/junit/category/Integration.java new file mode 100644 index 000000000..bcd0f6db8 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/junit/category/Integration.java @@ -0,0 +1,4 @@ +package com.github.dockerjava.junit.category; + +public interface Integration { +} diff --git a/docker-java/src/test/java/com/github/dockerjava/junit/category/SwarmModeIntegration.java b/docker-java/src/test/java/com/github/dockerjava/junit/category/SwarmModeIntegration.java new file mode 100644 index 000000000..3a08d3a4f --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/junit/category/SwarmModeIntegration.java @@ -0,0 +1,4 @@ +package com.github.dockerjava.junit.category; + +public interface SwarmModeIntegration { +} diff --git a/docker-java/src/test/java/com/github/dockerjava/junit/suite/IntegrationDockerTestSuite.java b/docker-java/src/test/java/com/github/dockerjava/junit/suite/IntegrationDockerTestSuite.java new file mode 100644 index 000000000..06f6a4d25 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/junit/suite/IntegrationDockerTestSuite.java @@ -0,0 +1,43 @@ +package com.github.dockerjava.junit.suite; + +import com.github.dockerjava.cmd.AttachContainerCmdIT; +import com.github.dockerjava.cmd.AuthCmdIT; +import com.github.dockerjava.cmd.BuildImageCmdIT; +import com.github.dockerjava.cmd.CommitCmdIT; +import com.github.dockerjava.cmd.ConnectToNetworkCmdIT; +import com.github.dockerjava.cmd.CopyArchiveFromContainerCmdIT; +import com.github.dockerjava.cmd.CopyArchiveToContainerCmdIT; +import com.github.dockerjava.cmd.CopyFileFromContainerCmdIT; +import com.github.dockerjava.cmd.CreateContainerCmdIT; +import com.github.dockerjava.cmd.CreateNetworkCmdIT; +import com.github.dockerjava.cmd.CreateVolumeCmdIT; +import com.github.dockerjava.cmd.EventsCmdIT; +import com.github.dockerjava.cmd.InfoCmdIT; +import com.github.dockerjava.cmd.RemoveNetworkCmdIT; +import org.junit.runner.RunWith; +import org.junit.runners.Suite; + +/** + * Tests that requires real connection to docker. + * + * @author Kanstantsin Shautsou + */ +@RunWith(Suite.class) +@Suite.SuiteClasses({ + AttachContainerCmdIT.class, + AuthCmdIT.class, + BuildImageCmdIT.class, + CommitCmdIT.class, + CopyArchiveFromContainerCmdIT.class, + CopyArchiveToContainerCmdIT.class, + CopyFileFromContainerCmdIT.class, + ConnectToNetworkCmdIT.class, + CreateContainerCmdIT.class, + CreateNetworkCmdIT.class, + CreateVolumeCmdIT.class, + EventsCmdIT.class, + InfoCmdIT.class, + RemoveNetworkCmdIT.class, +}) +public class IntegrationDockerTestSuite { +} diff --git a/docker-java/src/test/java/com/github/dockerjava/junit/suite/SwarmSuite.java b/docker-java/src/test/java/com/github/dockerjava/junit/suite/SwarmSuite.java new file mode 100644 index 000000000..7f3982240 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/junit/suite/SwarmSuite.java @@ -0,0 +1,9 @@ +package com.github.dockerjava.junit.suite; + +import com.github.dockerjava.junit.category.SwarmModeIntegration; +import org.junit.experimental.categories.Categories; + +//@RunWith(Categories.class) +@Categories.IncludeCategory(SwarmModeIntegration.class) +public class SwarmSuite { +} diff --git a/docker-java/src/test/java/com/github/dockerjava/netty/NettyDockerCmdExecFactoryConfigTest.java b/docker-java/src/test/java/com/github/dockerjava/netty/NettyDockerCmdExecFactoryConfigTest.java new file mode 100644 index 000000000..03019f383 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/netty/NettyDockerCmdExecFactoryConfigTest.java @@ -0,0 +1,156 @@ +package com.github.dockerjava.netty; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.core.DefaultDockerClientConfig; +import com.github.dockerjava.core.DefaultDockerClientConfig.Builder; +import com.github.dockerjava.core.DockerClientBuilder; +import io.netty.bootstrap.ServerBootstrap; +import io.netty.buffer.Unpooled; +import io.netty.channel.Channel; +import io.netty.channel.ChannelHandlerContext; +import io.netty.channel.ChannelInitializer; +import io.netty.channel.ChannelPipeline; +import io.netty.channel.SimpleChannelInboundHandler; +import io.netty.channel.nio.NioEventLoopGroup; +import io.netty.channel.socket.SocketChannel; +import io.netty.channel.socket.nio.NioServerSocketChannel; +import io.netty.handler.codec.http.DefaultFullHttpResponse; +import io.netty.handler.codec.http.FullHttpResponse; +import io.netty.handler.codec.http.HttpContent; +import io.netty.handler.codec.http.HttpRequest; +import io.netty.handler.codec.http.HttpServerCodec; +import io.netty.util.CharsetUtil; +import org.junit.Test; + +import java.io.IOException; +import java.net.ServerSocket; +import java.util.ArrayList; +import java.util.List; + +import static io.netty.handler.codec.http.HttpHeaderNames.CONTENT_LENGTH; +import static io.netty.handler.codec.http.HttpHeaderNames.CONTENT_TYPE; +import static io.netty.handler.codec.http.HttpResponseStatus.OK; +import static io.netty.handler.codec.http.HttpVersion.HTTP_1_1; +import static org.junit.Assert.assertEquals; + +public class NettyDockerCmdExecFactoryConfigTest { + + @Test + public void testNettyDockerCmdExecFactoryConfigWithApiVersion() throws Exception { + int dockerPort = getFreePort(); + + NettyDockerCmdExecFactory factory = new NettyDockerCmdExecFactory(); + Builder configBuilder = new DefaultDockerClientConfig.Builder() + .withDockerTlsVerify(false) + .withDockerHost("tcp://localhost:" + dockerPort) + .withApiVersion("1.44"); + + DockerClient client = DockerClientBuilder.getInstance(configBuilder) + .withDockerCmdExecFactory(factory) + .build(); + + FakeDockerServer server = new FakeDockerServer(dockerPort); + server.start(); + try { + client.versionCmd().exec(); + + List requests = server.getRequests(); + + assertEquals(1, requests.size()); + assertEquals("/v1.44/version", requests.get(0).uri()); + } finally { + server.stop(); + } + } + + @Test + public void testNettyDockerCmdExecFactoryConfigWithoutApiVersion() throws Exception { + int dockerPort = getFreePort(); + + NettyDockerCmdExecFactory factory = new NettyDockerCmdExecFactory(); + Builder configBuilder = new DefaultDockerClientConfig.Builder() + .withDockerTlsVerify(false) + .withDockerHost("tcp://localhost:" + dockerPort); + + DockerClient client = DockerClientBuilder.getInstance(configBuilder) + .withDockerCmdExecFactory(factory) + .build(); + + FakeDockerServer server = new FakeDockerServer(dockerPort); + server.start(); + try { + client.versionCmd().exec(); + + List requests = server.getRequests(); + + assertEquals(1, requests.size()); + assertEquals("/version", requests.get(0).uri()); + } finally { + server.stop(); + } + } + + private int getFreePort() throws IOException { + ServerSocket socket = new ServerSocket(0); + int freePort = socket.getLocalPort(); + socket.close(); + return freePort; + } + + private class FakeDockerServer { + private final int port; + private final NioEventLoopGroup parent; + private final NioEventLoopGroup child; + private final List requests = new ArrayList<>(); + private Channel channel; + + private FakeDockerServer(int port) { + this.port = port; + this.parent = new NioEventLoopGroup(); + this.child = new NioEventLoopGroup(); + } + + private void start() throws Exception { + ServerBootstrap bootstrap = new ServerBootstrap(); + bootstrap.group(parent, child) + .channel(NioServerSocketChannel.class) + .childHandler(new ChannelInitializer() { + @Override + protected void initChannel(SocketChannel socketChannel) throws Exception { + ChannelPipeline pipeline = socketChannel.pipeline(); + pipeline.addLast("codec", new HttpServerCodec()); + pipeline.addLast("httpHandler", new SimpleChannelInboundHandler() { + @Override + protected void channelRead0(ChannelHandlerContext context, Object message) throws Exception { + if (message instanceof HttpRequest) { + // Keep track of processed requests + HttpRequest request = (HttpRequest) message; + requests.add(request); + } + + if (message instanceof HttpContent) { + // Write an empty JSON response back to the client + FullHttpResponse response = new DefaultFullHttpResponse(HTTP_1_1, OK, Unpooled.copiedBuffer("{}", CharsetUtil.UTF_8)); + response.headers().set(CONTENT_TYPE, "application/json; charset=UTF-8"); + response.headers().set(CONTENT_LENGTH, response.content().readableBytes()); + context.writeAndFlush(response); + } + } + }); + } + }); + + channel = bootstrap.bind(port).syncUninterruptibly().channel(); + } + + private void stop() throws Exception { + parent.shutdownGracefully(); + child.shutdownGracefully(); + channel.closeFuture().sync(); + } + + private List getRequests() { + return requests; + } + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/netty/NettyWebTargetTest.java b/docker-java/src/test/java/com/github/dockerjava/netty/NettyWebTargetTest.java new file mode 100644 index 000000000..4e7bb1da2 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/netty/NettyWebTargetTest.java @@ -0,0 +1,42 @@ +package com.github.dockerjava.netty; + + +import com.github.dockerjava.test.serdes.JSONTestHelper; +import org.junit.Before; +import org.junit.Test; +import org.mockito.Mock; +import org.mockito.MockitoAnnotations; + +import static org.junit.Assert.assertEquals; + +/** + * @author Alexander Koshevoy + */ +public class NettyWebTargetTest { + @Mock + private ChannelProvider channelProvider; + + @Before + public void setUp() { + MockitoAnnotations.initMocks(this); + } + + @Test + public void verifyImmutability() { + NettyWebTarget emptyWebTarget = new NettyWebTarget(JSONTestHelper.getMapper(), channelProvider, "DUMMY"); + + NettyWebTarget initWebTarget = emptyWebTarget.path("/containers/{id}/attach").resolveTemplate("id", "d03da378b592") + .queryParam("logs", "true"); + + NettyWebTarget anotherWebTarget = emptyWebTarget.path("/containers/{id}/attach") + .resolveTemplate("id", "2cfada4e3c07").queryParam("stdin", "true"); + + assertEquals(emptyWebTarget, new NettyWebTarget(JSONTestHelper.getMapper(), channelProvider, "DUMMY")); + + assertEquals(initWebTarget, new NettyWebTarget(JSONTestHelper.getMapper(), channelProvider, "DUMMY").path("/containers/d03da378b592/attach") + .queryParam("logs", "true")); + + assertEquals(anotherWebTarget, new NettyWebTarget(JSONTestHelper.getMapper(), channelProvider, "DUMMY").path("/containers/2cfada4e3c07/attach") + .queryParam("stdin", "true")); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/netty/handler/FramedResponseStreamHandlerTest.java b/docker-java/src/test/java/com/github/dockerjava/netty/handler/FramedResponseStreamHandlerTest.java new file mode 100644 index 000000000..ef903f942 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/netty/handler/FramedResponseStreamHandlerTest.java @@ -0,0 +1,184 @@ +package com.github.dockerjava.netty.handler; + +import com.github.dockerjava.api.model.Frame; +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.netty.handler.FramedResponseStreamHandler; +import java.io.Closeable; +import java.util.ArrayList; +import java.util.List; +import io.netty.buffer.Unpooled; +import io.netty.channel.ChannelHandlerContext; +import org.junit.Test; +import org.mockito.Mockito; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertTrue; + +public class FramedResponseStreamHandlerTest { + + public class MockedResponseHandler implements ResultCallback { + + public List frames = new ArrayList<>(); + public List exceptions = new ArrayList<>(); + + @Override + public void close() { + } + + @Override + public void onStart(Closeable closeable) { + } + + @Override + public void onNext(Frame object) { + frames.add(object); + } + + @Override + public void onError(Throwable throwable) { + exceptions.add(throwable); + } + + @Override + public void onComplete() { + } + } + + + @Test + public void channelRead0emptyHeaderCount() throws Exception { + + // Arrange + final MockedResponseHandler responseHandler = new MockedResponseHandler(); + final FramedResponseStreamHandler objectUnderTest = new FramedResponseStreamHandler(responseHandler); + byte[] msg = {}; + + // Act + objectUnderTest.channelRead0(Mockito.mock(ChannelHandlerContext.class), Unpooled.wrappedBuffer(msg)); + + // Assert result + assertTrue(responseHandler.frames.isEmpty()); + } + + @Test + public void channelRead0headerTooSmall() throws Exception { + + // Arrange + final MockedResponseHandler responseHandler = new MockedResponseHandler(); + final FramedResponseStreamHandler objectUnderTest = new FramedResponseStreamHandler(responseHandler); + byte[] msg = {0}; + + // Act + objectUnderTest.channelRead0(Mockito.mock(ChannelHandlerContext.class), Unpooled.wrappedBuffer(msg)); + + // Assert result + assertTrue(responseHandler.frames.isEmpty()); + } + + @Test + public void channelRead0rawStream() throws Exception { + + // Arrange + final MockedResponseHandler responseHandler = new MockedResponseHandler(); + final FramedResponseStreamHandler objectUnderTest = new FramedResponseStreamHandler(responseHandler); + byte[] msg = {3, 0, 0, 0, 0, 0, 0, 0, 0}; + + // Act + objectUnderTest.channelRead0(Mockito.mock(ChannelHandlerContext.class), Unpooled.wrappedBuffer(msg)); + + // Assert result + assertEquals("RAW: ", responseHandler.frames.get(0).toString()); + } + + @Test + public void channelRead0emptyNonRaw() throws Exception { + + // Arrange + final MockedResponseHandler responseHandler = new MockedResponseHandler(); + final FramedResponseStreamHandler objectUnderTest = new FramedResponseStreamHandler(responseHandler); + byte[] msg = {0, 0, 0, 0, 0, 0, 0, 0, 0}; + + // Act + objectUnderTest.channelRead0(Mockito.mock(ChannelHandlerContext.class), Unpooled.wrappedBuffer(msg)); + + // Assert result + assertTrue(responseHandler.frames.isEmpty()); + } + + @Test + public void channelRead0stdIn() throws Exception { + + // Arrange + final MockedResponseHandler responseHandler = new MockedResponseHandler(); + final FramedResponseStreamHandler objectUnderTest = new FramedResponseStreamHandler(responseHandler); + byte[] msg = {0, 0, 0, 0, 0, 0, 0, 1, 0}; + + // Act + objectUnderTest.channelRead0(Mockito.mock(ChannelHandlerContext.class), Unpooled.wrappedBuffer(msg)); + + // Assert result + assertEquals("STDIN: ", responseHandler.frames.get(0).toString()); + } + + @Test + public void channelRead0stdOut() throws Exception { + + // Arrange + final MockedResponseHandler responseHandler = new MockedResponseHandler(); + final FramedResponseStreamHandler objectUnderTest = new FramedResponseStreamHandler(responseHandler); + byte[] msg = {1, 0, 0, 0, 0, 0, 0, 1, 0}; + + // Act + objectUnderTest.channelRead0(Mockito.mock(ChannelHandlerContext.class), Unpooled.wrappedBuffer(msg)); + + // Assert result + assertEquals("STDOUT: ", responseHandler.frames.get(0).toString()); + } + + @Test + public void channelRead0stdErr() throws Exception { + + // Arrange + final MockedResponseHandler responseHandler = new MockedResponseHandler(); + final FramedResponseStreamHandler objectUnderTest = new FramedResponseStreamHandler(responseHandler); + byte[] msg = {2, 0, 0, 0, 0, 0, 0, 1, 0}; + + // Act + objectUnderTest.channelRead0(Mockito.mock(ChannelHandlerContext.class), Unpooled.wrappedBuffer(msg)); + + // Assert result + assertEquals("STDERR: ", responseHandler.frames.get(0).toString()); + } + + @Test + public void channelRead0largePayload() throws Exception { + + // Arrange + final MockedResponseHandler responseHandler = new MockedResponseHandler(); + final FramedResponseStreamHandler objectUnderTest = new FramedResponseStreamHandler(responseHandler); + byte[] msg = {1, 0, 0, 0, 0, 0, 0, 1, 0, 2, 0, 0, 0, 0, 0, 0, 2, 0}; + + // Act + objectUnderTest.channelRead0(Mockito.mock(ChannelHandlerContext.class), Unpooled.wrappedBuffer(msg)); + + // Assert result + assertEquals("STDOUT: ", responseHandler.frames.get(0).toString()); + } + + @Test + public void exceptionCaught() throws Exception { + + // Arrange + final MockedResponseHandler responseHandler = new MockedResponseHandler(); + final FramedResponseStreamHandler objectUnderTest = new FramedResponseStreamHandler(responseHandler); + final Exception exception = new Exception(); + final Throwable throwable = new Throwable(); + throwable.initCause(exception); + + // Act + objectUnderTest.exceptionCaught(Mockito.mock(ChannelHandlerContext.class), throwable); + + // Assert result + assertEquals(exception, responseHandler.exceptions.get(0).getCause()); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandlerTest.java b/docker-java/src/test/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandlerTest.java new file mode 100644 index 000000000..9a2492062 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandlerTest.java @@ -0,0 +1,135 @@ +package com.github.dockerjava.netty.handler; + +import com.github.dockerjava.core.async.ResultCallbackTemplate; +import io.netty.buffer.ByteBuf; +import io.netty.buffer.ByteBufInputStream; +import io.netty.buffer.Unpooled; +import io.netty.channel.ChannelHandlerContext; +import io.netty.util.ReferenceCountUtil; +import org.apache.commons.io.IOUtils; +import org.junit.Test; +import org.mockito.Mockito; + +import java.io.IOException; +import java.io.InputStream; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.Future; + +import static com.github.dockerjava.netty.handler.HttpResponseStreamHandler.HttpResponseInputStream; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertTrue; + +/** + * @author Alexander Koshevoy + */ +public class HttpResponseStreamHandlerTest { + @Test + public void testNoBytesSkipped() throws Exception { + ResultCallbackTest callback = new ResultCallbackTest(); + HttpResponseStreamHandler streamHandler = new HttpResponseStreamHandler(callback); + ChannelHandlerContext ctx = Mockito.mock(ChannelHandlerContext.class); + ByteBuf buffer = generateByteBuf(); + ByteBuf readBuffer = buffer.copy(); + assertEquals(1, buffer.refCnt()); + streamHandler.channelRead(ctx, buffer); + streamHandler.channelInactive(ctx); + assertEquals(0, buffer.refCnt()); + try (InputStream inputStream = callback.getInputStream()) { + assertTrue(IOUtils.contentEquals(inputStream, new ByteBufInputStream(readBuffer))); + } + ReferenceCountUtil.release(readBuffer); + } + + @Test + public void testReadByteByByte() throws Exception { + ResultCallbackTest callback = new ResultCallbackTest(); + HttpResponseStreamHandler streamHandler = new HttpResponseStreamHandler(callback); + ChannelHandlerContext ctx = Mockito.mock(ChannelHandlerContext.class); + ByteBuf buffer = generateByteBuf(); + ByteBuf readBuffer = buffer.copy(); + assertEquals(1, buffer.refCnt()); + streamHandler.channelRead(ctx, buffer); + streamHandler.channelInactive(ctx); + assertEquals(0, buffer.refCnt()); + try (InputStream inputStream = callback.getInputStream()) { + for (int i = 0; i < readBuffer.readableBytes(); i++) { + int b = inputStream.read(); + assertEquals(b, readBuffer.getByte(i)); + } + assertTrue(inputStream.read() == -1); + } + ReferenceCountUtil.release(readBuffer); + } + + @Test + public void testCloseResponseStreamBeforeWrite() throws Exception { + HttpResponseInputStream inputStream = new HttpResponseInputStream(); + ByteBuf buffer = generateByteBuf(); + + inputStream.write(buffer); + inputStream.close(); + inputStream.write(buffer); + } + + @Test + public void testCloseResponseStreamOnWrite() throws Exception { + final HttpResponseInputStream inputStream = new HttpResponseInputStream(); + + final ByteBuf buffer = generateByteBuf(); + + final CountDownLatch firstWrite = new CountDownLatch(1); + + ExecutorService executor = Executors.newSingleThreadExecutor(); + Future submit = executor.submit(() -> { + try { + inputStream.write(buffer); + firstWrite.countDown(); + inputStream.write(buffer); + } catch (InterruptedException e) { + e.printStackTrace(); + } + }); + + firstWrite.await(); + assertTrue(inputStream.read() != -1); + + // second write should have started + Thread.sleep(500L); + inputStream.close(); + + submit.get(); + } + + @Test(expected = IOException.class) + public void testReadClosedResponseStream() throws Exception { + HttpResponseInputStream inputStream = new HttpResponseInputStream(); + ByteBuf buffer = generateByteBuf(); + + inputStream.write(buffer); + inputStream.close(); + inputStream.read(); + } + + private ByteBuf generateByteBuf() { + byte[] array = new byte[256]; + for (int i = 0; i < array.length; i++) { + array[i] = (byte) i; + } + return Unpooled.copiedBuffer(array); + } + + private static class ResultCallbackTest extends ResultCallbackTemplate { + private InputStream stream; + + @Override + public void onNext(InputStream stream) { + this.stream = stream; + } + + private InputStream getInputStream() { + return stream; + } + } +} diff --git a/src/test/java/com/github/dockerjava/test/serdes/AbstractJSONResourceRef.java b/docker-java/src/test/java/com/github/dockerjava/test/serdes/AbstractJSONResourceRef.java similarity index 100% rename from src/test/java/com/github/dockerjava/test/serdes/AbstractJSONResourceRef.java rename to docker-java/src/test/java/com/github/dockerjava/test/serdes/AbstractJSONResourceRef.java diff --git a/src/test/java/com/github/dockerjava/test/serdes/JSONResourceRef.java b/docker-java/src/test/java/com/github/dockerjava/test/serdes/JSONResourceRef.java similarity index 100% rename from src/test/java/com/github/dockerjava/test/serdes/JSONResourceRef.java rename to docker-java/src/test/java/com/github/dockerjava/test/serdes/JSONResourceRef.java diff --git a/docker-java/src/test/java/com/github/dockerjava/test/serdes/JSONSamples.java b/docker-java/src/test/java/com/github/dockerjava/test/serdes/JSONSamples.java new file mode 100644 index 000000000..231d1426a --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/test/serdes/JSONSamples.java @@ -0,0 +1,62 @@ +package com.github.dockerjava.test.serdes; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.JavaType; +import com.fasterxml.jackson.databind.JsonNode; +import com.github.dockerjava.core.RemoteApiVersion; +import org.apache.commons.io.FileUtils; + +import java.io.File; +import java.io.IOException; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotSame; + +/** + * Samples helper + * + * @author Kanstantsin Shautsou + */ +public class JSONSamples { + + /** + * Access to samples storage. + * + * @param version docker version of json sample + * @param context path to file for interested dump + * @return Content of JSON sample + * @throws IOException + */ + public static String getSampleContent(RemoteApiVersion version, String context) throws IOException { + File resource = new File("src/test/resources/samples/" + version.getVersion() + "/" + context); + return FileUtils.readFileToString(resource, "UTF-8"); + } + + public static TClass testRoundTrip(RemoteApiVersion version, String context, + JavaType type) throws IOException { + final TClass tObject = JSONTestHelper.getMapper().readValue(getSampleContent(version, context), type); + return testRoundTrip(tObject, type); + } + + /** + * Same as {@link JSONTestHelper#testRoundTrip(java.lang.Object, java.lang.Class)} + * but via {@link TypeReference} + */ + public static TClass testRoundTrip(TClass item, JavaType type) + throws IOException, AssertionError { + String serialized1 = JSONTestHelper.getMapper().writeValueAsString(item); + JsonNode json1 = JSONTestHelper.getMapper().readTree(serialized1); + + TClass deserialized1 = JSONTestHelper.getMapper().readValue(serialized1, type); + String serialized2 = JSONTestHelper.getMapper().writeValueAsString(deserialized1); + + JsonNode json2 = JSONTestHelper.getMapper().readTree(serialized2); + TClass deserialized2 = JSONTestHelper.getMapper().readValue(serialized2, type); + + assertEquals("JSONs must be equal after the second roundtrip", json2, json1); + assertEquals("Objects must be equal after the second roundtrip", deserialized2, deserialized2); + assertNotSame("Objects must be not the same", deserialized2, deserialized1); + + return deserialized2; + } +} diff --git a/src/test/java/com/github/dockerjava/test/serdes/JSONTestHelper.java b/docker-java/src/test/java/com/github/dockerjava/test/serdes/JSONTestHelper.java similarity index 80% rename from src/test/java/com/github/dockerjava/test/serdes/JSONTestHelper.java rename to docker-java/src/test/java/com/github/dockerjava/test/serdes/JSONTestHelper.java index 99a90f0af..0c03bdcc2 100644 --- a/src/test/java/com/github/dockerjava/test/serdes/JSONTestHelper.java +++ b/docker-java/src/test/java/com/github/dockerjava/test/serdes/JSONTestHelper.java @@ -15,16 +15,16 @@ */ package com.github.dockerjava.test.serdes; -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertNotNull; +import com.fasterxml.jackson.databind.JsonNode; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.dockerjava.core.DockerClientConfig; +import org.apache.commons.io.IOUtils; import java.io.IOException; import java.io.InputStream; -import org.apache.commons.io.IOUtils; - -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.ObjectMapper; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; /** * Provides helper methods for serialization-deserialization tests. @@ -33,6 +33,17 @@ */ public class JSONTestHelper { + static final ObjectMapper MAPPER; + + static { + try { + MAPPER = DockerClientConfig.getDefaultObjectMapper(); + } catch (Throwable e) { + e.printStackTrace(); + throw e; + } + } + /** * Reads JSON String from the specified resource * @@ -65,9 +76,8 @@ public static String readString(JSONResourceRef resource) throws IOException { * JSON conversion error */ public static TClass readObject(JSONResourceRef resource, Class tclass) throws IOException { - ObjectMapper mapper = new ObjectMapper(); String str = readString(resource); - return mapper.readValue(str, tclass); + return MAPPER.readValue(str, tclass); } /** @@ -126,16 +136,18 @@ public static TClass testRoundTrip(TClass item) throws IOException, Ass * Validation error */ public static TClass testRoundTrip(TClass item, Class asclass) throws IOException, AssertionError { - ObjectMapper mapper = new ObjectMapper(); + String serialized1 = MAPPER.writeValueAsString(item); + JsonNode json1 = MAPPER.readTree(serialized1); + TClass deserialized1 = MAPPER.readValue(serialized1, asclass); + String serialized2 = MAPPER.writeValueAsString(deserialized1); + JsonNode json2 = MAPPER.readTree(serialized2); + TClass deserialized2 = MAPPER.readValue(serialized2, asclass); - String serialized1 = mapper.writeValueAsString(item); - JsonNode json1 = mapper.readTree(serialized1); - TClass deserialized1 = mapper.readValue(serialized1, asclass); - String serialized2 = mapper.writeValueAsString(deserialized1); - JsonNode json2 = mapper.readTree(serialized2); - TClass deserialized2 = mapper.readValue(serialized2, asclass); - - assertEquals(json2, json1, "JSONs must be equal after the second roundtrip"); + assertEquals("JSONs must be equal after the second roundtrip", json2, json1); return deserialized2; } + + public static ObjectMapper getMapper() { + return MAPPER; + } } diff --git a/src/test/java/com/github/dockerjava/utils/ContainerUtils.java b/docker-java/src/test/java/com/github/dockerjava/utils/ContainerUtils.java similarity index 96% rename from src/test/java/com/github/dockerjava/utils/ContainerUtils.java rename to docker-java/src/test/java/com/github/dockerjava/utils/ContainerUtils.java index 75be6c4d9..206be0693 100644 --- a/src/test/java/com/github/dockerjava/utils/ContainerUtils.java +++ b/docker-java/src/test/java/com/github/dockerjava/utils/ContainerUtils.java @@ -60,7 +60,7 @@ public static void stopContainer(DockerClient dockerClient, CreateContainerRespo * @param dockerClient docker client * @param container container */ - public static void unpaseContainer(DockerClient dockerClient, CreateContainerResponse container) { + public static void unpauseContainer(DockerClient dockerClient, CreateContainerResponse container) { dockerClient.unpauseContainerCmd(container.getId()).exec(); InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); diff --git a/docker-java/src/test/java/com/github/dockerjava/utils/LogContainerTestCallback.java b/docker-java/src/test/java/com/github/dockerjava/utils/LogContainerTestCallback.java new file mode 100644 index 000000000..e21a55465 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/utils/LogContainerTestCallback.java @@ -0,0 +1,42 @@ +package com.github.dockerjava.utils; + +import com.github.dockerjava.api.async.ResultCallback; +import com.github.dockerjava.api.model.Frame; + +import java.util.ArrayList; +import java.util.List; + +/** + * @author Kanstantsin Shautsou + */ +public class LogContainerTestCallback extends ResultCallback.Adapter { + protected final StringBuffer log = new StringBuffer(); + + List collectedFrames = new ArrayList<>(); + + boolean collectFrames = false; + + public LogContainerTestCallback() { + this(false); + } + + public LogContainerTestCallback(boolean collectFrames) { + this.collectFrames = collectFrames; + } + + @Override + public void onNext(Frame frame) { + if (collectFrames) collectedFrames.add(frame); + log.append(new String(frame.getPayload())); + } + + @Override + public String toString() { + return log.toString(); + } + + + public List getCollectedFrames() { + return collectedFrames; + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/utils/TestResources.java b/docker-java/src/test/java/com/github/dockerjava/utils/TestResources.java new file mode 100644 index 000000000..2a56333f1 --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/utils/TestResources.java @@ -0,0 +1,15 @@ +package com.github.dockerjava.utils; + +import java.net.URISyntaxException; +import java.nio.file.Path; +import java.nio.file.Paths; + +public class TestResources { + + private TestResources() { + } + + public static Path getApiImagesLoadTestTarball() throws URISyntaxException { + return Paths.get(ClassLoader.getSystemResource("api/images/load/image.tar").toURI()); + } +} diff --git a/docker-java/src/test/java/com/github/dockerjava/utils/TestUtils.java b/docker-java/src/test/java/com/github/dockerjava/utils/TestUtils.java new file mode 100644 index 000000000..eb3af8deb --- /dev/null +++ b/docker-java/src/test/java/com/github/dockerjava/utils/TestUtils.java @@ -0,0 +1,76 @@ +package com.github.dockerjava.utils; + +import com.github.dockerjava.api.DockerClient; +import com.github.dockerjava.api.model.Network; +import com.github.dockerjava.core.RemoteApiVersion; +import org.apache.commons.io.IOUtils; +import org.apache.commons.io.LineIterator; +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; +import java.io.InputStream; +import java.io.StringWriter; +import java.util.List; + +/** + * @author Kanstantsin Shautsou + */ +public class TestUtils { + public static final Logger LOG = LoggerFactory.getLogger(TestUtils.class); + + private TestUtils() { + } + + public static RemoteApiVersion getVersion(DockerClient client) { + final String serverVersion = client.versionCmd().exec().getApiVersion(); + return RemoteApiVersion.parseConfig(serverVersion); + } + + public static boolean isSwarm(DockerClient client) { + final String serverVersion = client.versionCmd().exec().getVersion(); + return serverVersion.startsWith("swarm/"); + } + + public static boolean isNotSwarm(DockerClient client) { + return !isSwarm(client); + } + + public static Network findNetwork(List networks, String name) { + + for (Network network : networks) { + if (StringUtils.equals(network.getName(), name)) { + return network; + } + } + + throw new AssertionError("No network found."); + } + + public static String asString(InputStream response) { + return consumeAsString(response); + } + + public static String consumeAsString(InputStream response) { + + StringWriter logwriter = new StringWriter(); + + try { + LineIterator itr = IOUtils.lineIterator(response, "UTF-8"); + + while (itr.hasNext()) { + String line = itr.next(); + logwriter.write(line + (itr.hasNext() ? "\n" : "")); + LOG.info("line: " + line); + } + response.close(); + + return logwriter.toString(); + } catch (IOException e) { + throw new RuntimeException(e); + } finally { + IOUtils.closeQuietly(response); + } + } +} diff --git a/src/test/resources/api/images/load/image.tar b/docker-java/src/test/resources/api/images/load/image.tar similarity index 100% rename from src/test/resources/api/images/load/image.tar rename to docker-java/src/test/resources/api/images/load/image.tar diff --git a/docker-java/src/test/resources/attachContainerTestDockerfile/Dockerfile b/docker-java/src/test/resources/attachContainerTestDockerfile/Dockerfile new file mode 100644 index 000000000..4c8f6bd78 --- /dev/null +++ b/docker-java/src/test/resources/attachContainerTestDockerfile/Dockerfile @@ -0,0 +1,9 @@ +FROM busybox:latest + +ADD ./echo.sh /tmp/ + +RUN mkdir -p /usr/local/bin +RUN cp /tmp/echo.sh /usr/local/bin/ && chmod +x /usr/local/bin/echo.sh + +CMD ["echo.sh"] + diff --git a/docker-java/src/test/resources/attachContainerTestDockerfile/echo.sh b/docker-java/src/test/resources/attachContainerTestDockerfile/echo.sh new file mode 100644 index 000000000..370cda203 --- /dev/null +++ b/docker-java/src/test/resources/attachContainerTestDockerfile/echo.sh @@ -0,0 +1,2 @@ +#!/bin/sh +echo stdout && echo stderr >&2 diff --git a/docker-java/src/test/resources/buildTests/ADD/file/Dockerfile b/docker-java/src/test/resources/buildTests/ADD/file/Dockerfile new file mode 100644 index 000000000..48faa93c6 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/ADD/file/Dockerfile @@ -0,0 +1,9 @@ +FROM busybox:latest + +# Copy testrun.sh files into the container + +ADD ./testrun.sh /tmp/ +RUN mkdir -p /usr/local/bin +RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh + +CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/ADD/file/testrun.sh b/docker-java/src/test/resources/buildTests/ADD/file/testrun.sh similarity index 100% rename from src/test/resources/buildTests/ADD/file/testrun.sh rename to docker-java/src/test/resources/buildTests/ADD/file/testrun.sh diff --git a/docker-java/src/test/resources/buildTests/ADD/fileInSubfolder/Dockerfile b/docker-java/src/test/resources/buildTests/ADD/fileInSubfolder/Dockerfile new file mode 100644 index 000000000..440e61e81 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/ADD/fileInSubfolder/Dockerfile @@ -0,0 +1,10 @@ +FROM busybox:latest + +# Copy testrun.sh files into the container + +ADD ./files/testrun.sh /tmp/ + +RUN mkdir -p /usr/local/bin +RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh + +CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/ADD/fileInSubfolder/files/testrun.sh b/docker-java/src/test/resources/buildTests/ADD/fileInSubfolder/files/testrun.sh similarity index 100% rename from src/test/resources/buildTests/ADD/fileInSubfolder/files/testrun.sh rename to docker-java/src/test/resources/buildTests/ADD/fileInSubfolder/files/testrun.sh diff --git a/docker-java/src/test/resources/buildTests/ADD/files/Dockerfile b/docker-java/src/test/resources/buildTests/ADD/files/Dockerfile new file mode 100644 index 000000000..8de15aed4 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/ADD/files/Dockerfile @@ -0,0 +1,5 @@ +FROM busybox:latest + +# Copy multiple source files into the container + +ADD src1 src2 /tmp/ diff --git a/src/test/resources/buildTests/ADD/files/src1 b/docker-java/src/test/resources/buildTests/ADD/files/src1 similarity index 100% rename from src/test/resources/buildTests/ADD/files/src1 rename to docker-java/src/test/resources/buildTests/ADD/files/src1 diff --git a/src/test/resources/buildTests/ADD/files/src2 b/docker-java/src/test/resources/buildTests/ADD/files/src2 similarity index 100% rename from src/test/resources/buildTests/ADD/files/src2 rename to docker-java/src/test/resources/buildTests/ADD/files/src2 diff --git a/docker-java/src/test/resources/buildTests/ADD/filesViaWildcard/Dockerfile b/docker-java/src/test/resources/buildTests/ADD/filesViaWildcard/Dockerfile new file mode 100644 index 000000000..f62e18087 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/ADD/filesViaWildcard/Dockerfile @@ -0,0 +1,10 @@ +FROM busybox:latest + +# Copy testrun.sh files into the container + +ADD ./folder*/* /tmp/ + +RUN mkdir -p /usr/local/bin +RUN cp /tmp/*.sh /usr/local/bin/ && chmod +x /usr/local/bin/*.sh + +CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/ADD/filesViaWildcard/folder1/testrun.sh b/docker-java/src/test/resources/buildTests/ADD/filesViaWildcard/folder1/testrun.sh similarity index 100% rename from src/test/resources/buildTests/ADD/filesViaWildcard/folder1/testrun.sh rename to docker-java/src/test/resources/buildTests/ADD/filesViaWildcard/folder1/testrun.sh diff --git a/src/test/resources/buildTests/ADD/filesViaWildcard/folder2/testinclude1.sh b/docker-java/src/test/resources/buildTests/ADD/filesViaWildcard/folder2/testinclude1.sh similarity index 100% rename from src/test/resources/buildTests/ADD/filesViaWildcard/folder2/testinclude1.sh rename to docker-java/src/test/resources/buildTests/ADD/filesViaWildcard/folder2/testinclude1.sh diff --git a/src/test/resources/buildTests/ADD/filesViaWildcard/ignore/testinclude2.sh b/docker-java/src/test/resources/buildTests/ADD/filesViaWildcard/ignore/testinclude2.sh similarity index 100% rename from src/test/resources/buildTests/ADD/filesViaWildcard/ignore/testinclude2.sh rename to docker-java/src/test/resources/buildTests/ADD/filesViaWildcard/ignore/testinclude2.sh diff --git a/docker-java/src/test/resources/buildTests/ADD/folder/Dockerfile b/docker-java/src/test/resources/buildTests/ADD/folder/Dockerfile new file mode 100644 index 000000000..9d8ff411f --- /dev/null +++ b/docker-java/src/test/resources/buildTests/ADD/folder/Dockerfile @@ -0,0 +1,12 @@ +FROM busybox:latest + +# Copy testrun.sh files into the container + +ADD . /src/ + +RUN ls -la /src + +RUN mkdir -p /usr/local/bin +RUN cp /src/folderA/testAddFolder.sh /usr/local/bin/ && chmod +x /usr/local/bin/testAddFolder.sh + +CMD ["testAddFolder.sh"] diff --git a/src/test/resources/buildTests/ADD/folder/folderA/testAddFolder.sh b/docker-java/src/test/resources/buildTests/ADD/folder/folderA/testAddFolder.sh similarity index 100% rename from src/test/resources/buildTests/ADD/folder/folderA/testAddFolder.sh rename to docker-java/src/test/resources/buildTests/ADD/folder/folderA/testAddFolder.sh diff --git a/docker-java/src/test/resources/buildTests/ADD/url/Dockerfile b/docker-java/src/test/resources/buildTests/ADD/url/Dockerfile new file mode 100644 index 000000000..4fbfa3236 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/ADD/url/Dockerfile @@ -0,0 +1,10 @@ +FROM busybox:latest + +# Copy testrun.sh files into the container + +ADD http://www.example.com/index.html /tmp/some.html +ADD ./testrun.sh /tmp/ +RUN mkdir -p /usr/local/bin +RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh + +CMD ["testrun.sh"] \ No newline at end of file diff --git a/src/test/resources/buildTests/ADD/url/testrun.sh b/docker-java/src/test/resources/buildTests/ADD/url/testrun.sh similarity index 100% rename from src/test/resources/buildTests/ADD/url/testrun.sh rename to docker-java/src/test/resources/buildTests/ADD/url/testrun.sh diff --git a/docker-java/src/test/resources/buildTests/AUTHOR/Dockerfile b/docker-java/src/test/resources/buildTests/AUTHOR/Dockerfile new file mode 100644 index 000000000..86bbd1a4f --- /dev/null +++ b/docker-java/src/test/resources/buildTests/AUTHOR/Dockerfile @@ -0,0 +1,12 @@ +# Nginx +# +# VERSION 0.0.1 + +FROM busybox:latest +MAINTAINER Guillaume J. Charmes "guillaume@dotcloud.com" + +# make sure the package repository is up to date +#RUN echo "deb http://archive.ubuntu.com/ubuntu precise main universe" > /etc/apt/sources.list +#RUN apt-get update + +#RUN apt-get install -y nginx \ No newline at end of file diff --git a/docker-java/src/test/resources/buildTests/CacheFrom/test1/Dockerfile b/docker-java/src/test/resources/buildTests/CacheFrom/test1/Dockerfile new file mode 100644 index 000000000..b1319b7b8 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/CacheFrom/test1/Dockerfile @@ -0,0 +1,7 @@ +FROM busybox:latest + +RUN echo "Step 1" + +RUN echo "Step 2" + +RUN echo "Step 3" \ No newline at end of file diff --git a/docker-java/src/test/resources/buildTests/CacheFrom/test2/Dockerfile b/docker-java/src/test/resources/buildTests/CacheFrom/test2/Dockerfile new file mode 100644 index 000000000..f45391b10 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/CacheFrom/test2/Dockerfile @@ -0,0 +1,9 @@ +FROM busybox:latest + +RUN echo "Step 1" + +RUN echo "Step 2" + +RUN echo "Step 3" + +RUN echo "Step 4" \ No newline at end of file diff --git a/docker-java/src/test/resources/buildTests/ENV/Dockerfile b/docker-java/src/test/resources/buildTests/ENV/Dockerfile new file mode 100644 index 000000000..f116eb96e --- /dev/null +++ b/docker-java/src/test/resources/buildTests/ENV/Dockerfile @@ -0,0 +1,13 @@ +FROM busybox:latest + +# Copy testrun.sh files into the container + +ENV variable abc123 +ADD ./testrun.sh /tmp/ +ADD ./subst-file-$variable.txt /tmp/ +COPY ./subst-file-2-${variable}.txt /tmp/ + +RUN mkdir -p /usr/local/bin +RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh + +CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/ENV/subst-file-2-abc123.txt b/docker-java/src/test/resources/buildTests/ENV/subst-file-2-abc123.txt similarity index 100% rename from src/test/resources/buildTests/ENV/subst-file-2-abc123.txt rename to docker-java/src/test/resources/buildTests/ENV/subst-file-2-abc123.txt diff --git a/src/test/resources/buildTests/ENV/subst-file-abc123.txt b/docker-java/src/test/resources/buildTests/ENV/subst-file-abc123.txt similarity index 100% rename from src/test/resources/buildTests/ENV/subst-file-abc123.txt rename to docker-java/src/test/resources/buildTests/ENV/subst-file-abc123.txt diff --git a/src/test/resources/buildTests/ENV/testrun.sh b/docker-java/src/test/resources/buildTests/ENV/testrun.sh similarity index 100% rename from src/test/resources/buildTests/ENV/testrun.sh rename to docker-java/src/test/resources/buildTests/ENV/testrun.sh diff --git a/docker-java/src/test/resources/buildTests/FROM/privateRegistry/Dockerfile b/docker-java/src/test/resources/buildTests/FROM/privateRegistry/Dockerfile new file mode 100644 index 000000000..f6813ad40 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/FROM/privateRegistry/Dockerfile @@ -0,0 +1 @@ +FROM localhost:5050/testuser/busybox:latest diff --git a/src/test/resources/buildTests/ONBUILD/child/Dockerfile b/docker-java/src/test/resources/buildTests/ONBUILD/child/Dockerfile similarity index 100% rename from src/test/resources/buildTests/ONBUILD/child/Dockerfile rename to docker-java/src/test/resources/buildTests/ONBUILD/child/Dockerfile diff --git a/src/test/resources/buildTests/ONBUILD/child/testrun.sh b/docker-java/src/test/resources/buildTests/ONBUILD/child/testrun.sh similarity index 100% rename from src/test/resources/buildTests/ONBUILD/child/testrun.sh rename to docker-java/src/test/resources/buildTests/ONBUILD/child/testrun.sh diff --git a/docker-java/src/test/resources/buildTests/ONBUILD/parent/Dockerfile b/docker-java/src/test/resources/buildTests/ONBUILD/parent/Dockerfile new file mode 100644 index 000000000..f7721dafd --- /dev/null +++ b/docker-java/src/test/resources/buildTests/ONBUILD/parent/Dockerfile @@ -0,0 +1,10 @@ +FROM busybox:latest + +RUN mkdir -p /usr/local/bin +# Copy testrun.sh files into the container + +ONBUILD ADD ./testrun.sh /tmp/ + +ONBUILD RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh + +CMD ["testrun.sh"] diff --git a/docker-java/src/test/resources/buildTests/buildArgs/Dockerfile b/docker-java/src/test/resources/buildTests/buildArgs/Dockerfile new file mode 100644 index 000000000..038f98e81 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/buildArgs/Dockerfile @@ -0,0 +1,5 @@ +FROM busybox:latest + +ARG testArg + +LABEL "test"=$testArg diff --git a/docker-java/src/test/resources/buildTests/dockerfileNotInBaseDirectory/dockerfileFolder/Dockerfile b/docker-java/src/test/resources/buildTests/dockerfileNotInBaseDirectory/dockerfileFolder/Dockerfile new file mode 100644 index 000000000..15d8fd7a8 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerfileNotInBaseDirectory/dockerfileFolder/Dockerfile @@ -0,0 +1,8 @@ +FROM busybox:latest + +ADD testrunFolder/testrun.sh /tmp/ + +RUN mkdir -p /usr/local/bin +RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh + +CMD ["testrun.sh"] \ No newline at end of file diff --git a/src/test/resources/buildTests/dockerfileNotInBaseDirectory/testrunFolder/testrun.sh b/docker-java/src/test/resources/buildTests/dockerfileNotInBaseDirectory/testrunFolder/testrun.sh similarity index 100% rename from src/test/resources/buildTests/dockerfileNotInBaseDirectory/testrunFolder/testrun.sh rename to docker-java/src/test/resources/buildTests/dockerfileNotInBaseDirectory/testrunFolder/testrun.sh diff --git a/src/test/resources/buildTests/dockerignore/DockerfileIgnored/.dockerignore b/docker-java/src/test/resources/buildTests/dockerignore/DockerfileIgnored/.dockerignore similarity index 100% rename from src/test/resources/buildTests/dockerignore/DockerfileIgnored/.dockerignore rename to docker-java/src/test/resources/buildTests/dockerignore/DockerfileIgnored/.dockerignore diff --git a/docker-java/src/test/resources/buildTests/dockerignore/DockerfileIgnored/Dockerfile b/docker-java/src/test/resources/buildTests/dockerignore/DockerfileIgnored/Dockerfile new file mode 100644 index 000000000..cecc27462 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/DockerfileIgnored/Dockerfile @@ -0,0 +1,10 @@ +FROM busybox:latest + +# Copy testrun.sh files into the container + +ADD ./testrun.sh /tmp/ + +RUN mkdir -p /usr/local/bin +RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh + +CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/dockerignore/DockerfileIgnored/testrun.sh b/docker-java/src/test/resources/buildTests/dockerignore/DockerfileIgnored/testrun.sh similarity index 100% rename from src/test/resources/buildTests/dockerignore/DockerfileIgnored/testrun.sh rename to docker-java/src/test/resources/buildTests/dockerignore/DockerfileIgnored/testrun.sh diff --git a/src/test/resources/buildTests/dockerignore/DockerfileNotIgnored/.dockerignore b/docker-java/src/test/resources/buildTests/dockerignore/DockerfileNotIgnored/.dockerignore similarity index 100% rename from src/test/resources/buildTests/dockerignore/DockerfileNotIgnored/.dockerignore rename to docker-java/src/test/resources/buildTests/dockerignore/DockerfileNotIgnored/.dockerignore diff --git a/docker-java/src/test/resources/buildTests/dockerignore/DockerfileNotIgnored/Dockerfile b/docker-java/src/test/resources/buildTests/dockerignore/DockerfileNotIgnored/Dockerfile new file mode 100644 index 000000000..e5b29ce0a --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/DockerfileNotIgnored/Dockerfile @@ -0,0 +1,3 @@ +FROM busybox:latest + +CMD ["echo", "Success"] diff --git a/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/.dockerignore b/docker-java/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/.dockerignore similarity index 100% rename from src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/.dockerignore rename to docker-java/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/.dockerignore diff --git a/docker-java/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/Dockerfile b/docker-java/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/Dockerfile new file mode 100644 index 000000000..3c57c5061 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/Dockerfile @@ -0,0 +1,3 @@ +FROM busybox:latest + +CMD ["echo", "Success"] diff --git a/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/README-secret.md b/docker-java/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/README-secret.md similarity index 100% rename from src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/README-secret.md rename to docker-java/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/README-secret.md diff --git a/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/README.md b/docker-java/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/README.md similarity index 100% rename from src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/README.md rename to docker-java/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/README.md diff --git a/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/.dockerignore b/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/.dockerignore new file mode 100644 index 000000000..116300b83 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/.dockerignore @@ -0,0 +1,3 @@ +* +!Dockerfile +!build/libs/foo.jar \ No newline at end of file diff --git a/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/Dockerfile b/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/Dockerfile new file mode 100644 index 000000000..617801170 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/Dockerfile @@ -0,0 +1 @@ +FROM ubuntu:18.04 diff --git a/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/README.MD b/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/README.MD new file mode 100644 index 000000000..ac11abdc4 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/README.MD @@ -0,0 +1 @@ +DO NOT WANT THIS IN THE DOCKER diff --git a/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/build/libs/foo.jar b/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/build/libs/foo.jar new file mode 100644 index 000000000..4d6bf796e --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/build/libs/foo.jar @@ -0,0 +1 @@ +foo.jar \ No newline at end of file diff --git a/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/test/bar.txt b/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/test/bar.txt new file mode 100644 index 000000000..c7152c4d8 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/IgnoreAllBut/test/bar.txt @@ -0,0 +1 @@ +FILE diff --git a/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/.dockerignore b/docker-java/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/.dockerignore similarity index 100% rename from src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/.dockerignore rename to docker-java/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/.dockerignore diff --git a/docker-java/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/Dockerfile b/docker-java/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/Dockerfile new file mode 100644 index 000000000..3c57c5061 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/Dockerfile @@ -0,0 +1,3 @@ +FROM busybox:latest + +CMD ["echo", "Success"] diff --git a/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/README-secret.md b/docker-java/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/README-secret.md similarity index 100% rename from src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/README-secret.md rename to docker-java/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/README-secret.md diff --git a/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/README.md b/docker-java/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/README.md similarity index 100% rename from src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/README.md rename to docker-java/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/README.md diff --git a/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/.dockerignore b/docker-java/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/.dockerignore similarity index 100% rename from src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/.dockerignore rename to docker-java/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/.dockerignore diff --git a/docker-java/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/Dockerfile b/docker-java/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/Dockerfile new file mode 100644 index 000000000..cecc27462 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/Dockerfile @@ -0,0 +1,10 @@ +FROM busybox:latest + +# Copy testrun.sh files into the container + +ADD ./testrun.sh /tmp/ + +RUN mkdir -p /usr/local/bin +RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh + +CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/testrun.sh b/docker-java/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/testrun.sh similarity index 100% rename from src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/testrun.sh rename to docker-java/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/testrun.sh diff --git a/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/.dockerignore b/docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/.dockerignore similarity index 100% rename from src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/.dockerignore rename to docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/.dockerignore diff --git a/docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/Dockerfile b/docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/Dockerfile new file mode 100644 index 000000000..3c57c5061 --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/Dockerfile @@ -0,0 +1,3 @@ +FROM busybox:latest + +CMD ["echo", "Success"] diff --git a/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/README.md b/docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/README.md similarity index 100% rename from src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/README.md rename to docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/README.md diff --git a/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/a.txt b/docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/a.txt similarity index 100% rename from src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/a.txt rename to docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/a.txt diff --git a/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/README-child.md b/docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/README-child.md similarity index 100% rename from src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/README-child.md rename to docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/README-child.md diff --git a/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/grandChild/README-grand.md b/docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/grandChild/README-grand.md similarity index 100% rename from src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/grandChild/README-grand.md rename to docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/grandChild/README-grand.md diff --git a/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/grandChild/c.txt b/docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/grandChild/c.txt similarity index 100% rename from src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/grandChild/c.txt rename to docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/anotherChild/grandChild/c.txt diff --git a/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/child/b.txt b/docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/child/b.txt similarity index 100% rename from src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/child/b.txt rename to docker-java/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/parent/child/b.txt diff --git a/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/.dockerignore b/docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/.dockerignore similarity index 100% rename from src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/.dockerignore rename to docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/.dockerignore diff --git a/docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/Dockerfile b/docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/Dockerfile new file mode 100644 index 000000000..1268f56cf --- /dev/null +++ b/docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/Dockerfile @@ -0,0 +1,11 @@ +FROM busybox:latest + +# Copy testrun.sh files into the container + +ADD ./testrun.sh /tmp/ +ADD ./a /tmp/a + +RUN mkdir -p /usr/local/bin +RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh + +CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/a b/docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/a similarity index 100% rename from src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/a rename to docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/a diff --git a/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/b b/docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/b similarity index 100% rename from src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/b rename to docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/b diff --git a/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/c b/docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/c similarity index 100% rename from src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/c rename to docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/c diff --git a/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/d b/docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/d similarity index 100% rename from src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/d rename to docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/a/d diff --git a/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/testrun.sh b/docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/testrun.sh similarity index 100% rename from src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/testrun.sh rename to docker-java/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/testrun.sh diff --git a/docker-java/src/test/resources/buildTests/labels/Dockerfile b/docker-java/src/test/resources/buildTests/labels/Dockerfile new file mode 100644 index 000000000..e5b29ce0a --- /dev/null +++ b/docker-java/src/test/resources/buildTests/labels/Dockerfile @@ -0,0 +1,3 @@ +FROM busybox:latest + +CMD ["echo", "Success"] diff --git a/src/test/resources/busyboxDockerfile/Dockerfile b/docker-java/src/test/resources/busyboxDockerfile/Dockerfile similarity index 100% rename from src/test/resources/busyboxDockerfile/Dockerfile rename to docker-java/src/test/resources/busyboxDockerfile/Dockerfile diff --git a/docker-java/src/test/resources/com.github.dockerjava.core/registry.v1/.dockercfg b/docker-java/src/test/resources/com.github.dockerjava.core/registry.v1/.dockercfg new file mode 100644 index 000000000..3ab173fba --- /dev/null +++ b/docker-java/src/test/resources/com.github.dockerjava.core/registry.v1/.dockercfg @@ -0,0 +1,5 @@ +{ + "https://test.docker.io/v1/": { + "auth": "dXNlcjpwYXNzd29yZA==" + } +} diff --git a/docker-java/src/test/resources/com.github.dockerjava.core/registry.v2/config.json b/docker-java/src/test/resources/com.github.dockerjava.core/registry.v2/config.json new file mode 100644 index 000000000..73ab82aca --- /dev/null +++ b/docker-java/src/test/resources/com.github.dockerjava.core/registry.v2/config.json @@ -0,0 +1,7 @@ +{ + "auths": { + "https://test.docker.io/v2/": { + "auth": "dXNlcjpwYXNzd29yZA==" + } + } +} diff --git a/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_empty.json b/docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_empty.json similarity index 100% rename from src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_empty.json rename to docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_empty.json diff --git a/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full.json b/docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full.json similarity index 100% rename from src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full.json rename to docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full.json diff --git a/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full_1_21.json b/docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full_1_21.json similarity index 100% rename from src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full_1_21.json rename to docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full_1_21.json diff --git a/docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full_1_26a.json b/docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full_1_26a.json new file mode 100644 index 000000000..688ea2689 --- /dev/null +++ b/docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full_1_26a.json @@ -0,0 +1,156 @@ +[ + { + "Driver" : "aufs", + "Path" : "docker-entrypoint.sh", + "Args" : [ + "postgres" + ], + "SizeRootFs" : null, + "SizeRw" : null, + "HostConfig" : { + "KernelMemory" : 0, + "MemorySwappiness" : -1, + "PidMode" : "", + "CpuPeriod" : 0, + "LogConfig" : { + "Type" : "json-file", + "Config" : {} + }, + "ReadonlyRootfs" : false, + "CgroupParent" : "", + "PublishAllPorts" : false, + "VolumeDriver" : "", + "NetworkMode" : "default", + "BlkioWeight" : 0, + "OomKillDisable" : false, + "Privileged" : false, + "CpusetMems" : "", + "ContainerIDFile" : "", + "ShmSize" : 67108864, + "CpusetCpus" : "", + "CpuShares" : 0, + "PidsLimit" : 0, + "RestartPolicy" : { + "Name" : "", + "MaximumRetryCount" : 0 + }, + "Memory" : 0, + "MemorySwap" : 0, + "CpuQuota" : 0, + "OomScoreAdj" : 500, + "MemoryReservation" : 0 + }, + "Id" : "58fd1abe8e43a65fb6231b76a9678e7bb4e91686f838945e782a4b74119ce959", + "Volumes" : null, + "State" : { + "Pid" : 0, + "ExitCode" : 0, + "FinishedAt" : "0001-01-01T00:00:00Z", + "Paused" : false, + "Error" : "", + "Status" : "created", + "OOMKilled" : false, + "Health" : null, + "oomkilled" : false, + "StartedAt" : "0001-01-01T00:00:00Z", + "Dead" : false, + "Running" : false, + "Restarting" : false + }, + "Config" : { + "OpenStdin" : false, + "Hostname" : "58fd1abe8e43", + "StdinOnce" : false, + "Domainname" : "", + "Env" : [ + "PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/lib/postgresql/10/bin", + "GOSU_VERSION=1.10", + "LANG=en_US.utf8", + "PG_MAJOR=10", + "PG_VERSION=10.4-2.pgdg90+1", + "PGDATA=/var/lib/postgresql/data" + ], + "Entrypoint" : [ + "docker-entrypoint.sh" + ], + "User" : "", + "AttachStdout" : false, + "Cmd" : [ + "postgres" + ], + "ExposedPorts" : { + "5432/tcp" : {} + }, + "Image" : "postgres:latest", + "AttachStderr" : false, + "Labels" : {}, + "Volumes" : { + "/var/lib/postgresql/data" : {} + }, + "Tty" : false, + "WorkingDir" : "", + "AttachStdin" : false + }, + "ProcessLabel" : "", + "HostsPath" : "", + "Name" : "/fmi-test01", + "ExecDriver" : null, + "LogPath" : "", + "ExecIDs" : null, + "MountLabel" : "", + "Created" : "2018-05-30T08:37:12.308001081Z", + "Image" : "sha256:61d053fc271ce1313896a2edf7719fb3b68637b53397e61d8114793b39e9ae65", + "ResolvConfPath" : "", + "HostnamePath" : "", + "RestartCount" : 0, + "NetworkSettings" : { + "GlobalIPv6PrefixLen" : 0, + "LinkLocalIPv6PrefixLen" : 0, + "EndpointID" : "", + "SandboxID" : "", + "GlobalIPv6Address" : "", + "LinkLocalIPv6Address" : "", + "HairpinMode" : false, + "PortMapping" : null, + "SecondaryIPv6Addresses" : null, + "SecondaryIPAddresses" : null, + "Gateway" : "", + "Ports" : null, + "MacAddress" : "", + "Networks" : { + "bridge" : { + "IPv6Gateway" : "", + "IPPrefixLen" : 0, + "GlobalIPv6Address" : "", + "IPAddress" : "", + "EndpointID" : "", + "Aliases" : null, + "GlobalIPv6PrefixLen" : 0, + "Gateway" : "", + "NetworkID" : "", + "IPAMConfig" : null, + "Links" : null, + "MacAddress" : "" + } + }, + "IPAddress" : "", + "Bridge" : "", + "IPPrefixLen" : 0, + "IPv6Gateway" : "", + "SandboxKey" : "" + }, + "VolumesRW" : null, + "Mounts" : [ + { + "RW" : true, + "Driver" : "local", + "Name" : "d9e76cbb4f797b0b8d62f5f4cd46ee6502fc520c6ce1187a62d200bc8364dd74", + "Source" : "/var/lib/docker/volumes/d9e76cbb4f797b0b8d62f5f4cd46ee6502fc520c6ce1187a62d200bc8364dd74/_data", + "Mode" : "", + "Destination" : { + "path" : "/var/lib/postgresql/data" + } + } + ] + } +] diff --git a/docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full_1_26b.json b/docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full_1_26b.json new file mode 100644 index 000000000..916ab3e9c --- /dev/null +++ b/docker-java/src/test/resources/com/github/dockerjava/api/command/inspectContainerResponse_full_1_26b.json @@ -0,0 +1,180 @@ +[ + { + "HostsPath" : "", + "RestartCount" : 0, + "ProcessLabel" : "", + "Args" : [], + "GraphDriver" : { + "Name" : "aufs", + "Data" : null + }, + "Config" : { + "OpenStdin" : false, + "Volumes" : { + "/srv/test" : {} + }, + "User" : "", + "Image" : "busybox:latest", + "Entrypoint" : null, + "Tty" : false, + "Env" : [ + "PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin" + ], + "StdinOnce" : false, + "OnBuild" : null, + "Hostname" : "812f8c0d2342", + "Domainname" : "", + "AttachStdout" : false, + "Cmd" : [ + "true" + ], + "AttachStdin" : false, + "WorkingDir" : "", + "AttachStderr" : false, + "Labels" : {} + }, + "Driver" : "aufs", + "Image" : "sha256:8c811b4aec35f259572d0f79207bc0678df4c736eeec50bc9fec37ed936a472a", + "LogPath" : "", + "Mounts" : [ + { + "Source" : "/var/lib/docker/volumes/53690c8e68d587308c3b8c8c76a7912aecf5abd4851f555df687229ac625112f/_data", + "Name" : "53690c8e68d587308c3b8c8c76a7912aecf5abd4851f555df687229ac625112f", + "RW" : true, + "Type" : "volume", + "Destination" : "/srv/test", + "Driver" : "local", + "Propagation" : "", + "Mode" : "" + } + ], + "MountLabel" : "", + "ResolvConfPath" : "", + "AppArmorProfile" : "", + "NetworkSettings" : { + "MacAddress" : "", + "SecondaryIPv6Addresses" : null, + "LinkLocalIPv6PrefixLen" : 0, + "LinkLocalIPv6Address" : "", + "GlobalIPv6PrefixLen" : 0, + "Networks" : { + "bridge" : { + "GlobalIPv6Address" : "", + "NetworkID" : "", + "Gateway" : "", + "IPAddress" : "", + "IPPrefixLen" : 0, + "MacAddress" : "", + "EndpointID" : "", + "IPAMConfig" : null, + "IPv6Gateway" : "", + "GlobalIPv6PrefixLen" : 0, + "Aliases" : null, + "Links" : null + } + }, + "SandboxKey" : "", + "EndpointID" : "", + "Bridge" : "", + "IPPrefixLen" : 0, + "Ports" : null, + "IPAddress" : "", + "IPv6Gateway" : "", + "SandboxID" : "", + "Gateway" : "", + "GlobalIPv6Address" : "", + "SecondaryIPAddresses" : null, + "HairpinMode" : false + }, + "Name" : "/elegant_mccarthy", + "Id" : "812f8c0d2342c45bda537b856dd50f65f6c4cd13cded5231d0424edfb9132f38", + "HostConfig" : { + "Dns" : null, + "BlkioDeviceWriteBps" : null, + "BlkioWeight" : 0, + "CpuQuota" : 0, + "ExtraHosts" : null, + "NanoCpus" : 0, + "KernelMemory" : 0, + "MemorySwap" : 0, + "CgroupParent" : "", + "ShmSize" : 67108864, + "CpuPercent" : 0, + "DnsOptions" : null, + "PidMode" : "", + "BlkioDeviceWriteIOps" : null, + "Devices" : null, + "Binds" : null, + "Memory" : 0, + "RestartPolicy" : { + "MaximumRetryCount" : 0, + "Name" : "" + }, + "IOMaximumIOps" : 0, + "CpuRealtimeRuntime" : 0, + "PublishAllPorts" : false, + "CapAdd" : null, + "ReadonlyRootfs" : false, + "GroupAdd" : null, + "MemoryReservation" : 0, + "AutoRe ove" : false, + "UsernsMode" : "", + "Runtime" : "runc", + "BlkioWeightDevice" : null, + "Ulimits" : null, + "Cgroup" : "", + "NetworkMode" : "default", + "CpusetCpus" : "", + "Isolation" : "", + "CpuCount" : 0, + "CpuRealtimePeriod" : 0, + "IOMaximumBandwidth" : 0, + "IpcMode" : "", + "OomKillDisable" : false, + "OomScoreAdj" : 0, + "SecurityOpt" : null, + "MemorySwappiness" : -1, + "ContainerIDFile" : "", + "BlkioDeviceReadBps" : null, + "CpusetMems" : "", + "PortBindings" : null, + "BlkioDeviceReadIOps" : null, + "PidsLimit" : 0, + "VolumesFrom" : null, + "Links" : null, + "Privileged" : false, + "ConsoleSize" : [ + 0, + 0 + ], + "DnsSearch" : null, + "UTSMode" : "", + "LogConfig" : { + "Type" : "json-file", + "Config" : {} + }, + "CpuShares" : 0, + "VolumeDriver" : "", + "CpuPeriod" : 0, + "CapDrop" : null, + "DiskQuota" : 0 + }, + "Created" : "2018-06-04T08:09:33.086574793Z", + "HostnamePath" : "", + "ExecIDs" : null, + "Path" : "true", + "State" : { + "Error" : "", + "Pid" : 0, + "Running" : false, + "Dead" : false, + "OOMKilled" : false, + "ExitCode" : 0, + "StartedAt" : "0001-01-01T00:00:00Z", + "Paused" : false, + "Restarting" : false, + "FinishedAt" : "0001-01-01T00:00:00Z", + "Status" : "created" + } + } +] diff --git a/docker-java/src/test/resources/com/github/dockerjava/api/command/updateContainerResponse_empty.json b/docker-java/src/test/resources/com/github/dockerjava/api/command/updateContainerResponse_empty.json new file mode 100644 index 000000000..0967ef424 --- /dev/null +++ b/docker-java/src/test/resources/com/github/dockerjava/api/command/updateContainerResponse_empty.json @@ -0,0 +1 @@ +{} diff --git a/docker-java/src/test/resources/com/github/dockerjava/api/command/updateContainerResponse_warnings.json b/docker-java/src/test/resources/com/github/dockerjava/api/command/updateContainerResponse_warnings.json new file mode 100644 index 000000000..edeaedc7a --- /dev/null +++ b/docker-java/src/test/resources/com/github/dockerjava/api/command/updateContainerResponse_warnings.json @@ -0,0 +1,5 @@ +{ + "Warnings": [ + "Published ports are discarded when using host network mode" + ] +} diff --git a/docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_alreadyExists.json b/docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_alreadyExists.json new file mode 100644 index 000000000..ae318e29d --- /dev/null +++ b/docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_alreadyExists.json @@ -0,0 +1 @@ +{"status":"Already exists"} diff --git a/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_error.json b/docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_error.json similarity index 100% rename from src/test/resources/com/github/dockerjava/api/model/pullImageResponse_error.json rename to docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_error.json diff --git a/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_legacy.json b/docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_legacy.json similarity index 100% rename from src/test/resources/com/github/dockerjava/api/model/pullImageResponse_legacy.json rename to docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_legacy.json diff --git a/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_newerImage.json b/docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_newerImage.json similarity index 100% rename from src/test/resources/com/github/dockerjava/api/model/pullImageResponse_newerImage.json rename to docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_newerImage.json diff --git a/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_upToDate.json b/docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_upToDate.json similarity index 100% rename from src/test/resources/com/github/dockerjava/api/model/pullImageResponse_upToDate.json rename to docker-java/src/test/resources/com/github/dockerjava/api/model/pullImageResponse_upToDate.json diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/ca.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/ca.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/ca.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/ca.pem diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/cert.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/cert.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/cert.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/cert.pem diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/key.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/key.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/key.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist/key.pem diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caAndCertMissing/key.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caAndCertMissing/key.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caAndCertMissing/key.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caAndCertMissing/key.pem diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caAndKeyMissing/cert.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caAndKeyMissing/cert.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caAndKeyMissing/cert.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caAndKeyMissing/cert.pem diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caMissing/cert.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caMissing/cert.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caMissing/cert.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caMissing/cert.pem diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caMissing/key.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caMissing/key.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caMissing/key.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caMissing/key.pem diff --git a/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caTest/multiple_ca.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caTest/multiple_ca.pem new file mode 100644 index 000000000..53b571269 --- /dev/null +++ b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caTest/multiple_ca.pem @@ -0,0 +1,50 @@ +-----BEGIN CERTIFICATE----- +MIIENjCCAx6gAwIBAgIJALOOFdRswGmDMA0GCSqGSIb3DQEBBQUAMG8xCzAJBgNV +BAYTAlVTMRMwEQYDVQQIEwpTb21lLVN0YXRlMSEwHwYDVQQKExhJbnRlcm5ldCBX +aWRnaXRzIFB0eSBMdGQxDDAKBgNVBAMTA0NBMTEaMBgGCSqGSIb3DQEJARYLY2FA +dGVzdC5jb20wHhcNMTcwMTI1MTc0ODE5WhcNMjcwMTIzMTc0ODE5WjBvMQswCQYD +VQQGEwJVUzETMBEGA1UECBMKU29tZS1TdGF0ZTEhMB8GA1UEChMYSW50ZXJuZXQg +V2lkZ2l0cyBQdHkgTHRkMQwwCgYDVQQDEwNDQTExGjAYBgkqhkiG9w0BCQEWC2Nh +QHRlc3QuY29tMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA346h3pA5 +5Uu6t7I8oNsiwF9PNgY7FKL/fXAiGAi1GMcPvil5Sf+GOYenQNsh4unfMRu8C+Xn +0NSnQNRkgJ2eIR5w3qDDK+UFj34rLdBK+wNze3gMGOaTFiSZLeoxrXs6fGbJ7Jmd +GaW7u8+P1+1Iej2k7lz61SXRL7+fn5lo3Lpfi+7s44zzOHoPkbHYNyl3md5Wgsoc +U+VGcABl1cq956ycQU1/jWZgclaTL3sQjNvLJ+dnN4+IwbFiPE9tY+cJtdNn2YXd +jLqxkfr/MZXoxArASanzPa4S7sKl8nnGrYM8cnppMeFk2a/p1YT+507V6VEmLM1A +IlqtzOAWpZ4vSQIDAQABo4HUMIHRMB0GA1UdDgQWBBRTvtGtQ298/2Ukc6ncXNUI +plkkxjCBoQYDVR0jBIGZMIGWgBRTvtGtQ298/2Ukc6ncXNUIplkkxqFzpHEwbzEL +MAkGA1UEBhMCVVMxEzARBgNVBAgTClNvbWUtU3RhdGUxITAfBgNVBAoTGEludGVy +bmV0IFdpZGdpdHMgUHR5IEx0ZDEMMAoGA1UEAxMDQ0ExMRowGAYJKoZIhvcNAQkB +FgtjYUB0ZXN0LmNvbYIJALOOFdRswGmDMAwGA1UdEwQFMAMBAf8wDQYJKoZIhvcN +AQEFBQADggEBAIs8PjXPWvTWVDQ8xsZHLosD5BzCqM6ac+W7pjKEVNaiIo+UT3st +bl1rZS2U924M9xtOkmAOPCQx1ozMxqKk2SVYztll3/Nw7p3qbP2/7L1tOTjoNZqT +o2dd4Jf8txj9AaDus9FW8gmR6TLSgm7xeNb/kEa1b5ZW6l0zYRSBNT4I38d9wCa4 +QeohD8t1T1QOSc82dTHqccSiCSRiOX883I/fK+cTo8o1gdyUjYoCQlsJESr1rUun +C3zMTZ4Lkt72vM88Hf0OyK/+sw7sqVEf+95VMx+a/UaYYkJY3Bg6JTJb9vukbtAl +5GrDZOGM1AgW8iZyG/jSJlwkjiB9vy8RRnI= +-----END CERTIFICATE----- +-----BEGIN CERTIFICATE----- +MIIENjCCAx6gAwIBAgIJAL4WwpOexcT8MA0GCSqGSIb3DQEBBQUAMG8xCzAJBgNV +BAYTAlVTMRMwEQYDVQQIEwpTb21lLVN0YXRlMSEwHwYDVQQKExhJbnRlcm5ldCBX +aWRnaXRzIFB0eSBMdGQxDDAKBgNVBAMTA0NBMjEaMBgGCSqGSIb3DQEJARYLY2FA +dGVzdC5jb20wHhcNMTcwMTI1MTc0ODQ4WhcNMjcwMTIzMTc0ODQ4WjBvMQswCQYD +VQQGEwJVUzETMBEGA1UECBMKU29tZS1TdGF0ZTEhMB8GA1UEChMYSW50ZXJuZXQg +V2lkZ2l0cyBQdHkgTHRkMQwwCgYDVQQDEwNDQTIxGjAYBgkqhkiG9w0BCQEWC2Nh +QHRlc3QuY29tMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA346h3pA5 +5Uu6t7I8oNsiwF9PNgY7FKL/fXAiGAi1GMcPvil5Sf+GOYenQNsh4unfMRu8C+Xn +0NSnQNRkgJ2eIR5w3qDDK+UFj34rLdBK+wNze3gMGOaTFiSZLeoxrXs6fGbJ7Jmd +GaW7u8+P1+1Iej2k7lz61SXRL7+fn5lo3Lpfi+7s44zzOHoPkbHYNyl3md5Wgsoc +U+VGcABl1cq956ycQU1/jWZgclaTL3sQjNvLJ+dnN4+IwbFiPE9tY+cJtdNn2YXd +jLqxkfr/MZXoxArASanzPa4S7sKl8nnGrYM8cnppMeFk2a/p1YT+507V6VEmLM1A +IlqtzOAWpZ4vSQIDAQABo4HUMIHRMB0GA1UdDgQWBBRTvtGtQ298/2Ukc6ncXNUI +plkkxjCBoQYDVR0jBIGZMIGWgBRTvtGtQ298/2Ukc6ncXNUIplkkxqFzpHEwbzEL +MAkGA1UEBhMCVVMxEzARBgNVBAgTClNvbWUtU3RhdGUxITAfBgNVBAoTGEludGVy +bmV0IFdpZGdpdHMgUHR5IEx0ZDEMMAoGA1UEAxMDQ0EyMRowGAYJKoZIhvcNAQkB +FgtjYUB0ZXN0LmNvbYIJAL4WwpOexcT8MAwGA1UdEwQFMAMBAf8wDQYJKoZIhvcN +AQEFBQADggEBAKVIxgBQ76JQYfejcaK946J6VglSKaHuPZ8bDVM9e2KB1HFwTzAA +oRtMBIJaSO1+CSopTnzvzv/j8XeXbAUt6UHWiejGcIke1wW1CSVHeDQM7KUzXYtu +2jWzlGHXtnni4EeSwdE628Kfk82r/NU3+3+zo/3ASkYcSGBEYgIdvLBmbvQ8sviJ +rQZ1HCDP/TgP/ExBDi2QWBHe3XSTD0pBK153FC2YT1Q0AG5mxsIeq7C85m22aNX9 +fUmG5ORnVMkT/ktY+jg5aP+50o58MX24ZG1mh59vZtuBbvD74qZ2eNauStt1ji34 +V43XhW5Bigsp/z7GyFb2VzejH9n+VvPmTnU= +-----END CERTIFICATE----- diff --git a/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caTest/single_ca.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caTest/single_ca.pem new file mode 100644 index 000000000..8e67cb783 --- /dev/null +++ b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/caTest/single_ca.pem @@ -0,0 +1,25 @@ +-----BEGIN CERTIFICATE----- +MIIENjCCAx6gAwIBAgIJAL4WwpOexcT8MA0GCSqGSIb3DQEBBQUAMG8xCzAJBgNV +BAYTAlVTMRMwEQYDVQQIEwpTb21lLVN0YXRlMSEwHwYDVQQKExhJbnRlcm5ldCBX +aWRnaXRzIFB0eSBMdGQxDDAKBgNVBAMTA0NBMjEaMBgGCSqGSIb3DQEJARYLY2FA +dGVzdC5jb20wHhcNMTcwMTI1MTc0ODQ4WhcNMjcwMTIzMTc0ODQ4WjBvMQswCQYD +VQQGEwJVUzETMBEGA1UECBMKU29tZS1TdGF0ZTEhMB8GA1UEChMYSW50ZXJuZXQg +V2lkZ2l0cyBQdHkgTHRkMQwwCgYDVQQDEwNDQTIxGjAYBgkqhkiG9w0BCQEWC2Nh +QHRlc3QuY29tMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA346h3pA5 +5Uu6t7I8oNsiwF9PNgY7FKL/fXAiGAi1GMcPvil5Sf+GOYenQNsh4unfMRu8C+Xn +0NSnQNRkgJ2eIR5w3qDDK+UFj34rLdBK+wNze3gMGOaTFiSZLeoxrXs6fGbJ7Jmd +GaW7u8+P1+1Iej2k7lz61SXRL7+fn5lo3Lpfi+7s44zzOHoPkbHYNyl3md5Wgsoc +U+VGcABl1cq956ycQU1/jWZgclaTL3sQjNvLJ+dnN4+IwbFiPE9tY+cJtdNn2YXd +jLqxkfr/MZXoxArASanzPa4S7sKl8nnGrYM8cnppMeFk2a/p1YT+507V6VEmLM1A +IlqtzOAWpZ4vSQIDAQABo4HUMIHRMB0GA1UdDgQWBBRTvtGtQ298/2Ukc6ncXNUI +plkkxjCBoQYDVR0jBIGZMIGWgBRTvtGtQ298/2Ukc6ncXNUIplkkxqFzpHEwbzEL +MAkGA1UEBhMCVVMxEzARBgNVBAgTClNvbWUtU3RhdGUxITAfBgNVBAoTGEludGVy +bmV0IFdpZGdpdHMgUHR5IEx0ZDEMMAoGA1UEAxMDQ0EyMRowGAYJKoZIhvcNAQkB +FgtjYUB0ZXN0LmNvbYIJAL4WwpOexcT8MAwGA1UdEwQFMAMBAf8wDQYJKoZIhvcN +AQEFBQADggEBAKVIxgBQ76JQYfejcaK946J6VglSKaHuPZ8bDVM9e2KB1HFwTzAA +oRtMBIJaSO1+CSopTnzvzv/j8XeXbAUt6UHWiejGcIke1wW1CSVHeDQM7KUzXYtu +2jWzlGHXtnni4EeSwdE628Kfk82r/NU3+3+zo/3ASkYcSGBEYgIdvLBmbvQ8sviJ +rQZ1HCDP/TgP/ExBDi2QWBHe3XSTD0pBK153FC2YT1Q0AG5mxsIeq7C85m22aNX9 +fUmG5ORnVMkT/ktY+jg5aP+50o58MX24ZG1mh59vZtuBbvD74qZ2eNauStt1ji34 +V43XhW5Bigsp/z7GyFb2VzejH9n+VvPmTnU= +-----END CERTIFICATE----- diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certAndKeyMissing/ca.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certAndKeyMissing/ca.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certAndKeyMissing/ca.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certAndKeyMissing/ca.pem diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certMissing/ca.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certMissing/ca.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certMissing/ca.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certMissing/ca.pem diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certMissing/key.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certMissing/key.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certMissing/key.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/certMissing/key.pem diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/keyMissing/ca.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/keyMissing/ca.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/keyMissing/ca.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/keyMissing/ca.pem diff --git a/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/keyMissing/cert.pem b/docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/keyMissing/cert.pem similarity index 100% rename from src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/keyMissing/cert.pem rename to docker-java/src/test/resources/com/github/dockerjava/core/util/CertificateUtilsTest/keyMissing/cert.pem diff --git a/docker-java/src/test/resources/dockerContextHomeDir/.docker/config.json b/docker-java/src/test/resources/dockerContextHomeDir/.docker/config.json new file mode 100644 index 000000000..5bb2ebebe --- /dev/null +++ b/docker-java/src/test/resources/dockerContextHomeDir/.docker/config.json @@ -0,0 +1,4 @@ +{ + "auths": {}, + "currentContext": "configcontext" +} diff --git a/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/meta/51699a7c75211315f1dbf6ecc40dfb0ffdd4ee11ecb2ce7853c9751aea1f9444/meta.json b/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/meta/51699a7c75211315f1dbf6ecc40dfb0ffdd4ee11ecb2ce7853c9751aea1f9444/meta.json new file mode 100644 index 000000000..c6456d6b8 --- /dev/null +++ b/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/meta/51699a7c75211315f1dbf6ecc40dfb0ffdd4ee11ecb2ce7853c9751aea1f9444/meta.json @@ -0,0 +1,12 @@ +{ + "Name": "envvarcontext", + "Metadata": { + "Description": "envvarcontext" + }, + "Endpoints": { + "docker": { + "Host": "unix:///envvarcontext.sock", + "SkipTLSVerify": false + } + } +} diff --git a/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/meta/b71199ebd070b36beab7317920c2c2f1d777df8d05e5527d8458fda57cb17a7a/meta.json b/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/meta/b71199ebd070b36beab7317920c2c2f1d777df8d05e5527d8458fda57cb17a7a/meta.json new file mode 100644 index 000000000..a4ff5b460 --- /dev/null +++ b/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/meta/b71199ebd070b36beab7317920c2c2f1d777df8d05e5527d8458fda57cb17a7a/meta.json @@ -0,0 +1,15 @@ +{ + "Name": "remote", + "Metadata": { + "Description": "remote" + }, + "Endpoints": { + "docker": { + "Host": "tcp://remote:2376", + "SkipTLSVerify": false + } + }, + "Storage": { + "TLSPath": "target/test-classes/com/github/dockerjava/core/util/CertificateUtilsTest/allFilesExist" + } +} diff --git a/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/meta/d090e08f0c9167acd72adef6d9fa07ec2de3a873cdd545dd8cb7fc7a10a1331a/meta.json b/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/meta/d090e08f0c9167acd72adef6d9fa07ec2de3a873cdd545dd8cb7fc7a10a1331a/meta.json new file mode 100644 index 000000000..adff3b1c9 --- /dev/null +++ b/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/meta/d090e08f0c9167acd72adef6d9fa07ec2de3a873cdd545dd8cb7fc7a10a1331a/meta.json @@ -0,0 +1,12 @@ +{ + "Name": "configcontext", + "Metadata": { + "Description": "configcontext" + }, + "Endpoints": { + "docker": { + "Host": "unix:///configcontext.sock", + "SkipTLSVerify": false + } + } +} diff --git a/src/test/resources/someHomeDir/.docker/certs/dummy.txt b/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/tls/b71199ebd070b36beab7317920c2c2f1d777df8d05e5527d8458fda57cb17a7a/docker/ca.pem similarity index 100% rename from src/test/resources/someHomeDir/.docker/certs/dummy.txt rename to docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/tls/b71199ebd070b36beab7317920c2c2f1d777df8d05e5527d8458fda57cb17a7a/docker/ca.pem diff --git a/src/test/resources/testAuthConfigFile/emptyFile b/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/tls/b71199ebd070b36beab7317920c2c2f1d777df8d05e5527d8458fda57cb17a7a/docker/cert.pem similarity index 100% rename from src/test/resources/testAuthConfigFile/emptyFile rename to docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/tls/b71199ebd070b36beab7317920c2c2f1d777df8d05e5527d8458fda57cb17a7a/docker/cert.pem diff --git a/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/tls/b71199ebd070b36beab7317920c2c2f1d777df8d05e5527d8458fda57cb17a7a/docker/key.pem b/docker-java/src/test/resources/dockerContextHomeDir/.docker/contexts/tls/b71199ebd070b36beab7317920c2c2f1d777df8d05e5527d8458fda57cb17a7a/docker/key.pem new file mode 100644 index 000000000..e69de29bb diff --git a/src/test/resources/eventStreamReaderDockerfile/Dockerfile b/docker-java/src/test/resources/eventStreamReaderDockerfile/Dockerfile similarity index 100% rename from src/test/resources/eventStreamReaderDockerfile/Dockerfile rename to docker-java/src/test/resources/eventStreamReaderDockerfile/Dockerfile diff --git a/src/test/resources/frameReaderDockerfile/Dockerfile b/docker-java/src/test/resources/frameReaderDockerfile/Dockerfile similarity index 100% rename from src/test/resources/frameReaderDockerfile/Dockerfile rename to docker-java/src/test/resources/frameReaderDockerfile/Dockerfile diff --git a/docker-java/src/test/resources/logback.xml b/docker-java/src/test/resources/logback.xml new file mode 100644 index 000000000..8fb1a7a6d --- /dev/null +++ b/docker-java/src/test/resources/logback.xml @@ -0,0 +1,20 @@ + + + + + %d{HH:mm:ss.SSS} %-5level %logger{36} - %msg%n + + + + + + + + + + + + + + + diff --git a/src/test/resources/privateRegistry/Dockerfile b/docker-java/src/test/resources/privateRegistry/Dockerfile similarity index 100% rename from src/test/resources/privateRegistry/Dockerfile rename to docker-java/src/test/resources/privateRegistry/Dockerfile diff --git a/src/test/resources/privateRegistry/auth/htpasswd b/docker-java/src/test/resources/privateRegistry/auth/htpasswd similarity index 100% rename from src/test/resources/privateRegistry/auth/htpasswd rename to docker-java/src/test/resources/privateRegistry/auth/htpasswd diff --git a/src/test/resources/privateRegistry/certs/README.txt b/docker-java/src/test/resources/privateRegistry/certs/README.txt similarity index 100% rename from src/test/resources/privateRegistry/certs/README.txt rename to docker-java/src/test/resources/privateRegistry/certs/README.txt diff --git a/src/test/resources/privateRegistry/certs/domain.crt b/docker-java/src/test/resources/privateRegistry/certs/domain.crt similarity index 100% rename from src/test/resources/privateRegistry/certs/domain.crt rename to docker-java/src/test/resources/privateRegistry/certs/domain.crt diff --git a/src/test/resources/privateRegistry/certs/domain.key b/docker-java/src/test/resources/privateRegistry/certs/domain.key similarity index 100% rename from src/test/resources/privateRegistry/certs/domain.key rename to docker-java/src/test/resources/privateRegistry/certs/domain.key diff --git a/src/test/resources/samples/1.22/containers/container/json/1.json b/docker-java/src/test/resources/samples/1.22/containers/container/json/1.json similarity index 100% rename from src/test/resources/samples/1.22/containers/container/json/1.json rename to docker-java/src/test/resources/samples/1.22/containers/container/json/1.json diff --git a/src/test/resources/samples/1.22/containers/container/update/docs.json b/docker-java/src/test/resources/samples/1.22/containers/container/update/docs.json similarity index 100% rename from src/test/resources/samples/1.22/containers/container/update/docs.json rename to docker-java/src/test/resources/samples/1.22/containers/container/update/docs.json diff --git a/src/test/resources/samples/1.22/containers/create/docs.json b/docker-java/src/test/resources/samples/1.22/containers/create/docs.json similarity index 100% rename from src/test/resources/samples/1.22/containers/create/docs.json rename to docker-java/src/test/resources/samples/1.22/containers/create/docs.json diff --git a/src/test/resources/samples/1.22/containers/json/filter1.json b/docker-java/src/test/resources/samples/1.22/containers/json/filter1.json similarity index 98% rename from src/test/resources/samples/1.22/containers/json/filter1.json rename to docker-java/src/test/resources/samples/1.22/containers/json/filter1.json index 159e62da6..51329bb63 100644 --- a/src/test/resources/samples/1.22/containers/json/filter1.json +++ b/docker-java/src/test/resources/samples/1.22/containers/json/filter1.json @@ -10,6 +10,7 @@ "Created": 1455662451, "Ports": [], "SizeRootFs": 1113554, + "SizeRw": 0, "Labels": {}, "Status": "Up Less than a second", "HostConfig": { diff --git a/src/test/resources/samples/1.22/exec/ID/1.json b/docker-java/src/test/resources/samples/1.22/exec/ID/1.json similarity index 100% rename from src/test/resources/samples/1.22/exec/ID/1.json rename to docker-java/src/test/resources/samples/1.22/exec/ID/1.json diff --git a/src/test/resources/samples/1.22/images/docImage/doc.json b/docker-java/src/test/resources/samples/1.22/images/docImage/doc.json similarity index 100% rename from src/test/resources/samples/1.22/images/docImage/doc.json rename to docker-java/src/test/resources/samples/1.22/images/docImage/doc.json diff --git a/src/test/resources/samples/1.22/images/docImage/inspect_doc.json b/docker-java/src/test/resources/samples/1.22/images/docImage/inspect_doc.json similarity index 100% rename from src/test/resources/samples/1.22/images/docImage/inspect_doc.json rename to docker-java/src/test/resources/samples/1.22/images/docImage/inspect_doc.json diff --git a/src/test/resources/samples/1.22/images/image1/inspect1.json b/docker-java/src/test/resources/samples/1.22/images/image1/inspect1.json similarity index 100% rename from src/test/resources/samples/1.22/images/image1/inspect1.json rename to docker-java/src/test/resources/samples/1.22/images/image1/inspect1.json diff --git a/src/test/resources/samples/1.22/images/overlay/inspectOverlay.json b/docker-java/src/test/resources/samples/1.22/images/overlay/inspectOverlay.json similarity index 100% rename from src/test/resources/samples/1.22/images/overlay/inspectOverlay.json rename to docker-java/src/test/resources/samples/1.22/images/overlay/inspectOverlay.json diff --git a/src/test/resources/samples/1.22/info/1.json b/docker-java/src/test/resources/samples/1.22/info/1.json similarity index 100% rename from src/test/resources/samples/1.22/info/1.json rename to docker-java/src/test/resources/samples/1.22/info/1.json diff --git a/src/test/resources/samples/1.22/info/2.json b/docker-java/src/test/resources/samples/1.22/info/2.json similarity index 100% rename from src/test/resources/samples/1.22/info/2.json rename to docker-java/src/test/resources/samples/1.22/info/2.json diff --git a/src/test/resources/samples/1.22/info/docs.json b/docker-java/src/test/resources/samples/1.22/info/docs.json similarity index 100% rename from src/test/resources/samples/1.22/info/docs.json rename to docker-java/src/test/resources/samples/1.22/info/docs.json diff --git a/src/test/resources/samples/1.22/other/AuthConfig/docs1.json b/docker-java/src/test/resources/samples/1.22/other/AuthConfig/docs1.json similarity index 100% rename from src/test/resources/samples/1.22/other/AuthConfig/docs1.json rename to docker-java/src/test/resources/samples/1.22/other/AuthConfig/docs1.json diff --git a/src/test/resources/samples/1.22/other/AuthConfig/docs2.json b/docker-java/src/test/resources/samples/1.22/other/AuthConfig/docs2.json similarity index 100% rename from src/test/resources/samples/1.22/other/AuthConfig/docs2.json rename to docker-java/src/test/resources/samples/1.22/other/AuthConfig/docs2.json diff --git a/src/test/resources/samples/1.22/version/1.json b/docker-java/src/test/resources/samples/1.22/version/1.json similarity index 100% rename from src/test/resources/samples/1.22/version/1.json rename to docker-java/src/test/resources/samples/1.22/version/1.json diff --git a/docker-java/src/test/resources/samples/1.23/other/AuthConfig/docs1.json b/docker-java/src/test/resources/samples/1.23/other/AuthConfig/docs1.json new file mode 100644 index 000000000..1ebc88193 --- /dev/null +++ b/docker-java/src/test/resources/samples/1.23/other/AuthConfig/docs1.json @@ -0,0 +1,4 @@ +{ + "auth": "YWRtaW46", + "identitytoken": "1cba468e-8cbe-4c55-9098-2c2ed769e885" +} \ No newline at end of file diff --git a/docker-java/src/test/resources/samples/1.24/containers/inspect/1.json b/docker-java/src/test/resources/samples/1.24/containers/inspect/1.json new file mode 100644 index 000000000..54d23923f --- /dev/null +++ b/docker-java/src/test/resources/samples/1.24/containers/inspect/1.json @@ -0,0 +1,167 @@ +{ + "Id": "fc1243c01bbb791d9eca64204d76f72fc47fbebbca892d7dd0a5cd0e4e346045", + "Created": "2015-11-20T21:10:34.775649753Z", + "Path": "cat", + "Args": [], + "State": { + "Status": "running", + "Running": true, + "Paused": false, + "Restarting": false, + "OOMKilled": false, + "Dead": false, + "Pid": 10912, + "ExitCode": 0, + "Error": "", + "StartedAt": "2015-11-20T21:10:34.845546358Z", + "FinishedAt": "0001-01-01T00:00:00Z", + "Health": { + "Status": "healthy", + "FailingStreak": 0, + "Log": [ + { + "Start": "2016-11-12T03:23:03.351561Z", + "End": "2016-11-12T03:23:03.422176171Z", + "ExitCode": 0, + "Output" : "Hello" + }, + { + "Start": "2016-11-12T03:23:08.423558928Z", + "End": "2016-11-12T03:23:08.510122392Z", + "ExitCode": 0, + "Output" : "World" + } + ] + } + }, + "Image": "c51f86c283408d1749d066333f7acd5d33b053b003a61ff6a7b36819ddcbc7b7", + "ResolvConfPath": "/mnt/sda1/var/lib/docker/containers/fc1243c01bbb791d9eca64204d76f72fc47fbebbca892d7dd0a5cd0e4e346045/resolv.conf", + "HostnamePath": "/mnt/sda1/var/lib/docker/containers/fc1243c01bbb791d9eca64204d76f72fc47fbebbca892d7dd0a5cd0e4e346045/hostname", + "HostsPath": "/mnt/sda1/var/lib/docker/containers/fc1243c01bbb791d9eca64204d76f72fc47fbebbca892d7dd0a5cd0e4e346045/hosts", + "LogPath": "/mnt/sda1/var/lib/docker/containers/fc1243c01bbb791d9eca64204d76f72fc47fbebbca892d7dd0a5cd0e4e346045/fc1243c01bbb791d9eca64204d76f72fc47fbebbca892d7dd0a5cd0e4e346045-json.log", + "Name": "/small_hodgkin", + "RestartCount": 0, + "Driver": "aufs", + "ExecDriver": "native-0.2", + "MountLabel": "", + "ProcessLabel": "", + "AppArmorProfile": "", + "ExecIDs": null, + "HostConfig": { + "Binds": null, + "ContainerIDFile": "", + "LxcConf": [], + "Memory": 0, + "MemoryReservation": 0, + "MemorySwap": 0, + "KernelMemory": 0, + "CpuShares": 0, + "CpuPeriod": 0, + "CpusetCpus": "", + "CpusetMems": "", + "CpuQuota": 0, + "BlkioWeight": 0, + "OomKillDisable": false, + "MemorySwappiness": -1, + "Privileged": false, + "PortBindings": {}, + "Links": null, + "PublishAllPorts": false, + "Dns": null, + "DnsOptions": null, + "DnsSearch": null, + "ExtraHosts": null, + "VolumesFrom": null, + "Devices": [], + "NetworkMode": "default", + "IpcMode": "", + "PidMode": "", + "UTSMode": "", + "CapAdd": null, + "CapDrop": null, + "GroupAdd": null, + "RestartPolicy": { + "Name": "no", + "MaximumRetryCount": 0 + }, + "SecurityOpt": null, + "ReadonlyRootfs": false, + "Ulimits": null, + "LogConfig": { + "Type": "json-file", + "Config": {} + }, + "CgroupParent": "", + "ConsoleSize": [ + 0, + 0 + ], + "VolumeDriver": "" + }, + "GraphDriver": { + "Name": "aufs", + "Data": null + }, + "Mounts": [], + "Config": { + "Hostname": "fc1243c01bbb", + "Domainname": "", + "User": "", + "AttachStdin": false, + "AttachStdout": false, + "AttachStderr": false, + "Tty": false, + "OpenStdin": true, + "StdinOnce": false, + "Env": null, + "Cmd": [ + "cat" + ], + "Healthcheck": { + "Test": [ + "CMD-SHELL", + "mysqladmin ping --silent" + ], + "Interval": 2000000000, + "Timeout": 3000000000 + }, + "Image": "busybox", + "Volumes": null, + "WorkingDir": "", + "Entrypoint": null, + "OnBuild": null, + "Labels": {}, + "StopSignal": "SIGTERM" + }, + "NetworkSettings": { + "Bridge": "", + "SandboxID": "5a6ded01bf23cc180e8ba6a059449ac832f28fa1d8367127e316607f92d3228c", + "HairpinMode": false, + "LinkLocalIPv6Address": "", + "LinkLocalIPv6PrefixLen": 0, + "Ports": {}, + "SandboxKey": "/var/run/docker/netns/5a6ded01bf23", + "SecondaryIPAddresses": null, + "SecondaryIPv6Addresses": null, + "EndpointID": "6b4bb2aff9981d6e132cf37ebbfd370069061fab848ae56247b154717a99aba7", + "Gateway": "172.17.0.1", + "GlobalIPv6Address": "", + "GlobalIPv6PrefixLen": 0, + "IPAddress": "172.17.0.2", + "IPPrefixLen": 16, + "IPv6Gateway": "", + "MacAddress": "02:42:ac:11:00:02", + "Networks": { + "bridge": { + "EndpointID": "6b4bb2aff9981d6e132cf37ebbfd370069061fab848ae56247b154717a99aba7", + "Gateway": "172.17.0.1", + "IPAddress": "172.17.0.2", + "IPPrefixLen": 16, + "IPv6Gateway": "", + "GlobalIPv6Address": "", + "GlobalIPv6PrefixLen": 0, + "MacAddress": "02:42:ac:11:00:02" + } + } + } +} diff --git a/docker-java/src/test/resources/samples/1.24/events/docs1.json b/docker-java/src/test/resources/samples/1.24/events/docs1.json new file mode 100644 index 000000000..f62f2284e --- /dev/null +++ b/docker-java/src/test/resources/samples/1.24/events/docs1.json @@ -0,0 +1,17 @@ +{ + "status": "create", + "id": "ede54ee1afda366ab42f824e8a5ffd195155d853ceaec74a927f249ea270c743", + "from": "alpine", + "Type": "container", + "Action": "create", + "Actor": { + "ID": "ede54ee1afda366ab42f824e8a5ffd195155d853ceaec74a927f249ea270c743", + "Attributes": { + "com.example.some-label": "some-label-value", + "image": "alpine", + "name": "my-container" + } + }, + "time": 1461943101, + "timeNano": 1461943101381709551 +} diff --git a/docker-java/src/test/resources/samples/1.25/images/windowsImage/doc.json b/docker-java/src/test/resources/samples/1.25/images/windowsImage/doc.json new file mode 100644 index 000000000..e7b68c078 --- /dev/null +++ b/docker-java/src/test/resources/samples/1.25/images/windowsImage/doc.json @@ -0,0 +1,72 @@ +{ + "Id": "sha256:105d76d0f40e38427c63023ffe649bf36fa85058d3469551e43e4dcc2431fb31", + "RepoTags": [ + "microsoft/nanoserver:latest" + ], + "RepoDigests": [ + "microsoft/nanoserver@sha256:aee7d4330fe3dc5987c808f647441c16ed2fa1c7d9c6ef49d6498e5c9860b50b" + ], + "Parent": "", + "Comment": "", + "Created": "2016-09-22T02:39:30.9154862-07:00", + "Container": "", + "ContainerConfig": { + "Hostname": "", + "Domainname": "", + "User": "", + "AttachStdin": false, + "AttachStdout": false, + "AttachStderr": false, + "Tty": false, + "OpenStdin": false, + "StdinOnce": false, + "Env": null, + "Cmd": null, + "Image": "", + "Volumes": null, + "WorkingDir": "", + "Entrypoint": null, + "OnBuild": null, + "Labels": null + }, + "DockerVersion": "", + "Author": "", + "Config": { + "Hostname": "", + "Domainname": "", + "User": "", + "AttachStdin": false, + "AttachStdout": false, + "AttachStderr": false, + "Tty": false, + "OpenStdin": false, + "StdinOnce": false, + "Env": null, + "Cmd": [ + "c:\\windows\\system32\\cmd.exe" + ], + "Image": "", + "Volumes": null, + "WorkingDir": "", + "Entrypoint": null, + "OnBuild": null, + "Labels": null + }, + "Architecture": "", + "Os": "windows", + "OsVersion": "10.0.14393", + "Size": 651862727, + "VirtualSize": 651862727, + "GraphDriver": { + "Name": "windowsfilter", + "Data": { + "dir": "C:\\control\\windowsfilter\\6fe6a289b98276a6a5ca0345156ca61d7b38f3da6bb49ef95af1d0f1ac37e5bf" + } + }, + "RootFS": { + "Type": "layers", + "Layers": [ + "sha256:342d4e407550c52261edd20cd901b5ce438f0b1e940336de3978210612365063" + ] + } +} \ No newline at end of file diff --git a/docker-java/src/test/resources/samples/1.25/other/AuthConfig/orchestrators.json b/docker-java/src/test/resources/samples/1.25/other/AuthConfig/orchestrators.json new file mode 100644 index 000000000..3b67b1941 --- /dev/null +++ b/docker-java/src/test/resources/samples/1.25/other/AuthConfig/orchestrators.json @@ -0,0 +1,3 @@ +{ + "stackOrchestrator" : "kubernetes" +} \ No newline at end of file diff --git a/docker-java/src/test/resources/samples/1.27/containers/container/stats/stats1.json b/docker-java/src/test/resources/samples/1.27/containers/container/stats/stats1.json new file mode 100644 index 000000000..5a80cd99c --- /dev/null +++ b/docker-java/src/test/resources/samples/1.27/containers/container/stats/stats1.json @@ -0,0 +1,191 @@ +{ + "read":"2017-12-06T00:42:03.8352972Z", + "preread":"2017-12-06T00:42:02.8366736Z", + "pids_stats":{ + "current":2 + }, + "blkio_stats":{ + "io_service_bytes_recursive":[ + { + "major":259, + "minor":0, + "op":"Read", + "value":823296 + }, + { + "major":259, + "minor":0, + "op":"Write", + "value":122880 + }, + { + "major":259, + "minor":0, + "op":"Sync", + "value":835584 + }, + { + "major":259, + "minor":0, + "op":"Async", + "value":110592 + }, + { + "major":259, + "minor":0, + "op":"Total", + "value":946176 + } + ], + "io_serviced_recursive":[ + { + "major":259, + "minor":0, + "op":"Read", + "value":145 + }, + { + "major":259, + "minor":0, + "op":"Write", + "value":4 + }, + { + "major":259, + "minor":0, + "op":"Sync", + "value":148 + }, + { + "major":259, + "minor":0, + "op":"Async", + "value":1 + }, + { + "major":259, + "minor":0, + "op":"Total", + "value":149 + } + ], + "io_queue_recursive":[ + + ], + "io_service_time_recursive":[ + + ], + "io_wait_time_recursive":[ + + ], + "io_merged_recursive":[ + + ], + "io_time_recursive":[ + + ], + "sectors_recursive":[ + + ] + }, + "num_procs":0, + "storage_stats":{ + + }, + "cpu_stats":{ + "cpu_usage":{ + "total_usage":212198028, + "percpu_usage":[ + 71592953, + 42494761, + 59298344, + 38811970 + ], + "usage_in_kernelmode":170000000, + "usage_in_usermode":20000000 + }, + "system_cpu_usage":545941980000000, + "online_cpus":4, + "throttling_data":{ + "periods":0, + "throttled_periods":0, + "throttled_time":0 + } + }, + "precpu_stats":{ + "cpu_usage":{ + "total_usage":211307214, + "percpu_usage":[ + 71451389, + 42097782, + 59298344, + 38459699 + ], + "usage_in_kernelmode":170000000, + "usage_in_usermode":20000000 + }, + "system_cpu_usage":545937990000000, + "online_cpus":4, + "throttling_data":{ + "periods":0, + "throttled_periods":0, + "throttled_time":0 + } + }, + "memory_stats":{ + "usage":647168, + "max_usage":1703936, + "stats":{ + "active_anon":102400, + "active_file":0, + "cache":0, + "dirty":0, + "hierarchical_memory_limit":9223372036854771712, + "hierarchical_memsw_limit":9223372036854771712, + "inactive_anon":0, + "inactive_file":0, + "mapped_file":0, + "pgfault":9656, + "pgmajfault":0, + "pgpgin":3425, + "pgpgout":3400, + "rss":102400, + "rss_huge":0, + "swap":0, + "total_active_anon":102400, + "total_active_file":0, + "total_cache":0, + "total_dirty":0, + "total_inactive_anon":0, + "total_inactive_file":0, + "total_mapped_file":0, + "total_pgfault":9656, + "total_pgmajfault":0, + "total_pgpgin":3425, + "total_pgpgout":3400, + "total_rss":102400, + "total_rss_huge":0, + "total_swap":0, + "total_unevictable":0, + "total_writeback":0, + "unevictable":0, + "writeback":0 + }, + "limit":2095874048, + "failcnt":0 + }, + "name":"/gallant_hamilton", + "id":"b581d78b03e41d81c9fe941f03f5d35e23733ff96370456b58d2906e002b0deb", + "networks":{ + "eth0":{ + "rx_bytes":1230, + "rx_packets":19, + "rx_errors":0, + "rx_dropped":0, + "tx_bytes":0, + "tx_packets":0, + "tx_errors":0, + "tx_dropped":0 + } + } +} diff --git a/docker-java/src/test/resources/samples/1.38/containers/inspect/lcow.json b/docker-java/src/test/resources/samples/1.38/containers/inspect/lcow.json new file mode 100644 index 000000000..4e7725def --- /dev/null +++ b/docker-java/src/test/resources/samples/1.38/containers/inspect/lcow.json @@ -0,0 +1,169 @@ +{ + "AppArmorProfile": "", + "Args": [], + "Config": { + "AttachStderr": true, + "AttachStdin": true, + "AttachStdout": true, + "Cmd": [ + "cmd" + ], + "Domainname": "", + "Entrypoint": null, + "Env": null, + "Hostname": "35da02ca897b", + "Image": "microsoft/nanoserver", + "Labels": {}, + "OnBuild": null, + "OpenStdin": true, + "StdinOnce": true, + "Tty": true, + "User": "", + "Volumes": null, + "WorkingDir": "" + }, + "Created": "2018-09-18T10:37:25.0470753Z", + "Driver": "windowsfilter", + "ExecIDs": null, + "GraphDriver": { + "Data": { + "dir": "C:\\ProgramData\\Docker\\windowsfilter\\35da02ca897bd378ee52be3066c847fee396ba1a28a00b4be36f42c6686bf556" + }, + "Name": "windowsfilter" + }, + "HostConfig": { + "AutoRemove": true, + "Binds": null, + "BlkioDeviceReadBps": null, + "BlkioDeviceReadIOps": null, + "BlkioDeviceWriteBps": null, + "BlkioDeviceWriteIOps": null, + "BlkioWeight": 0, + "BlkioWeightDevice": [], + "CapAdd": null, + "CapDrop": null, + "Cgroup": "", + "CgroupParent": "", + "ConsoleSize": [ + 50, + 173 + ], + "ContainerIDFile": "", + "CpuCount": 0, + "CpuPercent": 0, + "CpuPeriod": 0, + "CpuQuota": 0, + "CpuRealtimePeriod": 0, + "CpuRealtimeRuntime": 0, + "CpuShares": 0, + "CpusetCpus": "", + "CpusetMems": "", + "DeviceCgroupRules": null, + "Devices": [], + "DiskQuota": 0, + "Dns": [], + "DnsOptions": [], + "DnsSearch": [], + "ExtraHosts": null, + "GroupAdd": null, + "IOMaximumBandwidth": 0, + "IOMaximumIOps": 0, + "IpcMode": "", + "Isolation": "hyperv", + "KernelMemory": 0, + "Links": null, + "LogConfig": { + "Config": {}, + "Type": "json-file" + }, + "MaskedPaths": null, + "Memory": 0, + "MemoryReservation": 0, + "MemorySwap": 0, + "MemorySwappiness": null, + "NanoCpus": 0, + "NetworkMode": "default", + "OomKillDisable": false, + "OomScoreAdj": 0, + "PidMode": "", + "PidsLimit": 0, + "PortBindings": {}, + "Privileged": false, + "PublishAllPorts": false, + "ReadonlyPaths": null, + "ReadonlyRootfs": false, + "RestartPolicy": { + "MaximumRetryCount": 0, + "Name": "no" + }, + "SecurityOpt": null, + "ShmSize": 0, + "UTSMode": "", + "Ulimits": null, + "UsernsMode": "", + "VolumeDriver": "", + "VolumesFrom": null + }, + "HostnamePath": "", + "HostsPath": "", + "Id": "35da02ca897bd378ee52be3066c847fee396ba1a28a00b4be36f42c6686bf556", + "Image": "sha256:1381511ec0122f197b6abff5bc0692bef19943ddafd6680eff41197afa3a6dda", + "LogPath": "C:\\ProgramData\\Docker\\containers\\35da02ca897bd378ee52be3066c847fee396ba1a28a00b4be36f42c6686bf556\\35da02ca897bd378ee52be3066c847fee396ba1a28a00b4be36f42c6686bf556-json.log", + "MountLabel": "", + "Mounts": [], + "Name": "/cranky_clarke", + "NetworkSettings": { + "Bridge": "", + "EndpointID": "", + "Gateway": "", + "GlobalIPv6Address": "", + "GlobalIPv6PrefixLen": 0, + "HairpinMode": false, + "IPAddress": "", + "IPPrefixLen": 0, + "IPv6Gateway": "", + "LinkLocalIPv6Address": "", + "LinkLocalIPv6PrefixLen": 0, + "MacAddress": "", + "Networks": { + "nat": { + "Aliases": null, + "DriverOpts": null, + "EndpointID": "493b77d6fe7e3b92435b1eb01461fde669781330deb84a9cbada360db8997ebc", + "Gateway": "172.17.18.1", + "GlobalIPv6Address": "", + "GlobalIPv6PrefixLen": 0, + "IPAMConfig": null, + "IPAddress": "172.17.18.123", + "IPPrefixLen": 16, + "IPv6Gateway": "", + "Links": null, + "MacAddress": "00:aa:ff:cf:dd:09", + "NetworkID": "398c0e206dd677ed4a6566f9de458311f5767d8c7a8b963275490ab64c5d10a7" + } + }, + "Ports": {}, + "SandboxID": "35da02ca897bd378ee52be3066c847fee396ba1a28a00b4be36f42c6686bf556", + "SandboxKey": "35da02ca897bd378ee52be3066c847fee396ba1a28a00b4be36f42c6686bf556", + "SecondaryIPAddresses": null, + "SecondaryIPv6Addresses": null + }, + "Path": "cmd", + "Platform": "windows", + "ProcessLabel": "", + "ResolvConfPath": "", + "RestartCount": 0, + "State": { + "Dead": false, + "Error": "", + "ExitCode": 0, + "FinishedAt": "0001-01-01T00:00:00Z", + "OOMKilled": false, + "Paused": false, + "Pid": 1588, + "Restarting": false, + "Running": true, + "StartedAt": "2018-09-18T10:37:28.3668368Z", + "Status": "running" + } +} \ No newline at end of file diff --git a/docker-java/src/test/resources/samples/1.38/info/lcow.json b/docker-java/src/test/resources/samples/1.38/info/lcow.json new file mode 100644 index 000000000..7ab600449 --- /dev/null +++ b/docker-java/src/test/resources/samples/1.38/info/lcow.json @@ -0,0 +1,93 @@ +{ + "Architecture": "x86_64", + "BridgeNfIp6tables": true, + "BridgeNfIptables": true, + "CPUSet": false, + "CPUShares": false, + "CgroupDriver": "", + "ClusterAdvertise": "", + "ClusterStore": "", + "ContainerdCommit": { + "Expected": "", + "ID": "" + }, + "Containers": 3, + "ContainersPaused": 0, + "ContainersRunning": 0, + "ContainersStopped": 3, + "CpuCfsPeriod": false, + "CpuCfsQuota": false, + "Debug": true, + "DefaultRuntime": "", + "DockerRootDir": "C:\\ProgramData\\Docker", + "Driver": "windowsfilter (windows) lcow (linux)", + "DriverStatus": [["Windows", ""], ["LCOW", ""]], + "ExperimentalBuild": true, + "GenericResources": null, + "HttpProxy": "", + "HttpsProxy": "", + "ID": "ZOGT:VB24:YEPZ:Y7HU:JHPB:WNUE:UYQG:7YRY:VLZV:FLWV:R65B:ICZG", + "IPv4Forwarding": true, + "Images": 2, + "IndexServerAddress": "https://index.docker.io/v1/", + "InitBinary": "", + "InitCommit": { + "Expected": "", + "ID": "" + }, + "Isolation": "hyperv", + "KernelMemory": false, + "KernelVersion": "10.0 17134 (17134.1.amd64fre.rs4_release.180410-1804)", + "Labels": [], + "LiveRestoreEnabled": false, + "LoggingDriver": "json-file", + "MemTotal": 68684476416, + "MemoryLimit": false, + "NCPU": 8, + "NEventsListener": 1, + "NFd": -1, + "NGoroutines": 28, + "Name": "somename", + "NoProxy": "", + "OSType": "windows", + "OomKillDisable": false, + "OperatingSystem": "Windows 10 Pro Version 1803 (OS Build 17134.228)", + "Plugins": { + "Authorization": null, + "Log": ["awslogs", "etwlogs", "fluentd", "gelf", "json-file", "logentries", "splunk", "syslog"], + "Network": ["ics", "l2bridge", "l2tunnel", "nat", "null", "overlay", "transparent"], + "Volume": ["local"] + }, + "RegistryConfig": { + "AllowNondistributableArtifactsCIDRs": [], + "AllowNondistributableArtifactsHostnames": [], + "IndexConfigs": { + "docker.io": { + "Mirrors": [], + "Name": "docker.io", + "Official": true, + "Secure": true + } + }, + "InsecureRegistryCIDRs": ["127.0.0.0/8"], + "Mirrors": [] + }, + "RuncCommit": { + "Expected": "", + "ID": "" + }, + "Runtimes": null, + "SecurityOptions": [], + "ServerVersion": "18.06.1-ce", + "SwapLimit": false, + "Swarm": { + "ControlAvailable": false, + "Error": "", + "LocalNodeState": "inactive", + "NodeAddr": "", + "NodeID": "", + "RemoteManagers": null + }, + "SystemStatus": null, + "SystemTime": "2018-09-14T09:40:05.1369294+02:00" +} diff --git a/docker-java/src/test/resources/samples/1.38/version/lcow.json b/docker-java/src/test/resources/samples/1.38/version/lcow.json new file mode 100644 index 000000000..21f496823 --- /dev/null +++ b/docker-java/src/test/resources/samples/1.38/version/lcow.json @@ -0,0 +1,31 @@ +{ + "ApiVersion": "1.38", + "Arch": "amd64", + "BuildTime": "2018-08-21T17:36:40.000000000+00:00", + "Components": [{ + "Details": { + "ApiVersion": "1.38", + "Arch": "amd64", + "BuildTime": "2018-08-21T17:36:40.000000000+00:00", + "Experimental": "true", + "GitCommit": "e68fc7a", + "GoVersion": "go1.10.3", + "KernelVersion": "10.0 17134 (17134.1.amd64fre.rs4_release.180410-1804)", + "MinAPIVersion": "1.24", + "Os": "windows" + }, + "Name": "Engine", + "Version": "18.06.1-ce" + } + ], + "Experimental": true, + "GitCommit": "e68fc7a", + "GoVersion": "go1.10.3", + "KernelVersion": "10.0 17134 (17134.1.amd64fre.rs4_release.180410-1804)", + "MinAPIVersion": "1.24", + "Os": "windows", + "Platform": { + "Name": "" + }, + "Version": "18.06.1-ce" +} diff --git a/docker-java/src/test/resources/someHomeDir/.docker/certs/dummy.txt b/docker-java/src/test/resources/someHomeDir/.docker/certs/dummy.txt new file mode 100644 index 000000000..e69de29bb diff --git a/docker-java/src/test/resources/someHomeDir/.docker/config.json b/docker-java/src/test/resources/someHomeDir/.docker/config.json new file mode 100644 index 000000000..02ed0cf7f --- /dev/null +++ b/docker-java/src/test/resources/someHomeDir/.docker/config.json @@ -0,0 +1,9 @@ +{ + "auths":{ + "https://index.docker.io/v1/":{ + "auth":"dXNlcm5hbWU6cGFzc3dvcmQ=", + "email":"foo.bar@test.com" + } + + } +} diff --git a/docker-java/src/test/resources/testAuthConfigFile/emptyFile/.dockercfg b/docker-java/src/test/resources/testAuthConfigFile/emptyFile/.dockercfg new file mode 100644 index 000000000..e69de29bb diff --git a/src/test/resources/testAuthConfigFile/invalidJsonInvalidAuth b/docker-java/src/test/resources/testAuthConfigFile/invalidJsonInvalidAuth/.dockercfg similarity index 100% rename from src/test/resources/testAuthConfigFile/invalidJsonInvalidAuth rename to docker-java/src/test/resources/testAuthConfigFile/invalidJsonInvalidAuth/.dockercfg diff --git a/src/test/resources/testAuthConfigFile/invalidLegacyAuthLine b/docker-java/src/test/resources/testAuthConfigFile/invalidLegacyAuthLine/.dockercfg similarity index 100% rename from src/test/resources/testAuthConfigFile/invalidLegacyAuthLine rename to docker-java/src/test/resources/testAuthConfigFile/invalidLegacyAuthLine/.dockercfg diff --git a/src/test/resources/testAuthConfigFile/invalidLegacyEmailLine b/docker-java/src/test/resources/testAuthConfigFile/invalidLegacyEmailLine/.dockercfg similarity index 100% rename from src/test/resources/testAuthConfigFile/invalidLegacyEmailLine rename to docker-java/src/test/resources/testAuthConfigFile/invalidLegacyEmailLine/.dockercfg diff --git a/src/test/resources/testAuthConfigFile/invalidLegacyInvalidAuth b/docker-java/src/test/resources/testAuthConfigFile/invalidLegacyInvalidAuth/.dockercfg similarity index 100% rename from src/test/resources/testAuthConfigFile/invalidLegacyInvalidAuth rename to docker-java/src/test/resources/testAuthConfigFile/invalidLegacyInvalidAuth/.dockercfg diff --git a/src/test/resources/testAuthConfigFile/tooSmallFile b/docker-java/src/test/resources/testAuthConfigFile/tooSmallFile/.dockercfg similarity index 100% rename from src/test/resources/testAuthConfigFile/tooSmallFile rename to docker-java/src/test/resources/testAuthConfigFile/tooSmallFile/.dockercfg diff --git a/docker-java/src/test/resources/testAuthConfigFile/validDockerConfig/config.json b/docker-java/src/test/resources/testAuthConfigFile/validDockerConfig/config.json new file mode 100644 index 000000000..ee32e91ed --- /dev/null +++ b/docker-java/src/test/resources/testAuthConfigFile/validDockerConfig/config.json @@ -0,0 +1,12 @@ +{ + "auths": { + "quay.io" : { + "auth" : "Zm9vOmJhcg==", + "email" :"foo@example.com" + }, + "https://index.docker.io/v1/" : { + "auth" : "Zm9vMTpiYXIx", + "email" : "moo@example.com" + } + } +} \ No newline at end of file diff --git a/docker-java/src/test/resources/testAuthConfigFile/validDockerConfigWithCurrentContext/config.json b/docker-java/src/test/resources/testAuthConfigFile/validDockerConfigWithCurrentContext/config.json new file mode 100644 index 000000000..8c5963f87 --- /dev/null +++ b/docker-java/src/test/resources/testAuthConfigFile/validDockerConfigWithCurrentContext/config.json @@ -0,0 +1,4 @@ +{ + "auths": {}, + "currentContext": "expectedContext" +} diff --git a/docker-java/src/test/resources/testAuthConfigFile/validJsonAuthsNull/config.json b/docker-java/src/test/resources/testAuthConfigFile/validJsonAuthsNull/config.json new file mode 100644 index 000000000..d104c357c --- /dev/null +++ b/docker-java/src/test/resources/testAuthConfigFile/validJsonAuthsNull/config.json @@ -0,0 +1,9 @@ +{ + "auths": null, + "credsStore": "desktop", + "plugins": { + "-x-cli-hints": { + "enabled": "true" + } + } +} diff --git a/docker-java/src/test/resources/testAuthConfigFile/validJsonWithOnlyUnknown/config.json b/docker-java/src/test/resources/testAuthConfigFile/validJsonWithOnlyUnknown/config.json new file mode 100644 index 000000000..cd5f7a124 --- /dev/null +++ b/docker-java/src/test/resources/testAuthConfigFile/validJsonWithOnlyUnknown/config.json @@ -0,0 +1,3 @@ +{ + "credsStore" : "osxkeychain" +} diff --git a/docker-java/src/test/resources/testAuthConfigFile/validJsonWithUnknown/config.json b/docker-java/src/test/resources/testAuthConfigFile/validJsonWithUnknown/config.json new file mode 100644 index 000000000..557714668 --- /dev/null +++ b/docker-java/src/test/resources/testAuthConfigFile/validJsonWithUnknown/config.json @@ -0,0 +1,12 @@ +{ + "auths": { + "192.168.99.100:32768": {}, + "https://index.docker.io/v1/": { + "auth": "Zm9vOmJhcg==", + "email": "foo@example.com" + } + }, + "HttpHeaders": { + "User-Agent": "user agent" + } +} diff --git a/src/test/resources/testAuthConfigFile/validLegacy b/docker-java/src/test/resources/testAuthConfigFile/validLegacy/.dockercfg similarity index 100% rename from src/test/resources/testAuthConfigFile/validLegacy rename to docker-java/src/test/resources/testAuthConfigFile/validLegacy/.dockercfg diff --git a/src/test/resources/testAuthConfigFile/validJson b/docker-java/src/test/resources/testAuthConfigFile/validLegacyJson/.dockercfg similarity index 100% rename from src/test/resources/testAuthConfigFile/validJson rename to docker-java/src/test/resources/testAuthConfigFile/validLegacyJson/.dockercfg diff --git a/src/test/resources/testCopyFromArchive/binary.dat b/docker-java/src/test/resources/testCopyFromArchive/binary.dat similarity index 100% rename from src/test/resources/testCopyFromArchive/binary.dat rename to docker-java/src/test/resources/testCopyFromArchive/binary.dat diff --git a/src/test/resources/testImportImageFromTar/empty.tar b/docker-java/src/test/resources/testImportImageFromTar/empty.tar similarity index 100% rename from src/test/resources/testImportImageFromTar/empty.tar rename to docker-java/src/test/resources/testImportImageFromTar/empty.tar diff --git a/docker-java/src/test/resources/testReadFile/Dockerfile b/docker-java/src/test/resources/testReadFile/Dockerfile new file mode 100644 index 000000000..af2575fa7 --- /dev/null +++ b/docker-java/src/test/resources/testReadFile/Dockerfile @@ -0,0 +1,11 @@ +FROM busybox:latest + +# Copy testrun.sh files into the container + +ADD ./testrun.sh /tmp/ +ADD ./oldFile.txt / + +RUN mkdir -p /usr/local/bin +RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh + +CMD ["testrun.sh"] diff --git a/src/test/resources/testReadFile/oldFile.txt b/docker-java/src/test/resources/testReadFile/oldFile.txt similarity index 100% rename from src/test/resources/testReadFile/oldFile.txt rename to docker-java/src/test/resources/testReadFile/oldFile.txt diff --git a/src/test/resources/testReadFile/testrun.sh b/docker-java/src/test/resources/testReadFile/testrun.sh similarity index 100% rename from src/test/resources/testReadFile/testrun.sh rename to docker-java/src/test/resources/testReadFile/testrun.sh diff --git a/src/test/resources/travis-logback.xml b/docker-java/src/test/resources/travis-logback.xml similarity index 86% rename from src/test/resources/travis-logback.xml rename to docker-java/src/test/resources/travis-logback.xml index 250cbd335..06574e1a5 100644 --- a/src/test/resources/travis-logback.xml +++ b/docker-java/src/test/resources/travis-logback.xml @@ -12,8 +12,8 @@ - - - + + + diff --git a/docker-java/template.mf b/docker-java/template.mf new file mode 100644 index 000000000..2ce26d092 --- /dev/null +++ b/docker-java/template.mf @@ -0,0 +1,30 @@ +Manifest-Version: 1.0 +Bundle-ManifestVersion: 2 +Bundle-SymbolicName: com.github.docker-java +Bundle-Version: ${parsedVersion.osgiVersion} +Bundle-Name: ${project.name} +Version-Patterns: + default;pattern="[=.=.=, =.+1.0)", + short;pattern="[=.=,+1.0)" +Import-Template: + com.fasterxml.jackson.*;version="${jackson-jaxrs.version}", + com.google.common.*;version="${guava.version:short}", + io.netty.*;version="${netty.version:default}";resolution:=optional, + javax.ws.rs.*;version="[2.0.0, 2.1.0)", + org.apache.commons.compress.*;version="${commons-compress.version:short}", + org.apache.commons.io.*;version="${commons-io.version:short}", + org.apache.commons.lang3.*;version="${commons-lang3.version:short}", + org.apache.http.*;version="[4.4.0, 4.6.0)", + org.bouncycastle.*;version="${bouncycastle.version:short}", + org.glassfish.jersey.*;version="${jersey.version:default}", + org.slf4j.*;version="[1.7.0, 1.8.0)", + org.newsclub.net.unix;version="${junixsocket.version:default}";resolution:=optional +Excluded-Exports: + org.apache.http.impl.io, + org.newsclub.net.unix, + com.github.dockerjava.jaxrs.*, + com.github.dockerjava.netty.* +Excluded-Imports: + javax.annotation, + javax.net.ssl, + edu.umd.cs.findbugs.annotations diff --git a/docs/README.md b/docs/README.md new file mode 100644 index 000000000..4b8ef0798 --- /dev/null +++ b/docs/README.md @@ -0,0 +1,2 @@ +* [Getting Started](./getting_started.md) +* [Available transports](./transports.md) diff --git a/docs/devel.adoc b/docs/devel.adoc deleted file mode 100644 index 2305b89d0..000000000 --- a/docs/devel.adoc +++ /dev/null @@ -1,33 +0,0 @@ -### Code Design - * Model is based on Objects and not primitives that allows nullify requests and have null values for data - that wasn't provided by docker daemon. - * For null safeness findbugs annotations are used. - ** Every method that may return `null` (and we are unsure in any fields as docker daemon may change something) - should be annotated with `@CheckForNull` return qualifier from `javax.annotation` package. - ** Methods that can't return `null` must be annotated with `@Nonnull`. - ** The same for Arguments. - ** `@Nullable` must be used only for changing inherited (other typed) qualifier. - * Setters in builder style must be prefixed with `withXX`. - * All classes should provide `toString()` `equals()` and `hashCode()` defined methods. - * Javadocs - ** Provide full information on field: - *** For models define API version with `@since {@link RemoteApiVersion#VERSION_1_X}`. - ** getters/setters should refernce to field `@see #$field`. - -### Coding style - * TBD, some initial styling already enforced with checkstyle. - IDEA/checkstyle file analogues will be provided soon. - -### Testing - * Unit tests for serder (serialization-deserialization). - * Integration tests for commands. - * If model object has builders, then fill it with data and compare by `equals()` with expected response - from docker daemon. If failed, then some fields mappings are wrong. - -### Debug - * When there are unreproducible Travis errors: - ** Try locally run test 10-20 times in IDE against the same docker daemon version and same connection type (tcp or socket). - ** Limit `.travis.yml` to single run (to not consume their resources with matrix run). - ** Remove `travis-logback.xml` replacement (build can't output everything in every run because travis has log limitation). - ** Set single test in `pom.xml` `for maven-failsafe-plugin` - ** Make PR or if you are maintainer push to branch, catch log and fix. diff --git a/docs/getting_started.md b/docs/getting_started.md new file mode 100644 index 000000000..7781e38ec --- /dev/null +++ b/docs/getting_started.md @@ -0,0 +1,129 @@ +# Getting Started + +## Dependencies + +To start using `docker-java` , you need to add at least two dependencies: +1. `com.github.docker-java:docker-java-core` for the `DockerClient` +1. one of `com.github.docker-java:docker-java-transport-*` to communicate with the Docker daemon. See [Available Transports](./transports.md) for more info. + +The latest available version: +[![Maven Central](https://img.shields.io/maven-central/v/com.github.docker-java/docker-java.svg)](https://mvnrepository.com/artifact/com.github.docker-java/docker-java) + + +## Instantiating a `DockerClientConfig` + +You will need an instance of `DockerClientConfig` to tell the library how to access Docker, which credentials to use to pull from Docker registries, etc etc. + +The builder is available and allows you to configure every property of the client: +```java +import com.github.dockerjava.core.DockerClientConfig +import com.github.dockerjava.core.DefaultDockerClientConfig +DockerClientConfig standard = DefaultDockerClientConfig.createDefaultConfigBuilder().build(); +``` + +```java +import com.github.dockerjava.core.DockerClientConfig +import com.github.dockerjava.core.DefaultDockerClientConfig + +DockerClientConfig custom = DefaultDockerClientConfig.createDefaultConfigBuilder() + .withDockerHost("tcp://docker.somewhere.tld:2376") + .withDockerTlsVerify(true) + .withDockerCertPath("/home/user/.docker") + .withRegistryUsername(registryUser) + .withRegistryPassword(registryPass) + .withRegistryEmail(registryMail) + .withRegistryUrl(registryUrl) + .build(); +``` + +Here you can tune registry auth, DOCKER_HOST and other options. + +There are a couple of configuration items, all of which have sensible defaults: + +* `DOCKER_HOST` The Docker Host URL, e.g. `tcp://localhost:2376` or `unix:///var/run/docker.sock` +* `DOCKER_TLS_VERIFY` enable/disable TLS verification (switch between `http` and `https` protocol) +* `DOCKER_CERT_PATH` Path to the certificates needed for TLS verification +* `DOCKER_CONFIG` Path for additional docker configuration files (like `.dockercfg`) +* `api.version` The API version, e.g. `1.23`. +* `registry.url` Your registry's address. +* `registry.username` Your registry username (required to push containers). +* `registry.password` Your registry password. +* `registry.email` Your registry email. + +There are three ways to configure, in descending order of precedence: + +##### Properties (docker-java.properties) + + DOCKER_HOST=tcp://localhost:2376 + DOCKER_TLS_VERIFY=1 + DOCKER_CERT_PATH=/home/user/.docker/certs + DOCKER_CONFIG=/home/user/.docker + api.version=1.23 + registry.url=https://index.docker.io/v1/ + registry.username=dockeruser + registry.password=ilovedocker + registry.email=dockeruser@github.com + +##### System Properties: + + java -DDOCKER_HOST=tcp://localhost:2375 -Dregistry.username=dockeruser pkg.Main + +##### System Environment + + export DOCKER_HOST=tcp://localhost:2376 + export DOCKER_TLS_VERIFY=1 + export DOCKER_CERT_PATH=/home/user/.docker/certs + export DOCKER_CONFIG=/home/user/.docker + +##### File System + +In `$HOME/.docker-java.properties` + +##### Class Path + +In the class path at `/docker-java.properties` + +### Jackson + +Should you need to customize the Jackson's `ObjectMapper` used by `docker-java`, you can create your own `DockerClientConfig` and override `DockerClientConfig#getObjectMapper()`. + +## Instantiating a `DockerHttpClient` +Once you decided which transport to use, you will need to instantiate an HTTP client: +```java +DockerClientConfig config = ...; + +DockerHttpClient httpClient = new ApacheDockerHttpClient.Builder() + .dockerHost(config.getDockerHost()) + .sslConfig(config.getSSLConfig()) + .maxConnections(100) + .connectionTimeout(Duration.ofSeconds(30)) + .responseTimeout(Duration.ofSeconds(45)) + .build(); +``` + +Please refer to selected transport's builder for other available configuration options (like timeouts). + +Once you have an HTTP client, you can make raw requests to the Docker daemon directly: +```java +Request request = Request.builder() + .method(Request.Method.GET) + .path("/_ping") + .build(); + +try (Response response = httpClient.execute(request)) { + assertThat(response.getStatusCode(), equalTo(200)); + assertThat(IOUtils.toString(response.getBody()), equalTo("OK")); +} +``` + +## Instantiating a `DockerClient` + +To get an instance of `DockerClient`, you need to pass both `DockerClientConfig` and `DockerHttpClient`: +```java +DockerClient dockerClient = DockerClientImpl.getInstance(config, httpClient); +``` + +Once you have it, you can start executing Docker commands: +```java +dockerClient.pingCmd().exec(); +``` diff --git a/docs/transports.md b/docs/transports.md new file mode 100644 index 000000000..18a93d9e0 --- /dev/null +++ b/docs/transports.md @@ -0,0 +1,74 @@ +# Available transports + +## Apache HttpClient 5 +| | | +|---|---| +| Maven coordinates | `com.github.docker-java:docker-java-transport-httpclient5` | +| Stability | 🙂| +| Long term support plans | ✅ | +| Unix sockets support | ✅ | +| Windows Npipe support | ✅ | +| Stdin attachment support | ✅ | + +This transport is based on Apache HttpClient library version 5, which has a great flexibility and allows us to implement all Docker-specific features and protocols required, without having to use internal APIs or anything. + +It has everything to become the default transport of docker-java in future releases. + +## "Zerodep" +| | | +|---|---| +| Maven coordinates | `com.github.docker-java:docker-java-transport-zerodep` | +| Stability | 🙂| +| Long term support plans | ✅ | +| Unix sockets support | ✅ | +| Windows Npipe support | ✅ | +| Stdin attachment support | ✅ | + +The idea of this transport is to provide a transport that supports 100% of the features without having to worry about transitive dependencies. + +Note: due to the implementation details, it cannot be true "0 dependencies" module, so it needs to depend on `slf4j-api` and JNA. + +## OkHttp +| | | +|---|---| +| Maven coordinates | `com.github.docker-java:docker-java-transport-okhttp` | +| Stability | 🧐| +| Long term support plans | ❓ | +| Unix sockets support | ✅ | +| Windows Npipe support | ✅ | +| Stdin attachment support | ✅ | + +The OkHttp transport was first implemented in [the Testcontainers library](http://github.com/testcontainers/testcontainers-java) as a replacement for Netty. The main motivation for it was to not have heavy-weight Netty-specific native dependencies and the lack of Npipe support in the Netty one. + +OkHttp's migration to Kotlin and the need to use internal APIs for doing stdin hijacking makes us question the future of this transport (still under the consideration). + +## Netty +| | | +|---|---| +| Maven coordinates | `com.github.docker-java:docker-java-transport-netty` | +| Stability | 🧐| +| Long term support plans | ❌ | +| Unix sockets support | ✅ | +| Windows Npipe support | ❌ | +| Stdin attachment support | ✅ | + +Netty was the first alternative transport introduced as an alternative to Jersey. + +Although it gives a very low level access to the protocol, the lack of Windows Npipe support and the native library dependency for Unix Sockets make it hard to maintain and there are no plans to continue including this transport option in future versions. + +The community may decide to pick it up and continue the development as a 3rd party transport based on the existing abstractions `docker-java` provides. + +## Jersey +| | | +|---|---| +| Maven coordinates | `com.github.docker-java:docker-java-transport-jersey` | +| Stability | 🙃| +| Long term support plans | ❌ | +| Unix sockets support | ✅ | +| Windows Npipe support | ❌ | +| Stdin attachment support | ❌ | + +Jersey was the initial transport of the project. And, while working well, it was lacking support for connection hijacking (e.g. stdin attachment) or Windows Npipes. +The big amount of dependencies was also causing issues. + +Since Apache HttpClient 5-based transport is available now, there is no reason to keep Jersey and it will eventually be removed. diff --git a/get-docker-com.sh b/get-docker-com.sh deleted file mode 100755 index 66ca4aefc..000000000 --- a/get-docker-com.sh +++ /dev/null @@ -1,313 +0,0 @@ -#!/bin/sh -set -e -# -# This script is meant for quick & easy install via: -# 'curl -sSL https://get.docker.com/ | sh' -# or: -# 'wget -qO- https://get.docker.com/ | sh' -# -# For test builds (ie. release candidates): -# 'curl -fsSL https://test.docker.com/ | sh' -# or: -# 'wget -qO- https://test.docker.com/ | sh' -# -# For experimental builds: -# 'curl -fsSL https://experimental.docker.com/ | sh' -# or: -# 'wget -qO- https://experimental.docker.com/ | sh' -# -# Docker Maintainers: -# To update this script on https://get.docker.com, -# use hack/release.sh during a normal release, -# or the following one-liner for script hotfixes: -# aws s3 cp --acl public-read hack/install.sh s3://get.docker.com/index -# - -url="https://get.docker.com/" -apt_url="https://apt.dockerproject.org" -yum_url="https://yum.dockerproject.org" -gpg_fingerprint="58118E89F3A912897C070ADBF76221572C52609D" - -key_servers=" -ha.pool.sks-keyservers.net -pgp.mit.edu -keyserver.ubuntu.com -" - -command_exists() { - command -v "$@" > /dev/null 2>&1 -} - -semverParse() { - major="${1%%.*}" - minor="${1#$major.}" - minor="${minor%%.*}" - patch="${1#$major.$minor.}" - patch="${patch%%[-.]*}" -} - -do_install() { - case "$(uname -m)" in - *64) - ;; - *) - cat >&2 <<-'EOF' - Error: you are not using a 64bit platform. - Docker currently only supports 64bit platforms. - EOF - exit 1 - ;; - esac - - user="$(id -un 2>/dev/null || true)" - - sh_c='sh -c' - if [ "$user" != 'root' ]; then - if command_exists sudo; then - sh_c='sudo -E sh -c' - elif command_exists su; then - sh_c='su -c' - else - cat >&2 <<-'EOF' - Error: this installer needs the ability to run commands as root. - We are unable to find either "sudo" or "su" available to make this happen. - EOF - exit 1 - fi - fi - - curl='' - if command_exists curl; then - curl='curl -sSL' - elif command_exists wget; then - curl='wget -qO-' - elif command_exists busybox && busybox --list-modules | grep -q wget; then - curl='busybox wget -qO-' - fi - - # check to see which repo they are trying to install from - if [ -z "$repo" ]; then - repo='main' - if [ "https://test.docker.com/" = "$url" ]; then - repo='testing' - elif [ "https://experimental.docker.com/" = "$url" ]; then - repo='experimental' - fi - fi - - # perform some very rudimentary platform detection - lsb_dist='' - dist_version='' - if command_exists lsb_release; then - lsb_dist="$(lsb_release -si)" - fi - if [ -z "$lsb_dist" ] && [ -r /etc/lsb-release ]; then - lsb_dist="$(. /etc/lsb-release && echo "$DISTRIB_ID")" - fi - if [ -z "$lsb_dist" ] && [ -r /etc/debian_version ]; then - lsb_dist='debian' - fi - if [ -z "$lsb_dist" ] && [ -r /etc/fedora-release ]; then - lsb_dist='fedora' - fi - if [ -z "$lsb_dist" ] && [ -r /etc/oracle-release ]; then - lsb_dist='oracleserver' - fi - if [ -z "$lsb_dist" ]; then - if [ -r /etc/centos-release ] || [ -r /etc/redhat-release ]; then - lsb_dist='centos' - fi - fi - if [ -z "$lsb_dist" ] && [ -r /etc/os-release ]; then - lsb_dist="$(. /etc/os-release && echo "$ID")" - fi - - lsb_dist="$(echo "$lsb_dist" | tr '[:upper:]' '[:lower:]')" - - case "$lsb_dist" in - - ubuntu) - if command_exists lsb_release; then - dist_version="$(lsb_release --codename | cut -f2)" - fi - if [ -z "$dist_version" ] && [ -r /etc/lsb-release ]; then - dist_version="$(. /etc/lsb-release && echo "$DISTRIB_CODENAME")" - fi - ;; - - debian) - dist_version="$(cat /etc/debian_version | sed 's/\/.*//' | sed 's/\..*//')" - case "$dist_version" in - 8) - dist_version="jessie" - ;; - 7) - dist_version="wheezy" - ;; - esac - ;; - - oracleserver) - # need to switch lsb_dist to match yum repo URL - lsb_dist="oraclelinux" - dist_version="$(rpm -q --whatprovides redhat-release --queryformat "%{VERSION}\n" | sed 's/\/.*//' | sed 's/\..*//' | sed 's/Server*//')" - ;; - - fedora|centos) - dist_version="$(rpm -q --whatprovides redhat-release --queryformat "%{VERSION}\n" | sed 's/\/.*//' | sed 's/\..*//' | sed 's/Server*//')" - ;; - - *) - if command_exists lsb_release; then - dist_version="$(lsb_release --codename | cut -f2)" - fi - if [ -z "$dist_version" ] && [ -r /etc/os-release ]; then - dist_version="$(. /etc/os-release && echo "$VERSION_ID")" - fi - ;; - - - esac - - - # Run setup for each distro accordingly - case "$lsb_dist" in - ubuntu|debian) - export DEBIAN_FRONTEND=noninteractive - - did_apt_get_update= - apt_get_update() { - if [ -z "$did_apt_get_update" ]; then - ( set -x; $sh_c 'sleep 3; apt-get update' ) - did_apt_get_update=1 - fi - } - - # aufs is preferred over devicemapper; try to ensure the driver is available. - if ! grep -q aufs /proc/filesystems && ! $sh_c 'modprobe aufs'; then - if uname -r | grep -q -- '-generic' && dpkg -l 'linux-image-*-generic' | grep -qE '^ii|^hi' 2>/dev/null; then - kern_extras="linux-image-extra-$(uname -r) linux-image-extra-virtual" - - apt_get_update - ( set -x; $sh_c 'sleep 3; apt-get install -y -q '"$kern_extras" ) || true - - if ! grep -q aufs /proc/filesystems && ! $sh_c 'modprobe aufs'; then - echo >&2 'Warning: tried to install '"$kern_extras"' (for AUFS)' - echo >&2 ' but we still have no AUFS. Docker may not work. Proceeding anyways!' - ( set -x; sleep 10 ) - fi - else - echo >&2 'Warning: current kernel is not supported by the linux-image-extra-virtual' - echo >&2 ' package. We have no AUFS support. Consider installing the packages' - echo >&2 ' linux-image-virtual kernel and linux-image-extra-virtual for AUFS support.' - ( set -x; sleep 10 ) - fi - fi - - # install apparmor utils if they're missing and apparmor is enabled in the kernel - # otherwise Docker will fail to start - if [ "$(cat /sys/module/apparmor/parameters/enabled 2>/dev/null)" = 'Y' ]; then - if command -v apparmor_parser >/dev/null 2>&1; then - echo 'apparmor is enabled in the kernel and apparmor utils were already installed' - else - echo 'apparmor is enabled in the kernel, but apparmor_parser missing' - apt_get_update - ( set -x; $sh_c 'sleep 3; apt-get install -y -q apparmor' ) - fi - fi - - if [ ! -e /usr/lib/apt/methods/https ]; then - apt_get_update - ( set -x; $sh_c 'sleep 3; apt-get install -y -q apt-transport-https ca-certificates' ) - fi - if [ -z "$curl" ]; then - apt_get_update - ( set -x; $sh_c 'sleep 3; apt-get install -y -q curl ca-certificates' ) - curl='curl -sSL' - fi - ( - set -x - for key_server in $key_servers ; do - $sh_c "apt-key adv --keyserver hkp://${key_server}:80 --recv-keys ${gpg_fingerprint}" && break - done - $sh_c "apt-key adv -k ${gpg_fingerprint} >/dev/null" - $sh_c "mkdir -p /etc/apt/sources.list.d" - $sh_c "echo deb [arch=$(dpkg --print-architecture)] ${apt_url}/repo ${lsb_dist}-${dist_version} ${repo} > /etc/apt/sources.list.d/docker.list" - $sh_c 'sleep 3; apt-get update' - if [ -z "$DOCKER_VERSION" ]; then - $sh_c 'apt-get -o Dpkg::Options::="--force-confnew" install -y -q docker-engine' - else - $sh_c "apt-get -o Dpkg::Options::=\"--force-confnew\" install -y -q docker-engine=$DOCKER_VERSION" - fi - ) - exit 0 - ;; - - fedora|centos|oraclelinux) - $sh_c "cat >/etc/yum.repos.d/docker-${repo}.repo" <<-EOF - [docker-${repo}-repo] - name=Docker ${repo} Repository - baseurl=${yum_url}/repo/${repo}/${lsb_dist}/${dist_version} - enabled=1 - gpgcheck=1 - gpgkey=${yum_url}/gpg - EOF - if [ "$lsb_dist" = "fedora" ] && [ "$dist_version" -ge "22" ]; then - ( - set -x - $sh_c 'sleep 3; dnf -y -q install docker-engine' - ) - else - ( - set -x - $sh_c 'sleep 3; yum -y -q install docker-engine' - ) - fi - exit 0 - ;; - gentoo) - if [ "$url" = "https://test.docker.com/" ]; then - # intentionally mixed spaces and tabs here -- tabs are stripped by "<<-'EOF'", spaces are kept in the output - cat >&2 <<-'EOF' - - You appear to be trying to install the latest nightly build in Gentoo.' - The portage tree should contain the latest stable release of Docker, but' - if you want something more recent, you can always use the live ebuild' - provided in the "docker" overlay available via layman. For more' - instructions, please see the following URL:' - - https://github.com/tianon/docker-overlay#using-this-overlay' - - After adding the "docker" overlay, you should be able to:' - - emerge -av =app-emulation/docker-9999' - - EOF - exit 1 - fi - - ( - set -x - $sh_c 'sleep 3; emerge app-emulation/docker' - ) - exit 0 - ;; - esac - - # intentionally mixed spaces and tabs here -- tabs are stripped by "<<-'EOF'", spaces are kept in the output - cat >&2 <<-'EOF' - - Either your platform is not easily detectable, is not supported by this - installer script (yet - PRs welcome! [hack/install.sh]), or does not yet have - a package for Docker. Please visit the following URL for more detailed - installation instructions: - - https://docs.docker.com/engine/installation/ - - EOF - exit 1 -} - -# wrapped up in a function so that we have some protection against only getting -# half the file during "curl | sh" -do_install \ No newline at end of file diff --git a/mvnw b/mvnw new file mode 100755 index 000000000..41c0f0c23 --- /dev/null +++ b/mvnw @@ -0,0 +1,310 @@ +#!/bin/sh +# ---------------------------------------------------------------------------- +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you 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. +# ---------------------------------------------------------------------------- + +# ---------------------------------------------------------------------------- +# Maven Start Up Batch script +# +# Required ENV vars: +# ------------------ +# JAVA_HOME - location of a JDK home dir +# +# Optional ENV vars +# ----------------- +# M2_HOME - location of maven2's installed home dir +# MAVEN_OPTS - parameters passed to the Java VM when running Maven +# e.g. to debug Maven itself, use +# set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000 +# MAVEN_SKIP_RC - flag to disable loading of mavenrc files +# ---------------------------------------------------------------------------- + +if [ -z "$MAVEN_SKIP_RC" ] ; then + + if [ -f /etc/mavenrc ] ; then + . /etc/mavenrc + fi + + if [ -f "$HOME/.mavenrc" ] ; then + . "$HOME/.mavenrc" + fi + +fi + +# OS specific support. $var _must_ be set to either true or false. +cygwin=false; +darwin=false; +mingw=false +case "`uname`" in + CYGWIN*) cygwin=true ;; + MINGW*) mingw=true;; + Darwin*) darwin=true + # Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home + # See https://developer.apple.com/library/mac/qa/qa1170/_index.html + if [ -z "$JAVA_HOME" ]; then + if [ -x "/usr/libexec/java_home" ]; then + export JAVA_HOME="`/usr/libexec/java_home`" + else + export JAVA_HOME="/Library/Java/Home" + fi + fi + ;; +esac + +if [ -z "$JAVA_HOME" ] ; then + if [ -r /etc/gentoo-release ] ; then + JAVA_HOME=`java-config --jre-home` + fi +fi + +if [ -z "$M2_HOME" ] ; then + ## resolve links - $0 may be a link to maven's home + PRG="$0" + + # need this for relative symlinks + while [ -h "$PRG" ] ; do + ls=`ls -ld "$PRG"` + link=`expr "$ls" : '.*-> \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG="`dirname "$PRG"`/$link" + fi + done + + saveddir=`pwd` + + M2_HOME=`dirname "$PRG"`/.. + + # make it fully qualified + M2_HOME=`cd "$M2_HOME" && pwd` + + cd "$saveddir" + # echo Using m2 at $M2_HOME +fi + +# For Cygwin, ensure paths are in UNIX format before anything is touched +if $cygwin ; then + [ -n "$M2_HOME" ] && + M2_HOME=`cygpath --unix "$M2_HOME"` + [ -n "$JAVA_HOME" ] && + JAVA_HOME=`cygpath --unix "$JAVA_HOME"` + [ -n "$CLASSPATH" ] && + CLASSPATH=`cygpath --path --unix "$CLASSPATH"` +fi + +# For Mingw, ensure paths are in UNIX format before anything is touched +if $mingw ; then + [ -n "$M2_HOME" ] && + M2_HOME="`(cd "$M2_HOME"; pwd)`" + [ -n "$JAVA_HOME" ] && + JAVA_HOME="`(cd "$JAVA_HOME"; pwd)`" +fi + +if [ -z "$JAVA_HOME" ]; then + javaExecutable="`which javac`" + if [ -n "$javaExecutable" ] && ! [ "`expr \"$javaExecutable\" : '\([^ ]*\)'`" = "no" ]; then + # readlink(1) is not available as standard on Solaris 10. + readLink=`which readlink` + if [ ! `expr "$readLink" : '\([^ ]*\)'` = "no" ]; then + if $darwin ; then + javaHome="`dirname \"$javaExecutable\"`" + javaExecutable="`cd \"$javaHome\" && pwd -P`/javac" + else + javaExecutable="`readlink -f \"$javaExecutable\"`" + fi + javaHome="`dirname \"$javaExecutable\"`" + javaHome=`expr "$javaHome" : '\(.*\)/bin'` + JAVA_HOME="$javaHome" + export JAVA_HOME + fi + fi +fi + +if [ -z "$JAVACMD" ] ; then + if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + else + JAVACMD="`which java`" + fi +fi + +if [ ! -x "$JAVACMD" ] ; then + echo "Error: JAVA_HOME is not defined correctly." >&2 + echo " We cannot execute $JAVACMD" >&2 + exit 1 +fi + +if [ -z "$JAVA_HOME" ] ; then + echo "Warning: JAVA_HOME environment variable is not set." +fi + +CLASSWORLDS_LAUNCHER=org.codehaus.plexus.classworlds.launcher.Launcher + +# traverses directory structure from process work directory to filesystem root +# first directory with .mvn subdirectory is considered project base directory +find_maven_basedir() { + + if [ -z "$1" ] + then + echo "Path not specified to find_maven_basedir" + return 1 + fi + + basedir="$1" + wdir="$1" + while [ "$wdir" != '/' ] ; do + if [ -d "$wdir"/.mvn ] ; then + basedir=$wdir + break + fi + # workaround for JBEAP-8937 (on Solaris 10/Sparc) + if [ -d "${wdir}" ]; then + wdir=`cd "$wdir/.."; pwd` + fi + # end of workaround + done + echo "${basedir}" +} + +# concatenates all lines of a file +concat_lines() { + if [ -f "$1" ]; then + echo "$(tr -s '\n' ' ' < "$1")" + fi +} + +BASE_DIR=`find_maven_basedir "$(pwd)"` +if [ -z "$BASE_DIR" ]; then + exit 1; +fi + +########################################################################################## +# Extension to allow automatically downloading the maven-wrapper.jar from Maven-central +# This allows using the maven wrapper in projects that prohibit checking in binary data. +########################################################################################## +if [ -r "$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" ]; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found .mvn/wrapper/maven-wrapper.jar" + fi +else + if [ "$MVNW_VERBOSE" = true ]; then + echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..." + fi + if [ -n "$MVNW_REPOURL" ]; then + jarUrl="$MVNW_REPOURL/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + else + jarUrl="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + fi + while IFS="=" read key value; do + case "$key" in (wrapperUrl) jarUrl="$value"; break ;; + esac + done < "$BASE_DIR/.mvn/wrapper/maven-wrapper.properties" + if [ "$MVNW_VERBOSE" = true ]; then + echo "Downloading from: $jarUrl" + fi + wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" + if $cygwin; then + wrapperJarPath=`cygpath --path --windows "$wrapperJarPath"` + fi + + if command -v wget > /dev/null; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found wget ... using wget" + fi + if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then + wget "$jarUrl" -O "$wrapperJarPath" + else + wget --http-user=$MVNW_USERNAME --http-password=$MVNW_PASSWORD "$jarUrl" -O "$wrapperJarPath" + fi + elif command -v curl > /dev/null; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found curl ... using curl" + fi + if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then + curl -o "$wrapperJarPath" "$jarUrl" -f + else + curl --user $MVNW_USERNAME:$MVNW_PASSWORD -o "$wrapperJarPath" "$jarUrl" -f + fi + + else + if [ "$MVNW_VERBOSE" = true ]; then + echo "Falling back to using Java to download" + fi + javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java" + # For Cygwin, switch paths to Windows format before running javac + if $cygwin; then + javaClass=`cygpath --path --windows "$javaClass"` + fi + if [ -e "$javaClass" ]; then + if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then + if [ "$MVNW_VERBOSE" = true ]; then + echo " - Compiling MavenWrapperDownloader.java ..." + fi + # Compiling the Java class + ("$JAVA_HOME/bin/javac" "$javaClass") + fi + if [ -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then + # Running the downloader + if [ "$MVNW_VERBOSE" = true ]; then + echo " - Running MavenWrapperDownloader.java ..." + fi + ("$JAVA_HOME/bin/java" -cp .mvn/wrapper MavenWrapperDownloader "$MAVEN_PROJECTBASEDIR") + fi + fi + fi +fi +########################################################################################## +# End of extension +########################################################################################## + +export MAVEN_PROJECTBASEDIR=${MAVEN_BASEDIR:-"$BASE_DIR"} +if [ "$MVNW_VERBOSE" = true ]; then + echo $MAVEN_PROJECTBASEDIR +fi +MAVEN_OPTS="$(concat_lines "$MAVEN_PROJECTBASEDIR/.mvn/jvm.config") $MAVEN_OPTS" + +# For Cygwin, switch paths to Windows format before running java +if $cygwin; then + [ -n "$M2_HOME" ] && + M2_HOME=`cygpath --path --windows "$M2_HOME"` + [ -n "$JAVA_HOME" ] && + JAVA_HOME=`cygpath --path --windows "$JAVA_HOME"` + [ -n "$CLASSPATH" ] && + CLASSPATH=`cygpath --path --windows "$CLASSPATH"` + [ -n "$MAVEN_PROJECTBASEDIR" ] && + MAVEN_PROJECTBASEDIR=`cygpath --path --windows "$MAVEN_PROJECTBASEDIR"` +fi + +# Provide a "standardized" way to retrieve the CLI args that will +# work with both Windows and non-Windows executions. +MAVEN_CMD_LINE_ARGS="$MAVEN_CONFIG $@" +export MAVEN_CMD_LINE_ARGS + +WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain + +exec "$JAVACMD" \ + $MAVEN_OPTS \ + -classpath "$MAVEN_PROJECTBASEDIR/.mvn/wrapper/maven-wrapper.jar" \ + "-Dmaven.home=${M2_HOME}" "-Dmaven.multiModuleProjectDirectory=${MAVEN_PROJECTBASEDIR}" \ + ${WRAPPER_LAUNCHER} $MAVEN_CONFIG "$@" diff --git a/mvnw.cmd b/mvnw.cmd new file mode 100644 index 000000000..86115719e --- /dev/null +++ b/mvnw.cmd @@ -0,0 +1,182 @@ +@REM ---------------------------------------------------------------------------- +@REM Licensed to the Apache Software Foundation (ASF) under one +@REM or more contributor license agreements. See the NOTICE file +@REM distributed with this work for additional information +@REM regarding copyright ownership. The ASF licenses this file +@REM to you under the Apache License, Version 2.0 (the +@REM "License"); you may not use this file except in compliance +@REM with the License. You may obtain a copy of the License at +@REM +@REM http://www.apache.org/licenses/LICENSE-2.0 +@REM +@REM Unless required by applicable law or agreed to in writing, +@REM software distributed under the License is distributed on an +@REM "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +@REM KIND, either express or implied. See the License for the +@REM specific language governing permissions and limitations +@REM under the License. +@REM ---------------------------------------------------------------------------- + +@REM ---------------------------------------------------------------------------- +@REM Maven Start Up Batch script +@REM +@REM Required ENV vars: +@REM JAVA_HOME - location of a JDK home dir +@REM +@REM Optional ENV vars +@REM M2_HOME - location of maven2's installed home dir +@REM MAVEN_BATCH_ECHO - set to 'on' to enable the echoing of the batch commands +@REM MAVEN_BATCH_PAUSE - set to 'on' to wait for a keystroke before ending +@REM MAVEN_OPTS - parameters passed to the Java VM when running Maven +@REM e.g. to debug Maven itself, use +@REM set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000 +@REM MAVEN_SKIP_RC - flag to disable loading of mavenrc files +@REM ---------------------------------------------------------------------------- + +@REM Begin all REM lines with '@' in case MAVEN_BATCH_ECHO is 'on' +@echo off +@REM set title of command window +title %0 +@REM enable echoing by setting MAVEN_BATCH_ECHO to 'on' +@if "%MAVEN_BATCH_ECHO%" == "on" echo %MAVEN_BATCH_ECHO% + +@REM set %HOME% to equivalent of $HOME +if "%HOME%" == "" (set "HOME=%HOMEDRIVE%%HOMEPATH%") + +@REM Execute a user defined script before this one +if not "%MAVEN_SKIP_RC%" == "" goto skipRcPre +@REM check for pre script, once with legacy .bat ending and once with .cmd ending +if exist "%HOME%\mavenrc_pre.bat" call "%HOME%\mavenrc_pre.bat" +if exist "%HOME%\mavenrc_pre.cmd" call "%HOME%\mavenrc_pre.cmd" +:skipRcPre + +@setlocal + +set ERROR_CODE=0 + +@REM To isolate internal variables from possible post scripts, we use another setlocal +@setlocal + +@REM ==== START VALIDATION ==== +if not "%JAVA_HOME%" == "" goto OkJHome + +echo. +echo Error: JAVA_HOME not found in your environment. >&2 +echo Please set the JAVA_HOME variable in your environment to match the >&2 +echo location of your Java installation. >&2 +echo. +goto error + +:OkJHome +if exist "%JAVA_HOME%\bin\java.exe" goto init + +echo. +echo Error: JAVA_HOME is set to an invalid directory. >&2 +echo JAVA_HOME = "%JAVA_HOME%" >&2 +echo Please set the JAVA_HOME variable in your environment to match the >&2 +echo location of your Java installation. >&2 +echo. +goto error + +@REM ==== END VALIDATION ==== + +:init + +@REM Find the project base dir, i.e. the directory that contains the folder ".mvn". +@REM Fallback to current working directory if not found. + +set MAVEN_PROJECTBASEDIR=%MAVEN_BASEDIR% +IF NOT "%MAVEN_PROJECTBASEDIR%"=="" goto endDetectBaseDir + +set EXEC_DIR=%CD% +set WDIR=%EXEC_DIR% +:findBaseDir +IF EXIST "%WDIR%"\.mvn goto baseDirFound +cd .. +IF "%WDIR%"=="%CD%" goto baseDirNotFound +set WDIR=%CD% +goto findBaseDir + +:baseDirFound +set MAVEN_PROJECTBASEDIR=%WDIR% +cd "%EXEC_DIR%" +goto endDetectBaseDir + +:baseDirNotFound +set MAVEN_PROJECTBASEDIR=%EXEC_DIR% +cd "%EXEC_DIR%" + +:endDetectBaseDir + +IF NOT EXIST "%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config" goto endReadAdditionalConfig + +@setlocal EnableExtensions EnableDelayedExpansion +for /F "usebackq delims=" %%a in ("%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config") do set JVM_CONFIG_MAVEN_PROPS=!JVM_CONFIG_MAVEN_PROPS! %%a +@endlocal & set JVM_CONFIG_MAVEN_PROPS=%JVM_CONFIG_MAVEN_PROPS% + +:endReadAdditionalConfig + +SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe" +set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar" +set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain + +set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + +FOR /F "tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO ( + IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B +) + +@REM Extension to allow automatically downloading the maven-wrapper.jar from Maven-central +@REM This allows using the maven wrapper in projects that prohibit checking in binary data. +if exist %WRAPPER_JAR% ( + if "%MVNW_VERBOSE%" == "true" ( + echo Found %WRAPPER_JAR% + ) +) else ( + if not "%MVNW_REPOURL%" == "" ( + SET DOWNLOAD_URL="%MVNW_REPOURL%/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + ) + if "%MVNW_VERBOSE%" == "true" ( + echo Couldn't find %WRAPPER_JAR%, downloading it ... + echo Downloading from: %DOWNLOAD_URL% + ) + + powershell -Command "&{"^ + "$webclient = new-object System.Net.WebClient;"^ + "if (-not ([string]::IsNullOrEmpty('%MVNW_USERNAME%') -and [string]::IsNullOrEmpty('%MVNW_PASSWORD%'))) {"^ + "$webclient.Credentials = new-object System.Net.NetworkCredential('%MVNW_USERNAME%', '%MVNW_PASSWORD%');"^ + "}"^ + "[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; $webclient.DownloadFile('%DOWNLOAD_URL%', '%WRAPPER_JAR%')"^ + "}" + if "%MVNW_VERBOSE%" == "true" ( + echo Finished downloading %WRAPPER_JAR% + ) +) +@REM End of extension + +@REM Provide a "standardized" way to retrieve the CLI args that will +@REM work with both Windows and non-Windows executions. +set MAVEN_CMD_LINE_ARGS=%* + +%MAVEN_JAVA_EXE% %JVM_CONFIG_MAVEN_PROPS% %MAVEN_OPTS% %MAVEN_DEBUG_OPTS% -classpath %WRAPPER_JAR% "-Dmaven.multiModuleProjectDirectory=%MAVEN_PROJECTBASEDIR%" %WRAPPER_LAUNCHER% %MAVEN_CONFIG% %* +if ERRORLEVEL 1 goto error +goto end + +:error +set ERROR_CODE=1 + +:end +@endlocal & set ERROR_CODE=%ERROR_CODE% + +if not "%MAVEN_SKIP_RC%" == "" goto skipRcPost +@REM check for post script, once with legacy .bat ending and once with .cmd ending +if exist "%HOME%\mavenrc_post.bat" call "%HOME%\mavenrc_post.bat" +if exist "%HOME%\mavenrc_post.cmd" call "%HOME%\mavenrc_post.cmd" +:skipRcPost + +@REM pause the script if MAVEN_BATCH_PAUSE is set to 'on' +if "%MAVEN_BATCH_PAUSE%" == "on" pause + +if "%MAVEN_TERMINATE_CMD%" == "on" exit %ERROR_CODE% + +exit /B %ERROR_CODE% diff --git a/pom.xml b/pom.xml index 208d52705..72add3980 100644 --- a/pom.xml +++ b/pom.xml @@ -1,18 +1,13 @@ - + 4.0.0 - - org.sonatype.oss - oss-parent - 9 - - com.github.docker-java - docker-java - jar - 3.0.3-SNAPSHOT + docker-java-parent + pom + 0-SNAPSHOT - docker-java + docker-java-parent https://github.com/docker-java/docker-java Java API Client for Docker @@ -32,11 +27,26 @@ + + marcuslinke + Marcus Linke + marcus.linke@gmx.de + + + kostyasha + Kanstantsin Shautsou + kanstantsin.sha@gmail.com + kpelykh Konstantin Pelykh kpelykh@gmail.com + + bsideup + Sergei Egorov + bsideup@gmail.com + @@ -44,232 +54,87 @@ UTF-8 true false - 1.7 - 1.7 + 1.8 + 1.8 - 2.23.1 - 2.6.4 - 4.5 - 1.12 - 1.10 - 2.5 - 2.6 - 1.7.21 + 2.47 + 4.5.12 + 1.28.0 + 2.21.0 + 3.19.0 + 1.7.30 - 1.54 - 2015-01-27T15-02-14 - 19.0 + 1.82 + 2.10.1 + 33.4.8-jre - 1.1.7 - 6.9.10 - 4.1.3.Final - 1.3 + 1.2.3 + 4.2.7.Final + 2.2 1.8 2.3.3 - 1.10.19 + 3.3.0 3.0.2 - 3.5.1 - 2.5.3 - 2.19.1 - 2.19.1 + 3.8.1 + 3.0.0-M1 + 3.0.0-M4 + 3.0.0-M4 1.8 + 1.1.2.RELEASE + 3.0.0 + 1.6.8 - - - com.fasterxml.jackson.jaxrs - jackson-jaxrs-json-provider - ${jackson-jaxrs.version} - - - org.glassfish.jersey.connectors - jersey-apache-connector - ${jersey.version} - - - org.apache.httpcomponents - httpcore - 4.4.5 - - - org.apache.httpcomponents - httpclient - ${httpclient.version} - - - org.glassfish.jersey.core - jersey-client - ${jersey.version} - - - de.gesellix - unix-socket-factory - ${unix-socket-factory.version} - - - - org.apache.commons - commons-compress - ${commons-compress.version} - - - commons-codec - commons-codec - ${commons-codec.version} - - - commons-lang - commons-lang - ${commons-lang.version} - - - commons-io - commons-io - ${commons-io.version} - - - org.slf4j - slf4j-api - ${slf4j-api.version} - - - org.slf4j - jcl-over-slf4j - 1.7.21 - - - - com.google.guava - guava - ${guava.version} - - - org.bouncycastle - bcpkix-jdk15on - ${bouncycastle.version} - - - - - ch.qos.logback - logback-core - ${logback.version} - test - - - - ch.qos.logback - logback-classic - ${logback.version} - test - - - - org.testng - testng - ${testng.version} - test - - - - org.hamcrest - hamcrest-library - ${hamcrest.library.version} - test - - - - com.googlecode.lambdaj - lambdaj - ${lambdaj.version} - test - - - org.hamcrest - hamcrest-all - - - - - - org.testinfected.hamcrest-matchers - jpa-matchers - ${hamcrest.jpa-matchers} - test - - - - org.mockito - mockito-core - ${mockito.version} - test - - - - com.google.code.findbugs - annotations - 3.0.1 - provided - - - - io.netty - netty-codec-http - ${netty.version} - - - io.netty - netty-handler - ${netty.version} - - - io.netty - netty-handler-proxy - ${netty.version} - - - io.netty - netty-transport-native-epoll - ${netty.version} - linux-x86_64 - - - junit - junit - 4.12 - test - - - - - - ossrh - https://oss.sonatype.org/content/repositories/snapshots - - - ossrh - https://oss.sonatype.org/service/local/staging/deploy/maven2/ - - + + docker-java-api + docker-java-bom + docker-java-core + docker-java-transport + docker-java-transport-tck + docker-java-transport-netty + docker-java-transport-jersey + docker-java-transport-okhttp + docker-java-transport-httpclient5 + docker-java-transport-zerodep + docker-java + - - - - - - - + + org.apache.maven.plugins + maven-clean-plugin + 3.1.0 + + + org.apache.maven.plugins + maven-deploy-plugin + 3.0.0-M1 + + + org.apache.maven.plugins + maven-resources-plugin + 3.1.0 + + + org.apache.maven.plugins + maven-gpg-plugin + 1.6 + + + org.apache.maven.plugins + maven-install-plugin + 3.0.0-M1 + org.apache.maven.plugins - maven-release-plugin - ${maven-release-plugin.version} + maven-surefire-plugin + ${maven-surefire-plugin.version} @@ -281,6 +146,7 @@ ${jdk.target} ${jdk.debug} ${jdk.optimize} + true @@ -295,6 +161,13 @@ + + + + ${automatic.module.name} + + + @@ -346,81 +219,96 @@ org.apache.maven.plugins maven-javadoc-plugin 2.10.4 + + -Xdoclint:none + attach-javadocs jar - - -Xdoclint:none - + + org.apache.felix + maven-bundle-plugin + 4.2.1 + + + + + + com.github.siom79.japicmp + japicmp-maven-plugin + 0.24.2 + + + + com.github.docker-java + ${project.artifactId} + 3.3.4 + jar + + + + + ${project.build.directory}/${project.artifactId}-${project.version}.jar + + + + true + public + true + + + METHOD_NEW_DEFAULT + true + true + + + METHOD_ABSTRACT_NOW_DEFAULT + true + true + + + + + + + verify + + cmp + + + + + + org.sonatype.central + central-publishing-maven-plugin + 0.9.0 + true + + central + docker-java-transport-tck + com.github.docker-java + + - - org.sonatype.plugins - nexus-staging-maven-plugin - 1.6.7 - true - - ossrh - https://oss.sonatype.org/ - true - - - org.apache.maven.plugins - maven-release-plugin - ${maven-release-plugin.version} - - true - false - release - deploy nexus-staging:release - + maven-source-plugin - org.apache.maven.plugins - maven-surefire-plugin - ${maven-surefire-plugin.version} - - integration,integration-auth - + maven-javadoc-plugin - - org.apache.maven.plugins - maven-failsafe-plugin - ${maven-failsafe-plugin.version} - - - - integration-test - verify - - - true - true - 1 - integration - integration-auth - - **/*Test.java - - - - - - - org.apache.maven.plugins - maven-source-plugin - + org.sonatype.central + central-publishing-maven-plugin @@ -432,6 +320,13 @@ org.apache.maven.plugins maven-gpg-plugin + + + + --pinentry-mode + loopback + + sign-artifacts @@ -473,65 +368,13 @@ true true false - + - src/test/resources/checkstyle/checkstyle-config.xml + ${maven.multiModuleProjectDirectory}/src/test/resources/checkstyle/checkstyle-config.xml - - org.codehaus.mojo - findbugs-maven-plugin - 3.0.3 - - Max - Low - true - - false - - - - - check - - - - - - org.jacoco - jacoco-maven-plugin - 0.7.7.201606060606 - - - - prepare-agent - - - - - post-unit-test - test - - report - - - - - pre-integration-test - pre-integration-test - - prepare-agent-integration - - - - report-integration - - report-integration - - - - diff --git a/src/main/java/com/github/dockerjava/api/DockerClient.java b/src/main/java/com/github/dockerjava/api/DockerClient.java deleted file mode 100644 index 491c11ac8..000000000 --- a/src/main/java/com/github/dockerjava/api/DockerClient.java +++ /dev/null @@ -1,256 +0,0 @@ -package com.github.dockerjava.api; - -import java.io.Closeable; -import java.io.File; -import java.io.IOException; -import java.io.InputStream; - -import javax.annotation.Nonnull; - -import com.github.dockerjava.api.command.AttachContainerCmd; -import com.github.dockerjava.api.command.AuthCmd; -import com.github.dockerjava.api.command.BuildImageCmd; -import com.github.dockerjava.api.command.CommitCmd; -import com.github.dockerjava.api.command.ConnectToNetworkCmd; -import com.github.dockerjava.api.command.ContainerDiffCmd; -import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; -import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; -import com.github.dockerjava.api.command.CopyFileFromContainerCmd; -import com.github.dockerjava.api.command.CreateContainerCmd; -import com.github.dockerjava.api.command.CreateImageCmd; -import com.github.dockerjava.api.command.CreateNetworkCmd; -import com.github.dockerjava.api.command.CreateVolumeCmd; -import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; -import com.github.dockerjava.api.command.EventsCmd; -import com.github.dockerjava.api.command.ExecCreateCmd; -import com.github.dockerjava.api.command.ExecStartCmd; -import com.github.dockerjava.api.command.InfoCmd; -import com.github.dockerjava.api.command.InspectContainerCmd; -import com.github.dockerjava.api.command.InspectExecCmd; -import com.github.dockerjava.api.command.InspectImageCmd; -import com.github.dockerjava.api.command.InspectNetworkCmd; -import com.github.dockerjava.api.command.InspectVolumeCmd; -import com.github.dockerjava.api.command.KillContainerCmd; -import com.github.dockerjava.api.command.ListContainersCmd; -import com.github.dockerjava.api.command.ListImagesCmd; -import com.github.dockerjava.api.command.ListNetworksCmd; -import com.github.dockerjava.api.command.ListVolumesCmd; -import com.github.dockerjava.api.command.LoadImageCmd; -import com.github.dockerjava.api.command.LogContainerCmd; -import com.github.dockerjava.api.command.PauseContainerCmd; -import com.github.dockerjava.api.command.PingCmd; -import com.github.dockerjava.api.command.PullImageCmd; -import com.github.dockerjava.api.command.PushImageCmd; -import com.github.dockerjava.api.command.RemoveContainerCmd; -import com.github.dockerjava.api.command.RemoveImageCmd; -import com.github.dockerjava.api.command.RemoveNetworkCmd; -import com.github.dockerjava.api.command.RemoveVolumeCmd; -import com.github.dockerjava.api.command.RestartContainerCmd; -import com.github.dockerjava.api.command.SaveImageCmd; -import com.github.dockerjava.api.command.SearchImagesCmd; -import com.github.dockerjava.api.command.StartContainerCmd; -import com.github.dockerjava.api.command.StatsCmd; -import com.github.dockerjava.api.command.StopContainerCmd; -import com.github.dockerjava.api.command.TagImageCmd; -import com.github.dockerjava.api.command.TopContainerCmd; -import com.github.dockerjava.api.command.UnpauseContainerCmd; -import com.github.dockerjava.api.command.UpdateContainerCmd; -import com.github.dockerjava.api.command.VersionCmd; -import com.github.dockerjava.api.command.WaitContainerCmd; -import com.github.dockerjava.api.command.RenameContainerCmd; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.Identifier; -import com.github.dockerjava.core.RemoteApiVersion; - -// https://godoc.org/github.com/fsouza/go-dockerclient -public interface DockerClient extends Closeable { - - AuthConfig authConfig() throws DockerException; - - /** - * Authenticate with the server, useful for checking authentication. - */ - AuthCmd authCmd(); - - InfoCmd infoCmd(); - - PingCmd pingCmd(); - - VersionCmd versionCmd(); - - /** - * * IMAGE API * - */ - - PullImageCmd pullImageCmd(@Nonnull String repository); - - PushImageCmd pushImageCmd(@Nonnull String name); - - PushImageCmd pushImageCmd(@Nonnull Identifier identifier); - - CreateImageCmd createImageCmd(@Nonnull String repository, @Nonnull InputStream imageStream); - - /** - * Loads a tarball with a set of images and tags into a Docker repository. - * - * Corresponds to POST /images/load API endpoint. - * - * @param imageStream - * stream of the tarball file - * @return created command - * @since {@link RemoteApiVersion#VERSION_1_7} - */ - LoadImageCmd loadImageCmd(@Nonnull InputStream imageStream); - - SearchImagesCmd searchImagesCmd(@Nonnull String term); - - RemoveImageCmd removeImageCmd(@Nonnull String imageId); - - ListImagesCmd listImagesCmd(); - - InspectImageCmd inspectImageCmd(@Nonnull String imageId); - - SaveImageCmd saveImageCmd(@Nonnull String name); - - /** - * * CONTAINER API * - */ - - ListContainersCmd listContainersCmd(); - - CreateContainerCmd createContainerCmd(@Nonnull String image); - - /** - * Creates a new {@link StartContainerCmd} for the container with the given ID. The command can then be further customized by using - * builder methods on it like {@link StartContainerCmd#withDns(String...)}. - *

- * If you customize the command, any existing configuration of the target container will get reset to its default before applying the - * new configuration. To preserve the existing configuration, use an unconfigured {@link StartContainerCmd}. - *

- * This command corresponds to the /containers/{id}/start endpoint of the Docker Remote API. - */ - StartContainerCmd startContainerCmd(@Nonnull String containerId); - - ExecCreateCmd execCreateCmd(@Nonnull String containerId); - - InspectContainerCmd inspectContainerCmd(@Nonnull String containerId); - - RemoveContainerCmd removeContainerCmd(@Nonnull String containerId); - - WaitContainerCmd waitContainerCmd(@Nonnull String containerId); - - AttachContainerCmd attachContainerCmd(@Nonnull String containerId); - - ExecStartCmd execStartCmd(@Nonnull String execId); - - InspectExecCmd inspectExecCmd(@Nonnull String execId); - - LogContainerCmd logContainerCmd(@Nonnull String containerId); - - /** - * Copy resource from container to local machine. - * - * @param containerId - * id of the container - * @param resource - * path to container's resource - * @return created command - * @since {@link RemoteApiVersion#VERSION_1_20} - */ - CopyArchiveFromContainerCmd copyArchiveFromContainerCmd(@Nonnull String containerId, @Nonnull String resource); - - /** - * Copy resource from container to local machine. - * - * @param containerId - * id of the container - * @param resource - * path to container's resource - * @return created command - * @see #copyArchiveFromContainerCmd(String, String) - * @deprecated since docker API version 1.20, replaced by {@link #copyArchiveFromContainerCmd(String, String)} - */ - @Deprecated - CopyFileFromContainerCmd copyFileFromContainerCmd(@Nonnull String containerId, @Nonnull String resource); - - /** - * Copy archive from local machine to remote container - * - * @param containerId - * id of the container - * @return created command - * @since {@link RemoteApiVersion#VERSION_1_20} - */ - CopyArchiveToContainerCmd copyArchiveToContainerCmd(@Nonnull String containerId); - - ContainerDiffCmd containerDiffCmd(@Nonnull String containerId); - - StopContainerCmd stopContainerCmd(@Nonnull String containerId); - - KillContainerCmd killContainerCmd(@Nonnull String containerId); - - /** - * Update container settings - * - * @param containerId id of the container - * @return command - * @since {@link RemoteApiVersion#VERSION_1_22} - */ - UpdateContainerCmd updateContainerCmd(@Nonnull String containerId); - - /** - * Rename container. - * - * @param containerId id of the container - * @return command - * @since {@link RemoteApiVersion#VERSION_1_17} - */ - RenameContainerCmd renameContainerCmd(@Nonnull String containerId); - - RestartContainerCmd restartContainerCmd(@Nonnull String containerId); - - CommitCmd commitCmd(@Nonnull String containerId); - - BuildImageCmd buildImageCmd(); - - BuildImageCmd buildImageCmd(File dockerFileOrFolder); - - BuildImageCmd buildImageCmd(InputStream tarInputStream); - - TopContainerCmd topContainerCmd(String containerId); - - TagImageCmd tagImageCmd(String imageId, String repository, String tag); - - PauseContainerCmd pauseContainerCmd(String containerId); - - UnpauseContainerCmd unpauseContainerCmd(String containerId); - - EventsCmd eventsCmd(); - - StatsCmd statsCmd(String containerId); - - CreateVolumeCmd createVolumeCmd(); - - InspectVolumeCmd inspectVolumeCmd(String name); - - RemoveVolumeCmd removeVolumeCmd(String name); - - ListVolumesCmd listVolumesCmd(); - - ListNetworksCmd listNetworksCmd(); - - InspectNetworkCmd inspectNetworkCmd(); - - CreateNetworkCmd createNetworkCmd(); - - RemoveNetworkCmd removeNetworkCmd(@Nonnull String networkId); - - ConnectToNetworkCmd connectToNetworkCmd(); - - DisconnectFromNetworkCmd disconnectFromNetworkCmd(); - - @Override - void close() throws IOException; - -} diff --git a/src/main/java/com/github/dockerjava/api/command/AsyncDockerCmd.java b/src/main/java/com/github/dockerjava/api/command/AsyncDockerCmd.java deleted file mode 100644 index 3218a8419..000000000 --- a/src/main/java/com/github/dockerjava/api/command/AsyncDockerCmd.java +++ /dev/null @@ -1,18 +0,0 @@ -/* - * Created on 17.06.2015 - */ -package com.github.dockerjava.api.command; - -import com.github.dockerjava.api.async.ResultCallback; - -/** - * - * - * @author Marcus Linke - * - */ -public interface AsyncDockerCmd, A_RES_T> extends DockerCmd { - - > T exec(T resultCallback); - -} diff --git a/src/main/java/com/github/dockerjava/api/command/BuildImageCmd.java b/src/main/java/com/github/dockerjava/api/command/BuildImageCmd.java deleted file mode 100644 index 20b23e903..000000000 --- a/src/main/java/com/github/dockerjava/api/command/BuildImageCmd.java +++ /dev/null @@ -1,151 +0,0 @@ -package com.github.dockerjava.api.command; - -import java.io.File; -import java.io.InputStream; -import java.net.URI; -import java.util.Map; - -import javax.annotation.CheckForNull; -import javax.annotation.Nonnull; - -import com.github.dockerjava.api.model.AuthConfigurations; -import com.github.dockerjava.api.model.BuildResponseItem; -import com.github.dockerjava.core.RemoteApiVersion; - -/** - * Build an image from Dockerfile. - *

- * TODO: http://docs.docker.com/reference/builder/#dockerignore - * - * @see build-image-from-a-dockerfile - */ -public interface BuildImageCmd extends AsyncDockerCmd { - - // lib specific - - @CheckForNull - InputStream getTarInputStream(); - - @CheckForNull - AuthConfigurations getBuildAuthConfigs(); - - // getters - - /** - * "t" in API - */ - @CheckForNull - String getTag(); - - /** - * "remote" in API - */ - @CheckForNull - URI getRemote(); - - /** - * "nocache" in API - */ - @CheckForNull - Boolean hasNoCacheEnabled(); - - /** - * "rm" in API - */ - @CheckForNull - Boolean hasRemoveEnabled(); - - /** - * "forcerm" in API - */ - @CheckForNull - Boolean isForcerm(); - - /** - * "q" in API - */ - @CheckForNull - Boolean isQuiet(); - - /** - * "pull" in API - */ - @CheckForNull - Boolean hasPullEnabled(); - - @CheckForNull - String getPathToDockerfile(); - - @CheckForNull - Long getMemory(); - - @CheckForNull - Long getMemswap(); - - @CheckForNull - String getCpushares(); - - @CheckForNull - String getCpusetcpus(); - - /** - * @since {@link RemoteApiVersion#VERSION_1_21} - */ - @CheckForNull - Map getBuildArgs(); - - /** - *@since {@link RemoteApiVersion#VERSION_1_22} - */ - @CheckForNull - Long getShmsize(); - - // setters - - BuildImageCmd withTag(String tag); - - BuildImageCmd withRemote(URI remote); - - BuildImageCmd withBaseDirectory(File baseDirectory); - - BuildImageCmd withDockerfile(File dockerfile); - - BuildImageCmd withNoCache(Boolean noCache); - - BuildImageCmd withRemove(Boolean rm); - - BuildImageCmd withForcerm(Boolean forcerm); - - BuildImageCmd withQuiet(Boolean quiet); - - BuildImageCmd withPull(Boolean pull); - - BuildImageCmd withMemory(Long memory); - - BuildImageCmd withMemswap(Long memswap); - - BuildImageCmd withCpushares(String cpushares); - - BuildImageCmd withCpusetcpus(String cpusetcpus); - - /** - * @since {@link RemoteApiVersion#VERSION_1_21} - */ - BuildImageCmd withBuildArg(String key, String value); - - // setters lib specific - - BuildImageCmd withBuildAuthConfigs(AuthConfigurations authConfig); - - BuildImageCmd withTarInputStream(@Nonnull InputStream tarInputStream); - - /** - *@since {@link RemoteApiVersion#VERSION_1_22} - */ - BuildImageCmd withShmsize(Long shmsize); - - interface Exec extends DockerCmdAsyncExec { - } - -} diff --git a/src/main/java/com/github/dockerjava/api/command/CreateContainerCmd.java b/src/main/java/com/github/dockerjava/api/command/CreateContainerCmd.java deleted file mode 100644 index 77bf100ac..000000000 --- a/src/main/java/com/github/dockerjava/api/command/CreateContainerCmd.java +++ /dev/null @@ -1,456 +0,0 @@ -package com.github.dockerjava.api.command; - -import com.github.dockerjava.api.exception.ConflictException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Bind; -import com.github.dockerjava.api.model.Capability; -import com.github.dockerjava.api.model.Device; -import com.github.dockerjava.api.model.ExposedPort; -import com.github.dockerjava.api.model.HostConfig; -import com.github.dockerjava.api.model.Link; -import com.github.dockerjava.api.model.LogConfig; -import com.github.dockerjava.api.model.LxcConf; -import com.github.dockerjava.api.model.PortBinding; -import com.github.dockerjava.api.model.Ports; -import com.github.dockerjava.api.model.RestartPolicy; -import com.github.dockerjava.api.model.Ulimit; -import com.github.dockerjava.api.model.Volume; -import com.github.dockerjava.api.model.VolumesFrom; - -import javax.annotation.CheckForNull; -import java.util.List; -import java.util.Map; - -public interface CreateContainerCmd extends SyncDockerCmd { - - @CheckForNull - List getAliases(); - - @CheckForNull - Bind[] getBinds(); - - /** - * @since 1.19 - */ - @CheckForNull - Integer getBlkioWeight(); - - @CheckForNull - Capability[] getCapAdd(); - - @CheckForNull - Capability[] getCapDrop(); - - @CheckForNull - String[] getCmd(); - - /** - * @since 1.19 - */ - @CheckForNull - Integer getCpuPeriod(); - - @CheckForNull - String getCpusetCpus(); - - /** - * @since 1.19 - */ - @CheckForNull - String getCpusetMems(); - - @CheckForNull - Integer getCpuShares(); - - @CheckForNull - Device[] getDevices(); - - @CheckForNull - String[] getDns(); - - @CheckForNull - String[] getDnsSearch(); - - @CheckForNull - String getDomainName(); - - @CheckForNull - String[] getEntrypoint(); - - @CheckForNull - String[] getEnv(); - - @CheckForNull - ExposedPort[] getExposedPorts(); - - @CheckForNull - String getStopSignal(); - - @CheckForNull - String[] getExtraHosts(); - - @CheckForNull - String getHostName(); - - @CheckForNull - String getImage(); - - @CheckForNull - String getIpv4Address(); - - @CheckForNull - String getIpv6Address(); - - @CheckForNull - Map getLabels(); - - @CheckForNull - Link[] getLinks(); - - @CheckForNull - LogConfig getLogConfig(); - - @CheckForNull - LxcConf[] getLxcConf(); - - @CheckForNull - String getMacAddress(); - - @CheckForNull - Long getMemory(); - - @CheckForNull - Long getMemorySwap(); - - @CheckForNull - String getName(); - - @CheckForNull - String getNetworkMode(); - - @CheckForNull - Ports getPortBindings(); - - @CheckForNull - String[] getPortSpecs(); - - @CheckForNull - RestartPolicy getRestartPolicy(); - - @CheckForNull - Ulimit[] getUlimits(); - - @CheckForNull - String getUser(); - - @CheckForNull - Volume[] getVolumes(); - - @CheckForNull - VolumesFrom[] getVolumesFrom(); - - @CheckForNull - String getWorkingDir(); - - @CheckForNull - Boolean isAttachStderr(); - - @CheckForNull - Boolean isAttachStdin(); - - @CheckForNull - Boolean isAttachStdout(); - - @CheckForNull - Boolean isNetworkDisabled(); - - /** - * @since 1.19 - */ - @CheckForNull - Boolean getOomKillDisable(); - - @CheckForNull - Boolean getPrivileged(); - - @CheckForNull - Boolean getPublishAllPorts(); - - @CheckForNull - Boolean getReadonlyRootfs(); - - @CheckForNull - Boolean isStdInOnce(); - - @CheckForNull - Boolean isStdinOpen(); - - @CheckForNull - String getPidMode(); - - @CheckForNull - HostConfig getHostConfig(); - - @CheckForNull - String getCgroupParent(); - - @CheckForNull - Boolean isTty(); - - /** - * Add network-scoped alias for the container - * @param aliases on ore more aliases - */ - CreateContainerCmd withAliases(String... aliases); - - /** - * Add network-scoped alias for the container - * @param aliases on ore more aliases - */ - CreateContainerCmd withAliases(List aliases); - - CreateContainerCmd withAttachStderr(Boolean attachStderr); - - CreateContainerCmd withAttachStdin(Boolean attachStdin); - - CreateContainerCmd withAttachStdout(Boolean attachStdout); - - CreateContainerCmd withBinds(Bind... binds); - - CreateContainerCmd withBinds(List binds); - - /** - * @since 1.19 - */ - CreateContainerCmd withBlkioWeight(Integer blkioWeight); - - /** - * Add linux kernel capability to the container. For example: - * adding {@link Capability#MKNOD} allows the container to create special files using the 'mknod' command. - */ - CreateContainerCmd withCapAdd(Capability... capAdd); - - /** - * Add linux kernel capability to the container. For example: - * adding {@link Capability#MKNOD} allows the container to create special files using the 'mknod' command. - */ - CreateContainerCmd withCapAdd(List capAdd); - - /** - * Drop linux kernel capability from the container. For example: - * dropping {@link Capability#CHOWN} prevents the container from changing the owner of any files. - */ - CreateContainerCmd withCapDrop(Capability... capDrop); - - /** - * Drop linux kernel capability from the container. For example: - * dropping {@link Capability#CHOWN} prevents the container from changing the owner of any files. - */ - CreateContainerCmd withCapDrop(List capDrop); - - CreateContainerCmd withCmd(String... cmd); - - CreateContainerCmd withCmd(List cmd); - - CreateContainerCmd withContainerIDFile(String containerIDFile); - - /** - * @since 1.19 - */ - CreateContainerCmd withCpuPeriod(Integer cpuPeriod); - - CreateContainerCmd withCpusetCpus(String cpusetCpus); - - /** - * @since 1.19 - */ - CreateContainerCmd withCpusetMems(String cpusetMems); - - CreateContainerCmd withCpuShares(Integer cpuShares); - - /** - * Add host devices to the container - */ - CreateContainerCmd withDevices(Device... devices); - - /** - * Add host devices to the container - */ - CreateContainerCmd withDevices(List devices); - - /** - * Set custom DNS servers - */ - CreateContainerCmd withDns(String... dns); - - /** - * Set custom DNS servers - */ - CreateContainerCmd withDns(List dns); - - /** - * Set custom DNS search domains - */ - CreateContainerCmd withDnsSearch(String... dnsSearch); - - /** - * Set custom DNS search domains - */ - CreateContainerCmd withDnsSearch(List dnsSearch); - - CreateContainerCmd withDomainName(String domainName); - - CreateContainerCmd withEntrypoint(String... entrypoint); - - CreateContainerCmd withEntrypoint(List entrypoint); - - CreateContainerCmd withEnv(String... env); - - CreateContainerCmd withEnv(List env); - - CreateContainerCmd withExposedPorts(ExposedPort... exposedPorts); - - CreateContainerCmd withStopSignal(String stopSignal); - - CreateContainerCmd withExposedPorts(List exposedPorts); - - /** - * Add hostnames to /etc/hosts in the container - */ - CreateContainerCmd withExtraHosts(String... extraHosts); - - /** - * Add hostnames to /etc/hosts in the container - */ - CreateContainerCmd withExtraHosts(List extraHosts); - - CreateContainerCmd withHostName(String hostName); - - CreateContainerCmd withImage(String image); - - CreateContainerCmd withIpv4Address(String ipv4Address); - - CreateContainerCmd withIpv6Address(String ipv6Address); - - CreateContainerCmd withLabels(Map labels); - - /** - * Add link to another container. - */ - CreateContainerCmd withLinks(Link... links); - - /** - * Add link to another container. - */ - CreateContainerCmd withLinks(List links); - - CreateContainerCmd withLogConfig(LogConfig logConfig); - - CreateContainerCmd withLxcConf(LxcConf... lxcConf); - - CreateContainerCmd withLxcConf(List lxcConf); - - CreateContainerCmd withMacAddress(String macAddress); - - CreateContainerCmd withMemory(Long memory); - - CreateContainerCmd withMemorySwap(Long memorySwap); - - CreateContainerCmd withName(String name); - - CreateContainerCmd withNetworkDisabled(Boolean disableNetwork); - - /** - * Set the Network mode for the container - *

    - *
  • 'bridge': creates a new network stack for the container on the docker bridge
  • - *
  • 'none': no networking for this container
  • - *
  • 'container:': reuses another container network stack
  • - *
  • 'host': use the host network stack inside the container. Note: the host mode gives the container full access to local system - * services such as D-bus and is therefore considered insecure.
  • - *
- */ - CreateContainerCmd withNetworkMode(String networkMode); - - /** - * @since 1.19 - */ - CreateContainerCmd withOomKillDisable(Boolean oomKillDisable); - - /** - * Add one or more {@link PortBinding}s. This corresponds to the --publish (-p) option of the - * docker run CLI command. - */ - CreateContainerCmd withPortBindings(PortBinding... portBindings); - - /** - * Add one or more {@link PortBinding}s. This corresponds to the --publish (-p) option of the - * docker run CLI command. - */ - CreateContainerCmd withPortBindings(List portBindings); - - /** - * Add the port bindings that are contained in the given {@link Ports} object. - * - * @see #withPortBindings(PortBinding...) - */ - CreateContainerCmd withPortBindings(Ports portBindings); - - CreateContainerCmd withPortSpecs(String... portSpecs); - - CreateContainerCmd withPortSpecs(List portSpecs); - - CreateContainerCmd withPrivileged(Boolean privileged); - - CreateContainerCmd withPublishAllPorts(Boolean publishAllPorts); - - CreateContainerCmd withReadonlyRootfs(Boolean readonlyRootfs); - - /** - * Set custom {@link RestartPolicy} for the container. Defaults to {@link RestartPolicy#noRestart()} - */ - CreateContainerCmd withRestartPolicy(RestartPolicy restartPolicy); - - CreateContainerCmd withStdInOnce(Boolean stdInOnce); - - CreateContainerCmd withStdinOpen(Boolean stdinOpen); - - CreateContainerCmd withTty(Boolean tty); - - CreateContainerCmd withUlimits(Ulimit... ulimits); - - CreateContainerCmd withUlimits(List ulimits); - - CreateContainerCmd withUser(String user); - - CreateContainerCmd withVolumes(Volume... volumes); - - CreateContainerCmd withVolumes(List volumes); - - CreateContainerCmd withVolumesFrom(VolumesFrom... volumesFrom); - - CreateContainerCmd withVolumesFrom(List volumesFrom); - - CreateContainerCmd withWorkingDir(String workingDir); - - CreateContainerCmd withCgroupParent(String cgroupParent); - - /** - * Set the PID (Process) Namespace mode for the container, 'host': use the host's PID namespace inside the container - */ - CreateContainerCmd withPidMode(String pidMode); - - CreateContainerCmd withHostConfig(HostConfig hostConfig); - - /** - * @throws NotFoundException - * No such container - * @throws ConflictException - * Named container already exists - */ - @Override - CreateContainerResponse exec() throws NotFoundException, ConflictException; - - interface Exec extends DockerCmdSyncExec { - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/CreateContainerResponse.java b/src/main/java/com/github/dockerjava/api/command/CreateContainerResponse.java deleted file mode 100644 index 65234945f..000000000 --- a/src/main/java/com/github/dockerjava/api/command/CreateContainerResponse.java +++ /dev/null @@ -1,42 +0,0 @@ -package com.github.dockerjava.api.command; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * - * @author Konstantin Pelykh (kpelykh@gmail.com) - * - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class CreateContainerResponse { - - @JsonProperty("Id") - private String id; - - @JsonProperty("Warnings") - private String[] warnings; - - public String getId() { - return id; - } - - public String[] getWarnings() { - return warnings; - } - - public void setId(String id) { - this.id = id; - } - - public void setWarnings(String[] warnings) { - this.warnings = warnings; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/CreateImageResponse.java b/src/main/java/com/github/dockerjava/api/command/CreateImageResponse.java deleted file mode 100644 index 1e8c8d2e9..000000000 --- a/src/main/java/com/github/dockerjava/api/command/CreateImageResponse.java +++ /dev/null @@ -1,28 +0,0 @@ -package com.github.dockerjava.api.command; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * Parse reponses from /images/create - * - * @author Ryan Campbell (ryan.campbell@gmail.com) - * - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class CreateImageResponse { - - @JsonProperty("status") - private String id; - - public String getId() { - return id; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/CreateNetworkCmd.java b/src/main/java/com/github/dockerjava/api/command/CreateNetworkCmd.java deleted file mode 100644 index 39989bd13..000000000 --- a/src/main/java/com/github/dockerjava/api/command/CreateNetworkCmd.java +++ /dev/null @@ -1,59 +0,0 @@ -package com.github.dockerjava.api.command; - -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.api.model.Network.Ipam; -import com.github.dockerjava.core.RemoteApiVersion; - -import javax.annotation.CheckForNull; -import javax.annotation.Nonnull; -import java.util.Map; - -/** - * Create a network. - * - * @since {@link RemoteApiVersion#VERSION_1_21} - */ -public interface CreateNetworkCmd extends SyncDockerCmd { - - @CheckForNull - String getName(); - - @CheckForNull - String getDriver(); - - @CheckForNull - Network.Ipam getIpam(); - - @CheckForNull - Map getOptions(); - - @CheckForNull - Boolean getCheckDuplicate(); - - @CheckForNull - Boolean getInternal(); - - @CheckForNull - Boolean getEnableIPv6(); - - /** The new network's name. Required. */ - CreateNetworkCmd withName(@Nonnull String name); - - /** Name of the network driver to use. Defaults to bridge. */ - CreateNetworkCmd withDriver(String driver); - - /** Ipam config, such es subnet, gateway and ip range of the network */ - CreateNetworkCmd withIpam(Ipam ipam); - - /** Driver specific options */ - CreateNetworkCmd withOptions(Map options); - - CreateNetworkCmd withCheckDuplicate(boolean checkForDuplicate); - - CreateNetworkCmd withInternal(boolean internal); - - CreateNetworkCmd withEnableIpv6(boolean enableIpv6); - - interface Exec extends DockerCmdSyncExec { - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/CreateNetworkResponse.java b/src/main/java/com/github/dockerjava/api/command/CreateNetworkResponse.java deleted file mode 100644 index ec4827476..000000000 --- a/src/main/java/com/github/dockerjava/api/command/CreateNetworkResponse.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.api.command; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import org.apache.commons.lang.builder.ToStringBuilder; - -@JsonIgnoreProperties(ignoreUnknown = true) -public class CreateNetworkResponse { - - @JsonProperty("Id") - private String id; - - @JsonProperty("Warnings") - private String[] warnings; - - public String getId() { - return id; - } - - public String[] getWarnings() { - return warnings; - } - - public void setId(String id) { - this.id = id; - } - - public void setWarnings(String[] warnings) { - this.warnings = warnings; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/CreateVolumeResponse.java b/src/main/java/com/github/dockerjava/api/command/CreateVolumeResponse.java deleted file mode 100644 index 3a541a0cd..000000000 --- a/src/main/java/com/github/dockerjava/api/command/CreateVolumeResponse.java +++ /dev/null @@ -1,40 +0,0 @@ -package com.github.dockerjava.api.command; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * - * @author Marcus Linke - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class CreateVolumeResponse { - - @JsonProperty("Name") - private String name; - - @JsonProperty("Driver") - private String driver; - - @JsonProperty("Mountpoint") - private String mountpoint; - - public String getName() { - return name; - } - - public String getDriver() { - return driver; - } - - public String getMountpoint() { - return mountpoint; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/DockerCmdExecFactory.java b/src/main/java/com/github/dockerjava/api/command/DockerCmdExecFactory.java deleted file mode 100644 index 671c0d535..000000000 --- a/src/main/java/com/github/dockerjava/api/command/DockerCmdExecFactory.java +++ /dev/null @@ -1,123 +0,0 @@ -package com.github.dockerjava.api.command; - -import java.io.Closeable; -import java.io.IOException; - -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.RemoteApiVersion; - -public interface DockerCmdExecFactory extends Closeable { - - void init(DockerClientConfig dockerClientConfig); - - AuthCmd.Exec createAuthCmdExec(); - - InfoCmd.Exec createInfoCmdExec(); - - PingCmd.Exec createPingCmdExec(); - - ExecCreateCmd.Exec createExecCmdExec(); - - VersionCmd.Exec createVersionCmdExec(); - - PullImageCmd.Exec createPullImageCmdExec(); - - PushImageCmd.Exec createPushImageCmdExec(); - - SaveImageCmd.Exec createSaveImageCmdExec(); - - CreateImageCmd.Exec createCreateImageCmdExec(); - - LoadImageCmd.Exec createLoadImageCmdExec(); - - SearchImagesCmd.Exec createSearchImagesCmdExec(); - - RemoveImageCmd.Exec createRemoveImageCmdExec(); - - ListImagesCmd.Exec createListImagesCmdExec(); - - InspectImageCmd.Exec createInspectImageCmdExec(); - - ListContainersCmd.Exec createListContainersCmdExec(); - - CreateContainerCmd.Exec createCreateContainerCmdExec(); - - StartContainerCmd.Exec createStartContainerCmdExec(); - - InspectContainerCmd.Exec createInspectContainerCmdExec(); - - RemoveContainerCmd.Exec createRemoveContainerCmdExec(); - - WaitContainerCmd.Exec createWaitContainerCmdExec(); - - AttachContainerCmd.Exec createAttachContainerCmdExec(); - - ExecStartCmd.Exec createExecStartCmdExec(); - - InspectExecCmd.Exec createInspectExecCmdExec(); - - LogContainerCmd.Exec createLogContainerCmdExec(); - - CopyFileFromContainerCmd.Exec createCopyFileFromContainerCmdExec(); - - CopyArchiveFromContainerCmd.Exec createCopyArchiveFromContainerCmdExec(); - - CopyArchiveToContainerCmd.Exec createCopyArchiveToContainerCmdExec(); - - StopContainerCmd.Exec createStopContainerCmdExec(); - - ContainerDiffCmd.Exec createContainerDiffCmdExec(); - - KillContainerCmd.Exec createKillContainerCmdExec(); - - UpdateContainerCmd.Exec createUpdateContainerCmdExec(); - - /** - * Rename container. - * - * @since {@link RemoteApiVersion#VERSION_1_17} - */ - RenameContainerCmd.Exec createRenameContainerCmdExec(); - - RestartContainerCmd.Exec createRestartContainerCmdExec(); - - CommitCmd.Exec createCommitCmdExec(); - - BuildImageCmd.Exec createBuildImageCmdExec(); - - TopContainerCmd.Exec createTopContainerCmdExec(); - - TagImageCmd.Exec createTagImageCmdExec(); - - PauseContainerCmd.Exec createPauseContainerCmdExec(); - - UnpauseContainerCmd.Exec createUnpauseContainerCmdExec(); - - EventsCmd.Exec createEventsCmdExec(); - - StatsCmd.Exec createStatsCmdExec(); - - CreateVolumeCmd.Exec createCreateVolumeCmdExec(); - - InspectVolumeCmd.Exec createInspectVolumeCmdExec(); - - RemoveVolumeCmd.Exec createRemoveVolumeCmdExec(); - - ListVolumesCmd.Exec createListVolumesCmdExec(); - - ListNetworksCmd.Exec createListNetworksCmdExec(); - - InspectNetworkCmd.Exec createInspectNetworkCmdExec(); - - CreateNetworkCmd.Exec createCreateNetworkCmdExec(); - - RemoveNetworkCmd.Exec createRemoveNetworkCmdExec(); - - ConnectToNetworkCmd.Exec createConnectToNetworkCmdExec(); - - DisconnectFromNetworkCmd.Exec createDisconnectFromNetworkCmdExec(); - - @Override - void close() throws IOException; - -} diff --git a/src/main/java/com/github/dockerjava/api/command/EventsCmd.java b/src/main/java/com/github/dockerjava/api/command/EventsCmd.java deleted file mode 100644 index c52706226..000000000 --- a/src/main/java/com/github/dockerjava/api/command/EventsCmd.java +++ /dev/null @@ -1,68 +0,0 @@ -package com.github.dockerjava.api.command; - -import java.util.List; -import java.util.Map; - -import javax.annotation.CheckForNull; - -import com.github.dockerjava.api.model.Event; - -/** - * Get events - */ -public interface EventsCmd extends AsyncDockerCmd { - - @CheckForNull - Map> getFilters(); - - @CheckForNull - String getSince(); - - @CheckForNull - String getUntil(); - - /** - * @param container - * - container to filter - */ - EventsCmd withContainerFilter(String... container); - - /** - * @param event - * - event to filter (pull | create | attach | start | stop | kill) - */ - EventsCmd withEventFilter(String... event); - - /** - * @param image - * - image to filter - */ - EventsCmd withImageFilter(String... image); - - /** - * @param label - * - label to filter - */ - EventsCmd withLabelFilter(String... label); - - /** - * @param labels - * - labels to filter (map of names and values) - */ - EventsCmd withLabelFilter(Map labels); - - /** - * @param since - * - Show all events created since timestamp - */ - EventsCmd withSince(String since); - - /** - * @param until - * - Show all events created until timestamp - */ - EventsCmd withUntil(String until); - - interface Exec extends DockerCmdAsyncExec { - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/ExecCreateCmd.java b/src/main/java/com/github/dockerjava/api/command/ExecCreateCmd.java deleted file mode 100644 index fe2329290..000000000 --- a/src/main/java/com/github/dockerjava/api/command/ExecCreateCmd.java +++ /dev/null @@ -1,38 +0,0 @@ -package com.github.dockerjava.api.command; - -import javax.annotation.CheckForNull; -import javax.annotation.Nonnull; - -public interface ExecCreateCmd extends SyncDockerCmd { - - @CheckForNull - String getContainerId(); - - @CheckForNull - Boolean hasAttachStderrEnabled(); - - @CheckForNull - Boolean hasAttachStdinEnabled(); - - @CheckForNull - Boolean hasAttachStdoutEnabled(); - - @CheckForNull - Boolean hasTtyEnabled(); - - ExecCreateCmd withAttachStderr(Boolean attachStderr); - - ExecCreateCmd withAttachStdin(Boolean attachStdin); - - ExecCreateCmd withAttachStdout(Boolean attachStdout); - - ExecCreateCmd withCmd(String... cmd); - - ExecCreateCmd withContainerId(@Nonnull String containerId); - - ExecCreateCmd withTty(Boolean tty); - - interface Exec extends DockerCmdSyncExec { - } - -} diff --git a/src/main/java/com/github/dockerjava/api/command/ExecCreateCmdResponse.java b/src/main/java/com/github/dockerjava/api/command/ExecCreateCmdResponse.java deleted file mode 100644 index 9c13912fc..000000000 --- a/src/main/java/com/github/dockerjava/api/command/ExecCreateCmdResponse.java +++ /dev/null @@ -1,23 +0,0 @@ -package com.github.dockerjava.api.command; - -import org.apache.commons.lang.builder.ReflectionToStringBuilder; -import org.apache.commons.lang.builder.ToStringStyle; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -@JsonIgnoreProperties(ignoreUnknown = true) -public class ExecCreateCmdResponse { - - @JsonProperty("Id") - private String id; - - public String getId() { - return id; - } - - @Override - public String toString() { - return ReflectionToStringBuilder.toString(this, ToStringStyle.SHORT_PREFIX_STYLE); - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/GraphData.java b/src/main/java/com/github/dockerjava/api/command/GraphData.java deleted file mode 100644 index 5a2944e0b..000000000 --- a/src/main/java/com/github/dockerjava/api/command/GraphData.java +++ /dev/null @@ -1,109 +0,0 @@ -package com.github.dockerjava.api.command; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -import javax.annotation.CheckForNull; - -/** - * part of {@link GraphDriver} - * @author Kanstantsin Shautsou - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class GraphData { - - @JsonProperty("RootDir") - private String rootDir; - - @JsonProperty("DeviceId") - private String deviceId; - - @JsonProperty("DeviceName") - private String deviceName; - - @JsonProperty("DeviceSize") - private String deviceSize; - - /** - * @see #rootDir - */ - @CheckForNull - public String getRootDir() { - return rootDir; - } - - /** - * @see #rootDir - */ - public GraphData withRootDir(String rootDir) { - this.rootDir = rootDir; - return this; - } - - /** - * @see #deviceId - */ - @CheckForNull - public String getDeviceId() { - return deviceId; - } - - /** - * @see #deviceId - */ - public GraphData withDeviceId(String deviceId) { - this.deviceId = deviceId; - return this; - } - - /** - * @see #deviceName - */ - @CheckForNull - public String getDeviceName() { - return deviceName; - } - - /** - * @see #deviceName - */ - public GraphData withDeviceName(String deviceName) { - this.deviceName = deviceName; - return this; - } - - /** - * @see #deviceSize - */ - @CheckForNull - public String getDeviceSize() { - return deviceSize; - } - - /** - * @see #deviceSize - */ - public GraphData withDeviceSize(String deviceSize) { - this.deviceSize = deviceSize; - return this; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/GraphDriver.java b/src/main/java/com/github/dockerjava/api/command/GraphDriver.java deleted file mode 100644 index 1394e866e..000000000 --- a/src/main/java/com/github/dockerjava/api/command/GraphDriver.java +++ /dev/null @@ -1,79 +0,0 @@ -package com.github.dockerjava.api.command; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -import javax.annotation.CheckForNull; - -/** - * Part of {@link InspectImageResponse} - * - * @author Kanstantsin Shautsou - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_21} - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class GraphDriver { - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_21} - */ - @JsonProperty("Name") - private String name; - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_21} - */ - @JsonProperty("Data") - private GraphData data; - - - /** - * @see #data - */ - @CheckForNull - public GraphData getData() { - return data; - } - - /** - * @see #data - */ - public GraphDriver withData(GraphData data) { - this.data = data; - return this; - } - - /** - * @see #name - */ - @CheckForNull - public String getName() { - return name; - } - - /** - * @see #name - */ - public GraphDriver withName(String name) { - this.name = name; - return this; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/InspectVolumeResponse.java b/src/main/java/com/github/dockerjava/api/command/InspectVolumeResponse.java deleted file mode 100644 index b5b35c563..000000000 --- a/src/main/java/com/github/dockerjava/api/command/InspectVolumeResponse.java +++ /dev/null @@ -1,40 +0,0 @@ -package com.github.dockerjava.api.command; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * - * @author Marcus Linke - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class InspectVolumeResponse { - - @JsonProperty("Name") - private String name; - - @JsonProperty("Driver") - private String driver; - - @JsonProperty("Mountpoint") - private String mountpoint; - - public String getName() { - return name; - } - - public String getDriver() { - return driver; - } - - public String getMountpoint() { - return mountpoint; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/ListContainersCmd.java b/src/main/java/com/github/dockerjava/api/command/ListContainersCmd.java deleted file mode 100644 index 7c6c9acec..000000000 --- a/src/main/java/com/github/dockerjava/api/command/ListContainersCmd.java +++ /dev/null @@ -1,91 +0,0 @@ -package com.github.dockerjava.api.command; - -import java.util.List; -import java.util.Map; - -import javax.annotation.CheckForNull; - -import com.github.dockerjava.api.model.Container; - -/** - * List containers - * - */ -public interface ListContainersCmd extends SyncDockerCmd> { - - @CheckForNull - String getBeforeId(); - - @CheckForNull - Map> getFilters(); - - @CheckForNull - Integer getLimit(); - - @CheckForNull - String getSinceId(); - - @CheckForNull - Boolean hasShowAllEnabled(); - - @CheckForNull - Boolean hasShowSizeEnabled(); - - /** - * @param before - * - Show only containers created before Id, include non-running ones. - */ - ListContainersCmd withBefore(String before); - - /** - * @param exitcode - * - Show only containers that exited with the passed exitcode. - */ - ListContainersCmd withExitcodeFilter(Integer exitcode); - - /** - * @param status - * - Show only containers with the passed status (created|restarting|running|paused|exited). - */ - ListContainersCmd withStatusFilter(String status); - - /** - * @param labels - * - Show only containers with the passed labels. - */ - ListContainersCmd withLabelFilter(String... labels); - - /** - * @param labels - * - Show only containers with the passed labels. Labels is a {@link Map} that contains label keys and values - */ - ListContainersCmd withLabelFilter(Map labels); - - /** - * @param limit - * - Show `limit` last created containers, include non-running ones. There is no limit by default. - */ - ListContainersCmd withLimit(Integer limit); - - /** - * @param showAll - * - Show all containers. Only running containers are shown by default. - */ - ListContainersCmd withShowAll(Boolean showAll); - - /** - * @param showSize - * - Show the containers sizes. This is false by default. - */ - ListContainersCmd withShowSize(Boolean showSize); - - /** - * @param since - * - Show only containers created since Id, include non-running ones. - */ - ListContainersCmd withSince(String since); - - interface Exec extends DockerCmdSyncExec> { - } - -} diff --git a/src/main/java/com/github/dockerjava/api/command/ListNetworksCmd.java b/src/main/java/com/github/dockerjava/api/command/ListNetworksCmd.java deleted file mode 100644 index 23d92151c..000000000 --- a/src/main/java/com/github/dockerjava/api/command/ListNetworksCmd.java +++ /dev/null @@ -1,27 +0,0 @@ -package com.github.dockerjava.api.command; - -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.core.RemoteApiVersion; - -import javax.annotation.CheckForNull; - -import java.util.List; -import java.util.Map; - -/** - * List networks. - * - * @since {@link RemoteApiVersion#VERSION_1_21} - */ -public interface ListNetworksCmd extends SyncDockerCmd> { - - @CheckForNull - Map> getFilters(); - - ListNetworksCmd withNameFilter(String... networkName); - - ListNetworksCmd withIdFilter(String... networkId); - - interface Exec extends DockerCmdSyncExec> { - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/ListVolumesCmd.java b/src/main/java/com/github/dockerjava/api/command/ListVolumesCmd.java deleted file mode 100644 index 88c6ee291..000000000 --- a/src/main/java/com/github/dockerjava/api/command/ListVolumesCmd.java +++ /dev/null @@ -1,26 +0,0 @@ -package com.github.dockerjava.api.command; - -import java.util.List; -import java.util.Map; - -import javax.annotation.CheckForNull; - -/** - * List volumes. - * - * @author Marcus Linke - */ -public interface ListVolumesCmd extends SyncDockerCmd { - - @CheckForNull - Map> getFilters(); - - /** - * @param dangling - * - Show dangling volumes filter - */ - ListVolumesCmd withDanglingFilter(Boolean dangling); - - interface Exec extends DockerCmdSyncExec { - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/ListVolumesResponse.java b/src/main/java/com/github/dockerjava/api/command/ListVolumesResponse.java deleted file mode 100644 index c45712d09..000000000 --- a/src/main/java/com/github/dockerjava/api/command/ListVolumesResponse.java +++ /dev/null @@ -1,28 +0,0 @@ -package com.github.dockerjava.api.command; - -import java.util.List; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * - * @author Marcus Linke - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class ListVolumesResponse { - - @JsonProperty("Volumes") - private List volumes; - - public List getVolumes() { - return volumes; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/PullImageCmd.java b/src/main/java/com/github/dockerjava/api/command/PullImageCmd.java deleted file mode 100644 index 8127f7b1f..000000000 --- a/src/main/java/com/github/dockerjava/api/command/PullImageCmd.java +++ /dev/null @@ -1,37 +0,0 @@ -package com.github.dockerjava.api.command; - -import javax.annotation.CheckForNull; -import javax.annotation.Nonnull; - -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.PullResponseItem; - -/** - * - * Pull image from repository. - * - */ -public interface PullImageCmd extends AsyncDockerCmd { - - @CheckForNull - String getRepository(); - - @CheckForNull - String getTag(); - - @CheckForNull - String getRegistry(); - - AuthConfig getAuthConfig(); - - PullImageCmd withRepository(@Nonnull String repository); - - PullImageCmd withTag(String tag); - - PullImageCmd withRegistry(String registry); - - PullImageCmd withAuthConfig(AuthConfig authConfig); - - interface Exec extends DockerCmdAsyncExec { - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/PushImageCmd.java b/src/main/java/com/github/dockerjava/api/command/PushImageCmd.java deleted file mode 100644 index abd86c53c..000000000 --- a/src/main/java/com/github/dockerjava/api/command/PushImageCmd.java +++ /dev/null @@ -1,51 +0,0 @@ -package com.github.dockerjava.api.command; - -import javax.annotation.CheckForNull; -import javax.annotation.Nonnull; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.PushResponseItem; - -/** - * Push the latest image to the repository. - * - * @param name - * The name, e.g. "alexec/busybox" or just "busybox" if you want to default. Not null. - */ -public interface PushImageCmd extends AsyncDockerCmd { - - @CheckForNull - AuthConfig getAuthConfig(); - - @CheckForNull - String getName(); - - @CheckForNull - String getTag(); - - /** - * @param name - * The name, e.g. "alexec/busybox" or just "busybox" if you want to default. Not null. - */ - PushImageCmd withName(@Nonnull String name); - - /** - * @param tag - * The image's tag. Not null. - */ - PushImageCmd withTag(String tag); - - PushImageCmd withAuthConfig(AuthConfig authConfig); - - /** - * @throws NotFoundException - * No such image - */ - @Override - > T exec(T resultCallback); - - interface Exec extends DockerCmdAsyncExec { - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/RestartContainerCmd.java b/src/main/java/com/github/dockerjava/api/command/RestartContainerCmd.java deleted file mode 100644 index 5dcc59c24..000000000 --- a/src/main/java/com/github/dockerjava/api/command/RestartContainerCmd.java +++ /dev/null @@ -1,37 +0,0 @@ -package com.github.dockerjava.api.command; - -import javax.annotation.CheckForNull; -import javax.annotation.Nonnull; - -import com.github.dockerjava.api.exception.NotFoundException; - -/** - * Restart a running container. - * - * @param timeout - * - Timeout in seconds before killing the container. Defaults to 10 seconds. - * - */ -public interface RestartContainerCmd extends SyncDockerCmd { - - @CheckForNull - String getContainerId(); - - @CheckForNull - Integer getTimeout(); - - RestartContainerCmd withContainerId(@Nonnull String containerId); - - RestartContainerCmd withtTimeout(Integer timeout); - - /** - * @throws NotFoundException - * No such container - */ - @Override - Void exec() throws NotFoundException; - - interface Exec extends DockerCmdSyncExec { - } - -} diff --git a/src/main/java/com/github/dockerjava/api/command/TopContainerResponse.java b/src/main/java/com/github/dockerjava/api/command/TopContainerResponse.java deleted file mode 100644 index cd6da6814..000000000 --- a/src/main/java/com/github/dockerjava/api/command/TopContainerResponse.java +++ /dev/null @@ -1,41 +0,0 @@ -package com.github.dockerjava.api.command; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.google.common.base.Joiner; - -/** - * - * @author Marcus Linke - * - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class TopContainerResponse { - - @JsonProperty("Titles") - private String[] titles; - - @JsonProperty("Processes") - private String[][] processes; - - public String[] getTitles() { - return titles; - } - - public String[][] getProcesses() { - return processes; - } - - @Override - public String toString() { - StringBuffer buffer = new StringBuffer(); - buffer.append("["); - for (String[] fields : processes) { - buffer.append("[" + Joiner.on("; ").skipNulls().join(fields) + "]"); - } - buffer.append("]"); - - return "TopContainerResponse{" + "titles=" + Joiner.on("; ").skipNulls().join(titles) + ", processes=" - + buffer.toString() + '}'; - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/UpdateContainerCmd.java b/src/main/java/com/github/dockerjava/api/command/UpdateContainerCmd.java deleted file mode 100644 index a834b7ba0..000000000 --- a/src/main/java/com/github/dockerjava/api/command/UpdateContainerCmd.java +++ /dev/null @@ -1,71 +0,0 @@ -package com.github.dockerjava.api.command; - -import com.github.dockerjava.api.model.UpdateContainerResponse; -import com.github.dockerjava.core.RemoteApiVersion; - -import javax.annotation.CheckForNull; -import javax.annotation.Nonnull; - -/** - * @author Kanstantsin Shautsou - * @since {@link RemoteApiVersion#VERSION_1_22} - */ -public interface UpdateContainerCmd extends SyncDockerCmd { - @CheckForNull - String getContainerId(); - - @CheckForNull - Integer getBlkioWeight(); - - UpdateContainerCmd withBlkioWeight(Integer blkioWeight); - - UpdateContainerCmd withContainerId(@Nonnull String containerId); - - @CheckForNull - Integer getCpuPeriod(); - - UpdateContainerCmd withCpuPeriod(Integer cpuPeriod); - - @CheckForNull - Integer getCpuQuota(); - - UpdateContainerCmd withCpuQuota(Integer cpuQuota); - - @CheckForNull - String getCpusetCpus(); - - UpdateContainerCmd withCpusetCpus(String cpusetCpus); - - @CheckForNull - String getCpusetMems(); - - UpdateContainerCmd withCpusetMems(String cpusetMems); - - @CheckForNull - Integer getCpuShares(); - - UpdateContainerCmd withCpuShares(Integer cpuShares); - - @CheckForNull - Long getKernelMemory(); - - UpdateContainerCmd withKernelMemory(Long kernelMemory); - - @CheckForNull - Long getMemory(); - - UpdateContainerCmd withMemory(Long memory); - - @CheckForNull - Long getMemoryReservation(); - - UpdateContainerCmd withMemoryReservation(Long memoryReservation); - - @CheckForNull - Long getMemorySwap(); - - UpdateContainerCmd withMemorySwap(Long memorySwap); - - interface Exec extends DockerCmdSyncExec { - } -} diff --git a/src/main/java/com/github/dockerjava/api/command/WaitContainerCmd.java b/src/main/java/com/github/dockerjava/api/command/WaitContainerCmd.java deleted file mode 100644 index daa4e5d3c..000000000 --- a/src/main/java/com/github/dockerjava/api/command/WaitContainerCmd.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.api.command; - -import javax.annotation.CheckForNull; -import javax.annotation.Nonnull; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.WaitResponse; - -/** - * Wait a container - * - * Block until container stops, then returns its exit code - */ -public interface WaitContainerCmd extends AsyncDockerCmd { - - @CheckForNull - String getContainerId(); - - WaitContainerCmd withContainerId(@Nonnull String containerId); - - /** - * @throws NotFoundException - * container not found - */ - @Override - > T exec(T resultCallback); - - interface Exec extends DockerCmdAsyncExec { - } - -} diff --git a/src/main/java/com/github/dockerjava/api/exception/DockerException.java b/src/main/java/com/github/dockerjava/api/exception/DockerException.java deleted file mode 100644 index 5b511ff96..000000000 --- a/src/main/java/com/github/dockerjava/api/exception/DockerException.java +++ /dev/null @@ -1,27 +0,0 @@ -package com.github.dockerjava.api.exception; - -/** - * - * @author Konstantin Pelykh (kpelykh@gmail.com) - * - */ - -public class DockerException extends RuntimeException { - - private static final long serialVersionUID = 7667768099261650608L; - - private int httpStatus = 0; - - public DockerException(String message, int httpStatus) { - super(message); - this.httpStatus = httpStatus; - } - - public DockerException(String message, int httpStatus, Throwable cause) { - super(message, cause); - } - - public int getHttpStatus() { - return httpStatus; - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/AuthConfig.java b/src/main/java/com/github/dockerjava/api/model/AuthConfig.java deleted file mode 100644 index 849f26a21..000000000 --- a/src/main/java/com/github/dockerjava/api/model/AuthConfig.java +++ /dev/null @@ -1,159 +0,0 @@ -package com.github.dockerjava.api.model; - -import org.apache.commons.lang.builder.ReflectionToStringBuilder; -import org.apache.commons.lang.builder.ToStringStyle; - -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; - -import javax.annotation.CheckForNull; - -@JsonInclude(Include.NON_NULL) -public class AuthConfig { - - /** - * For backwards compatibility. Make sure you update the properties if you change this. - * - * @see "/docker.io.properties" - */ - public static final String DEFAULT_SERVER_ADDRESS = "https://index.docker.io/v1/"; - - @JsonProperty - private String username; - - @JsonProperty - private String password; - - @JsonProperty - private String email; - - @JsonProperty("serveraddress") - private String registryAddress = DEFAULT_SERVER_ADDRESS; - - @JsonProperty("auth") - private String auth; - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("registrytoken") - private String registrytoken; - - public String getUsername() { - return username; - } - - public AuthConfig withUsername(String username) { - this.username = username; - return this; - } - - public String getPassword() { - return password; - } - - public AuthConfig withPassword(String password) { - this.password = password; - return this; - } - - public String getEmail() { - return email; - } - - public AuthConfig withEmail(String email) { - this.email = email; - return this; - } - - public String getRegistryAddress() { - return registryAddress; - } - - public AuthConfig withRegistryAddress(String registryAddress) { - this.registryAddress = registryAddress; - return this; - } - - public String getAuth() { - return auth; - } - - public AuthConfig withAuth(String auth) { - this.auth = auth; - return this; - } - - /** - * @see #registrytoken - */ - @CheckForNull - public String getRegistrytoken() { - return registrytoken; - } - - /** - * @see #registrytoken - */ - public AuthConfig withRegistrytoken(String registrytoken) { - this.registrytoken = registrytoken; - return this; - } - - @Override - public String toString() { - return ReflectionToStringBuilder.toString(this, ToStringStyle.SHORT_PREFIX_STYLE); - } - - // CHECKSTYLE:OFF - @Override - public int hashCode() { - final int prime = 31; - int result = 1; - result = prime * result + ((auth == null) ? 0 : auth.hashCode()); - result = prime * result + ((email == null) ? 0 : email.hashCode()); - result = prime * result + ((password == null) ? 0 : password.hashCode()); - result = prime * result + ((registryAddress == null) ? 0 : registryAddress.hashCode()); - result = prime * result + ((username == null) ? 0 : username.hashCode()); - return result; - } - - @Override - public boolean equals(Object obj) { - if (this == obj) - return true; - if (obj == null) - return false; - if (getClass() != obj.getClass()) - return false; - AuthConfig other = (AuthConfig) obj; - if (auth == null) { - if (other.auth != null) - return false; - } else if (!auth.equals(other.auth)) - return false; - if (email == null) { - if (other.email != null) - return false; - } else if (!email.equals(other.email)) - return false; - if (password == null) { - if (other.password != null) - return false; - } else if (!password.equals(other.password)) - return false; - if (registryAddress == null) { - if (other.registryAddress != null) - return false; - } else if (!registryAddress.equals(other.registryAddress)) - return false; - if (username == null) { - if (other.username != null) - return false; - } else if (!username.equals(other.username)) - return false; - return true; - } - // CHECKSTYLE:ON -} diff --git a/src/main/java/com/github/dockerjava/api/model/AuthConfigurations.java b/src/main/java/com/github/dockerjava/api/model/AuthConfigurations.java deleted file mode 100644 index e8edbc950..000000000 --- a/src/main/java/com/github/dockerjava/api/model/AuthConfigurations.java +++ /dev/null @@ -1,21 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.util.Map; -import java.util.TreeMap; - -import com.fasterxml.jackson.annotation.JsonProperty; - -public class AuthConfigurations { - - @JsonProperty("configs") - private Map configs = new TreeMap<>(); - - public void addConfig(AuthConfig authConfig) { - configs.put(authConfig.getRegistryAddress(), authConfig); - } - - public Map getConfigs() { - return this.configs; - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/AuthResponse.java b/src/main/java/com/github/dockerjava/api/model/AuthResponse.java deleted file mode 100644 index cb5d9df80..000000000 --- a/src/main/java/com/github/dockerjava/api/model/AuthResponse.java +++ /dev/null @@ -1,12 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonProperty; - -public class AuthResponse { - @JsonProperty("Status") - private String status; - - public String getStatus() { - return status; - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Bind.java b/src/main/java/com/github/dockerjava/api/model/Bind.java deleted file mode 100644 index bf02bbad2..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Bind.java +++ /dev/null @@ -1,133 +0,0 @@ -package com.github.dockerjava.api.model; - -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; - -/** - * Represents a host path being bind mounted as a {@link Volume} in a Docker container. - * The Bind can be in read only or read write access mode. - */ -public class Bind { - - private String path; - - private Volume volume; - - private AccessMode accessMode; - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_17} - */ - private SELContext secMode; - - public Bind(String path, Volume volume) { - this(path, volume, AccessMode.DEFAULT, SELContext.DEFAULT); - } - - public Bind(String path, Volume volume, AccessMode accessMode) { - this(path, volume, accessMode, SELContext.DEFAULT); - } - - public Bind(String path, Volume volume, AccessMode accessMode, SELContext secMode) { - this.path = path; - this.volume = volume; - this.accessMode = accessMode; - this.secMode = secMode; - } - - public String getPath() { - return path; - } - - public Volume getVolume() { - return volume; - } - - public AccessMode getAccessMode() { - return accessMode; - } - - public SELContext getSecMode() { - return secMode; - } - - /** - * Parses a bind mount specification to a {@link Bind}. - * - * @param serialized - * the specification, e.g. /host:/container:ro - * @return a {@link Bind} matching the specification - * @throws IllegalArgumentException - * if the specification cannot be parsed - */ - public static Bind parse(String serialized) { - try { - String[] parts = serialized.split(":"); - switch (parts.length) { - case 2: { - return new Bind(parts[0], new Volume(parts[1])); - } - case 3: { - String[] flags = parts[2].split(","); - AccessMode accessMode = AccessMode.DEFAULT; - SELContext seMode = SELContext.DEFAULT; - for (String p : flags) { - if (p.length() == 2) { - accessMode = AccessMode.valueOf(p.toLowerCase()); - } else { - seMode = SELContext.fromString(p); - } - } - - return new Bind(parts[0], new Volume(parts[1]), accessMode, seMode); - } - default: { - throw new IllegalArgumentException(); - } - } - } catch (Exception e) { - throw new IllegalArgumentException("Error parsing Bind '" + serialized + "'", e); - } - } - - @Override - public boolean equals(Object obj) { - if (obj instanceof Bind) { - Bind other = (Bind) obj; - return new EqualsBuilder() - .append(path, other.getPath()) - .append(volume, other.getVolume()) - .append(accessMode, other.getAccessMode()) - .append(secMode, other.getSecMode()) - .isEquals(); - } else { - return super.equals(obj); - } - } - - @Override - public int hashCode() { - return new HashCodeBuilder() - .append(path) - .append(volume) - .append(accessMode) - .append(secMode) - .toHashCode(); - } - - /** - * Returns a string representation of this {@link Bind} suitable for inclusion in a JSON message. - * The format is <host path>:<container path>:<access mode>, - * like the argument in {@link #parse(String)}. - * - * @return a string representation of this {@link Bind} - */ - @Override - public String toString() { - return String.format("%s:%s:%s%s", - path, - volume.getPath(), - accessMode.toString(), - secMode != SELContext.none ? "," + secMode.toString() : ""); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Binds.java b/src/main/java/com/github/dockerjava/api/model/Binds.java deleted file mode 100644 index 917cfdcbd..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Binds.java +++ /dev/null @@ -1,69 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.Iterator; -import java.util.List; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; - -@JsonSerialize(using = Binds.Serializer.class) -@JsonDeserialize(using = Binds.Deserializer.class) -public class Binds { - - private Bind[] binds; - - public Binds(Bind... binds) { - this.binds = binds; - } - - public Bind[] getBinds() { - return binds; - } - - public static class Serializer extends JsonSerializer { - - @Override - public void serialize(Binds binds, JsonGenerator jsonGen, SerializerProvider serProvider) throws IOException, - JsonProcessingException { - - // - jsonGen.writeStartArray(); - for (Bind bind : binds.getBinds()) { - jsonGen.writeString(bind.toString()); - } - jsonGen.writeEndArray(); - // - } - - } - - public static class Deserializer extends JsonDeserializer { - @Override - public Binds deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - - List binds = new ArrayList(); - ObjectCodec oc = jsonParser.getCodec(); - JsonNode node = oc.readTree(jsonParser); - for (Iterator it = node.elements(); it.hasNext();) { - - JsonNode field = it.next(); - binds.add(Bind.parse(field.asText())); - - } - return new Binds(binds.toArray(new Bind[0])); - } - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/BuildResponseItem.java b/src/main/java/com/github/dockerjava/api/model/BuildResponseItem.java deleted file mode 100644 index a53a1df9a..000000000 --- a/src/main/java/com/github/dockerjava/api/model/BuildResponseItem.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; - -/** - * Represents a build response stream item - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class BuildResponseItem extends ResponseItem { - - private static final long serialVersionUID = -1252904184236343612L; - - private static final String BUILD_SUCCESS = "Successfully built"; - - /** - * Returns whether the stream field indicates a successful build operation - */ - @JsonIgnore - public boolean isBuildSuccessIndicated() { - if (isErrorIndicated() || getStream() == null) { - return false; - } - - return getStream().contains(BUILD_SUCCESS); - } - - @JsonIgnore - public String getImageId() { - if (!isBuildSuccessIndicated()) { - return null; - } - - return getStream().replaceFirst(BUILD_SUCCESS, "").trim(); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/ChangeLog.java b/src/main/java/com/github/dockerjava/api/model/ChangeLog.java deleted file mode 100644 index 36804362c..000000000 --- a/src/main/java/com/github/dockerjava/api/model/ChangeLog.java +++ /dev/null @@ -1,34 +0,0 @@ -package com.github.dockerjava.api.model; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * - * @author Konstantin Pelykh (kpelykh@gmail.com) - * - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class ChangeLog { - - @JsonProperty("Path") - private String path; - - @JsonProperty("Kind") - private Integer kind; - - public String getPath() { - return path; - } - - public Integer getKind() { - return kind; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/ContainerHostConfig.java b/src/main/java/com/github/dockerjava/api/model/ContainerHostConfig.java deleted file mode 100644 index 43a5a94b1..000000000 --- a/src/main/java/com/github/dockerjava/api/model/ContainerHostConfig.java +++ /dev/null @@ -1,46 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -/** - * Used in {@link Container} - * - * @see Container - * @author Kanstantsin Shautsou - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class ContainerHostConfig { - @JsonProperty("NetworkMode") - private String networkMode; - - public String getNetworkMode() { - return networkMode; - } - - /** - * @see #networkMode - */ - public ContainerHostConfig withNetworkMode(String networkMode) { - this.networkMode = networkMode; - return this; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/ContainerNetworkSettings.java b/src/main/java/com/github/dockerjava/api/model/ContainerNetworkSettings.java deleted file mode 100644 index 7f4b17be5..000000000 --- a/src/main/java/com/github/dockerjava/api/model/ContainerNetworkSettings.java +++ /dev/null @@ -1,55 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.core.RemoteApiVersion; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -import java.util.Map; - -/** - * Sub-object in {@link Container} - * - * @see Container - * @since {@link RemoteApiVersion#VERSION_1_22} - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class ContainerNetworkSettings { - /** - * @since {@link RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("Networks") - private Map networks; - - /** - * @see #networks - */ - public Map getNetworks() { - return networks; - } - - /** - * @see #networks - */ - public ContainerNetworkSettings withNetworks(Map networks) { - this.networks = networks; - return this; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/ContainerPort.java b/src/main/java/com/github/dockerjava/api/model/ContainerPort.java deleted file mode 100644 index 09f718ef6..000000000 --- a/src/main/java/com/github/dockerjava/api/model/ContainerPort.java +++ /dev/null @@ -1,108 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -import javax.annotation.CheckForNull; - -/** - * @author Kanstantsin Shautsou - * @see Container - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class ContainerPort { - - @JsonProperty("IP") - private String ip; - - @JsonProperty("PrivatePort") - private Integer privatePort; - - @JsonProperty("PublicPort") - private Integer publicPort; - - @JsonProperty("Type") - private String type; - - /** - * @see #ip - */ - @CheckForNull - public String getIp() { - return ip; - } - - /** - * @see #ip - */ - public ContainerPort withIp(String ip) { - this.ip = ip; - return this; - } - - /** - * @see #privatePort - */ - @CheckForNull - public Integer getPrivatePort() { - return privatePort; - } - - /** - * @see #privatePort - */ - public ContainerPort withPrivatePort(Integer privatePort) { - this.privatePort = privatePort; - return this; - } - - /** - * @see #publicPort - */ - @CheckForNull - public Integer getPublicPort() { - return publicPort; - } - - /** - * @see #publicPort - */ - public ContainerPort withPublicPort(Integer publicPort) { - this.publicPort = publicPort; - return this; - } - - /** - * @see #type - */ - @CheckForNull - public String getType() { - return type; - } - - /** - * @see #type - */ - public ContainerPort withType(String type) { - this.type = type; - return this; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Device.java b/src/main/java/com/github/dockerjava/api/model/Device.java deleted file mode 100644 index f2b75e3f5..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Device.java +++ /dev/null @@ -1,141 +0,0 @@ -package com.github.dockerjava.api.model; - -import static com.google.common.base.Preconditions.checkNotNull; -import static org.apache.commons.lang.BooleanUtils.isNotTrue; -import static org.apache.commons.lang.StringUtils.isEmpty; - -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; - -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; - -import javax.annotation.Nonnull; -import java.util.HashMap; -import java.util.Map; -import java.util.StringTokenizer; - -@JsonInclude(Include.NON_NULL) -public class Device { - - @JsonProperty("CgroupPermissions") - private String cGroupPermissions = ""; - - @JsonProperty("PathOnHost") - private String pathOnHost = null; - - @JsonProperty("PathInContainer") - private String pathInContainer = null; - - public Device() { - } - - public Device(String cGroupPermissions, String pathInContainer, String pathOnHost) { - checkNotNull(cGroupPermissions, "cGroupPermissions is null"); - checkNotNull(pathInContainer, "pathInContainer is null"); - checkNotNull(pathOnHost, "pathOnHost is null"); - this.cGroupPermissions = cGroupPermissions; - this.pathInContainer = pathInContainer; - this.pathOnHost = pathOnHost; - } - - public String getcGroupPermissions() { - return cGroupPermissions; - } - - public String getPathInContainer() { - return pathInContainer; - } - - public String getPathOnHost() { - return pathOnHost; - } - - /** - * @link https://github.com/docker/docker/blob/6b4a46f28266031ce1a1315f17fb69113a06efe1/runconfig/opts/parse_test.go#L468 - */ - @Nonnull - public static Device parse(@Nonnull String deviceStr) { - String src = ""; - String dst = ""; - String permissions = "rwm"; - final String[] arr = deviceStr.trim().split(":"); - // java String.split() returns wrong length, use tokenizer instead - switch (new StringTokenizer(deviceStr, ":").countTokens()) { - case 3: { - // Mismatches docker code logic. While there is no validations after parsing, checking heregit - if (validDeviceMode(arr[2])) { - permissions = arr[2]; - } else { - throw new IllegalArgumentException("Invalid device specification: " + deviceStr); - } - } - case 2: { - if (validDeviceMode(arr[1])) { - permissions = arr[1]; - } else { - dst = arr[1]; - } - } - case 1: { - src = arr[0]; - break; - } - default: { - throw new IllegalArgumentException("Invalid device specification: " + deviceStr); - } - } - - if (isEmpty(dst)) { - dst = src; - } - - return new Device(permissions, dst, src); - } - - /** - * ValidDeviceMode checks if the mode for device is valid or not. - * Valid mode is a composition of r (read), w (write), and m (mknod). - * - * @link https://github.com/docker/docker/blob/6b4a46f28266031ce1a1315f17fb69113a06efe1/runconfig/opts/parse.go#L796 - */ - private static boolean validDeviceMode(String deviceMode) { - Map validModes = new HashMap<>(3); - validModes.put("r", true); - validModes.put("w", true); - validModes.put("m", true); - - if (isEmpty(deviceMode)) { - return false; - } - - for (char ch : deviceMode.toCharArray()) { - final String mode = String.valueOf(ch); - if (isNotTrue(validModes.get(mode))) { - return false; // wrong mode - } - validModes.put(mode, false); - } - - return true; - } - - @Override - public boolean equals(Object obj) { - if (obj instanceof Device) { - Device other = (Device) obj; - return new EqualsBuilder().append(cGroupPermissions, other.getcGroupPermissions()) - .append(pathInContainer, other.getPathInContainer()).append(pathOnHost, other.getPathOnHost()) - .isEquals(); - } else { - return super.equals(obj); - } - } - - @Override - public int hashCode() { - return new HashCodeBuilder().append(cGroupPermissions).append(pathInContainer).append(pathOnHost).toHashCode(); - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/DriverStatus.java b/src/main/java/com/github/dockerjava/api/model/DriverStatus.java deleted file mode 100644 index 3e2b5037a..000000000 --- a/src/main/java/com/github/dockerjava/api/model/DriverStatus.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.api.model; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * Created by ben on 12/12/13. - */ -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(Include.NON_NULL) -public class DriverStatus { - - @JsonProperty("Root Dir") - private String rootDir; - - @JsonProperty("Dirs") - private Integer dirs; - - public String getRootDir() { - return rootDir; - } - - public Integer getDirs() { - return dirs; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/ErrorDetail.java b/src/main/java/com/github/dockerjava/api/model/ErrorDetail.java deleted file mode 100644 index 62018a4e0..000000000 --- a/src/main/java/com/github/dockerjava/api/model/ErrorDetail.java +++ /dev/null @@ -1,15 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; - -@JsonInclude(Include.NON_NULL) -public class ErrorDetail { - @JsonProperty - private String message; - - public String getMessage() { - return message; - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/ErrorResponse.java b/src/main/java/com/github/dockerjava/api/model/ErrorResponse.java deleted file mode 100644 index 172bd5c79..000000000 --- a/src/main/java/com/github/dockerjava/api/model/ErrorResponse.java +++ /dev/null @@ -1,22 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; - -@JsonInclude(Include.NON_NULL) -public class ErrorResponse { - @JsonProperty - private ErrorDetail errorDetail; - - @JsonProperty - private String error; - - public ErrorDetail getErrorDetail() { - return errorDetail; - } - - public String getError() { - return error; - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Event.java b/src/main/java/com/github/dockerjava/api/model/Event.java deleted file mode 100644 index 202f5a350..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Event.java +++ /dev/null @@ -1,101 +0,0 @@ -package com.github.dockerjava.api.model; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; - -/** - * Representation of a Docker event. - */ -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(Include.NON_NULL) -public class Event { - private String status; - - private String id; - - private String from; - - private Long time; - - @JsonIgnoreProperties - private Node node; - - /** - * Default constructor for the deserialization. - */ - public Event() { - } - - /** - * Constructor. - * - * @param id - * Container ID - * @param status - * Status string. List of statuses is available in Docker API v.1.16 - * @param from - * Image, from which the container has been created - * @param time - * Event time The time is specified in milliseconds since January 1, 1970, 00:00:00 GMT - * @since TODO - */ - public Event(String status, String id, String from, Long time) { - this.status = status; - this.id = id; - this.from = from; - this.time = time; - } - - /** - * Status of docker image or container. List of statuses is available in Docker API v.1.16 - * - * @return Status string - */ - public String getStatus() { - return status; - } - - /** - * Get ID of docker container. - * - * @return Container ID - */ - public String getId() { - return id; - } - - /** - * Get source image of the container. - * - * @return Name of the parent container - */ - public String getFrom() { - return from; - } - - /** - * Get the event time. The time is specified in milliseconds since January 1, 1970, 00:00:00 GMT - * - * @return Event time in the specified format. - */ - public Long getTime() { - return time; - } - - /** - * Returns the node when working against docker swarm - */ - public Node getNode() { - return node; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/ExposedPort.java b/src/main/java/com/github/dockerjava/api/model/ExposedPort.java deleted file mode 100644 index ec95f6273..000000000 --- a/src/main/java/com/github/dockerjava/api/model/ExposedPort.java +++ /dev/null @@ -1,192 +0,0 @@ -package com.github.dockerjava.api.model; - -import static com.github.dockerjava.api.model.InternetProtocol.TCP; -import static com.github.dockerjava.api.model.InternetProtocol.UDP; - -import java.io.IOException; -import java.util.Map.Entry; - -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; -import com.fasterxml.jackson.databind.node.NullNode; -import com.github.dockerjava.api.model.Ports.Binding; - -/** - * Represents a container port that Docker exposes to external clients. The port is defined by its {@link #getPort() port number} and an - * {@link InternetProtocol}. It can be published by Docker by {@link Ports#bind(ExposedPort, Binding) binding} it to a host port, - * represented by a {@link Binding}. - */ -@JsonDeserialize(using = ExposedPort.Deserializer.class) -@JsonSerialize(using = ExposedPort.Serializer.class) -public class ExposedPort { - - private final InternetProtocol protocol; - - private final int port; - - /** - * Creates an {@link ExposedPort} for the given parameters. - * - * @param port - * the {@link #getPort() port number} - * @param protocol - * the {@link InternetProtocol} - */ - public ExposedPort(int port, InternetProtocol protocol) { - this.port = port; - this.protocol = protocol; - } - - /** - * Creates an {@link ExposedPort} for the given {@link #getPort() port number} and {@link InternetProtocol#DEFAULT}. - * - * @param port - * the {@link #getPort() port number} - */ - public ExposedPort(int port) { - this(port, InternetProtocol.DEFAULT); - } - - /** - * Creates an {@link ExposedPort} for the given parameters. - * - * @param scheme - * the {@link #getScheme() scheme}, tcp or udp - * @param port - * the {@link #getPort() port number} - * @deprecated use {@link #ExposedPort(int, InternetProtocol)} - */ - @Deprecated - public ExposedPort(String scheme, int port) { - this(port, InternetProtocol.valueOf(scheme)); - } - - /** - * @return the {@link InternetProtocol} of the {@link #getPort() port} that the container exposes - */ - public InternetProtocol getProtocol() { - return protocol; - } - - /** - * @return the scheme (internet protocol), tcp or udp - * @deprecated use {@link #getProtocol()} - */ - @Deprecated - public String getScheme() { - return protocol.toString(); - } - - /** @return the port number that the container exposes */ - public int getPort() { - return port; - } - - /** - * Creates an {@link ExposedPort} for {@link InternetProtocol#TCP}. This is a shortcut for - * new ExposedPort(port, {@link InternetProtocol#TCP}) - */ - public static ExposedPort tcp(int port) { - return new ExposedPort(port, TCP); - } - - /** - * Creates an {@link ExposedPort} for {@link InternetProtocol#UDP}. This is a shortcut for - * new ExposedPort(port, {@link InternetProtocol#UDP}) - */ - public static ExposedPort udp(int port) { - return new ExposedPort(port, UDP); - } - - /** - * Parses a textual port specification (as used by the Docker CLI) to an {@link ExposedPort}. - * - * @param serialized - * the specification, e.g. 80/tcp - * @return an {@link ExposedPort} matching the specification - * @throws IllegalArgumentException - * if the specification cannot be parsed - */ - public static ExposedPort parse(String serialized) throws IllegalArgumentException { - try { - String[] parts = serialized.split("/"); - switch (parts.length) { - case 1: - return new ExposedPort(Integer.parseInt(parts[0])); - case 2: - return new ExposedPort(Integer.parseInt(parts[0]), InternetProtocol.parse(parts[1])); - default: - throw new IllegalArgumentException(); - } - } catch (Exception e) { - throw new IllegalArgumentException("Error parsing ExposedPort '" + serialized + "'"); - } - } - - /** - * Returns a string representation of this {@link ExposedPort} suitable for inclusion in a JSON message. The format is - * port/protocol, like the argument in {@link #parse(String)}. - * - * @return a string representation of this {@link ExposedPort} - */ - @Override - public String toString() { - return port + "/" + protocol.toString(); - } - - @Override - public boolean equals(Object obj) { - if (obj instanceof ExposedPort) { - ExposedPort other = (ExposedPort) obj; - return new EqualsBuilder().append(protocol, other.getProtocol()).append(port, other.getPort()).isEquals(); - } else { - return super.equals(obj); - } - } - - @Override - public int hashCode() { - return new HashCodeBuilder().append(protocol).append(port).toHashCode(); - } - - public static class Deserializer extends JsonDeserializer { - @Override - public ExposedPort deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - ObjectCodec oc = jsonParser.getCodec(); - JsonNode node = oc.readTree(jsonParser); - if (!node.equals(NullNode.getInstance())) { - Entry field = node.fields().next(); - return ExposedPort.parse(field.getKey()); - } else { - return null; - } - } - } - - public static class Serializer extends JsonSerializer { - - @Override - public void serialize(ExposedPort exposedPort, JsonGenerator jsonGen, SerializerProvider serProvider) - throws IOException, JsonProcessingException { - - jsonGen.writeStartObject(); - jsonGen.writeFieldName(exposedPort.toString()); - jsonGen.writeEndObject(); - } - - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/ExposedPorts.java b/src/main/java/com/github/dockerjava/api/model/ExposedPorts.java deleted file mode 100644 index 0ecb879b6..000000000 --- a/src/main/java/com/github/dockerjava/api/model/ExposedPorts.java +++ /dev/null @@ -1,76 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.Iterator; -import java.util.List; -import java.util.Map; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; -import com.fasterxml.jackson.databind.node.NullNode; - -@JsonSerialize(using = ExposedPorts.Serializer.class) -@JsonDeserialize(using = ExposedPorts.Deserializer.class) -public class ExposedPorts { - - private ExposedPort[] exposedPorts; - - public ExposedPorts(ExposedPort... exposedPorts) { - this.exposedPorts = exposedPorts; - } - - public ExposedPorts(List exposedPorts) { - this.exposedPorts = exposedPorts.toArray(new ExposedPort[exposedPorts.size()]); - } - - public ExposedPort[] getExposedPorts() { - return exposedPorts; - } - - public static class Serializer extends JsonSerializer { - - @Override - public void serialize(ExposedPorts exposedPorts, JsonGenerator jsonGen, SerializerProvider serProvider) - throws IOException, JsonProcessingException { - - jsonGen.writeStartObject(); - for (ExposedPort exposedPort : exposedPorts.getExposedPorts()) { - jsonGen.writeFieldName(exposedPort.toString()); - jsonGen.writeStartObject(); - jsonGen.writeEndObject(); - } - jsonGen.writeEndObject(); - } - - } - - public static class Deserializer extends JsonDeserializer { - @Override - public ExposedPorts deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - - List exposedPorts = new ArrayList(); - ObjectCodec oc = jsonParser.getCodec(); - JsonNode node = oc.readTree(jsonParser); - for (Iterator> it = node.fields(); it.hasNext();) { - - Map.Entry field = it.next(); - if (!field.getValue().equals(NullNode.getInstance())) { - exposedPorts.add(ExposedPort.parse(field.getKey())); - } - } - return new ExposedPorts(exposedPorts.toArray(new ExposedPort[0])); - } - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/Frame.java b/src/main/java/com/github/dockerjava/api/model/Frame.java deleted file mode 100644 index 0ad16e8c6..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Frame.java +++ /dev/null @@ -1,52 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.util.Arrays; - -/** - * Represents a logging frame. - */ -public class Frame { - private final StreamType streamType; - - private final byte[] payload; - - public Frame(StreamType streamType, byte[] payload) { - this.streamType = streamType; - this.payload = payload; - } - - public StreamType getStreamType() { - return streamType; - } - - public byte[] getPayload() { - return payload; - } - - @Override - public String toString() { - return String.format("%s: %s", streamType, new String(payload).trim()); - } - - // CHECKSTYLE:OFF - @Override - public boolean equals(Object o) { - if (this == o) - return true; - if (o == null || getClass() != o.getClass()) - return false; - - Frame frame = (Frame) o; - - return streamType == frame.streamType && Arrays.equals(payload, frame.payload); - - } - - @Override - public int hashCode() { - int result = streamType.hashCode(); - result = 31 * result + Arrays.hashCode(payload); - return result; - } - // CHECKSTYLE:ON -} diff --git a/src/main/java/com/github/dockerjava/api/model/HostConfig.java b/src/main/java/com/github/dockerjava/api/model/HostConfig.java deleted file mode 100644 index 5dada65cc..000000000 --- a/src/main/java/com/github/dockerjava/api/model/HostConfig.java +++ /dev/null @@ -1,811 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.core.RemoteApiVersion; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -import javax.annotation.CheckForNull; -import java.util.Arrays; -import java.util.List; - -/** - * Used in `/containers/create`, and in inspect container. - * TODO exclude usage for 2 different models. - */ -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(Include.NON_NULL) -public class HostConfig { - - private static final List PREDEFINED_NETWORKS = Arrays.asList("bridge", "host", "none"); - - @JsonProperty("Binds") - private Binds binds; - - @JsonProperty("BlkioWeight") - private Integer blkioWeight; - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("BlkioWeightDevice") - private List blkioWeightDevice; - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("BlkioDeviceReadBps") - private List blkioDeviceReadBps; - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("BlkioDeviceReadIOps") - private List blkioDeviceReadIOps; - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("BlkioDeviceWriteBps") - private List blkioDeviceWriteBps; - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("BlkioDeviceWriteIOps") - private List blkioDeviceWriteIOps; - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_20} - */ - @JsonProperty("MemorySwappiness") - private Integer memorySwappiness; - - @JsonProperty("CapAdd") - private Capability[] capAdd; - - @JsonProperty("CapDrop") - private Capability[] capDrop; - - @JsonProperty("ContainerIDFile") - private String containerIDFile; - - @JsonProperty("CpuPeriod") - private Integer cpuPeriod; - - @JsonProperty("CpuShares") - private Integer cpuShares; - - /** - * @since ~{@link RemoteApiVersion#VERSION_1_20} - */ - @JsonProperty("CpuQuota") - private Integer cpuQuota; - - @JsonProperty("CpusetCpus") - private String cpusetCpus; - - @JsonProperty("CpusetMems") - private String cpusetMems; - - @JsonProperty("Devices") - private Device[] devices; - - @JsonProperty("Dns") - private String[] dns; - - @JsonProperty("DnsSearch") - private String[] dnsSearch; - - @JsonProperty("ExtraHosts") - private String[] extraHosts; - - @JsonProperty("Links") - private Links links; - - @JsonProperty("LogConfig") - private LogConfig logConfig; - - @JsonProperty("LxcConf") - private LxcConf[] lxcConf; - - @JsonProperty("Memory") - private Long memory; - - @JsonProperty("MemorySwap") - private Long memorySwap; - - /** - * @since {@link RemoteApiVersion#VERSION_1_21} - */ - @JsonProperty("MemoryReservation") - private Long memoryReservation; - - /** - * @since {@link RemoteApiVersion#VERSION_1_21} - */ - @JsonProperty("KernelMemory") - private Long kernelMemory; - - @JsonProperty("NetworkMode") - private String networkMode; - - @JsonProperty("OomKillDisable") - private Boolean oomKillDisable; - - /** - * @since {@link RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("OomScoreAdj") - private Boolean oomScoreAdj; - - @JsonProperty("PortBindings") - private Ports portBindings; - - @JsonProperty("Privileged") - private Boolean privileged; - - @JsonProperty("PublishAllPorts") - private Boolean publishAllPorts; - - @JsonProperty("ReadonlyRootfs") - private Boolean readonlyRootfs; - - @JsonProperty("RestartPolicy") - private RestartPolicy restartPolicy; - - @JsonProperty("Ulimits") - private Ulimit[] ulimits; - - @JsonProperty("VolumesFrom") - private VolumesFrom[] volumesFrom; - - @JsonProperty("PidMode") - private String pidMode; - - /** - * @since {@link RemoteApiVersion#VERSION_1_20} - */ - @JsonProperty("SecurityOpt") - private List securityOpts; - - /** - * @since {@link RemoteApiVersion#VERSION_1_20} - */ - @JsonProperty("CgroupParent") - private String cgroupParent; - - /** - * @since {@link RemoteApiVersion#VERSION_1_21} - */ - @JsonProperty("VolumeDriver") - private String volumeDriver; - - /** - * @since {@link RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("ShmSize") - private Long shmSize; - - - @JsonIgnore - public Bind[] getBinds() { - return (binds == null) ? new Bind[0] : binds.getBinds(); - } - - public Integer getBlkioWeight() { - return blkioWeight; - } - - public Capability[] getCapAdd() { - return capAdd; - } - - public Capability[] getCapDrop() { - return capDrop; - } - - public String getContainerIDFile() { - return containerIDFile; - } - - public Integer getCpuPeriod() { - return cpuPeriod; - } - - public Integer getCpuShares() { - return cpuShares; - } - - public String getCpusetCpus() { - return cpusetCpus; - } - - public String getCpusetMems() { - return cpusetMems; - } - - public Device[] getDevices() { - return devices; - } - - public String[] getDns() { - return dns; - } - - public String[] getDnsSearch() { - return dnsSearch; - } - - public String[] getExtraHosts() { - return extraHosts; - } - - @JsonIgnore - public Link[] getLinks() { - return (links == null) ? new Link[0] : links.getLinks(); - } - - @JsonIgnore - public LogConfig getLogConfig() { - return (logConfig == null) ? new LogConfig() : logConfig; - } - - public LxcConf[] getLxcConf() { - return lxcConf; - } - - public Long getMemory() { - return memory; - } - - public Long getMemorySwap() { - return memorySwap; - } - - public String getNetworkMode() { - return networkMode; - } - - public Ports getPortBindings() { - return portBindings; - } - - public RestartPolicy getRestartPolicy() { - return restartPolicy; - } - - public Ulimit[] getUlimits() { - return ulimits; - } - - public VolumesFrom[] getVolumesFrom() { - return volumesFrom; - } - - @CheckForNull - public String getPidMode() { - return pidMode; - } - - /** - * @see #blkioDeviceReadBps - */ - @CheckForNull - public List getBlkioDeviceReadBps() { - return blkioDeviceReadBps; - } - - /** - * @see #blkioDeviceReadIOps - */ - @CheckForNull - public List getBlkioDeviceReadIOps() { - return blkioDeviceReadIOps; - } - - /** - * @see #blkioDeviceWriteBps - */ - @CheckForNull - public List getBlkioDeviceWriteBps() { - return blkioDeviceWriteBps; - } - - /** - * @see #blkioDeviceWriteIOps - */ - @CheckForNull - public List getBlkioDeviceWriteIOps() { - return blkioDeviceWriteIOps; - } - - /** - * @see #blkioWeightDevice - */ - @CheckForNull - public List getBlkioWeightDevice() { - return blkioWeightDevice; - } - - /** - * @see #oomScoreAdj - */ - @CheckForNull - public Boolean getOomScoreAdj() { - return oomScoreAdj; - } - - /** - * @see #cpuQuota - */ - @CheckForNull - public Integer getCpuQuota() { - return cpuQuota; - } - - /** - * @see #kernelMemory - */ - @CheckForNull - public Long getKernelMemory() { - return kernelMemory; - } - - /** - * @see #memoryReservation - */ - @CheckForNull - public Long getMemoryReservation() { - return memoryReservation; - } - - /** - * @see #memorySwappiness - */ - @CheckForNull - public Integer getMemorySwappiness() { - return memorySwappiness; - } - - /** - * @see #oomKillDisable - */ - @CheckForNull - public Boolean getOomKillDisable() { - return oomKillDisable; - } - - /** - * @see #securityOpts - */ - @CheckForNull - public List getSecurityOpts() { - return securityOpts; - } - - /** - * @see #cgroupParent - */ - @CheckForNull - public String getCgroupParent() { - return cgroupParent; - } - - /** - * @see #shmSize - */ - @CheckForNull - public Long getShmSize() { - return shmSize; - } - - /** - * @see #volumeDriver - */ - @CheckForNull - public String getVolumeDriver() { - return volumeDriver; - } - - /** - * Parse the network mode as specified at - * {@see https://github.com/docker/engine-api/blob/master/types/container/hostconfig_unix.go} - */ - @JsonIgnore - public boolean isUserDefinedNetwork() { - return networkMode != null && !PREDEFINED_NETWORKS.contains(networkMode) && !networkMode.startsWith("container:"); - } - - @JsonIgnore - public void setBinds(Bind... binds) { - this.binds = new Binds(binds); - } - - @JsonIgnore - public void setLinks(Link... links) { - this.links = new Links(links); - } - - // auto-generated builder setters - /** - * @see #binds - */ - public HostConfig withBinds(Binds binds) { - this.binds = binds; - return this; - } - - /** - * @see #blkioDeviceReadBps - */ - public HostConfig withBlkioDeviceReadBps(List blkioDeviceReadBps) { - this.blkioDeviceReadBps = blkioDeviceReadBps; - return this; - } - - /** - * @see #blkioDeviceReadIOps - */ - public HostConfig withBlkioDeviceReadIOps(List blkioDeviceReadIOps) { - this.blkioDeviceReadIOps = blkioDeviceReadIOps; - return this; - } - - /** - * @see #blkioDeviceWriteBps - */ - public HostConfig withBlkioDeviceWriteBps(List blkioDeviceWriteBps) { - this.blkioDeviceWriteBps = blkioDeviceWriteBps; - return this; - } - - /** - * @see #blkioDeviceWriteIOps - */ - public HostConfig withBlkioDeviceWriteIOps(List blkioDeviceWriteIOps) { - this.blkioDeviceWriteIOps = blkioDeviceWriteIOps; - return this; - } - - /** - * @see #blkioWeight - */ - public HostConfig withBlkioWeight(Integer blkioWeight) { - this.blkioWeight = blkioWeight; - return this; - } - - /** - * @see #blkioWeightDevice - */ - public HostConfig withBlkioWeightDevice(List blkioWeightDevice) { - this.blkioWeightDevice = blkioWeightDevice; - return this; - } - - /** - * @see #capAdd - */ - public HostConfig withCapAdd(Capability[] capAdd) { - this.capAdd = capAdd; - return this; - } - - /** - * @see #capDrop - */ - public HostConfig withCapDrop(Capability[] capDrop) { - this.capDrop = capDrop; - return this; - } - - /** - * @see #cgroupParent - */ - public HostConfig withCgroupParent(String cgroupParent) { - this.cgroupParent = cgroupParent; - return this; - } - - /** - * @see #containerIDFile - */ - public HostConfig withContainerIDFile(String containerIDFile) { - this.containerIDFile = containerIDFile; - return this; - } - - /** - * @see #cpuPeriod - */ - public HostConfig withCpuPeriod(Integer cpuPeriod) { - this.cpuPeriod = cpuPeriod; - return this; - } - - /** - * @see #cpuQuota - */ - public HostConfig withCpuQuota(Integer cpuQuota) { - this.cpuQuota = cpuQuota; - return this; - } - - /** - * @see #cpusetCpus - */ - public HostConfig withCpusetCpus(String cpusetCpus) { - this.cpusetCpus = cpusetCpus; - return this; - } - - /** - * @see #cpusetMems - */ - public HostConfig withCpusetMems(String cpusetMems) { - this.cpusetMems = cpusetMems; - return this; - } - - /** - * @see #cpuShares - */ - public HostConfig withCpuShares(Integer cpuShares) { - this.cpuShares = cpuShares; - return this; - } - - /** - * @see #devices - */ - public HostConfig withDevices(Device[] devices) { - this.devices = devices; - return this; - } - - /** - * @see #dns - */ - public HostConfig withDns(String[] dns) { - this.dns = dns; - return this; - } - - /** - * @see #dnsSearch - */ - public HostConfig withDnsSearch(String[] dnsSearch) { - this.dnsSearch = dnsSearch; - return this; - } - - /** - * @see #extraHosts - */ - public HostConfig withExtraHosts(String[] extraHosts) { - this.extraHosts = extraHosts; - return this; - } - - /** - * @see #kernelMemory - */ - public HostConfig withKernelMemory(Long kernelMemory) { - this.kernelMemory = kernelMemory; - return this; - } - - /** - * @see #links - */ - public HostConfig withLinks(Links links) { - this.links = links; - return this; - } - - /** - * @see #logConfig - */ - public HostConfig withLogConfig(LogConfig logConfig) { - this.logConfig = logConfig; - return this; - } - - /** - * @see #lxcConf - */ - public HostConfig withLxcConf(LxcConf[] lxcConf) { - this.lxcConf = lxcConf; - return this; - } - - /** - * @see #memory - */ - public HostConfig withMemory(Long memory) { - this.memory = memory; - return this; - } - - /** - * @see #memoryReservation - */ - public HostConfig withMemoryReservation(Long memoryReservation) { - this.memoryReservation = memoryReservation; - return this; - } - - /** - * @see #memorySwap - */ - public HostConfig withMemorySwap(Long memorySwap) { - this.memorySwap = memorySwap; - return this; - } - - /** - * @see #memorySwappiness - */ - public HostConfig withMemorySwappiness(Integer memorySwappiness) { - this.memorySwappiness = memorySwappiness; - return this; - } - - /** - * @see #networkMode - */ - public HostConfig withNetworkMode(String networkMode) { - this.networkMode = networkMode; - return this; - } - - /** - * @see #oomKillDisable - */ - public HostConfig withOomKillDisable(Boolean oomKillDisable) { - this.oomKillDisable = oomKillDisable; - return this; - } - - /** - * @see #oomScoreAdj - */ - public HostConfig withOomScoreAdj(Boolean oomScoreAdj) { - this.oomScoreAdj = oomScoreAdj; - return this; - } - - /** - * @see #pidMode - */ - public HostConfig withPidMode(String pidMode) { - this.pidMode = pidMode; - return this; - } - - /** - * @see #portBindings - */ - public HostConfig withPortBindings(Ports portBindings) { - this.portBindings = portBindings; - return this; - } - - /** - * @see #privileged - */ - @CheckForNull - public Boolean getPrivileged() { - return privileged; - } - - /** - * @see #privileged - */ - public HostConfig withPrivileged(Boolean privileged) { - this.privileged = privileged; - return this; - } - - /** - * @see #publishAllPorts - */ - @CheckForNull - public Boolean getPublishAllPorts() { - return publishAllPorts; - } - - /** - * @see #publishAllPorts - */ - public HostConfig withPublishAllPorts(Boolean publishAllPorts) { - this.publishAllPorts = publishAllPorts; - return this; - } - - /** - * @see #readonlyRootfs - */ - @CheckForNull - public Boolean getReadonlyRootfs() { - return readonlyRootfs; - } - - /** - * @see #readonlyRootfs - */ - public HostConfig withReadonlyRootfs(Boolean readonlyRootfs) { - this.readonlyRootfs = readonlyRootfs; - return this; - } - - /** - * @see #restartPolicy - */ - public HostConfig withRestartPolicy(RestartPolicy restartPolicy) { - this.restartPolicy = restartPolicy; - return this; - } - - /** - * @see #securityOpts - */ - public HostConfig withSecurityOpts(List securityOpts) { - this.securityOpts = securityOpts; - return this; - } - - /** - * @see #shmSize - */ - public HostConfig withShmSize(Long shmSize) { - this.shmSize = shmSize; - return this; - } - - /** - * @see #ulimits - */ - public HostConfig withUlimits(Ulimit[] ulimits) { - this.ulimits = ulimits; - return this; - } - - /** - * @see #volumeDriver - */ - public HostConfig withVolumeDriver(String volumeDriver) { - this.volumeDriver = volumeDriver; - return this; - } - - /** - * @see #volumesFrom - */ - public HostConfig withVolumesFrom(VolumesFrom[] volumesFrom) { - this.volumesFrom = volumesFrom; - return this; - } - // end of auto-generated - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Identifier.java b/src/main/java/com/github/dockerjava/api/model/Identifier.java deleted file mode 100644 index 1051d857f..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Identifier.java +++ /dev/null @@ -1,56 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.google.common.base.Objects; -import com.google.common.base.Optional; - -/** - * Created by magnayn on 22/07/2014. - */ -public class Identifier { - public final Repository repository; - - public final Optional tag; - - public Identifier(Repository repository, String tag) { - this.repository = repository; - - if (tag == null) { - this.tag = Optional.absent(); - } else { - this.tag = Optional.of(tag); - } - } - - /** - * Return an identifier that correctly splits up the repository and tag. There can be > 1 ":" fred/jim --> fred/jim, [] - * fred/jim:123 --> fred/jim, 123 fred:123/jim:123 --> fred:123/jim, 123 - * - * - * @param identifier - * as a string - * @return parsed identifier. - */ - public static Identifier fromCompoundString(String identifier) { - String[] parts = identifier.split("/"); - if (parts.length != 2) { - String[] rhs = identifier.split(":"); - if (rhs.length != 2) { - return new Identifier(new Repository(identifier), null); - } else { - return new Identifier(new Repository(rhs[0]), rhs[1]); - } - } - - String[] rhs = parts[1].split(":"); - if (rhs.length != 2) { - return new Identifier(new Repository(identifier), null); - } - - return new Identifier(new Repository(parts[0] + "/" + rhs[0]), rhs[1]); - } - - @Override - public String toString() { - return Objects.toStringHelper(this).add("repository", repository).add("tag", tag).toString(); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Image.java b/src/main/java/com/github/dockerjava/api/model/Image.java deleted file mode 100644 index 496db275e..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Image.java +++ /dev/null @@ -1,65 +0,0 @@ -package com.github.dockerjava.api.model; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * - * @author Konstantin Pelykh (kpelykh@gmail.com) - * - */ -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(Include.NON_NULL) -public class Image { - - @JsonProperty("Created") - private Long created; - - @JsonProperty("Id") - private String id; - - @JsonProperty("ParentId") - private String parentId; - - @JsonProperty("RepoTags") - private String[] repoTags; - - @JsonProperty("Size") - private Long size; - - @JsonProperty("VirtualSize") - private Long virtualSize; - - public String getId() { - return id; - } - - public String[] getRepoTags() { - return repoTags; - } - - public String getParentId() { - return parentId; - } - - public Long getCreated() { - return created; - } - - public Long getSize() { - return size; - } - - public Long getVirtualSize() { - return virtualSize; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/InfoRegistryConfig.java b/src/main/java/com/github/dockerjava/api/model/InfoRegistryConfig.java deleted file mode 100644 index 956aaebe3..000000000 --- a/src/main/java/com/github/dockerjava/api/model/InfoRegistryConfig.java +++ /dev/null @@ -1,189 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -import javax.annotation.CheckForNull; -import java.util.List; -import java.util.Map; - -/** - * @since ~{@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_20} - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public final class InfoRegistryConfig { - @JsonProperty("IndexConfigs") - private Map indexConfigs; - - @JsonProperty("InsecureRegistryCIDRs") - private List insecureRegistryCIDRs; - - /** - * //FIXME unknown field - */ - @JsonProperty("Mirrors") - private Object mirrors; - - /** - * @see #indexConfigs - */ - @CheckForNull - public Map getIndexConfigs() { - return indexConfigs; - } - - /** - * @see #indexConfigs - */ - public InfoRegistryConfig withIndexConfigs(Map indexConfigs) { - this.indexConfigs = indexConfigs; - return this; - } - - /** - * @see #insecureRegistryCIDRs - */ - @CheckForNull - public List getInsecureRegistryCIDRs() { - return insecureRegistryCIDRs; - } - - /** - * @see #insecureRegistryCIDRs - */ - public InfoRegistryConfig withInsecureRegistryCIDRs(List insecureRegistryCIDRs) { - this.insecureRegistryCIDRs = insecureRegistryCIDRs; - return this; - } - - /** - * @see #mirrors - */ - @CheckForNull - public Object getMirrors() { - return mirrors; - } - - /** - * @see #mirrors - */ - public InfoRegistryConfig withMirrors(Object mirrors) { - this.mirrors = mirrors; - return this; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } - - /** - * @since ~{@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_20} - */ - @JsonIgnoreProperties(ignoreUnknown = true) - public static final class IndexConfig { - @JsonProperty("Mirrors") - private List mirrors; - - @JsonProperty("Name") - private String name; - - @JsonProperty("Official") - private Boolean official; - - @JsonProperty("Secure") - private Boolean secure; - - /** - * @see #mirrors - */ - @CheckForNull - public List getMirrors() { - return mirrors; - } - - /** - * @see #mirrors - */ - public IndexConfig withMirrors(List mirrors) { - this.mirrors = mirrors; - return this; - } - - /** - * @see #name - */ - @CheckForNull - public String getName() { - return name; - } - - /** - * @see #name - */ - public IndexConfig withName(String name) { - this.name = name; - return this; - } - - /** - * @see #official - */ - @CheckForNull - public Boolean getOfficial() { - return official; - } - - /** - * @see #official - */ - public IndexConfig withOfficial(Boolean official) { - this.official = official; - return this; - } - - /** - * @see #secure - */ - @CheckForNull - public Boolean getSecure() { - return secure; - } - - /** - * @see #secure - */ - public IndexConfig withSecure(Boolean secure) { - this.secure = secure; - return this; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Links.java b/src/main/java/com/github/dockerjava/api/model/Links.java deleted file mode 100644 index 2d678b3a3..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Links.java +++ /dev/null @@ -1,72 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.Iterator; -import java.util.List; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; -import com.fasterxml.jackson.databind.node.NullNode; - -@JsonSerialize(using = Links.Serializer.class) -@JsonDeserialize(using = Links.Deserializer.class) -public class Links { - - private final Link[] links; - - public Links(final Link... links) { - this.links = links; - } - - public Links(final List links) { - this.links = links.toArray(new Link[links.size()]); - } - - public Link[] getLinks() { - return links; - } - - public static class Serializer extends JsonSerializer { - - @Override - public void serialize(final Links links, final JsonGenerator jsonGen, final SerializerProvider serProvider) - throws IOException, JsonProcessingException { - jsonGen.writeStartArray(); - for (final Link link : links.getLinks()) { - jsonGen.writeString(link.toString()); - } - jsonGen.writeEndArray(); - } - - } - - public static class Deserializer extends JsonDeserializer { - - @Override - public Links deserialize(final JsonParser jsonParser, final DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - final List binds = new ArrayList(); - final ObjectCodec oc = jsonParser.getCodec(); - final JsonNode node = oc.readTree(jsonParser); - for (final Iterator it = node.elements(); it.hasNext();) { - - final JsonNode element = it.next(); - if (!element.equals(NullNode.getInstance())) { - binds.add(Link.parse(element.asText())); - } - } - return new Links(binds.toArray(new Link[0])); - } - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/LogConfig.java b/src/main/java/com/github/dockerjava/api/model/LogConfig.java deleted file mode 100644 index f4fd0e958..000000000 --- a/src/main/java/com/github/dockerjava/api/model/LogConfig.java +++ /dev/null @@ -1,118 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.io.IOException; -import java.util.Map; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; - -/** - * Log driver to use for a created/running container. The available types are: - * - * json-file (default) syslog journald none - * - * If a driver is specified that is NOT supported,docker will default to null. If configs are supplied that are not supported by the type - * docker will ignore them. In most cases setting the config option to null will suffice. Consult the docker remote API for a more detailed - * and up-to-date explanation of the available types and their options. - */ -public class LogConfig { - - @JsonProperty("Type") - public LoggingType type = null; - - @JsonProperty("Config") - public Map config; - - public LogConfig(LoggingType type, Map config) { - this.type = type; - this.config = config; - } - - public LogConfig(LoggingType type) { - this(type, null); - } - - public LogConfig() { - } - - public LoggingType getType() { - return type; - } - - public LogConfig setType(LoggingType type) { - this.type = type; - return this; - } - - @JsonIgnore - public Map getConfig() { - return config; - } - - @JsonIgnore - public LogConfig setConfig(Map config) { - this.config = config; - return this; - } - - @JsonDeserialize(using = LoggingType.Deserializer.class) - @JsonSerialize(using = LoggingType.Serializer.class) - public enum LoggingType { - DEFAULT("json-file"), - JSON_FILE("json-file"), - NONE("none"), - SYSLOG("syslog"), - JOURNALD("journald"), - GELF("gelf"), - FLUENTD("fluentd"), - AWSLOGS("awslogs"), - SPLUNK("splunk"); - - private String type; - - LoggingType(String type) { - this.type = type; - } - - public String getType() { - return type; - } - - public static final class Serializer extends JsonSerializer { - @Override - public void serialize(LoggingType value, JsonGenerator jgen, SerializerProvider provider) - throws IOException, JsonProcessingException { - jgen.writeString(value.getType()); - } - } - - public static final class Deserializer extends JsonDeserializer { - @Override - public LoggingType deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - - ObjectCodec oc = jsonParser.getCodec(); - JsonNode node = oc.readTree(jsonParser); - - for (LoggingType loggingType : values()) { - if (loggingType.getType().equals(node.asText())) { - return loggingType; - } - } - - throw new IllegalArgumentException("No enum constant " + LoggingType.class + "." + node.asText()); - } - } - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/LxcConf.java b/src/main/java/com/github/dockerjava/api/model/LxcConf.java deleted file mode 100644 index aeb854285..000000000 --- a/src/main/java/com/github/dockerjava/api/model/LxcConf.java +++ /dev/null @@ -1,41 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; - -@JsonInclude(Include.NON_NULL) -public class LxcConf { - @JsonProperty("Key") - public String key; - - @JsonProperty("Value") - public String value; - - public LxcConf(String key, String value) { - this.key = key; - this.value = value; - } - - public LxcConf() { - } - - public String getKey() { - return key; - } - - public LxcConf setKey(String key) { - this.key = key; - return this; - } - - public String getValue() { - return value; - } - - public LxcConf setValue(String value) { - this.value = value; - return this; - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/Network.java b/src/main/java/com/github/dockerjava/api/model/Network.java deleted file mode 100644 index 6f4114914..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Network.java +++ /dev/null @@ -1,187 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonProperty; -import org.apache.commons.lang.builder.ToStringBuilder; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.Map; - -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(JsonInclude.Include.NON_NULL) -public class Network { - - @JsonProperty("Id") - private String id; - - @JsonProperty("Name") - private String name; - - @JsonProperty("Scope") - private String scope; - - @JsonProperty("Driver") - private String driver; - - @JsonProperty("IPAM") - private Ipam ipam; - - @JsonProperty("Containers") - private Map containers; - - @JsonProperty("Options") - private Map options; - - public String getId() { - return id; - } - - public String getName() { - return name; - } - - public String getScope() { - return scope; - } - - public String getDriver() { - return driver; - } - - public Ipam getIpam() { - return ipam; - } - - public Map getContainers() { - return containers; - } - - public Map getOptions() { - return options; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @JsonIgnoreProperties(ignoreUnknown = true) - public static class ContainerNetworkConfig { - - @JsonProperty("EndpointID") - private String endpointId; - - @JsonProperty("MacAddress") - private String macAddress; - - @JsonProperty("IPv4Address") - private String ipv4Address; - - @JsonProperty("IPv6Address") - private String ipv6Address; - - public String getEndpointId() { - return endpointId; - } - - public String getMacAddress() { - return macAddress; - } - - public String getIpv4Address() { - return ipv4Address; - } - - public String getIpv6Address() { - return ipv6Address; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - } - - @JsonIgnoreProperties(ignoreUnknown = true) - public static class Ipam { - - @JsonProperty("Driver") - private String driver; - - @JsonProperty("Config") - private List config = new ArrayList<>(); - - @JsonProperty("Options") - private Map options = null; - - public String getDriver() { - return driver; - } - - public Map getOptions() { - return options; - } - - public List getConfig() { - return config; - } - - public Ipam withConfig(List ipamConfigs) { - config = ipamConfigs; - return this; - } - - public Ipam withConfig(Config... ipamConfigs) { - config = Arrays.asList(ipamConfigs); - return this; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @JsonIgnoreProperties(ignoreUnknown = true) - public static class Config { - - @JsonProperty("Subnet") - private String subnet; - - @JsonProperty("IPRange") - private String ipRange; - - @JsonProperty("Gateway") - private String gateway; - - public String getSubnet() { - return subnet; - } - - public String getIpRange() { - return ipRange; - } - - public String getGateway() { - return gateway; - } - - public Config withSubnet(String subnet) { - this.subnet = subnet; - return this; - } - - public Config withIpRange(String ipRange) { - this.ipRange = ipRange; - return this; - } - - public Config withGateway(String gateway) { - this.gateway = gateway; - return this; - } - } - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Node.java b/src/main/java/com/github/dockerjava/api/model/Node.java deleted file mode 100644 index 9e8580e0a..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Node.java +++ /dev/null @@ -1,40 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * A node as returned by the /events API, for instance, when Swarm is used. - */ -@JsonInclude(Include.NON_NULL) -public class Node { - - @JsonProperty("Name") - private String name; - - @JsonProperty("Id") - private String id; - - @JsonProperty("Addr") - private String addr; - - @JsonProperty("Ip") - private String ip; - - public String getName() { - return name; - } - - public String getId() { - return id; - } - - public String getAddr() { - return addr; - } - - public String getIp() { - return ip; - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/PortBinding.java b/src/main/java/com/github/dockerjava/api/model/PortBinding.java deleted file mode 100644 index e163272d7..000000000 --- a/src/main/java/com/github/dockerjava/api/model/PortBinding.java +++ /dev/null @@ -1,77 +0,0 @@ -package com.github.dockerjava.api.model; - -import org.apache.commons.lang.StringUtils; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; - -import com.github.dockerjava.api.model.Ports.Binding; - -/** - * In a {@link PortBinding}, a network socket on the Docker host, expressed as a {@link Binding}, is bound to an {@link ExposedPort} of a - * container. A {@link PortBinding} corresponds to the --publish (-p) option of the docker run (and - * similar) CLI command for adding port bindings to a container. - *

- * Note: This is an abstraction used for creating new port bindings. It is not to be confused with the abstraction used for querying - * existing port bindings from a container configuration in {@link NetworkSettings#getPorts()} and {@link HostConfig#getPortBindings()}. In - * that context, a Map<ExposedPort, Binding[]> is used. - */ -public class PortBinding { - private final Binding binding; - - private final ExposedPort exposedPort; - - public PortBinding(Binding binding, ExposedPort exposedPort) { - this.binding = binding; - this.exposedPort = exposedPort; - } - - public Binding getBinding() { - return binding; - } - - public ExposedPort getExposedPort() { - return exposedPort; - } - - public static PortBinding parse(String serialized) throws IllegalArgumentException { - try { - String[] parts = StringUtils.splitByWholeSeparator(serialized, ":"); - switch (parts.length) { - case 3: - // 127.0.0.1:80:8080/tcp - return createFromSubstrings(parts[0] + ":" + parts[1], parts[2]); - case 2: - // 80:8080 // 127.0.0.1::8080 - return createFromSubstrings(parts[0], parts[1]); - case 1: - // 8080 - return createFromSubstrings("", parts[0]); - default: - throw new IllegalArgumentException(); - } - } catch (Exception e) { - throw new IllegalArgumentException("Error parsing PortBinding '" + serialized + "'", e); - } - } - - private static PortBinding createFromSubstrings(String binding, String exposedPort) throws IllegalArgumentException { - return new PortBinding(Binding.parse(binding), ExposedPort.parse(exposedPort)); - } - - @Override - public boolean equals(Object obj) { - if (obj instanceof PortBinding) { - PortBinding other = (PortBinding) obj; - return new EqualsBuilder().append(binding, other.getBinding()).append(exposedPort, other.getExposedPort()) - .isEquals(); - } else { - return super.equals(obj); - } - } - - @Override - public int hashCode() { - return new HashCodeBuilder().append(binding).append(exposedPort).toHashCode(); - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/Ports.java b/src/main/java/com/github/dockerjava/api/model/Ports.java deleted file mode 100644 index 904a31c4e..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Ports.java +++ /dev/null @@ -1,326 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; -import com.fasterxml.jackson.databind.node.NullNode; -import org.apache.commons.lang.ArrayUtils; -import org.apache.commons.lang.builder.EqualsBuilder; - -import java.io.IOException; -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; -import java.util.Map.Entry; - -import static org.apache.commons.lang.StringUtils.isEmpty; - -/** - * A container for port bindings, made available as a {@link Map} via its {@link #getBindings()} method. - *

- * Note: This is an abstraction used for querying existing port bindings from a container configuration. It is not to be confused with - * the {@link PortBinding} abstraction used for adding new port bindings to a container. - * - * @see HostConfig#getPortBindings() - * @see NetworkSettings#getPorts() - */ -@SuppressWarnings(value = "checkstyle:equalshashcode") -@JsonDeserialize(using = Ports.Deserializer.class) -@JsonSerialize(using = Ports.Serializer.class) -public class Ports { - - private final Map ports = new HashMap(); - - /** - * Creates a {@link Ports} object with no {@link PortBinding}s. Use {@link #bind(ExposedPort, Binding)} or {@link #add(PortBinding...)} - * to add {@link PortBinding}s. - */ - public Ports() { - } - - /** - * Creates a {@link Ports} object with an initial {@link PortBinding} for the specified {@link ExposedPort} and {@link Binding}. Use - * {@link #bind(ExposedPort, Binding)} or {@link #add(PortBinding...)} to add more {@link PortBinding}s. - */ - public Ports(ExposedPort exposedPort, Binding host) { - bind(exposedPort, host); - } - - public Ports(PortBinding... portBindings) { - add(portBindings); - } - - /** - * Adds a new {@link PortBinding} for the specified {@link ExposedPort} and {@link Binding} to the current bindings. - */ - public void bind(ExposedPort exposedPort, Binding binding) { - if (ports.containsKey(exposedPort)) { - Binding[] bindings = ports.get(exposedPort); - ports.put(exposedPort, (Binding[]) ArrayUtils.add(bindings, binding)); - } else { - if (binding == null) { - ports.put(exposedPort, null); - } else { - ports.put(exposedPort, new Binding[] {binding}); - } - } - } - - /** - * Adds the specified {@link PortBinding}(s) to the list of {@link PortBinding}s. - */ - public void add(PortBinding... portBindings) { - for (PortBinding binding : portBindings) { - bind(binding.getExposedPort(), binding.getBinding()); - } - } - - @Override - public String toString() { - return ports.toString(); - } - - /** - * Returns the port bindings in the format used by the Docker remote API, i.e. the {@link Binding}s grouped by {@link ExposedPort}. - * - * @return the port bindings as a {@link Map} that contains one or more {@link Binding}s per {@link ExposedPort}. - */ - public Map getBindings() { - return ports; - } - - // public PortBinding[] getBindingsAsArray() { - // List bindings = new ArrayList<>(); - // for(Map.Entry entry: ports.entrySet()) { - // for(Ports.Binding binding : entry.getValue()) { - // bindings.add(new PortBinding(binding, entry.getKey())); - // } - // } - // return bindings.toArray(new PortBinding[bindings.size()]); - // } - - /** - * A {@link Binding} represents a socket on the Docker host that is used in a {@link PortBinding}. It is characterized by an - * {@link #getHostIp() IP address} and a {@link #getHostPortSpec() port spec}. Both properties may be null in order to - * let Docker assign them dynamically/using defaults. - * - * @see Ports#bind(ExposedPort, Binding) - * @see ExposedPort - */ - public static class Binding { - - /** - * Creates a {@link Binding} for the given {@link #getHostPortSpec() port spec}, leaving the {@link #getHostIp() IP address} - * undefined. - * - * @see Ports#bind(ExposedPort, Binding) - * @see ExposedPort - */ - public static Binding bindPortSpec(String portSpec) { - return new Binding(null, portSpec); - } - - /** - * Creates a {@link Binding} for the given {@link #getHostIp() IP address}, leaving the {@link #getHostPortSpec() port spec} - * undefined. - */ - public static Binding bindIp(String hostIp) { - return new Binding(hostIp, null); - } - - /** - * Creates a {@link Binding} for the given {@link #getHostIp() IP address} and port number. - */ - public static Binding bindIpAndPort(String hostIp, int port) { - return new Binding(hostIp, "" + port); - } - - /** - * Creates a {@link Binding} for the given {@link #getHostIp() IP address} and port range. - */ - public static Binding bindIpAndPortRange(String hostIp, int lowPort, int highPort) { - return new Binding(hostIp, lowPort + "-" + highPort); - } - - /** - * Creates a {@link Binding} for the given port range, leaving the {@link #getHostIp() IP address} - * undefined. - */ - public static Binding bindPortRange(int lowPort, int highPort) { - return bindIpAndPortRange(null, lowPort, highPort); - } - - /** - * Creates a {@link Binding} for the given port leaving the {@link #getHostIp() IP address} - * undefined. - */ - public static Binding bindPort(int port) { - return bindIpAndPort(null, port); - } - - /** - * Creates an empty {@link Binding}. - */ - public static Binding empty() { - return new Binding(null, null); - } - - private final String hostIp; - - private final String hostPortSpec; - - /** - * Creates a {@link Binding} for the given {@link #getHostIp() host IP address} and {@link #getHostPortSpec() host port spec}. - * - * @see Ports#bind(ExposedPort, Binding) - * @see ExposedPort - */ - public Binding(String hostIp, String hostPortSpec) { - this.hostIp = isEmpty(hostIp) ? null : hostIp; - this.hostPortSpec = hostPortSpec; - } - - /** - * @return the IP address on the Docker host. May be null, in which case Docker will bind the port to all interfaces ( - * 0.0.0.0). - */ - public String getHostIp() { - return hostIp; - } - - /** - * @return the port spec for the binding on the Docker host. May reference a single port ("1234"), a port range ("1234-2345") or - * null, in which case Docker will dynamically assign a port. - */ - public String getHostPortSpec() { - return hostPortSpec; - } - - /** - * Parses a textual host and port specification (as used by the Docker CLI) to a {@link Binding}. - *

- * Legal syntax: IP|IP:portSpec|portSpec where portSpec is either a single port or a port range - * - * @param serialized - * serialized the specification, e.g. 127.0.0.1:80 - * @return a {@link Binding} matching the specification - * @throws IllegalArgumentException - * if the specification cannot be parsed - */ - public static Binding parse(String serialized) throws IllegalArgumentException { - try { - if (serialized.isEmpty()) { - return Binding.empty(); - } - - String[] parts = serialized.split(":"); - switch (parts.length) { - case 2: { - return new Binding(parts[0], parts[1]); - } - case 1: { - return parts[0].contains(".") ? Binding.bindIp(parts[0]) : Binding.bindPortSpec(parts[0]); - } - default: { - throw new IllegalArgumentException(); - } - } - } catch (Exception e) { - throw new IllegalArgumentException("Error parsing Binding '" + serialized + "'"); - } - } - - /** - * Returns a string representation of this {@link Binding} suitable for inclusion in a JSON message. The format is - * [IP:]Port, like the argument in {@link #parse(String)}. - * - * @return a string representation of this {@link Binding} - */ - @Override - public String toString() { - if (isEmpty(hostIp)) { - return hostPortSpec; - } else if (hostPortSpec == null) { - return hostIp; - } else { - return hostIp + ":" + hostPortSpec; - } - } - - @Override - public boolean equals(Object obj) { - if (obj instanceof Binding) { - Binding other = (Binding) obj; - return new EqualsBuilder().append(hostIp, other.getHostIp()).append(hostPortSpec, other.getHostPortSpec()) - .isEquals(); - } else { - return super.equals(obj); - } - } - } - - public static class Deserializer extends JsonDeserializer { - @Override - public Ports deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - - Ports out = new Ports(); - ObjectCodec oc = jsonParser.getCodec(); - JsonNode node = oc.readTree(jsonParser); - for (Iterator> it = node.fields(); it.hasNext();) { - - Map.Entry portNode = it.next(); - JsonNode bindingsArray = portNode.getValue(); - if (bindingsArray.equals(NullNode.getInstance())) { - out.bind(ExposedPort.parse(portNode.getKey()), null); - } else { - for (int i = 0; i < bindingsArray.size(); i++) { - JsonNode bindingNode = bindingsArray.get(i); - if (!bindingNode.equals(NullNode.getInstance())) { - String hostIp = bindingNode.get("HostIp").textValue(); - String hostPort = bindingNode.get("HostPort").textValue(); - out.bind(ExposedPort.parse(portNode.getKey()), new Binding(hostIp, hostPort)); - } - } - } - } - return out; - } - } - - public static class Serializer extends JsonSerializer { - - @Override - public void serialize(Ports portBindings, JsonGenerator jsonGen, SerializerProvider serProvider) - throws IOException, JsonProcessingException { - - jsonGen.writeStartObject(); - for (Entry entry : portBindings.getBindings().entrySet()) { - jsonGen.writeFieldName(entry.getKey().toString()); - if (entry.getValue() != null) { - jsonGen.writeStartArray(); - for (Binding binding : entry.getValue()) { - jsonGen.writeStartObject(); - jsonGen.writeStringField("HostIp", binding.getHostIp() == null ? "" : binding.getHostIp()); - jsonGen.writeStringField("HostPort", binding.getHostPortSpec() == null ? "" : binding.getHostPortSpec()); - jsonGen.writeEndObject(); - } - jsonGen.writeEndArray(); - } else { - jsonGen.writeNull(); - } - } - jsonGen.writeEndObject(); - } - - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/PullResponseItem.java b/src/main/java/com/github/dockerjava/api/model/PullResponseItem.java deleted file mode 100644 index 90b3ee336..000000000 --- a/src/main/java/com/github/dockerjava/api/model/PullResponseItem.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; - -/** - * Represents a pull response stream item - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class PullResponseItem extends ResponseItem { - - private static final long serialVersionUID = -2575482839766823293L; - - private static final String LEGACY_REGISTRY = "this image was pulled from a legacy registry"; - - private static final String DOWNLOADED_NEWER_IMAGE = "Downloaded newer image"; - - private static final String IMAGE_UP_TO_DATE = "Image is up to date"; - - private static final String DOWNLOAD_COMPLETE = "Download complete"; - - /** - * Returns whether the status indicates a successful pull operation - * - * @returns true: status indicates that pull was successful, false: status doesn't indicate a successful pull - */ - @JsonIgnore - public boolean isPullSuccessIndicated() { - if (isErrorIndicated() || getStatus() == null) { - return false; - } - - return (getStatus().contains(DOWNLOAD_COMPLETE) || getStatus().contains(IMAGE_UP_TO_DATE) - || getStatus().contains(DOWNLOADED_NEWER_IMAGE) || getStatus().contains(LEGACY_REGISTRY)); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/PushResponseItem.java b/src/main/java/com/github/dockerjava/api/model/PushResponseItem.java deleted file mode 100644 index 829fbbbad..000000000 --- a/src/main/java/com/github/dockerjava/api/model/PushResponseItem.java +++ /dev/null @@ -1,13 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; - -/** - * Represents a push response stream item - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class PushResponseItem extends ResponseItem { - - private static final long serialVersionUID = 8256977108011295857L; - -} diff --git a/src/main/java/com/github/dockerjava/api/model/ResponseItem.java b/src/main/java/com/github/dockerjava/api/model/ResponseItem.java deleted file mode 100644 index 12a090db8..000000000 --- a/src/main/java/com/github/dockerjava/api/model/ResponseItem.java +++ /dev/null @@ -1,219 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.core.RemoteApiVersion; -import org.apache.commons.lang.builder.ToStringBuilder; -import org.apache.commons.lang.builder.ToStringStyle; - -import javax.annotation.CheckForNull; -import java.io.Serializable; - -/** - * Represents a pull response stream item - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class ResponseItem implements Serializable { - - private static final long serialVersionUID = -5187169652557467828L; - - @JsonProperty("stream") - private String stream; - - @JsonProperty("status") - private String status; - - @JsonProperty("progressDetail") - private ProgressDetail progressDetail; - - @Deprecated - @JsonProperty("progress") - private String progress; - - @JsonProperty("id") - private String id; - - @JsonProperty("from") - private String from; - - @JsonProperty("time") - private Long time; - - @JsonProperty("errorDetail") - private ErrorDetail errorDetail; - - @Deprecated - @JsonProperty("error") - private String error; - - /** - * @since {@link RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("aux") - private AuxDetail aux; - - @CheckForNull - public String getStream() { - return stream; - } - - @CheckForNull - public String getStatus() { - return status; - } - - @CheckForNull - public ProgressDetail getProgressDetail() { - return progressDetail; - } - - @CheckForNull - @Deprecated - public String getProgress() { - return progress; - } - - @CheckForNull - public String getId() { - return id; - } - - @CheckForNull - public String getFrom() { - return from; - } - - @CheckForNull - public Long getTime() { - return time; - } - - @CheckForNull - public ErrorDetail getErrorDetail() { - return errorDetail; - } - - @Deprecated - public String getError() { - return error; - } - - /** - * @see #aux - */ - @CheckForNull - public AuxDetail getAux() { - return aux; - } - - /** - * Returns whether the error field indicates an error - * - * @returns true: the error field indicates an error, false: the error field doesn't indicate an error - */ - @JsonIgnore - public boolean isErrorIndicated() { - // check both the deprecated and current error fields, just in case - return getError() != null || getErrorDetail() != null; - } - - @JsonIgnoreProperties(ignoreUnknown = true) - public static class ProgressDetail implements Serializable { - private static final long serialVersionUID = -1954994695645715264L; - - @JsonProperty("current") - Long current; - - @JsonProperty("total") - Long total; - - @JsonProperty("start") - Long start; - - @CheckForNull - public Long getCurrent() { - return current; - } - - @CheckForNull - public Long getTotal() { - return total; - } - - @CheckForNull - public Long getStart() { - return start; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this, ToStringStyle.SHORT_PREFIX_STYLE); - } - } - - @JsonIgnoreProperties(ignoreUnknown = true) - public static class ErrorDetail implements Serializable { - private static final long serialVersionUID = -9136704865403084083L; - - @JsonProperty("code") - Integer code; - - @JsonProperty("message") - String message; - - @CheckForNull - public Integer getCode() { - return code; - } - - @CheckForNull - public String getMessage() { - return message; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this, ToStringStyle.SHORT_PREFIX_STYLE); - } - } - - @JsonIgnoreProperties(ignoreUnknown = true) - public static class AuxDetail implements Serializable { - private static final long serialVersionUID = 1L; - - @JsonProperty("Size") - private Integer size; - - @JsonProperty("Tag") - private String tag; - - @JsonProperty("Digest") - private String digest; - - @CheckForNull - public Integer getSize() { - return size; - } - - @CheckForNull - public String getTag() { - return tag; - } - - @CheckForNull - public String getDigest() { - return digest; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this, ToStringStyle.SHORT_PREFIX_STYLE); - } - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this, ToStringStyle.SHORT_PREFIX_STYLE); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/SearchItem.java b/src/main/java/com/github/dockerjava/api/model/SearchItem.java deleted file mode 100644 index 0a5843838..000000000 --- a/src/main/java/com/github/dockerjava/api/model/SearchItem.java +++ /dev/null @@ -1,55 +0,0 @@ -package com.github.dockerjava.api.model; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * - * @author Konstantin Pelykh (kpelykh@gmail.com) - * - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class SearchItem { - - @JsonProperty("star_count") - private Integer starCount; - - @JsonProperty("is_official") - private Boolean isOfficial; - - @JsonProperty("is_trusted") - private Boolean isTrusted; - - @JsonProperty("name") - private String name; - - @JsonProperty("description") - private String description; - - public Integer getStarCount() { - return starCount; - } - - public Boolean isOfficial() { - return isOfficial; - } - - public Boolean isTrusted() { - return isTrusted; - } - - public String getName() { - return name; - } - - public String getDescription() { - return description; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Statistics.java b/src/main/java/com/github/dockerjava/api/model/Statistics.java deleted file mode 100644 index 21b9ca923..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Statistics.java +++ /dev/null @@ -1,79 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.util.Map; - -import javax.annotation.CheckForNull; - -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * Representation of a Docker statistics. - */ -@JsonIgnoreProperties(ignoreUnknown = true) -@JsonInclude(Include.NON_NULL) -public class Statistics { - - @JsonProperty("read") - private String read; - - /** - * @since Docker Remote API 1.21 - */ - @CheckForNull - @JsonProperty("networks") - private Map networks; - - /** - * @deprecated as of Docker Remote API 1.21, replaced by {@link #networks} - */ - @Deprecated - @JsonProperty("network") - private Map network; - - @JsonProperty("memory_stats") - private Map memoryStats; - - @JsonProperty("blkio_stats") - private Map blkioStats; - - @JsonProperty("cpu_stats") - private Map cpuStats; - - /** - * @since Docker Remote API 1.21 - */ - @CheckForNull - public Map getNetworks() { - return networks; - } - - /** - * @deprecated as of Docker Remote API 1.21, replaced by {@link #getNetworks()} - */ - @Deprecated - public Map getNetwork() { - return network; - } - - public Map getCpuStats() { - return cpuStats; - } - - public Map getMemoryStats() { - return memoryStats; - } - - public Map getBlkioStats() { - return blkioStats; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Ulimit.java b/src/main/java/com/github/dockerjava/api/model/Ulimit.java deleted file mode 100644 index 7427fb0cc..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Ulimit.java +++ /dev/null @@ -1,65 +0,0 @@ -package com.github.dockerjava.api.model; - -import static com.google.common.base.Preconditions.checkNotNull; - -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; - -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * @author Vangie Du (duwan@live.com) - */ -public class Ulimit { - - @JsonProperty("Name") - private String name; - - @JsonProperty("Soft") - private Integer soft; - - @JsonProperty("Hard") - private Integer hard; - - public Ulimit() { - - } - - public Ulimit(String name, int soft, int hard) { - checkNotNull(name, "Name is null"); - - this.name = name; - this.soft = soft; - this.hard = hard; - } - - public String getName() { - return name; - } - - public Integer getSoft() { - return soft; - } - - public Integer getHard() { - return hard; - } - - // CHECKSTYLE:OFF - @Override - public boolean equals(Object obj) { - if (obj instanceof Ulimit) { - Ulimit other = (Ulimit) obj; - return new EqualsBuilder().append(name, other.getName()).append(soft, other.getSoft()) - .append(hard, other.getHard()).isEquals(); - } else - return super.equals(obj); - - } - - @Override - public int hashCode() { - return new HashCodeBuilder().append(name).append(soft).append(hard).toHashCode(); - } - // CHECKSTYLE:ON -} diff --git a/src/main/java/com/github/dockerjava/api/model/Version.java b/src/main/java/com/github/dockerjava/api/model/Version.java deleted file mode 100644 index 004c85f22..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Version.java +++ /dev/null @@ -1,112 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.api.command.VersionCmd; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -import javax.annotation.CheckForNull; - -/** - * Used for `/version` - * - * @author Konstantin Pelykh (kpelykh@gmail.com) - * @see VersionCmd - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class Version { - - @JsonProperty("ApiVersion") - private String apiVersion; - - @JsonProperty("Arch") - private String arch; - - @JsonProperty("GitCommit") - private String gitCommit; - - @JsonProperty("GoVersion") - private String goVersion; - - @JsonProperty("KernelVersion") - private String kernelVersion; - - @JsonProperty("Os") - private String operatingSystem; - - @JsonProperty("Version") - private String version; - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_22} - */ - @JsonProperty("BuildTime") - private String buildTime; - - /** - * @since ~{@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_20} - */ - @JsonProperty("Experimental") - private Boolean experimental; - - public String getVersion() { - return version; - } - - public String getGitCommit() { - return gitCommit; - } - - public String getGoVersion() { - return goVersion; - } - - public String getKernelVersion() { - return kernelVersion; - } - - public String getArch() { - return arch; - } - - public String getOperatingSystem() { - return operatingSystem; - } - - public String getApiVersion() { - return apiVersion; - } - - /** - * @see #buildTime - */ - @CheckForNull - public String getBuildTime() { - return buildTime; - } - - /** - * @see #experimental - */ - @CheckForNull - public Boolean getExperimental() { - return experimental; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/Volume.java b/src/main/java/com/github/dockerjava/api/model/Volume.java deleted file mode 100644 index 655a8dbfc..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Volume.java +++ /dev/null @@ -1,43 +0,0 @@ -package com.github.dockerjava.api.model; - -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; - -/** - * Represents a bind mounted volume in a Docker container. - * - * @see Bind - */ -public class Volume { - - private String path; - - public Volume(String path) { - this.path = path; - } - - public String getPath() { - return path; - } - - @Override - public String toString() { - return getPath(); - } - - @Override - public boolean equals(Object obj) { - if (obj instanceof Volume) { - Volume other = (Volume) obj; - return new EqualsBuilder().append(path, other.getPath()).isEquals(); - } else { - return super.equals(obj); - } - } - - @Override - public int hashCode() { - return new HashCodeBuilder().append(path).toHashCode(); - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/VolumeBind.java b/src/main/java/com/github/dockerjava/api/model/VolumeBind.java deleted file mode 100644 index cf9c077ab..000000000 --- a/src/main/java/com/github/dockerjava/api/model/VolumeBind.java +++ /dev/null @@ -1,25 +0,0 @@ -package com.github.dockerjava.api.model; - -public class VolumeBind { - private final String hostPath; - - private final String containerPath; - - public VolumeBind(String hostPath, String containerPath) { - this.hostPath = hostPath; - this.containerPath = containerPath; - } - - public String getContainerPath() { - return containerPath; - } - - public String getHostPath() { - return hostPath; - } - - @Override - public String toString() { - return hostPath + ":" + containerPath; - } -} diff --git a/src/main/java/com/github/dockerjava/api/model/VolumeBinds.java b/src/main/java/com/github/dockerjava/api/model/VolumeBinds.java deleted file mode 100644 index d8303ec84..000000000 --- a/src/main/java/com/github/dockerjava/api/model/VolumeBinds.java +++ /dev/null @@ -1,73 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.Iterator; -import java.util.List; -import java.util.Map; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; -import com.fasterxml.jackson.databind.node.NullNode; - -// This is not going to be serialized -@JsonDeserialize(using = VolumeBinds.Deserializer.class) -@JsonSerialize(using = VolumeBinds.Serializer.class) -public class VolumeBinds { - private final VolumeBind[] binds; - - public VolumeBinds(VolumeBind... binds) { - this.binds = binds; - } - - public VolumeBind[] getBinds() { - return binds; - } - - public static final class Serializer extends JsonSerializer { - - @Override - public void serialize(VolumeBinds value, JsonGenerator jgen, SerializerProvider provider) throws IOException, - JsonProcessingException { - jgen.writeStartObject(); - for (final VolumeBind bind : value.binds) { - jgen.writeStringField(bind.getContainerPath(), bind.getHostPath()); - } - jgen.writeEndObject(); - } - } - - public static final class Deserializer extends JsonDeserializer { - @Override - public VolumeBinds deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - - List binds = new ArrayList(); - ObjectCodec oc = jsonParser.getCodec(); - JsonNode node = oc.readTree(jsonParser); - for (Iterator> it = node.fields(); it.hasNext();) { - Map.Entry field = it.next(); - JsonNode value = field.getValue(); - if (!value.equals(NullNode.getInstance())) { - if (!value.isTextual()) { - throw deserializationContext.mappingException("Expected path for '" + field.getKey() - + "'in host but got '" + value + "'."); - } - VolumeBind bind = new VolumeBind(value.asText(), field.getKey()); - binds.add(bind); - } - } - return new VolumeBinds(binds.toArray(new VolumeBind[binds.size()])); - } - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/VolumeRW.java b/src/main/java/com/github/dockerjava/api/model/VolumeRW.java deleted file mode 100644 index 7c4bdfb26..000000000 --- a/src/main/java/com/github/dockerjava/api/model/VolumeRW.java +++ /dev/null @@ -1,113 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.io.IOException; -import java.util.Map.Entry; - -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; -import com.fasterxml.jackson.databind.node.NullNode; -import com.github.dockerjava.core.RemoteApiVersion; - -/** - * Represents a bind mounted volume in a Docker container. - * - * @see Bind - * @deprecated since {@link RemoteApiVersion#VERSION_1_20} - */ -@JsonDeserialize(using = VolumeRW.Deserializer.class) -@JsonSerialize(using = VolumeRW.Serializer.class) -@Deprecated -public class VolumeRW { - - private Volume volume; - - private AccessMode accessMode = AccessMode.rw; - - public VolumeRW(Volume volume) { - this.volume = volume; - } - - public VolumeRW(Volume volume, AccessMode accessMode) { - this.volume = volume; - this.accessMode = accessMode; - } - - public Volume getVolume() { - return volume; - } - - public AccessMode getAccessMode() { - return accessMode; - } - - /** - * Returns a string representation of this {@link VolumeRW} suitable for inclusion in a JSON message. The returned String is simply the - * container path, {@link #getPath()}. - * - * @return a string representation of this {@link VolumeRW} - */ - @Override - public String toString() { - return getVolume() + ":" + getAccessMode(); - } - - @Override - public boolean equals(Object obj) { - if (obj instanceof VolumeRW) { - VolumeRW other = (VolumeRW) obj; - return new EqualsBuilder().append(getVolume(), other.getVolume()).append(accessMode, other.getAccessMode()) - .isEquals(); - } else { - return super.equals(obj); - } - } - - @Override - public int hashCode() { - return new HashCodeBuilder().append(getVolume()).append(getAccessMode()).toHashCode(); - } - - public static class Serializer extends JsonSerializer { - - @Override - public void serialize(VolumeRW volumeRW, JsonGenerator jsonGen, SerializerProvider serProvider) - throws IOException, JsonProcessingException { - - jsonGen.writeStartObject(); - jsonGen.writeFieldName(volumeRW.getVolume().getPath()); - jsonGen.writeString(Boolean.toString(volumeRW.getAccessMode().toBoolean())); - jsonGen.writeEndObject(); - } - - } - - public static class Deserializer extends JsonDeserializer { - @Override - public VolumeRW deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - ObjectCodec oc = jsonParser.getCodec(); - JsonNode node = oc.readTree(jsonParser); - if (!node.equals(NullNode.getInstance())) { - Entry field = node.fields().next(); - String volume = field.getKey(); - AccessMode accessMode = AccessMode.fromBoolean(field.getValue().asBoolean()); - return new VolumeRW(new Volume(volume), accessMode); - } else { - return null; - } - } - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/Volumes.java b/src/main/java/com/github/dockerjava/api/model/Volumes.java deleted file mode 100644 index 46175548d..000000000 --- a/src/main/java/com/github/dockerjava/api/model/Volumes.java +++ /dev/null @@ -1,78 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.Iterator; -import java.util.List; -import java.util.Map; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; -import com.fasterxml.jackson.databind.node.NullNode; - -@JsonSerialize(using = Volumes.Serializer.class) -@JsonDeserialize(using = Volumes.Deserializer.class) -public class Volumes { - - private Volume[] volumes; - - public Volumes(Volume... volumes) { - this.volumes = volumes; - } - - public Volumes(List volumes) { - this.volumes = volumes.toArray(new Volume[volumes.size()]); - } - - public Volume[] getVolumes() { - return volumes; - } - - public static class Serializer extends JsonSerializer { - - @Override - public void serialize(Volumes volumes, JsonGenerator jsonGen, SerializerProvider serProvider) - throws IOException, JsonProcessingException { - - jsonGen.writeStartObject(); - for (Volume volume : volumes.getVolumes()) { - jsonGen.writeFieldName(volume.getPath()); - jsonGen.writeStartObject(); - jsonGen.writeEndObject(); - } - jsonGen.writeEndObject(); - } - - } - - public static class Deserializer extends JsonDeserializer { - @Override - public Volumes deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - - List volumes = new ArrayList(); - ObjectCodec oc = jsonParser.getCodec(); - JsonNode node = oc.readTree(jsonParser); - for (Iterator> it = node.fields(); it.hasNext();) { - - Map.Entry field = it.next(); - if (!field.getValue().equals(NullNode.getInstance())) { - String path = field.getKey(); - Volume volume = new Volume(path); - volumes.add(volume); - } - } - return new Volumes(volumes.toArray(new Volume[0])); - } - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/VolumesFrom.java b/src/main/java/com/github/dockerjava/api/model/VolumesFrom.java deleted file mode 100644 index f027b6d71..000000000 --- a/src/main/java/com/github/dockerjava/api/model/VolumesFrom.java +++ /dev/null @@ -1,125 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.io.IOException; - -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; - -@JsonSerialize(using = VolumesFrom.Serializer.class) -@JsonDeserialize(using = VolumesFrom.Deserializer.class) -public class VolumesFrom { - - private String container; - - private AccessMode accessMode; - - public VolumesFrom(String container) { - this(container, AccessMode.DEFAULT); - } - - public VolumesFrom(String container, AccessMode accessMode) { - this.container = container; - this.accessMode = accessMode; - } - - public String getContainer() { - return container; - } - - public AccessMode getAccessMode() { - return accessMode; - } - - /** - * Parses a volume from specification to a {@link VolumesFrom}. - * - * @param serialized - * the specification, e.g. container:ro - * @return a {@link VolumesFrom} matching the specification - * @throws IllegalArgumentException - * if the specification cannot be parsed - */ - public static VolumesFrom parse(String serialized) { - try { - String[] parts = serialized.split(":"); - switch (parts.length) { - case 1: { - return new VolumesFrom(parts[0]); - } - case 2: { - return new VolumesFrom(parts[0], AccessMode.valueOf(parts[1])); - } - - default: { - throw new IllegalArgumentException(); - } - } - } catch (Exception e) { - throw new IllegalArgumentException("Error parsing Bind '" + serialized + "'"); - } - } - - @Override - public boolean equals(Object obj) { - if (obj instanceof VolumesFrom) { - VolumesFrom other = (VolumesFrom) obj; - return new EqualsBuilder().append(container, other.getContainer()) - .append(accessMode, other.getAccessMode()).isEquals(); - } else { - return super.equals(obj); - } - } - - @Override - public int hashCode() { - return new HashCodeBuilder().append(container).append(accessMode).toHashCode(); - } - - /** - * Returns a string representation of this {@link VolumesFrom} suitable for inclusion in a JSON message. The format is - * <container>:<access mode>, like the argument in {@link #parse(String)}. - * - * @return a string representation of this {@link VolumesFrom} - */ - @Override - public String toString() { - return container + ":" + accessMode.toString(); - } - - public static class Serializer extends JsonSerializer { - - @Override - public void serialize(VolumesFrom volumeFrom, JsonGenerator jsonGen, SerializerProvider serProvider) - throws IOException, JsonProcessingException { - - jsonGen.writeString(volumeFrom.toString()); - - } - - } - - public static class Deserializer extends JsonDeserializer { - @Override - public VolumesFrom deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - - ObjectCodec oc = jsonParser.getCodec(); - JsonNode node = oc.readTree(jsonParser); - return VolumesFrom.parse(node.asText()); - - } - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/VolumesRW.java b/src/main/java/com/github/dockerjava/api/model/VolumesRW.java deleted file mode 100644 index f54f8242c..000000000 --- a/src/main/java/com/github/dockerjava/api/model/VolumesRW.java +++ /dev/null @@ -1,77 +0,0 @@ -package com.github.dockerjava.api.model; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.Iterator; -import java.util.List; -import java.util.Map; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.ObjectCodec; -import com.fasterxml.jackson.databind.DeserializationContext; -import com.fasterxml.jackson.databind.JsonDeserializer; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.JsonSerializer; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.annotation.JsonDeserialize; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; -import com.fasterxml.jackson.databind.node.NullNode; - -// This is not going to be serialized -@JsonSerialize(using = VolumesRW.Serializer.class) -@JsonDeserialize(using = VolumesRW.Deserializer.class) -public class VolumesRW { - private final VolumeRW[] volumesRW; - - public VolumesRW(VolumeRW... binds) { - this.volumesRW = binds; - } - - public VolumeRW[] getVolumesRW() { - return volumesRW; - } - - public static final class Serializer extends JsonSerializer { - - @Override - public void serialize(VolumesRW value, JsonGenerator jgen, SerializerProvider provider) throws IOException, - JsonProcessingException { - jgen.writeStartObject(); - for (final VolumeRW volumeRW : value.volumesRW) { - jgen.writeBooleanField(volumeRW.getVolume().getPath(), volumeRW.getAccessMode().toBoolean()); - } - jgen.writeEndObject(); - } - - } - - public static final class Deserializer extends JsonDeserializer { - @Override - public VolumesRW deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) - throws IOException, JsonProcessingException { - - List volumesRW = new ArrayList(); - ObjectCodec oc = jsonParser.getCodec(); - JsonNode node = oc.readTree(jsonParser); - - for (Iterator> it = node.fields(); it.hasNext();) { - Map.Entry field = it.next(); - JsonNode value = field.getValue(); - - if (!value.equals(NullNode.getInstance())) { - if (!value.isBoolean()) { - throw deserializationContext.mappingException("Expected access mode for '" + field.getKey() - + "' in host but got '" + value + "'."); - } - - VolumeRW bind = new VolumeRW(new Volume(field.getKey()), AccessMode.fromBoolean(value.asBoolean())); - volumesRW.add(bind); - } - } - return new VolumesRW(volumesRW.toArray(new VolumeRW[volumesRW.size()])); - } - } - -} diff --git a/src/main/java/com/github/dockerjava/api/model/WaitResponse.java b/src/main/java/com/github/dockerjava/api/model/WaitResponse.java deleted file mode 100644 index a2ab16f6f..000000000 --- a/src/main/java/com/github/dockerjava/api/model/WaitResponse.java +++ /dev/null @@ -1,18 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * Represents a wait container command response - */ -@JsonIgnoreProperties(ignoreUnknown = true) -public class WaitResponse { - - @JsonProperty("StatusCode") - private Integer statusCode; - - public Integer getStatusCode() { - return statusCode; - } -} diff --git a/src/main/java/com/github/dockerjava/core/AuthConfigFile.java b/src/main/java/com/github/dockerjava/core/AuthConfigFile.java deleted file mode 100644 index f796fd7c9..000000000 --- a/src/main/java/com/github/dockerjava/core/AuthConfigFile.java +++ /dev/null @@ -1,161 +0,0 @@ -package com.github.dockerjava.core; - -import java.io.File; -import java.io.IOException; -import java.nio.charset.Charset; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import org.apache.commons.codec.binary.Base64; -import org.apache.commons.io.FileUtils; -import org.apache.commons.lang.StringUtils; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.AuthConfigurations; - -public class AuthConfigFile { - private static final ObjectMapper MAPPER = new ObjectMapper(); - - private static final TypeReference> CONFIG_MAP_TYPE = new TypeReference>() { - }; - - private final Map authConfigMap; - - public AuthConfigFile() { - authConfigMap = new HashMap(); - } - - void addConfig(AuthConfig config) { - authConfigMap.put(config.getRegistryAddress(), config); - } - - public AuthConfig resolveAuthConfig(String hostname) { - if (StringUtils.isEmpty(hostname) || AuthConfig.DEFAULT_SERVER_ADDRESS.equals(hostname)) { - return authConfigMap.get(AuthConfig.DEFAULT_SERVER_ADDRESS); - } - AuthConfig c = authConfigMap.get(hostname); - if (c != null) { - return c; - } - - // Maybe they have a legacy config file, we will iterate the keys converting - // them to the new format and testing - String normalizedHostname = convertToHostname(hostname); - for (Map.Entry entry : authConfigMap.entrySet()) { - String registry = entry.getKey(); - AuthConfig config = entry.getValue(); - if (convertToHostname(registry).equals(normalizedHostname)) { - return config; - } - } - return null; - } - - public AuthConfigurations getAuthConfigurations() { - final AuthConfigurations authConfigurations = new AuthConfigurations(); - for (Map.Entry authConfigEntry : authConfigMap.entrySet()) { - authConfigurations.addConfig(authConfigEntry.getValue()); - } - - return authConfigurations; - } - - // CHECKSTYLE:OFF - @Override - public int hashCode() { - final int prime = 31; - int result = 1; - result = prime * result + ((authConfigMap == null) ? 0 : authConfigMap.hashCode()); - return result; - } - - @Override - public boolean equals(Object obj) { - if (this == obj) - return true; - if (obj == null) - return false; - if (getClass() != obj.getClass()) - return false; - AuthConfigFile other = (AuthConfigFile) obj; - if (authConfigMap == null) { - if (other.authConfigMap != null) - return false; - } else if (!authConfigMap.equals(other.authConfigMap)) - return false; - return true; - } - - // CHECKSTYLE:ON - - @Override - public String toString() { - return "AuthConfigFile [authConfigMap=" + authConfigMap + "]"; - } - - public static AuthConfigFile loadConfig(File confFile) throws IOException { - AuthConfigFile configFile = new AuthConfigFile(); - if (!confFile.exists()) { - return new AuthConfigFile(); - } - Map configMap = null; - try { - configMap = MAPPER.readValue(confFile, CONFIG_MAP_TYPE); - } catch (IOException e) { - // pass - } - if (configMap != null) { - for (Map.Entry entry : configMap.entrySet()) { - AuthConfig authConfig = entry.getValue(); - decodeAuth(authConfig.getAuth(), authConfig); - authConfig.withAuth(null); - authConfig.withRegistryAddress(entry.getKey()); - configFile.addConfig(authConfig); - } - } else { - List authFileContent = FileUtils.readLines(confFile); - if (authFileContent.size() < 2) { - throw new IOException("The Auth Config file is empty"); - } - AuthConfig config = new AuthConfig(); - String[] origAuth = authFileContent.get(0).split(" = "); - if (origAuth.length != 2) { - throw new IOException("Invalid Auth config file"); - } - decodeAuth(origAuth[1], config); - - String[] origEmail = authFileContent.get(1).split(" = "); - if (origEmail.length != 2) { - throw new IOException("Invalid Auth config file"); - } - config.withEmail(origEmail[1]); - configFile.addConfig(config); - } - return configFile; - - } - - static void decodeAuth(String auth, AuthConfig config) throws IOException { - String str = new String(Base64.decodeBase64(auth), Charset.forName("UTF-8")); - String[] parts = str.split(":", 2); - if (parts.length != 2) { - throw new IOException("Invalid auth configuration file"); - } - config.withUsername(parts[0]); - config.withPassword(parts[1]); - } - - static String convertToHostname(String server) { - String stripped = server; - if (server.startsWith("http://")) { - stripped = server.substring(7); - } else if (server.startsWith("https://")) { - stripped = server.substring(8); - } - String[] numParts = stripped.split("/", 2); - return numParts[0]; - } -} diff --git a/src/main/java/com/github/dockerjava/core/DefaultDockerClientConfig.java b/src/main/java/com/github/dockerjava/core/DefaultDockerClientConfig.java deleted file mode 100644 index a9a15bb36..000000000 --- a/src/main/java/com/github/dockerjava/core/DefaultDockerClientConfig.java +++ /dev/null @@ -1,456 +0,0 @@ -package com.github.dockerjava.core; - -import static com.google.common.base.Preconditions.checkNotNull; -import static org.apache.commons.lang.BooleanUtils.isTrue; - -import java.io.File; -import java.io.FileInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.io.Serializable; -import java.net.URI; -import java.util.HashSet; -import java.util.Map; -import java.util.Properties; -import java.util.Set; - -import org.apache.commons.lang.StringUtils; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; -import org.apache.commons.lang.builder.ToStringStyle; - -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.AuthConfigurations; -import com.github.dockerjava.core.NameParser.HostnameReposName; -import com.github.dockerjava.core.NameParser.ReposTag; - -/** - * Respects some of the docker CLI options. See https://docs.docker.com/engine/reference/commandline/cli/#environment-variables - */ -public class DefaultDockerClientConfig implements Serializable, DockerClientConfig { - - private static final long serialVersionUID = 1L; - - public static final String DOCKER_HOST = "DOCKER_HOST"; - - public static final String DOCKER_TLS_VERIFY = "DOCKER_TLS_VERIFY"; - - public static final String DOCKER_CONFIG = "DOCKER_CONFIG"; - - public static final String DOCKER_CERT_PATH = "DOCKER_CERT_PATH"; - - public static final String API_VERSION = "api.version"; - - public static final String REGISTRY_USERNAME = "registry.username"; - - public static final String REGISTRY_PASSWORD = "registry.password"; - - public static final String REGISTRY_EMAIL = "registry.email"; - - public static final String REGISTRY_URL = "registry.url"; - - private static final String DOCKER_JAVA_PROPERTIES = "docker-java.properties"; - - private static final String DOCKER_CFG = ".dockercfg"; - - private static final Set CONFIG_KEYS = new HashSet(); - - static { - CONFIG_KEYS.add(DOCKER_HOST); - CONFIG_KEYS.add(DOCKER_TLS_VERIFY); - CONFIG_KEYS.add(DOCKER_CONFIG); - CONFIG_KEYS.add(DOCKER_CERT_PATH); - CONFIG_KEYS.add(API_VERSION); - CONFIG_KEYS.add(REGISTRY_USERNAME); - CONFIG_KEYS.add(REGISTRY_PASSWORD); - CONFIG_KEYS.add(REGISTRY_EMAIL); - CONFIG_KEYS.add(REGISTRY_URL); - } - - private final URI dockerHost; - - private final String registryUsername, registryPassword, registryEmail, registryUrl, dockerConfig; - - private final SSLConfig sslConfig; - - private final RemoteApiVersion apiVersion; - - DefaultDockerClientConfig(URI dockerHost, String dockerConfig, String apiVersion, String registryUrl, - String registryUsername, String registryPassword, String registryEmail, SSLConfig sslConfig) { - this.dockerHost = checkDockerHostScheme(dockerHost); - this.dockerConfig = dockerConfig; - this.apiVersion = RemoteApiVersion.parseConfigWithDefault(apiVersion); - this.sslConfig = sslConfig; - this.registryUsername = registryUsername; - this.registryPassword = registryPassword; - this.registryEmail = registryEmail; - this.registryUrl = registryUrl; - } - - private URI checkDockerHostScheme(URI dockerHost) { - if ("tcp".equals(dockerHost.getScheme()) || "unix".equals(dockerHost.getScheme())) { - return dockerHost; - } else { - throw new DockerClientException("Unsupported protocol scheme found: '" + dockerHost - + "'. Only 'tcp://' or 'unix://' supported."); - } - } - - private static Properties loadIncludedDockerProperties(Properties systemProperties) { - try (InputStream is = DefaultDockerClientConfig.class.getResourceAsStream("/" + DOCKER_JAVA_PROPERTIES)) { - Properties p = new Properties(); - p.load(is); - replaceProperties(p, systemProperties); - return p; - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - private static void replaceProperties(Properties properties, Properties replacements) { - for (Object objectKey : properties.keySet()) { - String key = objectKey.toString(); - properties.setProperty(key, replaceProperties(properties.getProperty(key), replacements)); - } - } - - private static String replaceProperties(String s, Properties replacements) { - for (Map.Entry entry : replacements.entrySet()) { - String key = "${" + entry.getKey() + "}"; - while (s.contains(key)) { - s = s.replace(key, String.valueOf(entry.getValue())); - } - } - return s; - } - - /** - * Creates a new Properties object containing values overridden from ${user.home}/.docker.io.properties - * - * @param p - * The original set of properties to override - * @return A copy of the original Properties with overridden values - */ - private static Properties overrideDockerPropertiesWithSettingsFromUserHome(Properties p, Properties systemProperties) { - Properties overriddenProperties = new Properties(); - overriddenProperties.putAll(p); - - final File usersDockerPropertiesFile = new File(systemProperties.getProperty("user.home"), - "." + DOCKER_JAVA_PROPERTIES); - if (usersDockerPropertiesFile.isFile()) { - try (FileInputStream in = new FileInputStream(usersDockerPropertiesFile)) { - overriddenProperties.load(in); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - return overriddenProperties; - } - - private static Properties overrideDockerPropertiesWithEnv(Properties properties, Map env) { - Properties overriddenProperties = new Properties(); - overriddenProperties.putAll(properties); - - // special case which is a sensible default - if (env.containsKey(DOCKER_HOST)) { - overriddenProperties.setProperty(DOCKER_HOST, env.get(DOCKER_HOST)); - } - - for (Map.Entry envEntry : env.entrySet()) { - String envKey = envEntry.getKey(); - if (CONFIG_KEYS.contains(envKey)) { - overriddenProperties.setProperty(envKey, envEntry.getValue()); - } - } - - return overriddenProperties; - } - - /** - * Creates a new Properties object containing values overridden from the System properties - * - * @param p - * The original set of properties to override - * @return A copy of the original Properties with overridden values - */ - private static Properties overrideDockerPropertiesWithSystemProperties(Properties p, Properties systemProperties) { - Properties overriddenProperties = new Properties(); - overriddenProperties.putAll(p); - - for (String key : CONFIG_KEYS) { - if (systemProperties.containsKey(key)) { - overriddenProperties.setProperty(key, systemProperties.getProperty(key)); - } - } - return overriddenProperties; - } - - public static Builder createDefaultConfigBuilder() { - return createDefaultConfigBuilder(System.getenv(), System.getProperties()); - } - - /** - * Allows you to build the config without system environment interfering for more robust testing - */ - static Builder createDefaultConfigBuilder(Map env, Properties systemProperties) { - Properties properties = loadIncludedDockerProperties(systemProperties); - properties = overrideDockerPropertiesWithSettingsFromUserHome(properties, systemProperties); - properties = overrideDockerPropertiesWithEnv(properties, env); - properties = overrideDockerPropertiesWithSystemProperties(properties, systemProperties); - return new Builder().withProperties(properties); - } - - @Override - public URI getDockerHost() { - return dockerHost; - } - - @Override - public RemoteApiVersion getApiVersion() { - return apiVersion; - } - - @Override - public String getRegistryUsername() { - return registryUsername; - } - - @Override - public String getRegistryPassword() { - return registryPassword; - } - - @Override - public String getRegistryEmail() { - return registryEmail; - } - - @Override - public String getRegistryUrl() { - return registryUrl; - } - - public String getDockerConfig() { - return dockerConfig; - } - - private AuthConfig getAuthConfig() { - AuthConfig authConfig = null; - if (getRegistryUsername() != null && getRegistryPassword() != null && getRegistryEmail() != null - && getRegistryUrl() != null) { - authConfig = new AuthConfig() - .withUsername(getRegistryUsername()) - .withPassword(getRegistryPassword()) - .withEmail(getRegistryEmail()) - .withRegistryAddress(getRegistryUrl()); - } - return authConfig; - } - - @Override - public AuthConfig effectiveAuthConfig(String imageName) { - AuthConfig authConfig = null; - - File dockerCfgFile = new File(getDockerConfig() + File.separator + DOCKER_CFG); - - if (dockerCfgFile.exists() && dockerCfgFile.isFile() && imageName != null) { - AuthConfigFile authConfigFile; - try { - authConfigFile = AuthConfigFile.loadConfig(dockerCfgFile); - } catch (IOException e) { - throw new DockerClientException("Failed to parse dockerCfgFile", e); - } - ReposTag reposTag = NameParser.parseRepositoryTag(imageName); - HostnameReposName hostnameReposName = NameParser.resolveRepositoryName(reposTag.repos); - - authConfig = authConfigFile.resolveAuthConfig(hostnameReposName.hostname); - } - - AuthConfig otherAuthConfig = getAuthConfig(); - - if (otherAuthConfig != null) { - authConfig = otherAuthConfig; - } - - return authConfig; - } - - @Override - public AuthConfigurations getAuthConfigurations() { - File dockerCfgFile = new File(getDockerConfig() + File.separator + DOCKER_CFG); - if (dockerCfgFile.exists() && dockerCfgFile.isFile()) { - AuthConfigFile authConfigFile; - try { - authConfigFile = AuthConfigFile.loadConfig(dockerCfgFile); - } catch (IOException e) { - throw new DockerClientException("Failed to parse dockerCfgFile", e); - } - - return authConfigFile.getAuthConfigurations(); - } - - return new AuthConfigurations(); - } - - @Override - public SSLConfig getSSLConfig() { - return sslConfig; - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this, ToStringStyle.SHORT_PREFIX_STYLE); - } - - public static class Builder { - private URI dockerHost; - - private String apiVersion, registryUsername, registryPassword, registryEmail, registryUrl, dockerConfig, - dockerCertPath; - - private Boolean dockerTlsVerify; - - private SSLConfig customSslConfig = null; - - /** - * This will set all fields in the builder to those contained in the Properties object. The Properties object should contain the - * following docker-java configuration keys: DOCKER_HOST, DOCKER_TLS_VERIFY, api.version, registry.username, registry.password, - * registry.email, DOCKER_CERT_PATH, and DOCKER_CONFIG. - */ - public Builder withProperties(Properties p) { - return withDockerHost(p.getProperty(DOCKER_HOST)) - .withDockerTlsVerify(p.getProperty(DOCKER_TLS_VERIFY)) - .withDockerConfig(p.getProperty(DOCKER_CONFIG)) - .withDockerCertPath(p.getProperty(DOCKER_CERT_PATH)) - .withApiVersion(p.getProperty(API_VERSION)) - .withRegistryUsername(p.getProperty(REGISTRY_USERNAME)) - .withRegistryPassword(p.getProperty(REGISTRY_PASSWORD)) - .withRegistryEmail(p.getProperty(REGISTRY_EMAIL)) - .withRegistryUrl(p.getProperty(REGISTRY_URL)); - } - - /** - * configure DOCKER_HOST - */ - public final Builder withDockerHost(String dockerHost) { - checkNotNull(dockerHost, "uri was not specified"); - this.dockerHost = URI.create(dockerHost); - return this; - } - - public final Builder withApiVersion(RemoteApiVersion apiVersion) { - this.apiVersion = apiVersion.getVersion(); - return this; - } - - public final Builder withApiVersion(String apiVersion) { - this.apiVersion = apiVersion; - return this; - } - - public final Builder withRegistryUsername(String registryUsername) { - this.registryUsername = registryUsername; - return this; - } - - public final Builder withRegistryPassword(String registryPassword) { - this.registryPassword = registryPassword; - return this; - } - - public final Builder withRegistryEmail(String registryEmail) { - this.registryEmail = registryEmail; - return this; - } - - public Builder withRegistryUrl(String registryUrl) { - this.registryUrl = registryUrl; - return this; - } - - public final Builder withDockerCertPath(String dockerCertPath) { - this.dockerCertPath = dockerCertPath; - return this; - } - - public final Builder withDockerConfig(String dockerConfig) { - this.dockerConfig = dockerConfig; - return this; - } - - public final Builder withDockerTlsVerify(String dockerTlsVerify) { - if (dockerTlsVerify != null) { - String trimmed = dockerTlsVerify.trim(); - this.dockerTlsVerify = "true".equalsIgnoreCase(trimmed) || "1".equals(trimmed); - } else { - this.dockerTlsVerify = false; - } - return this; - } - - public final Builder withDockerTlsVerify(Boolean dockerTlsVerify) { - this.dockerTlsVerify = dockerTlsVerify; - return this; - } - - /** - * Overrides the default {@link SSLConfig} that is used when calling {@link Builder#withDockerTlsVerify(java.lang.Boolean)} and - * {@link Builder#withDockerCertPath(String)}. This way it is possible to pass a custom {@link SSLConfig} to the resulting - * {@link DockerClientConfig} that may be created by other means than the local file system. - */ - public final Builder withCustomSslConfig(SSLConfig customSslConfig) { - this.customSslConfig = customSslConfig; - return this; - } - - public DefaultDockerClientConfig build() { - - SSLConfig sslConfig = null; - - if (customSslConfig == null) { - if (isTrue(dockerTlsVerify)) { - dockerCertPath = checkDockerCertPath(dockerCertPath); - sslConfig = new LocalDirectorySSLConfig(dockerCertPath); - } - } else { - sslConfig = customSslConfig; - } - - return new DefaultDockerClientConfig(dockerHost, dockerConfig, apiVersion, registryUrl, registryUsername, - registryPassword, registryEmail, sslConfig); - } - - private String checkDockerCertPath(String dockerCertPath) { - if (StringUtils.isEmpty(dockerCertPath)) { - throw new DockerClientException( - "Enabled TLS verification (DOCKER_TLS_VERIFY=1) but certifate path (DOCKER_CERT_PATH) is not defined."); - } - - File certPath = new File(dockerCertPath); - - if (!certPath.exists()) { - throw new DockerClientException( - "Enabled TLS verification (DOCKER_TLS_VERIFY=1) but certificate path (DOCKER_CERT_PATH) '" - + dockerCertPath + "' doesn't exist."); - } else if (!certPath.isDirectory()) { - throw new DockerClientException( - "Enabled TLS verification (DOCKER_TLS_VERIFY=1) but certificate path (DOCKER_CERT_PATH) '" - + dockerCertPath + "' doesn't point to a directory."); - } - - return dockerCertPath; - } - } -} diff --git a/src/main/java/com/github/dockerjava/core/DockerClientBuilder.java b/src/main/java/com/github/dockerjava/core/DockerClientBuilder.java deleted file mode 100644 index 991bdbb86..000000000 --- a/src/main/java/com/github/dockerjava/core/DockerClientBuilder.java +++ /dev/null @@ -1,52 +0,0 @@ -package com.github.dockerjava.core; - -import com.github.dockerjava.api.DockerClient; -import com.github.dockerjava.api.command.DockerCmdExecFactory; -import com.github.dockerjava.core.DefaultDockerClientConfig.Builder; -import com.github.dockerjava.jaxrs.JerseyDockerCmdExecFactory; - -public class DockerClientBuilder { - - private DockerClientImpl dockerClient = null; - - private DockerCmdExecFactory dockerCmdExecFactory = null; - - private DockerClientBuilder(DockerClientImpl dockerClient) { - this.dockerClient = dockerClient; - } - - public static DockerClientBuilder getInstance() { - return new DockerClientBuilder(DockerClientImpl.getInstance()); - } - - public static DockerClientBuilder getInstance(Builder dockerClientConfigBuilder) { - return getInstance(dockerClientConfigBuilder.build()); - } - - public static DockerClientBuilder getInstance(DockerClientConfig dockerClientConfig) { - return new DockerClientBuilder(DockerClientImpl.getInstance(dockerClientConfig)); - } - - public static DockerClientBuilder getInstance(String serverUrl) { - return new DockerClientBuilder(DockerClientImpl.getInstance(serverUrl)); - } - - public static DockerCmdExecFactory getDefaultDockerCmdExecFactory() { - return new JerseyDockerCmdExecFactory(); - } - - public DockerClientBuilder withDockerCmdExecFactory(DockerCmdExecFactory dockerCmdExecFactory) { - this.dockerCmdExecFactory = dockerCmdExecFactory; - return this; - } - - public DockerClient build() { - if (dockerCmdExecFactory != null) { - dockerClient.withDockerCmdExecFactory(dockerCmdExecFactory); - } else { - dockerClient.withDockerCmdExecFactory(getDefaultDockerCmdExecFactory()); - } - - return dockerClient; - } -} diff --git a/src/main/java/com/github/dockerjava/core/DockerClientConfig.java b/src/main/java/com/github/dockerjava/core/DockerClientConfig.java deleted file mode 100644 index 532fe6b43..000000000 --- a/src/main/java/com/github/dockerjava/core/DockerClientConfig.java +++ /dev/null @@ -1,40 +0,0 @@ -/* - * Created on 08.06.2016 - */ -package com.github.dockerjava.core; - -import java.net.URI; - -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.AuthConfigurations; - -/** - * Interface that describes the docker client configuration. - * - * @author Marcus Linke - * - */ -public interface DockerClientConfig { - - URI getDockerHost(); - - RemoteApiVersion getApiVersion(); - - String getRegistryUsername(); - - String getRegistryPassword(); - - String getRegistryEmail(); - - String getRegistryUrl(); - - AuthConfig effectiveAuthConfig(String imageName); - - AuthConfigurations getAuthConfigurations(); - - /** - * Returns an {@link SSLConfig} when secure connection is configured or null if not. - */ - SSLConfig getSSLConfig(); - -} diff --git a/src/main/java/com/github/dockerjava/core/DockerClientImpl.java b/src/main/java/com/github/dockerjava/core/DockerClientImpl.java deleted file mode 100644 index 2d0b5caab..000000000 --- a/src/main/java/com/github/dockerjava/core/DockerClientImpl.java +++ /dev/null @@ -1,491 +0,0 @@ -package com.github.dockerjava.core; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.io.Closeable; -import java.io.File; -import java.io.IOException; -import java.io.InputStream; - -import com.github.dockerjava.api.DockerClient; -import com.github.dockerjava.api.command.AttachContainerCmd; -import com.github.dockerjava.api.command.AuthCmd; -import com.github.dockerjava.api.command.BuildImageCmd; -import com.github.dockerjava.api.command.CommitCmd; -import com.github.dockerjava.api.command.ConnectToNetworkCmd; -import com.github.dockerjava.api.command.ContainerDiffCmd; -import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; -import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; -import com.github.dockerjava.api.command.CopyFileFromContainerCmd; -import com.github.dockerjava.api.command.CreateContainerCmd; -import com.github.dockerjava.api.command.CreateImageCmd; -import com.github.dockerjava.api.command.CreateNetworkCmd; -import com.github.dockerjava.api.command.CreateVolumeCmd; -import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; -import com.github.dockerjava.api.command.DockerCmdExecFactory; -import com.github.dockerjava.api.command.EventsCmd; -import com.github.dockerjava.api.command.ExecCreateCmd; -import com.github.dockerjava.api.command.ExecStartCmd; -import com.github.dockerjava.api.command.InfoCmd; -import com.github.dockerjava.api.command.InspectContainerCmd; -import com.github.dockerjava.api.command.InspectExecCmd; -import com.github.dockerjava.api.command.InspectImageCmd; -import com.github.dockerjava.api.command.InspectNetworkCmd; -import com.github.dockerjava.api.command.InspectVolumeCmd; -import com.github.dockerjava.api.command.KillContainerCmd; -import com.github.dockerjava.api.command.ListContainersCmd; -import com.github.dockerjava.api.command.ListImagesCmd; -import com.github.dockerjava.api.command.ListNetworksCmd; -import com.github.dockerjava.api.command.ListVolumesCmd; -import com.github.dockerjava.api.command.LoadImageCmd; -import com.github.dockerjava.api.command.LogContainerCmd; -import com.github.dockerjava.api.command.PauseContainerCmd; -import com.github.dockerjava.api.command.PingCmd; -import com.github.dockerjava.api.command.PullImageCmd; -import com.github.dockerjava.api.command.PushImageCmd; -import com.github.dockerjava.api.command.RemoveContainerCmd; -import com.github.dockerjava.api.command.RemoveImageCmd; -import com.github.dockerjava.api.command.RemoveNetworkCmd; -import com.github.dockerjava.api.command.RemoveVolumeCmd; -import com.github.dockerjava.api.command.RestartContainerCmd; -import com.github.dockerjava.api.command.SaveImageCmd; -import com.github.dockerjava.api.command.SearchImagesCmd; -import com.github.dockerjava.api.command.StartContainerCmd; -import com.github.dockerjava.api.command.StatsCmd; -import com.github.dockerjava.api.command.StopContainerCmd; -import com.github.dockerjava.api.command.TagImageCmd; -import com.github.dockerjava.api.command.TopContainerCmd; -import com.github.dockerjava.api.command.UnpauseContainerCmd; -import com.github.dockerjava.api.command.UpdateContainerCmd; -import com.github.dockerjava.api.command.VersionCmd; -import com.github.dockerjava.api.command.WaitContainerCmd; -import com.github.dockerjava.api.command.RenameContainerCmd; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.Identifier; -import com.github.dockerjava.core.command.AttachContainerCmdImpl; -import com.github.dockerjava.core.command.AuthCmdImpl; -import com.github.dockerjava.core.command.BuildImageCmdImpl; -import com.github.dockerjava.core.command.CommitCmdImpl; -import com.github.dockerjava.core.command.ConnectToNetworkCmdImpl; -import com.github.dockerjava.core.command.ContainerDiffCmdImpl; -import com.github.dockerjava.core.command.CopyArchiveFromContainerCmdImpl; -import com.github.dockerjava.core.command.CopyArchiveToContainerCmdImpl; -import com.github.dockerjava.core.command.CopyFileFromContainerCmdImpl; -import com.github.dockerjava.core.command.CreateContainerCmdImpl; -import com.github.dockerjava.core.command.CreateImageCmdImpl; -import com.github.dockerjava.core.command.CreateNetworkCmdImpl; -import com.github.dockerjava.core.command.CreateVolumeCmdImpl; -import com.github.dockerjava.core.command.DisconnectFromNetworkCmdImpl; -import com.github.dockerjava.core.command.EventsCmdImpl; -import com.github.dockerjava.core.command.ExecCreateCmdImpl; -import com.github.dockerjava.core.command.ExecStartCmdImpl; -import com.github.dockerjava.core.command.InfoCmdImpl; -import com.github.dockerjava.core.command.InpectNetworkCmdImpl; -import com.github.dockerjava.core.command.InspectContainerCmdImpl; -import com.github.dockerjava.core.command.InspectExecCmdImpl; -import com.github.dockerjava.core.command.InspectImageCmdImpl; -import com.github.dockerjava.core.command.InspectVolumeCmdImpl; -import com.github.dockerjava.core.command.KillContainerCmdImpl; -import com.github.dockerjava.core.command.ListContainersCmdImpl; -import com.github.dockerjava.core.command.ListImagesCmdImpl; -import com.github.dockerjava.core.command.ListNetworksCmdImpl; -import com.github.dockerjava.core.command.ListVolumesCmdImpl; -import com.github.dockerjava.core.command.LoadImageCmdImpl; -import com.github.dockerjava.core.command.LogContainerCmdImpl; -import com.github.dockerjava.core.command.PauseContainerCmdImpl; -import com.github.dockerjava.core.command.PingCmdImpl; -import com.github.dockerjava.core.command.PullImageCmdImpl; -import com.github.dockerjava.core.command.PushImageCmdImpl; -import com.github.dockerjava.core.command.RemoveContainerCmdImpl; -import com.github.dockerjava.core.command.RemoveImageCmdImpl; -import com.github.dockerjava.core.command.RemoveNetworkCmdImpl; -import com.github.dockerjava.core.command.RemoveVolumeCmdImpl; -import com.github.dockerjava.core.command.RestartContainerCmdImpl; -import com.github.dockerjava.core.command.SaveImageCmdImpl; -import com.github.dockerjava.core.command.SearchImagesCmdImpl; -import com.github.dockerjava.core.command.StartContainerCmdImpl; -import com.github.dockerjava.core.command.StatsCmdImpl; -import com.github.dockerjava.core.command.StopContainerCmdImpl; -import com.github.dockerjava.core.command.TagImageCmdImpl; -import com.github.dockerjava.core.command.TopContainerCmdImpl; -import com.github.dockerjava.core.command.UnpauseContainerCmdImpl; -import com.github.dockerjava.core.command.UpdateContainerCmdImpl; -import com.github.dockerjava.core.command.VersionCmdImpl; -import com.github.dockerjava.core.command.WaitContainerCmdImpl; -import com.github.dockerjava.core.command.RenameContainerCmdImpl; - -import javax.annotation.Nonnull; - -/** - * @author Konstantin Pelykh (kpelykh@gmail.com) - * @see "https://github.com/docker/docker/blob/master/api/client/commands.go" - */ -public class DockerClientImpl implements Closeable, DockerClient { - - private final DockerClientConfig dockerClientConfig; - - private DockerCmdExecFactory dockerCmdExecFactory; - - private DockerClientImpl() { - this(DefaultDockerClientConfig.createDefaultConfigBuilder().build()); - } - - private DockerClientImpl(String serverUrl) { - this(configWithServerUrl(serverUrl)); - } - - private DockerClientImpl(DockerClientConfig dockerClientConfig) { - checkNotNull(dockerClientConfig, "config was not specified"); - this.dockerClientConfig = dockerClientConfig; - } - - private static DockerClientConfig configWithServerUrl(String serverUrl) { - return DefaultDockerClientConfig.createDefaultConfigBuilder().withDockerHost(serverUrl).build(); - } - - public static DockerClientImpl getInstance() { - return new DockerClientImpl(); - } - - public static DockerClientImpl getInstance(DockerClientConfig dockerClientConfig) { - return new DockerClientImpl(dockerClientConfig); - } - - public static DockerClientImpl getInstance(String serverUrl) { - return new DockerClientImpl(serverUrl); - } - - public DockerClientImpl withDockerCmdExecFactory(DockerCmdExecFactory dockerCmdExecFactory) { - checkNotNull(dockerCmdExecFactory, "dockerCmdExecFactory was not specified"); - this.dockerCmdExecFactory = dockerCmdExecFactory; - this.dockerCmdExecFactory.init(dockerClientConfig); - return this; - } - - private DockerCmdExecFactory getDockerCmdExecFactory() { - checkNotNull(dockerCmdExecFactory, "dockerCmdExecFactory was not specified"); - return dockerCmdExecFactory; - } - - @Override - public AuthConfig authConfig() { - checkNotNull(dockerClientConfig.getRegistryUsername(), "Configured username is null."); - checkNotNull(dockerClientConfig.getRegistryUrl(), "Configured serverAddress is null."); - - return new AuthConfig() - .withUsername(dockerClientConfig.getRegistryUsername()) - .withPassword(dockerClientConfig.getRegistryPassword()) - .withEmail(dockerClientConfig.getRegistryEmail()) - .withRegistryAddress(dockerClientConfig.getRegistryUrl()); - } - - /** - * * MISC API * - */ - - /** - * Authenticate with the server, useful for checking authentication. - */ - @Override - public AuthCmd authCmd() { - return new AuthCmdImpl(getDockerCmdExecFactory().createAuthCmdExec(), authConfig()); - } - - @Override - public InfoCmd infoCmd() { - return new InfoCmdImpl(getDockerCmdExecFactory().createInfoCmdExec()); - } - - @Override - public PingCmd pingCmd() { - return new PingCmdImpl(getDockerCmdExecFactory().createPingCmdExec()); - } - - @Override - public VersionCmd versionCmd() { - return new VersionCmdImpl(getDockerCmdExecFactory().createVersionCmdExec()); - } - - /** - * * IMAGE API * - */ - @Override - public PullImageCmd pullImageCmd(String repository) { - return new PullImageCmdImpl(getDockerCmdExecFactory().createPullImageCmdExec(), - dockerClientConfig.effectiveAuthConfig(repository), repository); - } - - @Override - public PushImageCmd pushImageCmd(String name) { - PushImageCmd cmd = new PushImageCmdImpl(getDockerCmdExecFactory().createPushImageCmdExec(), name); - - AuthConfig cfg = dockerClientConfig.effectiveAuthConfig(name); - if (cfg != null) { - cmd.withAuthConfig(cfg); - } - return cmd; - } - - @Override - public PushImageCmd pushImageCmd(Identifier identifier) { - PushImageCmd cmd = pushImageCmd(identifier.repository.name); - if (identifier.tag.isPresent()) { - cmd.withTag(identifier.tag.get()); - } - - AuthConfig cfg = dockerClientConfig.effectiveAuthConfig(identifier.repository.name); - if (cfg != null) { - cmd.withAuthConfig(cfg); - } - - return cmd; - } - - @Override - public SaveImageCmd saveImageCmd(String name) { - return new SaveImageCmdImpl(getDockerCmdExecFactory().createSaveImageCmdExec(), name); - } - - @Override - public CreateImageCmd createImageCmd(String repository, InputStream imageStream) { - return new CreateImageCmdImpl(getDockerCmdExecFactory().createCreateImageCmdExec(), repository, imageStream); - } - - @Override - public LoadImageCmd loadImageCmd(@Nonnull InputStream imageStream) { - return new LoadImageCmdImpl(getDockerCmdExecFactory().createLoadImageCmdExec(), imageStream); - } - - @Override - public SearchImagesCmd searchImagesCmd(String term) { - return new SearchImagesCmdImpl(getDockerCmdExecFactory().createSearchImagesCmdExec(), term); - } - - @Override - public RemoveImageCmd removeImageCmd(String imageId) { - return new RemoveImageCmdImpl(getDockerCmdExecFactory().createRemoveImageCmdExec(), imageId); - } - - @Override - public ListImagesCmd listImagesCmd() { - return new ListImagesCmdImpl(getDockerCmdExecFactory().createListImagesCmdExec()); - } - - @Override - public InspectImageCmd inspectImageCmd(String imageId) { - return new InspectImageCmdImpl(getDockerCmdExecFactory().createInspectImageCmdExec(), imageId); - } - - /** - * * CONTAINER API * - */ - - @Override - public ListContainersCmd listContainersCmd() { - return new ListContainersCmdImpl(getDockerCmdExecFactory().createListContainersCmdExec()); - } - - @Override - public CreateContainerCmd createContainerCmd(String image) { - return new CreateContainerCmdImpl(getDockerCmdExecFactory().createCreateContainerCmdExec(), image); - } - - @Override - public StartContainerCmd startContainerCmd(String containerId) { - return new StartContainerCmdImpl(getDockerCmdExecFactory().createStartContainerCmdExec(), containerId); - } - - @Override - public InspectContainerCmd inspectContainerCmd(String containerId) { - return new InspectContainerCmdImpl(getDockerCmdExecFactory().createInspectContainerCmdExec(), containerId); - } - - @Override - public ExecCreateCmd execCreateCmd(String containerId) { - return new ExecCreateCmdImpl(getDockerCmdExecFactory().createExecCmdExec(), containerId); - } - - @Override - public RemoveContainerCmd removeContainerCmd(String containerId) { - return new RemoveContainerCmdImpl(getDockerCmdExecFactory().createRemoveContainerCmdExec(), containerId); - } - - @Override - public WaitContainerCmd waitContainerCmd(String containerId) { - return new WaitContainerCmdImpl(getDockerCmdExecFactory().createWaitContainerCmdExec(), containerId); - } - - @Override - public AttachContainerCmd attachContainerCmd(String containerId) { - return new AttachContainerCmdImpl(getDockerCmdExecFactory().createAttachContainerCmdExec(), containerId); - } - - @Override - public ExecStartCmd execStartCmd(String execId) { - return new ExecStartCmdImpl(getDockerCmdExecFactory().createExecStartCmdExec(), execId); - } - - @Override - public InspectExecCmd inspectExecCmd(String execId) { - return new InspectExecCmdImpl(getDockerCmdExecFactory().createInspectExecCmdExec(), execId); - } - - @Override - public LogContainerCmd logContainerCmd(String containerId) { - return new LogContainerCmdImpl(getDockerCmdExecFactory().createLogContainerCmdExec(), containerId); - } - - @Override - public CopyFileFromContainerCmd copyFileFromContainerCmd(String containerId, String resource) { - return new CopyFileFromContainerCmdImpl(getDockerCmdExecFactory().createCopyFileFromContainerCmdExec(), - containerId, resource); - } - - @Override - public CopyArchiveFromContainerCmd copyArchiveFromContainerCmd(String containerId, String resource) { - return new CopyArchiveFromContainerCmdImpl(getDockerCmdExecFactory().createCopyArchiveFromContainerCmdExec(), - containerId, resource); - } - - @Override - public CopyArchiveToContainerCmd copyArchiveToContainerCmd(String containerId) { - return new CopyArchiveToContainerCmdImpl(getDockerCmdExecFactory().createCopyArchiveToContainerCmdExec(), - containerId); - } - - @Override - public ContainerDiffCmd containerDiffCmd(String containerId) { - return new ContainerDiffCmdImpl(getDockerCmdExecFactory().createContainerDiffCmdExec(), containerId); - } - - @Override - public StopContainerCmd stopContainerCmd(String containerId) { - return new StopContainerCmdImpl(getDockerCmdExecFactory().createStopContainerCmdExec(), containerId); - } - - @Override - public KillContainerCmd killContainerCmd(String containerId) { - return new KillContainerCmdImpl(getDockerCmdExecFactory().createKillContainerCmdExec(), containerId); - } - - @Override - public UpdateContainerCmd updateContainerCmd(@Nonnull String containerId) { - return new UpdateContainerCmdImpl(getDockerCmdExecFactory().createUpdateContainerCmdExec(), containerId); - } - - @Override - public RenameContainerCmd renameContainerCmd(@Nonnull String containerId) { - return new RenameContainerCmdImpl(getDockerCmdExecFactory().createRenameContainerCmdExec(), containerId); - } - - @Override - public RestartContainerCmd restartContainerCmd(String containerId) { - return new RestartContainerCmdImpl(getDockerCmdExecFactory().createRestartContainerCmdExec(), containerId); - } - - @Override - public CommitCmd commitCmd(String containerId) { - return new CommitCmdImpl(getDockerCmdExecFactory().createCommitCmdExec(), containerId); - } - - @Override - public BuildImageCmd buildImageCmd() { - return new BuildImageCmdImpl(getDockerCmdExecFactory().createBuildImageCmdExec()); - } - - @Override - public BuildImageCmd buildImageCmd(File dockerFileOrFolder) { - return new BuildImageCmdImpl(getDockerCmdExecFactory().createBuildImageCmdExec(), dockerFileOrFolder); - } - - @Override - public BuildImageCmd buildImageCmd(InputStream tarInputStream) { - return new BuildImageCmdImpl(getDockerCmdExecFactory().createBuildImageCmdExec(), tarInputStream); - } - - @Override - public TopContainerCmd topContainerCmd(String containerId) { - return new TopContainerCmdImpl(getDockerCmdExecFactory().createTopContainerCmdExec(), containerId); - } - - @Override - public TagImageCmd tagImageCmd(String imageId, String repository, String tag) { - return new TagImageCmdImpl(getDockerCmdExecFactory().createTagImageCmdExec(), imageId, repository, tag); - } - - @Override - public PauseContainerCmd pauseContainerCmd(String containerId) { - return new PauseContainerCmdImpl(getDockerCmdExecFactory().createPauseContainerCmdExec(), containerId); - } - - @Override - public UnpauseContainerCmd unpauseContainerCmd(String containerId) { - return new UnpauseContainerCmdImpl(getDockerCmdExecFactory().createUnpauseContainerCmdExec(), containerId); - } - - @Override - public EventsCmd eventsCmd() { - return new EventsCmdImpl(getDockerCmdExecFactory().createEventsCmdExec()); - } - - @Override - public StatsCmd statsCmd(String containerId) { - return new StatsCmdImpl(getDockerCmdExecFactory().createStatsCmdExec(), containerId); - } - - @Override - public CreateVolumeCmd createVolumeCmd() { - return new CreateVolumeCmdImpl(getDockerCmdExecFactory().createCreateVolumeCmdExec()); - } - - @Override - public InspectVolumeCmd inspectVolumeCmd(String name) { - return new InspectVolumeCmdImpl(getDockerCmdExecFactory().createInspectVolumeCmdExec(), name); - } - - @Override - public RemoveVolumeCmd removeVolumeCmd(String name) { - return new RemoveVolumeCmdImpl(getDockerCmdExecFactory().createRemoveVolumeCmdExec(), name); - } - - @Override - public ListVolumesCmd listVolumesCmd() { - return new ListVolumesCmdImpl(getDockerCmdExecFactory().createListVolumesCmdExec()); - } - - @Override - public ListNetworksCmd listNetworksCmd() { - return new ListNetworksCmdImpl(getDockerCmdExecFactory().createListNetworksCmdExec()); - } - - @Override - public InspectNetworkCmd inspectNetworkCmd() { - return new InpectNetworkCmdImpl(getDockerCmdExecFactory().createInspectNetworkCmdExec()); - } - - @Override - public CreateNetworkCmd createNetworkCmd() { - return new CreateNetworkCmdImpl(getDockerCmdExecFactory().createCreateNetworkCmdExec()); - } - - @Override - public RemoveNetworkCmd removeNetworkCmd(String networkId) { - return new RemoveNetworkCmdImpl(getDockerCmdExecFactory().createRemoveNetworkCmdExec(), networkId); - } - - @Override - public ConnectToNetworkCmd connectToNetworkCmd() { - return new ConnectToNetworkCmdImpl(getDockerCmdExecFactory().createConnectToNetworkCmdExec()); - } - - @Override - public DisconnectFromNetworkCmd disconnectFromNetworkCmd() { - return new DisconnectFromNetworkCmdImpl(getDockerCmdExecFactory().createDisconnectFromNetworkCmdExec()); - } - - @Override - public void close() throws IOException { - getDockerCmdExecFactory().close(); - } - -} diff --git a/src/main/java/com/github/dockerjava/core/LocalDirectorySSLConfig.java b/src/main/java/com/github/dockerjava/core/LocalDirectorySSLConfig.java deleted file mode 100644 index 961c282de..000000000 --- a/src/main/java/com/github/dockerjava/core/LocalDirectorySSLConfig.java +++ /dev/null @@ -1,108 +0,0 @@ -package com.github.dockerjava.core; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.io.File; -import java.io.Serializable; -import java.nio.file.Files; -import java.nio.file.Paths; -import java.security.Security; - -import javax.net.ssl.SSLContext; - -import org.bouncycastle.jce.provider.BouncyCastleProvider; -import org.glassfish.jersey.SslConfigurator; - -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.core.util.CertificateUtils; - -/** - * SSL Config from local files. - */ -public class LocalDirectorySSLConfig implements SSLConfig, Serializable { - - private static final long serialVersionUID = -4736328026418377358L; - - private final String dockerCertPath; - - public LocalDirectorySSLConfig(String dockerCertPath) { - checkNotNull(dockerCertPath); - this.dockerCertPath = dockerCertPath; - } - - public String getDockerCertPath() { - return dockerCertPath; - } - - @Override - public SSLContext getSSLContext() { - - boolean certificatesExist = CertificateUtils.verifyCertificatesExist(dockerCertPath); - - if (certificatesExist) { - - try { - - Security.addProvider(new BouncyCastleProvider()); - - // properties acrobatics not needed for java > 1.6 - String httpProtocols = System.getProperty("https.protocols"); - System.setProperty("https.protocols", "TLSv1"); - SslConfigurator sslConfig = SslConfigurator.newInstance(true); - if (httpProtocols != null) { - System.setProperty("https.protocols", httpProtocols); - } - - String caPemPath = dockerCertPath + File.separator + "ca.pem"; - String keyPemPath = dockerCertPath + File.separator + "key.pem"; - String certPemPath = dockerCertPath + File.separator + "cert.pem"; - - String keypem = new String(Files.readAllBytes(Paths.get(keyPemPath))); - String certpem = new String(Files.readAllBytes(Paths.get(certPemPath))); - String capem = new String(Files.readAllBytes(Paths.get(caPemPath))); - - sslConfig.keyStore(CertificateUtils.createKeyStore(keypem, certpem)); - sslConfig.keyStorePassword("docker"); - sslConfig.trustStore(CertificateUtils.createTrustStore(capem)); - - return sslConfig.createSSLContext(); - - } catch (Exception e) { - throw new DockerClientException(e.getMessage(), e); - } - - } - - return null; - - } - - @Override - public boolean equals(Object o) { - if (this == o) { - return true; - } - if (o == null || getClass() != o.getClass()) { - return false; - } - - LocalDirectorySSLConfig that = (LocalDirectorySSLConfig) o; - - if (!dockerCertPath.equals(that.dockerCertPath)) { - return false; - } - - return true; - } - - @Override - public int hashCode() { - return dockerCertPath.hashCode(); - } - - @Override - public String toString() { - return new StringBuilder().append(this.getClass().getSimpleName()).append("{").append("dockerCertPath=") - .append(dockerCertPath).append("}").toString(); - } -} diff --git a/src/main/java/com/github/dockerjava/core/RemoteApiVersion.java b/src/main/java/com/github/dockerjava/core/RemoteApiVersion.java deleted file mode 100644 index f40b5b4d5..000000000 --- a/src/main/java/com/github/dockerjava/core/RemoteApiVersion.java +++ /dev/null @@ -1,176 +0,0 @@ -package com.github.dockerjava.core; - -import com.google.common.base.MoreObjects; -import com.google.common.base.Objects; -import com.google.common.base.Preconditions; -import com.google.common.base.Strings; - -import java.io.Serializable; -import java.util.regex.Matcher; -import java.util.regex.Pattern; - -/** - * Bean to encapsulate the version of the Docker Remote (REST) - * API - *

- * Contains the minor and major version of the API as well as operations to compare API versions. - * - * @author Marcus Thiesen - */ -public class RemoteApiVersion implements Serializable { - private static final long serialVersionUID = -5382212999262115459L; - - private static final Pattern VERSION_REGEX = Pattern.compile("v?(\\d+)\\.(\\d+)"); - - /** - * Online documentation is not available anymore. - */ - public static final RemoteApiVersion VERSION_1_7 = RemoteApiVersion.create(1, 7); - - /** - * @see Docker API 1.16 - */ - public static final RemoteApiVersion VERSION_1_16 = RemoteApiVersion.create(1, 16); - - /** - * @see Docker API 1.17 - */ - public static final RemoteApiVersion VERSION_1_17 = RemoteApiVersion.create(1, 17); - - /** - * @see Docker API 1.18 - */ - public static final RemoteApiVersion VERSION_1_18 = RemoteApiVersion.create(1, 18); - - /** - * @see Docker API 1.19 - */ - public static final RemoteApiVersion VERSION_1_19 = RemoteApiVersion.create(1, 19); - - /** - * @see Docker API 1.20 - */ - public static final RemoteApiVersion VERSION_1_20 = RemoteApiVersion.create(1, 20); - - /** - * @see Docker API 1.21 - */ - public static final RemoteApiVersion VERSION_1_21 = RemoteApiVersion.create(1, 21); - - /** - * @see Docker API 1.22 - */ - public static final RemoteApiVersion VERSION_1_22 = RemoteApiVersion.create(1, 22); - - /** - * @see Docker API 1.22 - */ - public static final RemoteApiVersion VERSION_1_23 = RemoteApiVersion.create(1, 23); - - /** - * @see Docker API 1.22 - */ - public static final RemoteApiVersion VERSION_1_24 = RemoteApiVersion.create(1, 24); - - /** - * Unknown, docker doesn't reflect reality. I.e. we implemented method, but for javadoc it not clear when it was added. - */ - public static final RemoteApiVersion UNKNOWN_VERSION = new RemoteApiVersion(0, 0) { - - @Override - public boolean isGreaterOrEqual(final RemoteApiVersion other) { - return false; - } - - @Override - public String toString() { - return MoreObjects.toStringHelper(this).addValue("UNKNOWN_VERSION").toString(); - } - - @Override - public String asWebPathPart() { - return ""; - } - }; - - private final int major; - - private final int minor; - - private RemoteApiVersion(final int major, final int minor) { - this.major = major; - this.minor = minor; - } - - public static RemoteApiVersion create(final int major, final int minor) { - Preconditions.checkArgument(major > 0, "Major version must be bigger than 0 but is " + major); - Preconditions.checkArgument(minor > 0, "Minor version must be bigger than 0 but is " + minor); - return new RemoteApiVersion(major, minor); - } - - public static RemoteApiVersion unknown() { - return UNKNOWN_VERSION; - } - - public static RemoteApiVersion parseConfig(final String version) { - Preconditions.checkArgument(version != null, "Version must not be null"); - final Matcher matcher = VERSION_REGEX.matcher(version); - if (matcher.matches()) { - return create(Integer.parseInt(matcher.group(1)), Integer.parseInt(matcher.group(2))); - } - throw new IllegalArgumentException(version + " can not be parsed"); - } - - public static RemoteApiVersion parseConfigWithDefault(final String version) { - if (Strings.isNullOrEmpty(version)) { - return UNKNOWN_VERSION; - } - - try { - return parseConfig(version); - } catch (IllegalArgumentException e) { - return UNKNOWN_VERSION; - } - } - - public boolean isGreaterOrEqual(final RemoteApiVersion other) { - if (major >= other.major && minor >= other.minor) { - return true; - } - return false; - } - - /** - * @return String representation of version. i.e. "1.22" - */ - public String getVersion() { - return major + "." + minor; - } - - // CHECKSTYLE:OFF - @Override - public boolean equals(final Object o) { - if (this == o) - return true; - if (o == null || getClass() != o.getClass()) - return false; - final RemoteApiVersion that = (RemoteApiVersion) o; - return Objects.equal(major, that.major) && Objects.equal(minor, that.minor); - } - - // CHECKSTYLE:ON - - @Override - public int hashCode() { - return Objects.hashCode(major, minor); - } - - @Override - public String toString() { - return MoreObjects.toStringHelper(this).add("major", major).add("minor", minor).toString(); - } - - public String asWebPathPart() { - return "v" + major + "." + minor; - } -} diff --git a/src/main/java/com/github/dockerjava/core/SSLConfig.java b/src/main/java/com/github/dockerjava/core/SSLConfig.java deleted file mode 100644 index 0346aa610..000000000 --- a/src/main/java/com/github/dockerjava/core/SSLConfig.java +++ /dev/null @@ -1,22 +0,0 @@ -package com.github.dockerjava.core; - -import java.security.KeyManagementException; -import java.security.KeyStoreException; -import java.security.NoSuchAlgorithmException; -import java.security.UnrecoverableKeyException; - -import javax.net.ssl.SSLContext; - -/** - * Get an SSL Config. Allows for various different implementations. - */ -public interface SSLConfig { - - /** - * Get the SSL Context, from wherever it comes (file, keystore). - * - * @return an SSL context. - */ - SSLContext getSSLContext() throws KeyManagementException, UnrecoverableKeyException, NoSuchAlgorithmException, - KeyStoreException; -} diff --git a/src/main/java/com/github/dockerjava/core/async/JsonStreamProcessor.java b/src/main/java/com/github/dockerjava/core/async/JsonStreamProcessor.java deleted file mode 100644 index 9da43b62b..000000000 --- a/src/main/java/com/github/dockerjava/core/async/JsonStreamProcessor.java +++ /dev/null @@ -1,71 +0,0 @@ -/* - * Created on 18.06.2015 - */ -package com.github.dockerjava.core.async; - -import java.io.IOException; -import java.io.InputStream; - -import com.fasterxml.jackson.core.JsonFactory; -import com.fasterxml.jackson.core.JsonParser; -import com.fasterxml.jackson.core.JsonToken; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.fasterxml.jackson.databind.node.ObjectNode; -import com.github.dockerjava.api.async.ResultCallback; - -/** - * - * @author Marcus Linke - * - */ -public class JsonStreamProcessor implements ResponseStreamProcessor { - - private static final JsonFactory JSON_FACTORY = new JsonFactory(); - - private static final ObjectMapper OBJECT_MAPPER = new ObjectMapper(); - - private final Class clazz; - - public JsonStreamProcessor(Class clazz) { - this.clazz = clazz; - } - - @Override - public void processResponseStream(InputStream response, ResultCallback resultCallback) { - - resultCallback.onStart(response); - OBJECT_MAPPER.configure(JsonParser.Feature.AUTO_CLOSE_SOURCE, true); - - try { - JsonParser jp = JSON_FACTORY.createParser(response); - Boolean closed = jp.isClosed(); - JsonToken nextToken = jp.nextToken(); - while (!closed && nextToken != null && nextToken != JsonToken.END_OBJECT) { - try { - ObjectNode objectNode = OBJECT_MAPPER.readTree(jp); - // exclude empty item serialization into class #461 - if (!objectNode.isEmpty(null)) { - T next = OBJECT_MAPPER.treeToValue(objectNode, clazz); - resultCallback.onNext(next); - } - } catch (Exception e) { - resultCallback.onError(e); - } - - closed = jp.isClosed(); - nextToken = jp.nextToken(); - } - } catch (Throwable t) { - resultCallback.onError(t); - } finally { - try { - response.close(); - } catch (IOException e) { - resultCallback.onError(e); - } finally { - resultCallback.onComplete(); - } - } - - } -} diff --git a/src/main/java/com/github/dockerjava/core/async/ResultCallbackTemplate.java b/src/main/java/com/github/dockerjava/core/async/ResultCallbackTemplate.java deleted file mode 100644 index f120012af..000000000 --- a/src/main/java/com/github/dockerjava/core/async/ResultCallbackTemplate.java +++ /dev/null @@ -1,134 +0,0 @@ -/* - * Created on 16.06.2015 - */ -package com.github.dockerjava.core.async; - -import java.io.Closeable; -import java.io.IOException; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.TimeUnit; - -import javax.annotation.CheckForNull; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.google.common.base.Throwables; - -/** - * Abstract template implementation of {@link ResultCallback} - * - * @author Marcus Linke - * - */ -public abstract class ResultCallbackTemplate, A_RES_T> implements - ResultCallback { - - private static final Logger LOGGER = LoggerFactory.getLogger(ResultCallbackTemplate.class); - - private final CountDownLatch started = new CountDownLatch(1); - - private final CountDownLatch completed = new CountDownLatch(1); - - private Closeable stream; - - private boolean closed = false; - - private Throwable firstError = null; - - @Override - public void onStart(Closeable stream) { - this.stream = stream; - this.closed = false; - started.countDown(); - } - - @Override - public void onError(Throwable throwable) { - - if (closed) return; - - if (this.firstError == null) { - this.firstError = throwable; - } - - try { - LOGGER.error("Error during callback", throwable); - } finally { - try { - close(); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - } - - @Override - public void onComplete() { - try { - close(); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - @Override - public void close() throws IOException { - closed = true; - if (stream != null) { - stream.close(); - } - completed.countDown(); - } - - /** - * Blocks until {@link ResultCallback#onComplete()} was called - */ - @SuppressWarnings("unchecked") - public RC_T awaitCompletion() throws InterruptedException { - completed.await(); - // eventually (re)throws RuntimeException - getFirstError(); - return (RC_T) this; - } - - /** - * Blocks until {@link ResultCallback#onComplete()} was called or the given timeout occurs - * @return {@code true} if completed and {@code false} if the waiting time elapsed - * before {@link ResultCallback#onComplete()} was called. - */ - public boolean awaitCompletion(long timeout, TimeUnit timeUnit) throws InterruptedException { - return completed.await(timeout, timeUnit); - } - - /** - * Blocks until {@link ResultCallback#onStart()} was called. {@link ResultCallback#onStart()} is called when the request was processed - * on the server side and the response is incoming. - */ - @SuppressWarnings("unchecked") - public RC_T awaitStarted() throws InterruptedException { - started.await(); - return (RC_T) this; - } - - /** - * Blocks until {@link ResultCallback#onStart()} was called or the given timeout occurs. {@link ResultCallback#onStart()} is called when - * the request was processed on the server side and the response is incoming. - * @return {@code true} if started and {@code false} if the waiting time elapsed - * before {@link ResultCallback#onStart()} was called. - */ - public boolean awaitStarted(long timeout, TimeUnit timeUnit) throws InterruptedException { - return started.await(timeout, timeUnit); - } - - @CheckForNull - protected RuntimeException getFirstError() { - if (firstError != null) { - // this call throws a RuntimeException - return Throwables.propagate(firstError); - } else { - return null; - } - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/AbstrAuthCfgDockerCmd.java b/src/main/java/com/github/dockerjava/core/command/AbstrAuthCfgDockerCmd.java deleted file mode 100644 index 64e0e27f8..000000000 --- a/src/main/java/com/github/dockerjava/core/command/AbstrAuthCfgDockerCmd.java +++ /dev/null @@ -1,37 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import com.github.dockerjava.api.command.DockerCmd; -import com.github.dockerjava.api.command.DockerCmdSyncExec; -import com.github.dockerjava.api.model.AuthConfig; - -public abstract class AbstrAuthCfgDockerCmd, RES_T> extends AbstrDockerCmd { - - public AbstrAuthCfgDockerCmd(DockerCmdSyncExec execution, AuthConfig authConfig) { - super(execution); - withOptionalAuthConfig(authConfig); - } - - public AbstrAuthCfgDockerCmd(DockerCmdSyncExec execution) { - super(execution); - } - - private AuthConfig authConfig; - - public AuthConfig getAuthConfig() { - return authConfig; - } - - public T withAuthConfig(AuthConfig authConfig) { - checkNotNull(authConfig, "authConfig was not specified"); - return withOptionalAuthConfig(authConfig); - } - - @SuppressWarnings("unchecked") - private T withOptionalAuthConfig(AuthConfig authConfig) { - this.authConfig = authConfig; - return (T) this; - } - -} diff --git a/src/main/java/com/github/dockerjava/core/command/AbstrDockerCmd.java b/src/main/java/com/github/dockerjava/core/command/AbstrDockerCmd.java deleted file mode 100644 index 9e1e775c5..000000000 --- a/src/main/java/com/github/dockerjava/core/command/AbstrDockerCmd.java +++ /dev/null @@ -1,54 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.io.IOException; - -import org.apache.commons.codec.binary.Base64; -import org.apache.commons.lang.builder.ReflectionToStringBuilder; -import org.apache.commons.lang.builder.ToStringStyle; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.api.command.DockerCmd; -import com.github.dockerjava.api.command.DockerCmdSyncExec; -import com.github.dockerjava.api.command.SyncDockerCmd; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.AuthConfig; - -public abstract class AbstrDockerCmd, RES_T> implements SyncDockerCmd { - - private static final Logger LOGGER = LoggerFactory.getLogger(AbstrDockerCmd.class); - - protected DockerCmdSyncExec execution; - - public AbstrDockerCmd(DockerCmdSyncExec execution) { - checkNotNull(execution, "execution was not specified"); - this.execution = execution; - } - - @Override - @SuppressWarnings("unchecked") - public RES_T exec() throws DockerException { - LOGGER.debug("Cmd: {}", this); - return execution.exec((CMD_T) this); - } - - @Override - public void close() { - } - - @Override - public String toString() { - return ReflectionToStringBuilder.toString(this, ToStringStyle.SIMPLE_STYLE); - } - - protected String registryAuth(AuthConfig authConfig) { - try { - return Base64.encodeBase64String(new ObjectMapper().writeValueAsString(authConfig).getBytes()); - } catch (IOException e) { - throw new RuntimeException(e); - } - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/AttachContainerCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/AttachContainerCmdImpl.java deleted file mode 100644 index 89e27c8ea..000000000 --- a/src/main/java/com/github/dockerjava/core/command/AttachContainerCmdImpl.java +++ /dev/null @@ -1,119 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.io.InputStream; - -import com.github.dockerjava.api.command.AttachContainerCmd; -import com.github.dockerjava.api.model.Frame; - -/** - * Attach to container - * - * @param logs - * - true or false, includes logs. Defaults to false. - * - * @param followStream - * - true or false, return stream. Defaults to false. - * @param stdout - * - true or false, includes stdout log. Defaults to false. - * @param stderr - * - true or false, includes stderr log. Defaults to false. - * @param stdin - * - null or {@link InputStream}, pass stream to stdin of the container. - * @param timestamps - * - true or false, if true, print timestamps for every log line. Defaults to false. - */ -public class AttachContainerCmdImpl extends AbstrAsyncDockerCmd implements - AttachContainerCmd { - - private String containerId; - - private Boolean logs, followStream, timestamps, stdout, stderr; - - private InputStream stdin; - - public AttachContainerCmdImpl(AttachContainerCmd.Exec exec, String containerId) { - super(exec); - withContainerId(containerId); - } - - @Override - public String getContainerId() { - return containerId; - } - - @Override - public Boolean hasLogsEnabled() { - return logs; - } - - @Override - public Boolean hasFollowStreamEnabled() { - return followStream; - } - - @Override - public Boolean hasTimestampsEnabled() { - return timestamps; - } - - @Override - public Boolean hasStdoutEnabled() { - return stdout; - } - - @Override - public Boolean hasStderrEnabled() { - return stderr; - } - - @Override - public InputStream getStdin() { - return stdin; - } - - @Override - public AttachContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; - return this; - } - - @Override - public AttachContainerCmd withFollowStream(Boolean followStream) { - this.followStream = followStream; - return this; - } - - @Override - public AttachContainerCmd withTimestamps(Boolean timestamps) { - this.timestamps = timestamps; - return this; - } - - @Override - public AttachContainerCmd withStdOut(Boolean stdout) { - this.stdout = stdout; - return this; - } - - @Override - public AttachContainerCmd withStdErr(Boolean stderr) { - this.stderr = stderr; - return this; - } - - @Override - public AttachContainerCmd withStdIn(InputStream stdin) { - checkNotNull(stdin, "stdin was not specified"); - this.stdin = stdin; - return this; - } - - @Override - public AttachContainerCmd withLogs(Boolean logs) { - this.logs = logs; - return this; - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/AuthCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/AuthCmdImpl.java deleted file mode 100644 index e0a790c32..000000000 --- a/src/main/java/com/github/dockerjava/core/command/AuthCmdImpl.java +++ /dev/null @@ -1,24 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.AuthCmd; -import com.github.dockerjava.api.exception.UnauthorizedException; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.AuthResponse; - -/** - * - * Authenticate with the server, useful for checking authentication. - * - */ -public class AuthCmdImpl extends AbstrAuthCfgDockerCmd implements AuthCmd { - - public AuthCmdImpl(AuthCmd.Exec exec, AuthConfig authConfig) { - super(exec); - withAuthConfig(authConfig); - } - - @Override - public AuthResponse exec() throws UnauthorizedException { - return super.exec(); - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/BuildImageCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/BuildImageCmdImpl.java deleted file mode 100644 index 487f0b264..000000000 --- a/src/main/java/com/github/dockerjava/core/command/BuildImageCmdImpl.java +++ /dev/null @@ -1,317 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.io.File; -import java.io.IOException; -import java.io.InputStream; -import java.net.URI; -import java.util.HashMap; -import java.util.Map; - -import com.github.dockerjava.api.command.BuildImageCmd; -import com.github.dockerjava.api.model.AuthConfigurations; -import com.github.dockerjava.api.model.BuildResponseItem; -import com.github.dockerjava.core.dockerfile.Dockerfile; -import com.github.dockerjava.core.util.FilePathUtil; - -/** - * - * Build an image from Dockerfile. - * - */ -public class BuildImageCmdImpl extends AbstrAsyncDockerCmd implements BuildImageCmd { - - private InputStream tarInputStream; - - private String tag; - - private Boolean noCache; - - private Boolean remove = true; - - private Boolean quiet; - - private Boolean pull; - - private AuthConfigurations buildAuthConfigs; - - private File dockerFile; - - private File baseDirectory; - - private String cpusetcpus; - - private Long memory; - - private String cpushares; - - private Boolean forcerm; - - private Long memswap; - - private Long shmsize; - - private URI remote; - - private Map buildArgs; - - public BuildImageCmdImpl(BuildImageCmd.Exec exec) { - super(exec); - } - - public BuildImageCmdImpl(BuildImageCmd.Exec exec, File dockerFileOrFolder) { - super(exec); - checkNotNull(dockerFileOrFolder, "dockerFolder is null"); - - if (dockerFileOrFolder.isDirectory()) { - withBaseDirectory(dockerFileOrFolder); - withDockerfile(new File(dockerFileOrFolder, "Dockerfile")); - } else { - withDockerfile(dockerFileOrFolder); - } - } - - public BuildImageCmdImpl(BuildImageCmd.Exec exec, InputStream tarInputStream) { - super(exec); - checkNotNull(tarInputStream, "tarInputStream is null"); - withTarInputStream(tarInputStream); - } - - // getters API - - @Override - public String getTag() { - return tag; - } - - @Override - public URI getRemote() { - return remote; - } - - @Override - public Boolean hasNoCacheEnabled() { - return noCache; - } - - @Override - public Boolean hasRemoveEnabled() { - return remove; - } - - @Override - public Boolean isForcerm() { - return forcerm; - } - - @Override - public Boolean isQuiet() { - return quiet; - } - - @Override - public Boolean hasPullEnabled() { - return pull; - } - - @Override - public String getPathToDockerfile() { - if (baseDirectory != null && dockerFile != null) { - return FilePathUtil.relativize(baseDirectory, dockerFile); - } else { - return null; - } - } - - @Override - public Long getMemory() { - return memory; - } - - @Override - public Long getMemswap() { - return memswap; - } - - @Override - public String getCpushares() { - return cpushares; - } - - @Override - public String getCpusetcpus() { - return cpusetcpus; - } - - @Override - public Map getBuildArgs() { - return buildArgs; - } - - // getter lib specific - - @Override - public AuthConfigurations getBuildAuthConfigs() { - return buildAuthConfigs; - } - - @Override - public InputStream getTarInputStream() { - return tarInputStream; - } - - /** - * @see #shmsize - */ - @Override - public Long getShmsize() { - return shmsize; - } - - // setters - - @Override - public BuildImageCmdImpl withTag(String tag) { - checkNotNull(tag, "Tag is null"); - this.tag = tag; - return this; - } - - @Override - public BuildImageCmd withRemote(URI remote) { - this.remote = remote; - return this; - } - - @Override - public BuildImageCmdImpl withNoCache(Boolean noCache) { - this.noCache = noCache; - return this; - } - - @Override - public BuildImageCmdImpl withRemove(Boolean rm) { - this.remove = rm; - return this; - } - - @Override - public BuildImageCmd withForcerm(Boolean forcerm) { - this.forcerm = forcerm; - return this; - } - - @Override - public BuildImageCmdImpl withQuiet(Boolean quiet) { - this.quiet = quiet; - return this; - } - - @Override - public BuildImageCmdImpl withPull(Boolean pull) { - this.pull = pull; - return this; - } - - @Override - public BuildImageCmd withMemory(Long memory) { - this.memory = memory; - return this; - } - - @Override - public BuildImageCmd withMemswap(Long memswap) { - this.memswap = memswap; - return this; - } - - @Override - public BuildImageCmd withCpushares(String cpushares) { - this.cpushares = cpushares; - return this; - } - - @Override - public BuildImageCmd withCpusetcpus(String cpusetcpus) { - this.cpusetcpus = cpusetcpus; - return this; - } - - @Override - public BuildImageCmd withBuildArg(String key, String value) { - if (this.buildArgs == null) { - this.buildArgs = new HashMap(); - } - this.buildArgs.put(key, value); - return this; - } - - // lib specific - - @Override - public BuildImageCmd withBaseDirectory(File baseDirectory) { - this.baseDirectory = baseDirectory; - return this; - } - - @Override - public BuildImageCmdImpl withDockerfile(File dockerfile) { - checkNotNull(dockerfile); - if (!dockerfile.exists()) { - throw new IllegalArgumentException("Dockerfile does not exist"); - } - if (!dockerfile.isFile()) { - throw new IllegalArgumentException("Not a directory"); - } - - if (baseDirectory == null) { - withBaseDirectory(dockerfile.getParentFile()); - } - - this.dockerFile = dockerfile; - - try { - withTarInputStream(new Dockerfile(dockerfile, baseDirectory).parse().buildDockerFolderTar()); - } catch (IOException e) { - // we just created the file this should never happen. - throw new RuntimeException(e); - } - return this; - } - - @Override - public BuildImageCmdImpl withTarInputStream(InputStream tarInputStream) { - checkNotNull(tarInputStream, "tarInputStream is null"); - this.tarInputStream = tarInputStream; - return this; - } - - @Override - public BuildImageCmd withBuildAuthConfigs(AuthConfigurations authConfigs) { - checkNotNull(authConfigs, "authConfig is null"); - this.buildAuthConfigs = authConfigs; - return this; - } - - /** - * @see #shmsize - */ - @Override - public BuildImageCmd withShmsize(Long shmsize) { - this.shmsize = shmsize; - return this; - } - - @Override - public void close() { - super.close(); - - try { - tarInputStream.close(); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - -} diff --git a/src/main/java/com/github/dockerjava/core/command/BuildImageResultCallback.java b/src/main/java/com/github/dockerjava/core/command/BuildImageResultCallback.java deleted file mode 100644 index 1ca276434..000000000 --- a/src/main/java/com/github/dockerjava/core/command/BuildImageResultCallback.java +++ /dev/null @@ -1,77 +0,0 @@ -/* - * Created on 21.07.2015 - */ -package com.github.dockerjava.core.command; - -import java.util.concurrent.TimeUnit; - -import javax.annotation.CheckForNull; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.model.BuildResponseItem; -import com.github.dockerjava.core.async.ResultCallbackTemplate; - -/** - * - * @author Marcus Linke - * - */ -public class BuildImageResultCallback extends ResultCallbackTemplate { - - private static final Logger LOGGER = LoggerFactory.getLogger(BuildImageResultCallback.class); - - @CheckForNull - private BuildResponseItem latestItem = null; - - @Override - public void onNext(BuildResponseItem item) { - this.latestItem = item; - LOGGER.debug(item.toString()); - } - - /** - * Awaits the image id from the response stream. - * - * @throws DockerClientException - * if the build fails. - */ - public String awaitImageId() { - try { - awaitCompletion(); - } catch (InterruptedException e) { - throw new DockerClientException("", e); - } - - return getImageId(); - } - - /** - * Awaits the image id from the response stream. - * - * @throws DockerClientException - * if the build fails or the timeout occurs. - */ - public String awaitImageId(long timeout, TimeUnit timeUnit) { - try { - awaitCompletion(timeout, timeUnit); - } catch (InterruptedException e) { - throw new DockerClientException("Awaiting image id interrupted: ", e); - } - - return getImageId(); - } - - private String getImageId() { - if (latestItem == null) { - throw new DockerClientException("Could not build image"); - } else if (!latestItem.isBuildSuccessIndicated()) { - throw new DockerClientException("Could not build image: " + latestItem.getError()); - } else { - return latestItem.getImageId(); - } - } - -} diff --git a/src/main/java/com/github/dockerjava/core/command/CopyArchiveToContainerCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/CopyArchiveToContainerCmdImpl.java deleted file mode 100644 index 0f734fde9..000000000 --- a/src/main/java/com/github/dockerjava/core/command/CopyArchiveToContainerCmdImpl.java +++ /dev/null @@ -1,147 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.io.IOException; -import java.io.InputStream; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; - -import org.apache.commons.lang.StringUtils; -import org.apache.commons.lang.builder.ToStringBuilder; - -import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.core.util.CompressArchiveUtil; - -public class CopyArchiveToContainerCmdImpl extends AbstrDockerCmd implements - CopyArchiveToContainerCmd { - - private String containerId; - - private String remotePath = "."; - - private InputStream tarInputStream; - - private String hostResource; - - private boolean noOverwriteDirNonDir = false; - - private boolean dirChildrenOnly = false; - - public CopyArchiveToContainerCmdImpl(CopyArchiveToContainerCmd.Exec exec, String containerId) { - super(exec); - withContainerId(containerId); - } - - @Override - public CopyArchiveToContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; - return this; - } - - @Override - public CopyArchiveToContainerCmd withHostResource(String hostResource) { - checkNotNull(hostResource, "hostResource was not specified"); - this.hostResource = hostResource; - return this; - } - - @Override - public CopyArchiveToContainerCmd withNoOverwriteDirNonDir(boolean noOverwriteDirNonDir) { - this.noOverwriteDirNonDir = noOverwriteDirNonDir; - return this; - } - - @Override - public CopyArchiveToContainerCmd withRemotePath(String remotePath) { - checkNotNull(remotePath, "remotePath was not specified"); - this.remotePath = remotePath; - return this; - } - - @Override - public CopyArchiveToContainerCmd withTarInputStream(InputStream tarInputStream) { - checkNotNull(tarInputStream, "tarInputStream was not specified"); - this.tarInputStream = tarInputStream; - return this; - } - - @Override - public CopyArchiveToContainerCmd withDirChildrenOnly(boolean dirChildrenOnly) { - this.dirChildrenOnly = dirChildrenOnly; - return this; - } - - @Override - public InputStream getTarInputStream() { - return tarInputStream; - } - - @Override - public String getContainerId() { - return this.containerId; - } - - @Override - public String getHostResource() { - return this.hostResource; - } - - @Override - public boolean isNoOverwriteDirNonDir() { - return this.noOverwriteDirNonDir; - } - - @Override - public String getRemotePath() { - return this.remotePath; - } - - @Override - public boolean isDirChildrenOnly() { - return this.dirChildrenOnly; - } - - @Override - public String toString() { - return new ToStringBuilder(this).append("cp ").append(hostResource).append(" ").append(containerId).append(":") - .append(remotePath).toString(); - } - - private InputStream buildUploadStream(String hostResource, boolean dirChildrenOnly) throws IOException { - Path toUpload = Files.createTempFile("docker-java", ".tar.gz"); - CompressArchiveUtil.tar(Paths.get(hostResource), toUpload, true, dirChildrenOnly); - return Files.newInputStream(toUpload); - } - - /** - * @throws com.github.dockerjava.api.exception.NotFoundException - * No such container - */ - @Override - public Void exec() throws NotFoundException { - if (StringUtils.isNotEmpty(this.hostResource)) { - // User set host resource and not directly a stream - if (this.tarInputStream != null) { - throw new DockerClientException( - "Only one of host resource or tar input stream should be defined to perform the copy, not both"); - } - // We compress the given path, call exec so that the stream is consumed and then close it our self - try (InputStream uploadStream = buildUploadStream(this.hostResource, this.dirChildrenOnly)) { - this.tarInputStream = uploadStream; - return super.exec(); - } catch (IOException e) { - throw new DockerClientException("Unable to perform tar on host resource " + this.hostResource, e); - } - } else if (this.tarInputStream == null) { - throw new DockerClientException( - "One of host resource or tar input stream must be defined to perform the copy"); - } - // User set a stream, so we will just consume it and let the user close it by him self - return super.exec(); - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/CreateContainerCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/CreateContainerCmdImpl.java deleted file mode 100644 index f3cfefa05..000000000 --- a/src/main/java/com/github/dockerjava/core/command/CreateContainerCmdImpl.java +++ /dev/null @@ -1,1018 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.api.command.CreateContainerCmd; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.ConflictException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Bind; -import com.github.dockerjava.api.model.Capability; -import com.github.dockerjava.api.model.ContainerNetwork; -import com.github.dockerjava.api.model.Device; -import com.github.dockerjava.api.model.ExposedPort; -import com.github.dockerjava.api.model.ExposedPorts; -import com.github.dockerjava.api.model.HostConfig; -import com.github.dockerjava.api.model.Link; -import com.github.dockerjava.api.model.LogConfig; -import com.github.dockerjava.api.model.LxcConf; -import com.github.dockerjava.api.model.PortBinding; -import com.github.dockerjava.api.model.Ports; -import com.github.dockerjava.api.model.RestartPolicy; -import com.github.dockerjava.api.model.Ulimit; -import com.github.dockerjava.api.model.Volume; -import com.github.dockerjava.api.model.Volumes; -import com.github.dockerjava.api.model.VolumesFrom; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -import java.util.Arrays; -import java.util.List; -import java.util.Map; - -import static com.google.common.base.Preconditions.checkNotNull; -import static java.util.Collections.singletonMap; - -/** - * Creates a new container. - * `/containers/create` - */ -@JsonInclude(Include.NON_NULL) -public class CreateContainerCmdImpl extends AbstrDockerCmd implements - CreateContainerCmd { - - private String name; - - @JsonProperty("Hostname") - private String hostName; - - @JsonProperty("Domainname") - private String domainName; - - @JsonProperty("User") - private String user; - - @JsonProperty("AttachStdin") - private Boolean attachStdin; - - @JsonProperty("AttachStdout") - private Boolean attachStdout; - - @JsonProperty("AttachStderr") - private Boolean attachStderr; - - @JsonProperty("PortSpecs") - private String[] portSpecs; - - @JsonProperty("Tty") - private Boolean tty; - - @JsonProperty("OpenStdin") - private Boolean stdinOpen; - - @JsonProperty("StdinOnce") - private Boolean stdInOnce; - - @JsonProperty("Env") - private String[] env; - - @JsonProperty("Cmd") - private String[] cmd; - - @JsonProperty("Entrypoint") - private String[] entrypoint; - - @JsonProperty("Image") - private String image; - - @JsonProperty("Volumes") - private Volumes volumes = new Volumes(); - - @JsonProperty("WorkingDir") - private String workingDir; - - @JsonProperty("MacAddress") - private String macAddress; - - @JsonProperty("NetworkDisabled") - private Boolean networkDisabled; - - @JsonProperty("ExposedPorts") - private ExposedPorts exposedPorts = new ExposedPorts(); - - /** - * @since {@link com.github.dockerjava.core.RemoteApiVersion#VERSION_1_21} - */ - @JsonProperty("StopSignal") - private String stopSignal; - - @JsonProperty("HostConfig") - private HostConfig hostConfig = new HostConfig(); - - @JsonProperty("Labels") - private Map labels; - - @JsonProperty("NetworkingConfig") - private NetworkingConfig networkingConfig; - - @JsonIgnore - private String ipv4Address = null; - - @JsonIgnore - private String ipv6Address = null; - - @JsonIgnore - private List aliases = null; - - public CreateContainerCmdImpl(CreateContainerCmd.Exec exec, String image) { - super(exec); - checkNotNull(image, "image was not specified"); - withImage(image); - } - - /** - * @throws NotFoundException - * No such container - * @throws ConflictException - * Named container already exists - */ - @Override - public CreateContainerResponse exec() throws NotFoundException, ConflictException { - //code flow taken from https://github.com/docker/docker/blob/master/runconfig/opts/parse.go - ContainerNetwork containerNetwork = null; - - if (ipv4Address != null || ipv6Address != null) { - containerNetwork = new ContainerNetwork() - .withIpamConfig(new ContainerNetwork.Ipam() - .withIpv4Address(ipv4Address) - .withIpv6Address(ipv6Address) - ); - - } - - if (hostConfig.isUserDefinedNetwork() && hostConfig.getLinks().length > 0) { - if (containerNetwork == null) { - containerNetwork = new ContainerNetwork(); - } - - containerNetwork.withLinks(hostConfig.getLinks()); - } - - if (aliases != null) { - if (containerNetwork == null) { - containerNetwork = new ContainerNetwork(); - } - - containerNetwork.withAliases(aliases); - } - - if (containerNetwork != null) { - networkingConfig = new NetworkingConfig() - .withEndpointsConfig(singletonMap(hostConfig.getNetworkMode(), containerNetwork)); - } - - return super.exec(); - } - - @Override - @JsonIgnore - public List getAliases() { - return aliases; - } - - @Override - @JsonIgnore - public Bind[] getBinds() { - return hostConfig.getBinds(); - } - - @Override - @JsonIgnore - public Integer getBlkioWeight() { - return hostConfig.getBlkioWeight(); - } - - @Override - @JsonIgnore - public Capability[] getCapAdd() { - return hostConfig.getCapAdd(); - } - - @Override - @JsonIgnore - public Capability[] getCapDrop() { - return hostConfig.getCapDrop(); - } - - @Override - public String[] getCmd() { - return cmd; - } - - @Override - @JsonIgnore - public Integer getCpuPeriod() { - return hostConfig.getCpuPeriod(); - } - - @Override - @JsonIgnore - public String getCpusetCpus() { - return hostConfig.getCpusetCpus(); - } - - @Override - @JsonIgnore - public String getCpusetMems() { - return hostConfig.getCpusetMems(); - } - - @Override - @JsonIgnore - public Integer getCpuShares() { - return hostConfig.getCpuShares(); - } - - @Override - @JsonIgnore - public Device[] getDevices() { - return hostConfig.getDevices(); - } - - @Override - @JsonIgnore - public String[] getDns() { - return hostConfig.getDns(); - } - - @Override - @JsonIgnore - public String[] getDnsSearch() { - return hostConfig.getDnsSearch(); - } - - @Override - public String getDomainName() { - return domainName; - } - - @Override - public String[] getEntrypoint() { - return entrypoint; - } - - @Override - public String[] getEnv() { - return env; - } - - @Override - @JsonIgnore - public ExposedPort[] getExposedPorts() { - return exposedPorts.getExposedPorts(); - } - - /** - * @see #stopSignal - */ - @JsonIgnore - @Override - public String getStopSignal() { - return stopSignal; - } - - @Override - @JsonIgnore - public String[] getExtraHosts() { - return hostConfig.getExtraHosts(); - } - - @Override - public String getHostName() { - return hostName; - } - - @Override - public String getImage() { - return image; - } - - @Override - public String getIpv4Address() { - return ipv4Address; - } - - @Override - public String getIpv6Address() { - return ipv6Address; - } - - @Override - @JsonIgnore - public Map getLabels() { - return labels; - } - - @Override - @JsonIgnore - public Link[] getLinks() { - return hostConfig.getLinks(); - } - - @Override - @JsonIgnore - public LxcConf[] getLxcConf() { - return hostConfig.getLxcConf(); - } - - @Override - @JsonIgnore - public LogConfig getLogConfig() { - return hostConfig.getLogConfig(); - } - - @Override - public String getMacAddress() { - return macAddress; - } - - @Override - @JsonIgnore - public Long getMemory() { - return hostConfig.getMemory(); - } - - @Override - @JsonIgnore - public Long getMemorySwap() { - return hostConfig.getMemorySwap(); - } - - @Override - public String getName() { - return name; - } - - @Override - @JsonIgnore - public String getNetworkMode() { - return hostConfig.getNetworkMode(); - } - - @Override - @JsonIgnore - public Ports getPortBindings() { - return hostConfig.getPortBindings(); - } - - @Override - public String[] getPortSpecs() { - return portSpecs; - } - - @Override - @JsonIgnore - public RestartPolicy getRestartPolicy() { - return hostConfig.getRestartPolicy(); - } - - @Override - @JsonIgnore - public Ulimit[] getUlimits() { - return hostConfig.getUlimits(); - } - - @Override - public String getUser() { - return user; - } - - @Override - @JsonIgnore - public Volume[] getVolumes() { - return volumes.getVolumes(); - } - - @Override - @JsonIgnore - public VolumesFrom[] getVolumesFrom() { - return hostConfig.getVolumesFrom(); - } - - @Override - public String getWorkingDir() { - return workingDir; - } - - @Override - public Boolean isAttachStderr() { - return attachStderr; - } - - @Override - public Boolean isAttachStdin() { - return attachStdin; - } - - @Override - public Boolean isAttachStdout() { - return attachStdout; - } - - @Override - public Boolean isNetworkDisabled() { - return networkDisabled; - } - - @Override - @JsonIgnore - public Boolean getOomKillDisable() { - return hostConfig.getOomKillDisable(); - } - - @Override - @JsonIgnore - public Boolean getPrivileged() { - return hostConfig.getPrivileged(); - } - - @Override - @JsonIgnore - public Boolean getPublishAllPorts() { - return hostConfig.getPublishAllPorts(); - } - - @Override - @JsonIgnore - public Boolean getReadonlyRootfs() { - return hostConfig.getReadonlyRootfs(); - } - - @Override - public Boolean isStdInOnce() { - return stdInOnce; - } - - @Override - public Boolean isStdinOpen() { - return stdinOpen; - } - - @Override - public Boolean isTty() { - return tty; - } - - @Override - @JsonIgnore - public String getPidMode() { - return hostConfig.getPidMode(); - } - - @Override - public HostConfig getHostConfig() { - return hostConfig; - } - - @Override - public String getCgroupParent() { - return hostConfig.getCgroupParent(); - } - - @Override - public CreateContainerCmd withAliases(String... aliases) { - this.aliases = Arrays.asList(aliases); - return this; - } - - @Override - public CreateContainerCmd withAliases(List aliases) { - checkNotNull(aliases, "aliases was not specified"); - this.aliases = aliases; - return this; - } - - @Override - public CreateContainerCmd withAttachStderr(Boolean attachStderr) { - checkNotNull(attachStderr, "attachStderr was not specified"); - this.attachStderr = attachStderr; - return this; - } - - @Override - public CreateContainerCmd withAttachStdin(Boolean attachStdin) { - checkNotNull(attachStdin, "attachStdin was not specified"); - this.attachStdin = attachStdin; - return this; - } - - @Override - public CreateContainerCmd withAttachStdout(Boolean attachStdout) { - checkNotNull(attachStdout, "attachStdout was not specified"); - this.attachStdout = attachStdout; - return this; - } - - @Override - public CreateContainerCmd withBinds(Bind... binds) { - checkNotNull(binds, "binds was not specified"); - hostConfig.setBinds(binds); - return this; - } - - @Override - public CreateContainerCmd withBinds(List binds) { - checkNotNull(binds, "binds was not specified"); - return withBinds(binds.toArray(new Bind[binds.size()])); - } - - @Override - public CreateContainerCmd withBlkioWeight(Integer blkioWeight) { - checkNotNull(blkioWeight, "blkioWeight was not specified"); - hostConfig.withBlkioWeight(blkioWeight); - return this; - } - - @Override - public CreateContainerCmd withCapAdd(Capability... capAdd) { - checkNotNull(capAdd, "capAdd was not specified"); - hostConfig.withCapAdd(capAdd); - return this; - } - - @Override - public CreateContainerCmd withCapAdd(List capAdd) { - checkNotNull(capAdd, "capAdd was not specified"); - return withCapAdd(capAdd.toArray(new Capability[capAdd.size()])); - } - - @Override - public CreateContainerCmd withCapDrop(Capability... capDrop) { - checkNotNull(capDrop, "capDrop was not specified"); - hostConfig.withCapDrop(capDrop); - return this; - } - - @Override - public CreateContainerCmd withCapDrop(List capDrop) { - checkNotNull(capDrop, "capDrop was not specified"); - return withCapDrop(capDrop.toArray(new Capability[capDrop.size()])); - } - - @Override - public CreateContainerCmd withCmd(String... cmd) { - checkNotNull(cmd, "cmd was not specified"); - this.cmd = cmd; - return this; - } - - @Override - public CreateContainerCmd withCmd(List cmd) { - checkNotNull(cmd, "cmd was not specified"); - return withCmd(cmd.toArray(new String[cmd.size()])); - } - - @Override - public CreateContainerCmd withContainerIDFile(String containerIDFile) { - checkNotNull(containerIDFile, "no containerIDFile was specified"); - hostConfig.withContainerIDFile(containerIDFile); - return this; - } - - @Override - public CreateContainerCmd withCpuPeriod(Integer cpuPeriod) { - checkNotNull(cpuPeriod, "cpuPeriod was not specified"); - hostConfig.withCpuPeriod(cpuPeriod); - return this; - } - - @Override - public CreateContainerCmd withCpusetCpus(String cpusetCpus) { - checkNotNull(cpusetCpus, "cpusetCpus was not specified"); - hostConfig.withCpusetCpus(cpusetCpus); - return this; - } - - @Override - public CreateContainerCmd withCpusetMems(String cpusetMems) { - checkNotNull(cpusetMems, "cpusetMems was not specified"); - hostConfig.withCpusetMems(cpusetMems); - return this; - } - - @Override - public CreateContainerCmd withCpuShares(Integer cpuShares) { - checkNotNull(cpuShares, "cpuShares was not specified"); - hostConfig.withCpuShares(cpuShares); - return this; - } - - @Override - public CreateContainerCmd withDevices(Device... devices) { - checkNotNull(devices, "devices was not specified"); - this.hostConfig.withDevices(devices); - return this; - } - - @Override - public CreateContainerCmd withDevices(List devices) { - checkNotNull(devices, "devices was not specified"); - return withDevices(devices.toArray(new Device[devices.size()])); - } - - @Override - public CreateContainerCmd withDns(String... dns) { - checkNotNull(dns, "dns was not specified"); - this.hostConfig.withDns(dns); - return this; - } - - @Override - public CreateContainerCmd withDns(List dns) { - checkNotNull(dns, "dns was not specified"); - return withDns(dns.toArray(new String[dns.size()])); - } - - @Override - public CreateContainerCmd withDnsSearch(String... dnsSearch) { - checkNotNull(dnsSearch, "dnsSearch was not specified"); - this.hostConfig.withDnsSearch(dnsSearch); - return this; - } - - @Override - public CreateContainerCmd withDnsSearch(List dnsSearch) { - checkNotNull(dnsSearch, "dnsSearch was not specified"); - return withDnsSearch(dnsSearch.toArray(new String[0])); - } - - @Override - public CreateContainerCmd withDomainName(String domainName) { - checkNotNull(domainName, "no domainName was specified"); - this.domainName = domainName; - return this; - } - - @Override - public CreateContainerCmd withEntrypoint(String... entrypoint) { - checkNotNull(entrypoint, "entrypoint was not specified"); - this.entrypoint = entrypoint; - return this; - } - - @Override - public CreateContainerCmd withEntrypoint(List entrypoint) { - checkNotNull(entrypoint, "entrypoint was not specified"); - return withEntrypoint(entrypoint.toArray(new String[entrypoint.size()])); - } - - @Override - public CreateContainerCmd withEnv(String... env) { - checkNotNull(env, "env was not specified"); - this.env = env; - return this; - } - - @Override - public CreateContainerCmd withEnv(List env) { - checkNotNull(env, "env was not specified"); - return withEnv(env.toArray(new String[env.size()])); - } - - @Override - public CreateContainerCmd withExposedPorts(ExposedPort... exposedPorts) { - checkNotNull(exposedPorts, "exposedPorts was not specified"); - this.exposedPorts = new ExposedPorts(exposedPorts); - return this; - } - - @Override - public CreateContainerCmd withStopSignal(String stopSignal) { - checkNotNull(stopSignal, "stopSignal wasn't specified."); - this.stopSignal = stopSignal; - return this; - } - - @Override - public CreateContainerCmd withExposedPorts(List exposedPorts) { - checkNotNull(exposedPorts, "exposedPorts was not specified"); - return withExposedPorts(exposedPorts.toArray(new ExposedPort[exposedPorts.size()])); - } - - @Override - public CreateContainerCmd withExtraHosts(String... extraHosts) { - checkNotNull(extraHosts, "extraHosts was not specified"); - this.hostConfig.withExtraHosts(extraHosts); - return this; - } - - @Override - public CreateContainerCmd withExtraHosts(List extraHosts) { - checkNotNull(extraHosts, "extraHosts was not specified"); - return withExtraHosts(extraHosts.toArray(new String[extraHosts.size()])); - } - - @Override - public CreateContainerCmd withHostName(String hostName) { - checkNotNull(hostConfig, "no hostName was specified"); - this.hostName = hostName; - return this; - } - - @Override - public CreateContainerCmd withImage(String image) { - checkNotNull(image, "no image was specified"); - this.image = image; - return this; - } - - @Override - public CreateContainerCmd withIpv4Address(String ipv4Address) { - checkNotNull(ipv4Address, "no ipv4Address was specified"); - this.ipv4Address = ipv4Address; - return this; - } - - @Override - public CreateContainerCmd withIpv6Address(String ipv6Address) { - checkNotNull(ipv6Address, "no ipv6Address was specified"); - this.ipv6Address = ipv6Address; - return this; - } - - @Override - public CreateContainerCmd withLabels(Map labels) { - checkNotNull(labels, "labels was not specified"); - this.labels = labels; - return this; - } - - @Override - public CreateContainerCmd withLinks(Link... links) { - checkNotNull(links, "links was not specified"); - this.hostConfig.setLinks(links); - return this; - } - - @Override - public CreateContainerCmd withLinks(List links) { - checkNotNull(links, "links was not specified"); - return withLinks(links.toArray(new Link[links.size()])); - } - - @Override - public CreateContainerCmd withLxcConf(LxcConf... lxcConf) { - checkNotNull(lxcConf, "lxcConf was not specified"); - this.hostConfig.withLxcConf(lxcConf); - return this; - } - - @Override - public CreateContainerCmd withLxcConf(List lxcConf) { - checkNotNull(lxcConf, "lxcConf was not specified"); - return withLxcConf(lxcConf.toArray(new LxcConf[0])); - } - - @Override - public CreateContainerCmd withLogConfig(LogConfig logConfig) { - checkNotNull(logConfig, "logConfig was not specified"); - this.hostConfig.withLogConfig(logConfig); - return this; - } - - @Override - public CreateContainerCmd withMacAddress(String macAddress) { - checkNotNull(macAddress, "macAddress was not specified"); - this.macAddress = macAddress; - return this; - } - - @Override - public CreateContainerCmd withMemory(Long memory) { - checkNotNull(memory, "memory was not specified"); - hostConfig.withMemory(memory); - return this; - } - - @Override - public CreateContainerCmd withMemorySwap(Long memorySwap) { - checkNotNull(memorySwap, "memorySwap was not specified"); - hostConfig.withMemorySwap(memorySwap); - return this; - } - - @Override - public CreateContainerCmd withName(String name) { - checkNotNull(name, "name was not specified"); - this.name = name; - return this; - } - - @Override - public CreateContainerCmd withNetworkDisabled(Boolean disableNetwork) { - checkNotNull(disableNetwork, "disableNetwork was not specified"); - this.networkDisabled = disableNetwork; - return this; - } - - @Override - public CreateContainerCmd withNetworkMode(String networkMode) { - checkNotNull(networkMode, "networkMode was not specified"); - this.hostConfig.withNetworkMode(networkMode); - return this; - } - - @Override - public CreateContainerCmd withOomKillDisable(Boolean oomKillDisable) { - checkNotNull(oomKillDisable, "oomKillDisable was not specified"); - hostConfig.withOomKillDisable(oomKillDisable); - return this; - } - - @Override - public CreateContainerCmd withPortBindings(PortBinding... portBindings) { - checkNotNull(portBindings, "portBindings was not specified"); - this.hostConfig.withPortBindings(new Ports(portBindings)); - return this; - } - - @Override - public CreateContainerCmd withPortBindings(List portBindings) { - checkNotNull(portBindings, "portBindings was not specified"); - return withPortBindings(portBindings.toArray(new PortBinding[0])); - } - - @Override - public CreateContainerCmd withPortBindings(Ports portBindings) { - checkNotNull(portBindings, "portBindings was not specified"); - this.hostConfig.withPortBindings(portBindings); - return this; - } - - @Override - public CreateContainerCmd withPortSpecs(String... portSpecs) { - checkNotNull(portSpecs, "portSpecs was not specified"); - this.portSpecs = portSpecs; - return this; - } - - @Override - public CreateContainerCmd withPortSpecs(List portSpecs) { - checkNotNull(portSpecs, "portSpecs was not specified"); - return withPortSpecs(portSpecs.toArray(new String[portSpecs.size()])); - } - - @Override - public CreateContainerCmd withPrivileged(Boolean privileged) { - checkNotNull(privileged, "no privileged was specified"); - this.hostConfig.withPrivileged(privileged); - return this; - } - - @Override - public CreateContainerCmd withPublishAllPorts(Boolean publishAllPorts) { - checkNotNull(publishAllPorts, "no publishAllPorts was specified"); - this.hostConfig.withPublishAllPorts(publishAllPorts); - return this; - } - - @Override - public CreateContainerCmd withReadonlyRootfs(Boolean readonlyRootfs) { - checkNotNull(readonlyRootfs, "no readonlyRootfs was specified"); - hostConfig.withReadonlyRootfs(readonlyRootfs); - return this; - } - - @Override - public CreateContainerCmd withRestartPolicy(RestartPolicy restartPolicy) { - checkNotNull(restartPolicy, "restartPolicy was not specified"); - this.hostConfig.withRestartPolicy(restartPolicy); - return this; - } - - @Override - public CreateContainerCmd withStdInOnce(Boolean stdInOnce) { - checkNotNull(stdInOnce, "no stdInOnce was specified"); - this.stdInOnce = stdInOnce; - return this; - } - - @Override - public CreateContainerCmd withStdinOpen(Boolean stdinOpen) { - checkNotNull(stdinOpen, "no stdinOpen was specified"); - this.stdinOpen = stdinOpen; - return this; - } - - @Override - public CreateContainerCmd withTty(Boolean tty) { - checkNotNull(tty, "no tty was specified"); - this.tty = tty; - return this; - } - - @Override - public CreateContainerCmd withUlimits(Ulimit... ulimits) { - checkNotNull(ulimits, "no ulimits was specified"); - hostConfig.withUlimits(ulimits); - return this; - } - - @Override - public CreateContainerCmd withUlimits(List ulimits) { - checkNotNull(ulimits, "no ulimits was specified"); - return withUlimits(ulimits.toArray(new Ulimit[ulimits.size()])); - } - - @Override - public CreateContainerCmd withUser(String user) { - checkNotNull(user, "user was not specified"); - this.user = user; - return this; - } - - @Override - public CreateContainerCmd withVolumes(Volume... volumes) { - checkNotNull(volumes, "volumes was not specified"); - this.volumes = new Volumes(volumes); - return this; - } - - @Override - public CreateContainerCmd withVolumes(List volumes) { - checkNotNull(volumes, "volumes was not specified"); - return withVolumes(volumes.toArray(new Volume[volumes.size()])); - } - - @Override - public CreateContainerCmd withVolumesFrom(VolumesFrom... volumesFrom) { - checkNotNull(volumesFrom, "volumesFrom was not specified"); - this.hostConfig.withVolumesFrom(volumesFrom); - return this; - } - - @Override - public CreateContainerCmd withVolumesFrom(List volumesFrom) { - checkNotNull(volumesFrom, "volumesFrom was not specified"); - return withVolumesFrom(volumesFrom.toArray(new VolumesFrom[volumesFrom.size()])); - } - - @Override - public CreateContainerCmd withWorkingDir(String workingDir) { - checkNotNull(workingDir, "workingDir was not specified"); - this.workingDir = workingDir; - return this; - } - - @Override - public CreateContainerCmd withCgroupParent(final String cgroupParent) { - checkNotNull(cgroupParent, "cgroupParent was not specified"); - this.hostConfig.withCgroupParent(cgroupParent); - return this; - } - - @Override - public CreateContainerCmd withPidMode(String pidMode) { - checkNotNull(pidMode, "pidMode was not specified"); - this.hostConfig.withPidMode(pidMode); - return this; - } - - @Override - public CreateContainerCmd withHostConfig(HostConfig hostConfig) { - this.hostConfig = hostConfig; - return this; - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } - - public static class NetworkingConfig { - @JsonProperty("EndpointsConfig") - public Map endpointsConfig; - - public Map getEndpointsConfig() { - return endpointsConfig; - } - - public NetworkingConfig withEndpointsConfig(Map endpointsConfig) { - this.endpointsConfig = endpointsConfig; - return this; - } - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/CreateVolumeCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/CreateVolumeCmdImpl.java deleted file mode 100644 index 46b12ebea..000000000 --- a/src/main/java/com/github/dockerjava/core/command/CreateVolumeCmdImpl.java +++ /dev/null @@ -1,67 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.util.Map; - -import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.api.command.CreateVolumeCmd; -import com.github.dockerjava.api.command.CreateVolumeResponse; - -/** - * Create a volume. - * - * @author Marcus Linke - */ -public class CreateVolumeCmdImpl extends AbstrDockerCmd implements - CreateVolumeCmd { - - @JsonProperty("Name") - private String name; - - @JsonProperty("Driver") - private String driver; - - @JsonProperty("DriverOpts") - private Map driverOpts; - - public CreateVolumeCmdImpl(CreateVolumeCmd.Exec exec) { - super(exec); - } - - @Override - public String getName() { - return name; - } - - @Override - public String getDriver() { - return driver; - } - - @Override - public Map getDriverOpts() { - return driverOpts; - } - - @Override - public CreateVolumeCmdImpl withName(String name) { - checkNotNull(name, "name was not specified"); - this.name = name; - return this; - } - - @Override - public CreateVolumeCmdImpl withDriver(String driver) { - checkNotNull(driver, "driver was not specified"); - this.driver = driver; - return this; - } - - @Override - public CreateVolumeCmd withDriverOpts(Map driverOpts) { - checkNotNull(driverOpts, "driverOpts was not specified"); - this.driverOpts = driverOpts; - return this; - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/EventsCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/EventsCmdImpl.java deleted file mode 100644 index 230f6dd5a..000000000 --- a/src/main/java/com/github/dockerjava/core/command/EventsCmdImpl.java +++ /dev/null @@ -1,89 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.util.List; -import java.util.Map; - -import com.github.dockerjava.api.command.EventsCmd; -import com.github.dockerjava.api.model.Event; -import com.github.dockerjava.core.util.FiltersBuilder; - -/** - * Stream docker events - */ -public class EventsCmdImpl extends AbstrAsyncDockerCmd implements EventsCmd { - - private String since; - - private String until; - - private FiltersBuilder filters = new FiltersBuilder(); - - public EventsCmdImpl(EventsCmd.Exec exec) { - super(exec); - } - - @Override - public EventsCmd withSince(String since) { - this.since = since; - return this; - } - - @Override - public EventsCmd withUntil(String until) { - this.until = until; - return this; - } - - @Override - public EventsCmd withContainerFilter(String... container) { - checkNotNull(container, "container have not been specified"); - this.filters.withContainers(container); - return this; - } - - @Override - public EventsCmd withImageFilter(String... image) { - checkNotNull(image, "image have not been specified"); - this.filters.withImages(image); - return this; - } - - @Override - public EventsCmd withEventFilter(String... event) { - checkNotNull(event, "event have not been specified"); - this.filters.withFilter("event", event); - return this; - } - - @Override - public EventsCmd withLabelFilter(String... label) { - checkNotNull(label, "label have not been specified"); - this.filters.withLabels(label); - return this; - } - - @Override - public EventsCmd withLabelFilter(Map labels) { - checkNotNull(labels, "labels have not been specified"); - this.filters.withLabels(labels); - return this; - } - - @Override - public String getSince() { - return since; - } - - @Override - public String getUntil() { - return until; - } - - @Override - public Map> getFilters() { - return filters.build(); - } - -} diff --git a/src/main/java/com/github/dockerjava/core/command/ExecCreateCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/ExecCreateCmdImpl.java deleted file mode 100644 index 14da4626a..000000000 --- a/src/main/java/com/github/dockerjava/core/command/ExecCreateCmdImpl.java +++ /dev/null @@ -1,108 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.api.command.ExecCreateCmd; -import com.github.dockerjava.api.command.ExecCreateCmdResponse; -import com.github.dockerjava.api.exception.NotFoundException; - -@JsonInclude(Include.NON_NULL) -public class ExecCreateCmdImpl extends AbstrDockerCmd implements ExecCreateCmd { - - private String containerId; - - @JsonProperty("AttachStdin") - private Boolean attachStdin; - - @JsonProperty("AttachStdout") - private Boolean attachStdout; - - @JsonProperty("AttachStderr") - private Boolean attachStderr; - - @JsonProperty("Tty") - private Boolean tty; - - @JsonProperty("Cmd") - private String[] cmd; - - public ExecCreateCmdImpl(ExecCreateCmd.Exec exec, String containerId) { - super(exec); - withContainerId(containerId); - } - - @Override - public ExecCreateCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; - return this; - } - - @Override - public ExecCreateCmd withAttachStdin(Boolean attachStdin) { - this.attachStdin = attachStdin; - return this; - } - - @Override - public ExecCreateCmd withAttachStdout(Boolean attachStdout) { - this.attachStdout = attachStdout; - return this; - } - - @Override - public ExecCreateCmd withAttachStderr(Boolean attachStderr) { - this.attachStderr = attachStderr; - return this; - } - - @Override - public ExecCreateCmd withTty(Boolean tty) { - this.tty = tty; - return this; - } - - @Override - public ExecCreateCmd withCmd(String... cmd) { - this.cmd = cmd; - return this; - } - - @Override - public String getContainerId() { - return containerId; - } - - @Override - public Boolean hasAttachStdinEnabled() { - return attachStdin; - } - - @Override - public Boolean hasAttachStdoutEnabled() { - return attachStdout; - } - - @Override - public Boolean hasAttachStderrEnabled() { - return attachStderr; - } - - @Override - public Boolean hasTtyEnabled() { - return tty; - } - - /** - * @throws NotFoundException - * No such container - */ - @Override - public ExecCreateCmdResponse exec() throws NotFoundException { - return super.exec(); - } - -} diff --git a/src/main/java/com/github/dockerjava/core/command/ListContainersCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/ListContainersCmdImpl.java deleted file mode 100644 index 8e8e194da..000000000 --- a/src/main/java/com/github/dockerjava/core/command/ListContainersCmdImpl.java +++ /dev/null @@ -1,122 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkArgument; -import static com.google.common.base.Preconditions.checkNotNull; - -import java.util.List; -import java.util.Map; - -import com.github.dockerjava.api.command.ListContainersCmd; -import com.github.dockerjava.api.model.Container; -import com.github.dockerjava.core.util.FiltersBuilder; - -/** - * List containers. - */ -public class ListContainersCmdImpl extends AbstrDockerCmd> implements - ListContainersCmd { - - private Integer limit = -1; - - private Boolean showSize, showAll = false; - - private String sinceId, beforeId; - - private FiltersBuilder filters = new FiltersBuilder(); - - public ListContainersCmdImpl(ListContainersCmd.Exec exec) { - super(exec); - } - - @Override - public Integer getLimit() { - return limit; - } - - @Override - public Boolean hasShowSizeEnabled() { - return showSize; - } - - @Override - public Boolean hasShowAllEnabled() { - return showAll; - } - - @Override - public String getSinceId() { - return sinceId; - } - - @Override - public String getBeforeId() { - return beforeId; - } - - @Override - public Map> getFilters() { - return filters.build(); - } - - @Override - public ListContainersCmd withShowAll(Boolean showAll) { - this.showAll = showAll; - return this; - } - - @Override - public ListContainersCmd withShowSize(Boolean showSize) { - this.showSize = showSize; - return this; - } - - @Override - public ListContainersCmd withLimit(Integer limit) { - checkNotNull(limit, "limit was not specified"); - checkArgument(limit > 0, "limit must be greater 0"); - this.limit = limit; - return this; - } - - @Override - public ListContainersCmd withSince(String since) { - checkNotNull(since, "since was not specified"); - this.sinceId = since; - return this; - } - - @Override - public ListContainersCmd withBefore(String before) { - checkNotNull(before, "before was not specified"); - this.beforeId = before; - return this; - } - - @Override - public ListContainersCmd withLabelFilter(String... labels) { - checkNotNull(labels, "labels was not specified"); - this.filters.withLabels(labels); - return this; - } - - @Override - public ListContainersCmd withLabelFilter(Map labels) { - checkNotNull(labels, "labels was not specified"); - this.filters.withLabels(labels); - return this; - } - - @Override - public ListContainersCmd withExitcodeFilter(Integer exitcode) { - checkNotNull(exitcode, "exitcode was not specified"); - this.filters.withFilter("exitcode", exitcode.toString()); - return this; - } - - @Override - public ListContainersCmd withStatusFilter(String status) { - checkNotNull(status, "status was not specified"); - this.filters.withFilter("status", status); - return this; - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/ListImagesCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/ListImagesCmdImpl.java deleted file mode 100644 index 1033b43ec..000000000 --- a/src/main/java/com/github/dockerjava/core/command/ListImagesCmdImpl.java +++ /dev/null @@ -1,83 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.util.List; -import java.util.Map; - -import org.apache.commons.lang.builder.ReflectionToStringBuilder; -import org.apache.commons.lang.builder.ToStringStyle; - -import com.github.dockerjava.api.command.ListImagesCmd; -import com.github.dockerjava.api.model.Image; -import com.github.dockerjava.core.util.FiltersBuilder; - -/** - * List images - */ -public class ListImagesCmdImpl extends AbstrDockerCmd> implements ListImagesCmd { - - private String imageNameFilter; - - private Boolean showAll = false; - - private FiltersBuilder filters = new FiltersBuilder(); - - public ListImagesCmdImpl(ListImagesCmd.Exec exec) { - super(exec); - } - - @Override - public Map> getFilters() { - return filters.build(); - } - - @Override - public Boolean hasShowAllEnabled() { - return showAll; - } - - @Override - public ListImagesCmd withShowAll(Boolean showAll) { - this.showAll = showAll; - return this; - } - - @Override - public ListImagesCmd withDanglingFilter(Boolean dangling) { - checkNotNull(dangling, "dangling have not been specified"); - filters.withFilter("dangling", dangling.toString()); - return this; - } - - @Override - public ListImagesCmd withLabelFilter(String... labels) { - checkNotNull(labels, "labels have not been specified"); - filters.withLabels(labels); - return this; - } - - @Override - public ListImagesCmd withLabelFilter(Map labels) { - checkNotNull(labels, "labels have not been specified"); - filters.withLabels(labels); - return this; - } - - @Override - public ListImagesCmd withImageNameFilter(String imageNameFilter) { - checkNotNull(imageNameFilter, "image name filter not specified"); - this.imageNameFilter = imageNameFilter; - return this; - } - - @Override - public String getImageNameFilter() { - return this.imageNameFilter; - } - - @Override - public String toString() { - return ReflectionToStringBuilder.toString(this, ToStringStyle.SHORT_PREFIX_STYLE); - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/ListNetworksCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/ListNetworksCmdImpl.java deleted file mode 100644 index 44a9d2cfb..000000000 --- a/src/main/java/com/github/dockerjava/core/command/ListNetworksCmdImpl.java +++ /dev/null @@ -1,34 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.ListNetworksCmd; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.core.util.FiltersBuilder; - -import java.util.List; -import java.util.Map; - -public class ListNetworksCmdImpl extends AbstrDockerCmd> implements ListNetworksCmd { - - private FiltersBuilder filtersBuilder = new FiltersBuilder(); - - public ListNetworksCmdImpl(ListNetworksCmd.Exec exec) { - super(exec); - } - - @Override - public Map> getFilters() { - return filtersBuilder.build(); - } - - @Override - public ListNetworksCmd withIdFilter(String... networkId) { - this.filtersBuilder.withFilter("id", networkId); - return this; - } - - @Override - public ListNetworksCmd withNameFilter(String... networkName) { - this.filtersBuilder.withFilter("name", networkName); - return this; - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/ListVolumesCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/ListVolumesCmdImpl.java deleted file mode 100644 index d9efe7b17..000000000 --- a/src/main/java/com/github/dockerjava/core/command/ListVolumesCmdImpl.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.util.List; -import java.util.Map; - -import com.github.dockerjava.api.command.ListVolumesCmd; -import com.github.dockerjava.api.command.ListVolumesResponse; -import com.github.dockerjava.core.util.FiltersBuilder; - -/** - * - * @author Marcus Linke - * - */ -public class ListVolumesCmdImpl extends AbstrDockerCmd implements ListVolumesCmd { - - private FiltersBuilder filters = new FiltersBuilder(); - - public ListVolumesCmdImpl(ListVolumesCmd.Exec exec) { - super(exec); - } - - @Override - public Map> getFilters() { - return filters.build(); - } - - @Override - public ListVolumesCmd withDanglingFilter(Boolean dangling) { - checkNotNull(dangling, "dangling have not been specified"); - this.filters.withFilter("dangling", dangling.toString()); - return this; - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/PullImageCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/PullImageCmdImpl.java deleted file mode 100644 index 547fc6519..000000000 --- a/src/main/java/com/github/dockerjava/core/command/PullImageCmdImpl.java +++ /dev/null @@ -1,76 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import com.github.dockerjava.api.command.PullImageCmd; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.PullResponseItem; - -/** - * - * Pull image from repository. - * - */ -public class PullImageCmdImpl extends AbstrAsyncDockerCmd implements PullImageCmd { - - private String repository, tag, registry; - - private AuthConfig authConfig; - - public PullImageCmdImpl(PullImageCmd.Exec exec, AuthConfig authConfig, String repository) { - super(exec); - withOptionalAuthConfig(authConfig); - withRepository(repository); - } - - public AuthConfig getAuthConfig() { - return authConfig; - } - - public PullImageCmd withAuthConfig(AuthConfig authConfig) { - checkNotNull(authConfig, "authConfig was not specified"); - return withOptionalAuthConfig(authConfig); - } - - private PullImageCmd withOptionalAuthConfig(AuthConfig authConfig) { - this.authConfig = authConfig; - return this; - } - - @Override - public String getRepository() { - return repository; - } - - @Override - public String getTag() { - return tag; - } - - @Override - public String getRegistry() { - return registry; - } - - @Override - public PullImageCmd withRepository(String repository) { - checkNotNull(repository, "repository was not specified"); - this.repository = repository; - return this; - } - - @Override - public PullImageCmd withTag(String tag) { - checkNotNull(tag, "tag was not specified"); - this.tag = tag; - return this; - } - - @Override - public PullImageCmd withRegistry(String registry) { - checkNotNull(registry, "registry was not specified"); - this.registry = registry; - return this; - } - -} diff --git a/src/main/java/com/github/dockerjava/core/command/PullImageResultCallback.java b/src/main/java/com/github/dockerjava/core/command/PullImageResultCallback.java deleted file mode 100644 index ce274dbe8..000000000 --- a/src/main/java/com/github/dockerjava/core/command/PullImageResultCallback.java +++ /dev/null @@ -1,53 +0,0 @@ -/* - * Created on 21.07.2015 - */ -package com.github.dockerjava.core.command; - -import javax.annotation.CheckForNull; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.model.PullResponseItem; -import com.github.dockerjava.core.async.ResultCallbackTemplate; - -/** - * - * @author Marcus Linke - * - */ -public class PullImageResultCallback extends ResultCallbackTemplate { - - private static final Logger LOGGER = LoggerFactory.getLogger(PullImageResultCallback.class); - - @CheckForNull - private PullResponseItem latestItem = null; - - @Override - public void onNext(PullResponseItem item) { - this.latestItem = item; - LOGGER.debug(item.toString()); - } - - /** - * Awaits the image to be pulled successful. - * - * @throws DockerClientException - * if the pull fails. - */ - public void awaitSuccess() { - try { - awaitCompletion(); - } catch (InterruptedException e) { - throw new DockerClientException("", e); - } - - if (latestItem == null) { - throw new DockerClientException("Could not pull image"); - } else if (!latestItem.isPullSuccessIndicated()) { - String message = (latestItem.getError() != null) ? latestItem.getError() : latestItem.getStatus(); - throw new DockerClientException("Could not pull image: " + message); - } - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/PushImageCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/PushImageCmdImpl.java deleted file mode 100644 index 621c4a7a4..000000000 --- a/src/main/java/com/github/dockerjava/core/command/PushImageCmdImpl.java +++ /dev/null @@ -1,74 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import com.github.dockerjava.api.command.PushImageCmd; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.PushResponseItem; - -/** - * Push the latest image to the repository. - * - * @param name - * The name, e.g. "alexec/busybox" or just "busybox" if you want to default. Not null. - */ -public class PushImageCmdImpl extends AbstrAsyncDockerCmd implements PushImageCmd { - - private String name; - - private String tag; - - private AuthConfig authConfig; - - public PushImageCmdImpl(PushImageCmd.Exec exec, String name) { - super(exec); - withName(name); - } - - @Override - public String getName() { - return name; - } - - @Override - public String getTag() { - return tag; - } - - /** - * @param name - * The name, e.g. "alexec/busybox" or just "busybox" if you want to default. Not null. - */ - @Override - public PushImageCmd withName(String name) { - checkNotNull(name, "name was not specified"); - this.name = name; - return this; - } - - /** - * @param tag - * The image's tag. Can be null or empty. - */ - @Override - public PushImageCmd withTag(String tag) { - checkNotNull(tag, "tag was not specified"); - this.tag = tag; - return this; - } - - public AuthConfig getAuthConfig() { - return authConfig; - } - - public PushImageCmd withAuthConfig(AuthConfig authConfig) { - checkNotNull(authConfig, "authConfig was not specified"); - return withOptionalAuthConfig(authConfig); - } - - private PushImageCmd withOptionalAuthConfig(AuthConfig authConfig) { - this.authConfig = authConfig; - return this; - } - -} diff --git a/src/main/java/com/github/dockerjava/core/command/RestartContainerCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/RestartContainerCmdImpl.java deleted file mode 100644 index e3621af4f..000000000 --- a/src/main/java/com/github/dockerjava/core/command/RestartContainerCmdImpl.java +++ /dev/null @@ -1,60 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkArgument; -import static com.google.common.base.Preconditions.checkNotNull; - -import com.github.dockerjava.api.command.RestartContainerCmd; -import com.github.dockerjava.api.exception.NotFoundException; - -/** - * Restart a running container. - * - * @param timeout - * - Timeout in seconds before killing the container. Defaults to 10 seconds. - * - */ -public class RestartContainerCmdImpl extends AbstrDockerCmd implements RestartContainerCmd { - - private String containerId; - - private Integer timeout = 10; - - public RestartContainerCmdImpl(RestartContainerCmd.Exec exec, String containerId) { - super(exec); - withContainerId(containerId); - } - - @Override - public String getContainerId() { - return containerId; - } - - @Override - public Integer getTimeout() { - return timeout; - } - - @Override - public RestartContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; - return this; - } - - @Override - public RestartContainerCmd withtTimeout(Integer timeout) { - checkNotNull(timeout, "timeout was not specified"); - checkArgument(timeout >= 0, "timeout must be greater or equal 0"); - this.timeout = timeout; - return this; - } - - /** - * @throws NotFoundException - * No such container - */ - @Override - public Void exec() throws NotFoundException { - return super.exec(); - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/SearchImagesCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/SearchImagesCmdImpl.java deleted file mode 100644 index 342a7fbd2..000000000 --- a/src/main/java/com/github/dockerjava/core/command/SearchImagesCmdImpl.java +++ /dev/null @@ -1,38 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.util.List; - -import com.github.dockerjava.api.command.SearchImagesCmd; -import com.github.dockerjava.api.model.SearchItem; - -/** - * Search images - * - * @param term - * - search term - * - */ -public class SearchImagesCmdImpl extends AbstrDockerCmd> implements SearchImagesCmd { - - private String term; - - public SearchImagesCmdImpl(SearchImagesCmd.Exec exec, String term) { - super(exec); - withTerm(term); - } - - @Override - public String getTerm() { - return term; - } - - @Override - public SearchImagesCmd withTerm(String term) { - checkNotNull(term, "term was not specified"); - this.term = term; - return this; - } - -} diff --git a/src/main/java/com/github/dockerjava/core/command/StatsCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/StatsCmdImpl.java deleted file mode 100644 index 69d9a02d7..000000000 --- a/src/main/java/com/github/dockerjava/core/command/StatsCmdImpl.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import com.github.dockerjava.api.command.StatsCmd; -import com.github.dockerjava.api.model.Statistics; - -/** - * Container stats - */ -public class StatsCmdImpl extends AbstrAsyncDockerCmd implements StatsCmd { - - private String containerId; - - public StatsCmdImpl(StatsCmd.Exec exec, String containerId) { - super(exec); - withContainerId(containerId); - } - - @Override - public StatsCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; - return this; - } - - @Override - public String getContainerId() { - return containerId; - } - -} diff --git a/src/main/java/com/github/dockerjava/core/command/UpdateContainerCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/UpdateContainerCmdImpl.java deleted file mode 100644 index b8915a802..000000000 --- a/src/main/java/com/github/dockerjava/core/command/UpdateContainerCmdImpl.java +++ /dev/null @@ -1,265 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; -import com.github.dockerjava.api.command.UpdateContainerCmd; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.UpdateContainerResponse; -import com.github.dockerjava.core.RemoteApiVersion; -import org.apache.commons.lang.builder.EqualsBuilder; -import org.apache.commons.lang.builder.HashCodeBuilder; -import org.apache.commons.lang.builder.ToStringBuilder; - -import javax.annotation.CheckForNull; -import javax.annotation.Nonnull; - -/** - * @author Kanstantsin Shautsou - * @see - * https://docs.docker.com/engine/reference/api/docker_remote_api_v1.22/ - * @since {@link RemoteApiVersion#VERSION_1_22} - */ -public class UpdateContainerCmdImpl extends AbstrDockerCmd - implements UpdateContainerCmd { - - @JsonIgnoreProperties - private String containerId; - - @JsonProperty("BlkioWeight") - private Integer blkioWeight; - - @JsonProperty("CpuShares") - private Integer cpuShares; - - @JsonProperty("CpuPeriod") - private Integer cpuPeriod; - - @JsonProperty("CpuQuota") - private Integer cpuQuota; - - @JsonProperty("CpusetCpus") - private String cpusetCpus; - - @JsonProperty("CpusetMems") - private String cpusetMems; - - @JsonProperty("Memory") - private Long memory; - - @JsonProperty("MemorySwap") - private Long memorySwap; - - @JsonProperty("MemoryReservation") - private Long memoryReservation; - - @JsonProperty("KernelMemory") - private Long kernelMemory; - - public UpdateContainerCmdImpl() { - super(null); - } - - public UpdateContainerCmdImpl(UpdateContainerCmd.Exec exec, String containerId) { - super(exec); - withContainerId(containerId); - } - - /** - * @see #blkioWeight - */ - @CheckForNull - public Integer getBlkioWeight() { - return blkioWeight; - } - - /** - * @see #blkioWeight - */ - public UpdateContainerCmd withBlkioWeight(Integer blkioWeight) { - this.blkioWeight = blkioWeight; - return this; - } - - /** - * @see #containerId - */ - @CheckForNull - public String getContainerId() { - return containerId; - } - - /** - * @see #containerId - */ - public UpdateContainerCmd withContainerId(@Nonnull String containerId) { - this.containerId = containerId; - return this; - } - - /** - * @see #cpuPeriod - */ - @CheckForNull - public Integer getCpuPeriod() { - return cpuPeriod; - } - - /** - * @see #cpuPeriod - */ - public UpdateContainerCmd withCpuPeriod(Integer cpuPeriod) { - this.cpuPeriod = cpuPeriod; - return this; - } - - /** - * @see #cpuQuota - */ - @CheckForNull - public Integer getCpuQuota() { - return cpuQuota; - } - - /** - * @see #cpuQuota - */ - public UpdateContainerCmd withCpuQuota(Integer cpuQuota) { - this.cpuQuota = cpuQuota; - return this; - } - - /** - * @see #cpusetCpus - */ - @CheckForNull - public String getCpusetCpus() { - return cpusetCpus; - } - - /** - * @see #cpusetCpus - */ - public UpdateContainerCmd withCpusetCpus(String cpusetCpus) { - this.cpusetCpus = cpusetCpus; - return this; - } - - /** - * @see #cpusetMems - */ - @CheckForNull - public String getCpusetMems() { - return cpusetMems; - } - - /** - * @see #cpusetMems - */ - public UpdateContainerCmd withCpusetMems(String cpusetMems) { - this.cpusetMems = cpusetMems; - return this; - } - - /** - * @see #cpuShares - */ - @CheckForNull - public Integer getCpuShares() { - return cpuShares; - } - - /** - * @see #cpuShares - */ - public UpdateContainerCmd withCpuShares(Integer cpuShares) { - this.cpuShares = cpuShares; - return this; - } - - /** - * @see #kernelMemory - */ - @CheckForNull - public Long getKernelMemory() { - return kernelMemory; - } - - /** - * @see #kernelMemory - */ - public UpdateContainerCmd withKernelMemory(Long kernelMemory) { - this.kernelMemory = kernelMemory; - return this; - } - - /** - * @see #memory - */ - @CheckForNull - public Long getMemory() { - return memory; - } - - /** - * @see #memory - */ - public UpdateContainerCmd withMemory(Long memory) { - this.memory = memory; - return this; - } - - /** - * @see #memoryReservation - */ - @CheckForNull - public Long getMemoryReservation() { - return memoryReservation; - } - - /** - * @see #memoryReservation - */ - public UpdateContainerCmd withMemoryReservation(Long memoryReservation) { - this.memoryReservation = memoryReservation; - return this; - } - - /** - * @see #memorySwap - */ - @CheckForNull - public Long getMemorySwap() { - return memorySwap; - } - - /** - * @see #memorySwap - */ - public UpdateContainerCmd withMemorySwap(Long memorySwap) { - this.memorySwap = memorySwap; - return this; - } - - /** - * @throws NotFoundException No such container - */ - @Override - public UpdateContainerResponse exec() throws NotFoundException { - return super.exec(); - } - - @Override - public String toString() { - return ToStringBuilder.reflectionToString(this); - } - - @Override - public boolean equals(Object o) { - return EqualsBuilder.reflectionEquals(this, o); - } - - @Override - public int hashCode() { - return HashCodeBuilder.reflectionHashCode(this); - } -} diff --git a/src/main/java/com/github/dockerjava/core/command/WaitContainerCmdImpl.java b/src/main/java/com/github/dockerjava/core/command/WaitContainerCmdImpl.java deleted file mode 100644 index eeb05ff60..000000000 --- a/src/main/java/com/github/dockerjava/core/command/WaitContainerCmdImpl.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.google.common.base.Preconditions.checkNotNull; - -import com.github.dockerjava.api.command.WaitContainerCmd; -import com.github.dockerjava.api.model.WaitResponse; - -/** - * Wait a container - * - * Block until container stops, then returns its exit code - */ -public class WaitContainerCmdImpl extends AbstrAsyncDockerCmd implements - WaitContainerCmd { - - private String containerId; - - public WaitContainerCmdImpl(WaitContainerCmd.Exec exec, String containerId) { - super(exec); - withContainerId(containerId); - } - - @Override - public String getContainerId() { - return containerId; - } - - @Override - public WaitContainerCmd withContainerId(String containerId) { - checkNotNull(containerId, "containerId was not specified"); - this.containerId = containerId; - return this; - } - -} diff --git a/src/main/java/com/github/dockerjava/core/command/WaitContainerResultCallback.java b/src/main/java/com/github/dockerjava/core/command/WaitContainerResultCallback.java deleted file mode 100644 index df415d555..000000000 --- a/src/main/java/com/github/dockerjava/core/command/WaitContainerResultCallback.java +++ /dev/null @@ -1,76 +0,0 @@ -/* - * Created on 21.07.2015 - */ -package com.github.dockerjava.core.command; - -import java.util.concurrent.TimeUnit; - -import javax.annotation.CheckForNull; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.model.WaitResponse; -import com.github.dockerjava.core.async.ResultCallbackTemplate; - -/** - * - * @author Marcus Linke - * - */ -public class WaitContainerResultCallback extends ResultCallbackTemplate { - - private static final Logger LOGGER = LoggerFactory.getLogger(WaitContainerResultCallback.class); - - @CheckForNull - private WaitResponse waitResponse = null; - - @Override - public void onNext(WaitResponse waitResponse) { - this.waitResponse = waitResponse; - LOGGER.debug(waitResponse.toString()); - } - - /** - * Awaits the status code from the container. - * - * @throws DockerClientException - * if the wait operation fails. - */ - public Integer awaitStatusCode() { - try { - awaitCompletion(); - } catch (InterruptedException e) { - throw new DockerClientException("", e); - } - - return getStatusCode(); - } - - /** - * Awaits the status code from the container. - * - * @throws DockerClientException - * if the wait operation fails. - */ - public Integer awaitStatusCode(long timeout, TimeUnit timeUnit) { - try { - if (!awaitCompletion(timeout, timeUnit)) { - throw new DockerClientException("Awaiting status code timeout."); - } - } catch (InterruptedException e) { - throw new DockerClientException("Awaiting status code interrupted: ", e); - } - - return getStatusCode(); - } - - private Integer getStatusCode() { - if (waitResponse == null) { - throw new DockerClientException("Error while wait container"); - } else { - return waitResponse.getStatusCode(); - } - } -} diff --git a/src/main/java/com/github/dockerjava/core/util/CertificateUtils.java b/src/main/java/com/github/dockerjava/core/util/CertificateUtils.java deleted file mode 100644 index c760331d6..000000000 --- a/src/main/java/com/github/dockerjava/core/util/CertificateUtils.java +++ /dev/null @@ -1,197 +0,0 @@ -package com.github.dockerjava.core.util; - -import java.io.BufferedReader; -import java.io.File; -import java.io.IOException; -import java.io.Reader; -import java.io.StringReader; -import java.security.KeyFactory; -import java.security.KeyStore; -import java.security.KeyStoreException; -import java.security.NoSuchAlgorithmException; -import java.security.PrivateKey; -import java.security.cert.Certificate; -import java.security.cert.CertificateException; -import java.security.spec.InvalidKeySpecException; -import java.security.spec.PKCS8EncodedKeySpec; -import java.util.ArrayList; -import java.util.List; - -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; - -import org.bouncycastle.asn1.ASN1ObjectIdentifier; -import org.bouncycastle.asn1.pkcs.PrivateKeyInfo; -import org.bouncycastle.cert.X509CertificateHolder; -import org.bouncycastle.cert.jcajce.JcaX509CertificateConverter; -import org.bouncycastle.openssl.PEMKeyPair; -import org.bouncycastle.openssl.PEMParser; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import javax.annotation.CheckForNull; - -import static java.util.Objects.requireNonNull; - -public class CertificateUtils { - private static final Logger LOG = LoggerFactory.getLogger(CertificateUtils.class); - - private CertificateUtils() { - // utility class - } - - public static boolean verifyCertificatesExist(String dockerCertPath) { - String[] files = {"ca.pem", "cert.pem", "key.pem"}; - boolean result = true; - for (String file : files) { - File path = new File(dockerCertPath, file); - result &= path.exists(); - } - - return result; - } - - @SuppressFBWarnings(value = "NP_NULL_ON_SOME_PATH_FROM_RETURN_VALUE") - public static KeyStore createKeyStore(final String keypem, final String certpem) throws NoSuchAlgorithmException, - InvalidKeySpecException, IOException, CertificateException, KeyStoreException { - PrivateKey privateKey = loadPrivateKey(keypem); - requireNonNull(privateKey); - List privateCertificates = loadCertificates(certpem); - - KeyStore keyStore = KeyStore.getInstance("JKS"); - keyStore.load(null); - - keyStore.setKeyEntry("docker", - privateKey, - "docker".toCharArray(), - privateCertificates.toArray(new Certificate[privateCertificates.size()]) - ); - - return keyStore; - } - - /** - * from "cert.pem" String - */ - private static List loadCertificates(final String certpem) throws IOException, - CertificateException { - final StringReader certReader = new StringReader(certpem); - try (BufferedReader reader = new BufferedReader(certReader)) { - return loadCertificates(reader); - } - } - - /** - * "cert.pem" from reader - */ - private static List loadCertificates(final Reader reader) throws IOException, - CertificateException { - try (PEMParser pemParser = new PEMParser(reader)) { - List certificates = new ArrayList<>(); - - JcaX509CertificateConverter certificateConverter = new JcaX509CertificateConverter().setProvider("BC"); - Object certObj = pemParser.readObject(); - - if (certObj instanceof X509CertificateHolder) { - X509CertificateHolder certificateHolder = (X509CertificateHolder) certObj; - certificates.add(certificateConverter.getCertificate(certificateHolder)); - } - - return certificates; - } - } - - - /** - * Return private key ("key.pem") from Reader - */ - @CheckForNull - private static PrivateKey loadPrivateKey(final Reader reader) throws IOException, NoSuchAlgorithmException, - InvalidKeySpecException { - try (PEMParser pemParser = new PEMParser(reader)) { - Object readObject = pemParser.readObject(); - while (readObject != null) { - if (readObject instanceof PEMKeyPair) { - PEMKeyPair pemKeyPair = (PEMKeyPair) readObject; - PrivateKey privateKey = guessKey(pemKeyPair.getPrivateKeyInfo().getEncoded()); - if (privateKey != null) { - return privateKey; - } - } else if (readObject instanceof PrivateKeyInfo) { - PrivateKeyInfo privateKeyInfo = (PrivateKeyInfo) readObject; - PrivateKey privateKey = guessKey(privateKeyInfo.getEncoded()); - if (privateKey != null) { - return privateKey; - } - } else if (readObject instanceof ASN1ObjectIdentifier) { - // no idea how it can be used - final ASN1ObjectIdentifier asn1ObjectIdentifier = (ASN1ObjectIdentifier) readObject; - LOG.trace("Ignoring asn1ObjectIdentifier {}", asn1ObjectIdentifier); - } else { - LOG.warn("Unknown object '{}' from PEMParser", readObject); - } - - readObject = pemParser.readObject(); - } - } - - return null; - } - - @CheckForNull - private static PrivateKey guessKey(byte[] encodedKey) throws NoSuchAlgorithmException { - //no way to know, so iterate - for (String guessFactory : new String[]{"RSA", "ECDSA"}) { - try { - KeyFactory factory = KeyFactory.getInstance(guessFactory); - - PKCS8EncodedKeySpec privateKeySpec = new PKCS8EncodedKeySpec(encodedKey); - return factory.generatePrivate(privateKeySpec); - } catch (InvalidKeySpecException ignore) { - } - } - - return null; - } - - /** - * Return KeyPair from "key.pem" - */ - @CheckForNull - private static PrivateKey loadPrivateKey(final String keypem) throws IOException, NoSuchAlgorithmException, - InvalidKeySpecException { - try (StringReader certReader = new StringReader(keypem); - BufferedReader reader = new BufferedReader(certReader)) { - return loadPrivateKey(reader); - } - } - - /** - * "ca.pem" from String - */ - public static KeyStore createTrustStore(String capem) throws IOException, CertificateException, - KeyStoreException, NoSuchAlgorithmException { - try (Reader certReader = new StringReader(capem)) { - return createTrustStore(certReader); - } - } - - /** - * "ca.pem" from Reader - */ - public static KeyStore createTrustStore(final Reader certReader) throws IOException, CertificateException, - KeyStoreException, NoSuchAlgorithmException { - try (PEMParser pemParser = new PEMParser(certReader)) { - X509CertificateHolder certificateHolder = (X509CertificateHolder) pemParser.readObject(); - Certificate caCertificate = new JcaX509CertificateConverter() - .setProvider("BC") - .getCertificate(certificateHolder); - - KeyStore trustStore = KeyStore.getInstance("JKS"); - trustStore.load(null); - trustStore.setCertificateEntry("ca", caCertificate); - - return trustStore; - } - } - -} diff --git a/src/main/java/com/github/dockerjava/core/util/CompressArchiveUtil.java b/src/main/java/com/github/dockerjava/core/util/CompressArchiveUtil.java deleted file mode 100644 index 2e75bcc41..000000000 --- a/src/main/java/com/github/dockerjava/core/util/CompressArchiveUtil.java +++ /dev/null @@ -1,113 +0,0 @@ -package com.github.dockerjava.core.util; - -import static com.github.dockerjava.core.util.FilePathUtil.relativize; - -import java.io.BufferedInputStream; -import java.io.BufferedOutputStream; -import java.io.File; -import java.io.FileNotFoundException; -import java.io.FileOutputStream; -import java.io.IOException; -import java.io.InputStream; -import java.io.OutputStream; -import java.nio.file.Files; -import java.nio.file.Path; -import java.util.zip.GZIPOutputStream; - -import org.apache.commons.compress.archivers.tar.TarArchiveEntry; -import org.apache.commons.compress.archivers.tar.TarArchiveOutputStream; -import org.apache.commons.compress.compressors.gzip.GzipCompressorOutputStream; -import org.apache.commons.io.FileUtils; - -import com.google.common.io.ByteStreams; - -public class CompressArchiveUtil { - private CompressArchiveUtil() { - // utility class - } - - static void putTarEntry(TarArchiveOutputStream tarOutputStream, TarArchiveEntry tarEntry, Path file) - throws IOException { - tarEntry.setSize(Files.size(file)); - tarOutputStream.putArchiveEntry(tarEntry); - try (InputStream input = new BufferedInputStream(Files.newInputStream(file))) { - ByteStreams.copy(input, tarOutputStream); - tarOutputStream.closeArchiveEntry(); - } - } - - private static TarArchiveOutputStream buildTarStream(Path outputPath, boolean gZipped) throws IOException { - OutputStream outputStream = new BufferedOutputStream(Files.newOutputStream(outputPath)); - if (gZipped) { - outputStream = new GzipCompressorOutputStream(outputStream); - } - TarArchiveOutputStream tarArchiveOutputStream = new TarArchiveOutputStream(outputStream); - tarArchiveOutputStream.setLongFileMode(TarArchiveOutputStream.LONGFILE_GNU); - return tarArchiveOutputStream; - } - - /** - * Recursively tar file - * - * @param inputPath - * file path can be directory - * @param outputPath - * where to put the archived file - * @param childrenOnly - * if inputPath is directory and if childrenOnly is true, the archive will contain all of its children, else the archive - * contains unique entry which is the inputPath itself - * @param gZipped - * compress with gzip algorithm - */ - public static void tar(Path inputPath, Path outputPath, boolean gZipped, boolean childrenOnly) throws IOException { - if (!Files.exists(inputPath)) { - throw new FileNotFoundException("File not found " + inputPath); - } - FileUtils.touch(outputPath.toFile()); - - try (TarArchiveOutputStream tarArchiveOutputStream = buildTarStream(outputPath, gZipped)) { - if (!Files.isDirectory(inputPath)) { - TarArchiveEntry tarEntry = new TarArchiveEntry(inputPath.getFileName().toString()); - if (inputPath.toFile().canExecute()) { - tarEntry.setMode(tarEntry.getMode() | 0755); - } - putTarEntry(tarArchiveOutputStream, tarEntry, inputPath); - } else { - Path sourcePath = inputPath; - if (!childrenOnly) { - // In order to have the dossier as the root entry - sourcePath = inputPath.getParent(); - } - Files.walkFileTree(inputPath, new TarDirWalker(sourcePath, tarArchiveOutputStream)); - } - tarArchiveOutputStream.flush(); - } - } - - public static File archiveTARFiles(File base, Iterable files, String archiveNameWithOutExtension) - throws IOException { - File tarFile = new File(FileUtils.getTempDirectoryPath(), archiveNameWithOutExtension + ".tar"); - tarFile.deleteOnExit(); - try (TarArchiveOutputStream tos = new TarArchiveOutputStream(new GZIPOutputStream(new BufferedOutputStream( - new FileOutputStream(tarFile))))) { - tos.setLongFileMode(TarArchiveOutputStream.LONGFILE_GNU); - for (File file : files) { - TarArchiveEntry tarEntry = new TarArchiveEntry(file); - tarEntry.setName(relativize(base, file)); - - if (!file.isDirectory() && file.canExecute()) { - tarEntry.setMode(tarEntry.getMode() | 0755); - } - - tos.putArchiveEntry(tarEntry); - - if (!file.isDirectory()) { - FileUtils.copyFile(file, tos); - } - tos.closeArchiveEntry(); - } - } - - return tarFile; - } -} diff --git a/src/main/java/com/github/dockerjava/core/util/FiltersBuilder.java b/src/main/java/com/github/dockerjava/core/util/FiltersBuilder.java deleted file mode 100644 index 4d71014dd..000000000 --- a/src/main/java/com/github/dockerjava/core/util/FiltersBuilder.java +++ /dev/null @@ -1,108 +0,0 @@ -package com.github.dockerjava.core.util; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Map.Entry; - -/** - * Representation of Docker filters. - * - * @author Carlos Sanchez - * - */ -public class FiltersBuilder { - - private Map> filters = new HashMap>(); - - public FiltersBuilder() { - } - - public FiltersBuilder withFilter(String key, String... value) { - filters.put(key, Arrays.asList(value)); - return this; - } - - public List getFilter(String key) { - return filters.get(key); - } - - public FiltersBuilder withImages(String... image) { - withFilter("image", image); - return this; - } - - public List getImage() { - return getFilter("image"); - } - - public FiltersBuilder withContainers(String... container) { - withFilter("container", container); - return this; - } - - public List getContainer() { - return getFilter("container"); - } - - /** - * Filter by labels - * - * @param labels - * string array in the form ["key"] or ["key=value"] or a mix of both - */ - public FiltersBuilder withLabels(String... labels) { - withFilter("label", labels); - return this; - } - - /** - * Filter by labels - * - * @param labels - * {@link Map} of labels that contains label keys and values - */ - public FiltersBuilder withLabels(Map labels) { - withFilter("label", labelsMapToList(labels).toArray(new String[labels.size()])); - return this; - } - - private static List labelsMapToList(Map labels) { - List result = new ArrayList(); - for (Entry entry : labels.entrySet()) { - String rest = (entry.getValue() != null & !entry.getValue().isEmpty()) ? "=" + entry.getValue() : ""; - - String label = entry.getKey() + rest; - - result.add(label); - } - return result; - } - - // CHECKSTYLE:OFF - @Override - public boolean equals(Object o) { - if (this == o) - return true; - if (o == null || getClass() != o.getClass()) - return false; - - FiltersBuilder filters1 = (FiltersBuilder) o; - - return filters.equals(filters1.filters); - - } - - // CHECKSTYLE:ON - - @Override - public int hashCode() { - return filters.hashCode(); - } - - public Map> build() { - return filters; - } -} diff --git a/src/main/java/com/github/dockerjava/core/util/FiltersEncoder.java b/src/main/java/com/github/dockerjava/core/util/FiltersEncoder.java deleted file mode 100644 index 280daad45..000000000 --- a/src/main/java/com/github/dockerjava/core/util/FiltersEncoder.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.core.util; - -import java.util.List; -import java.util.Map; - -import javax.ws.rs.core.MediaType; - -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.fasterxml.jackson.jaxrs.json.JacksonJaxbJsonProvider; - -/** - * JSON Encoder for docker filters. - * - * @author Carlos Sanchez - * - */ -public class FiltersEncoder { - private FiltersEncoder() { - } - - private static final ObjectMapper OBJECT_MAPPER = new JacksonJaxbJsonProvider().locateMapper(Map.class, - MediaType.APPLICATION_JSON_TYPE); - - public static String jsonEncode(Map> filters) { - try { - return OBJECT_MAPPER.writeValueAsString(filters); - } catch (JsonProcessingException e) { - throw new RuntimeException(e); - } - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/AbstrAsyncDockerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/AbstrAsyncDockerCmdExec.java deleted file mode 100644 index eac18ae8d..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/AbstrAsyncDockerCmdExec.java +++ /dev/null @@ -1,68 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import java.io.Closeable; -import java.io.IOException; - -import javax.ws.rs.client.WebTarget; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.AsyncDockerCmd; -import com.github.dockerjava.api.command.DockerCmdAsyncExec; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier; - -public abstract class AbstrAsyncDockerCmdExec, A_RES_T> extends - AbstrDockerCmdExec implements DockerCmdAsyncExec { - - public AbstrAsyncDockerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - public Void exec(CMD_T command, ResultCallback resultCallback) { - return execute(command, resultCallback); - } - - protected final Void execute(final CMD_T command, final ResultCallback resultCallback) { - - ResultCallback delegatingResultCallback = new ResultCallback() { - - @Override - public void close() throws IOException { - resultCallback.close(); - command.close(); - } - - @Override - public void onStart(Closeable closeable) { - resultCallback.onStart(closeable); - } - - @Override - public void onNext(A_RES_T object) { - resultCallback.onNext(object); - } - - @Override - public void onError(Throwable throwable) { - resultCallback.onError(throwable); - } - - @Override - public void onComplete() { - resultCallback.onComplete(); - command.close(); - } - }; - - AbstractCallbackNotifier callbackNotifier = callbackNotifier(command, delegatingResultCallback); - - AbstractCallbackNotifier.startAsyncProcessing(callbackNotifier); - - return null; - } - - protected abstract AbstractCallbackNotifier callbackNotifier(CMD_T command, - ResultCallback resultCallback); - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/AbstrDockerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/AbstrDockerCmdExec.java deleted file mode 100644 index ad819468a..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/AbstrDockerCmdExec.java +++ /dev/null @@ -1,84 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static com.github.dockerjava.core.RemoteApiVersion.UNKNOWN_VERSION; -import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_19; -import static com.google.common.base.Preconditions.checkNotNull; - -import java.io.IOException; - -import javax.ws.rs.client.WebTarget; - -import com.fasterxml.jackson.databind.node.ObjectNode; -import org.apache.commons.codec.binary.Base64; - -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.AuthConfigurations; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.RemoteApiVersion; - -public abstract class AbstrDockerCmdExec { - - private final DockerClientConfig dockerClientConfig; - - private final WebTarget baseResource; - - public AbstrDockerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - checkNotNull(baseResource, "baseResource was not specified"); - checkNotNull(dockerClientConfig, "dockerClientConfig was not specified"); - this.baseResource = baseResource; - this.dockerClientConfig = dockerClientConfig; - } - - protected WebTarget getBaseResource() { - return baseResource; - } - - protected AuthConfigurations getBuildAuthConfigs() { - return dockerClientConfig.getAuthConfigurations(); - } - - protected String registryAuth(AuthConfig authConfig) { - try { - return Base64.encodeBase64String(new ObjectMapper().writeValueAsString(authConfig).getBytes()); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - protected String registryConfigs(AuthConfigurations authConfigs) { - try { - final String json; - final ObjectMapper objectMapper = new ObjectMapper(); - final RemoteApiVersion apiVersion = dockerClientConfig.getApiVersion(); - - if (apiVersion.equals(UNKNOWN_VERSION)) { - ObjectNode rootNode = objectMapper.valueToTree(authConfigs.getConfigs()); // all registries - final ObjectNode authNodes = objectMapper.valueToTree(authConfigs); // wrapped in "configs":{} - rootNode.setAll(authNodes); // merge 2 variants - json = rootNode.toString(); - } else if (apiVersion.isGreaterOrEqual(VERSION_1_19)) { - json = objectMapper.writeValueAsString(authConfigs.getConfigs()); - } else { - json = objectMapper.writeValueAsString(authConfigs); - } - - return Base64.encodeBase64String(json.getBytes()); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - protected boolean bool(Boolean bool) { - return bool != null && bool; - } - - protected WebTarget booleanQueryParam(WebTarget webTarget, String name, Boolean value) { - if (bool(value)) { - webTarget = webTarget.queryParam(name, bool(value) + ""); - } - - return webTarget; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/AbstrSyncDockerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/AbstrSyncDockerCmdExec.java deleted file mode 100644 index ed7d3adf0..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/AbstrSyncDockerCmdExec.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.ProcessingException; -import javax.ws.rs.client.WebTarget; - -import com.github.dockerjava.api.command.DockerCmd; -import com.github.dockerjava.api.command.DockerCmdSyncExec; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.core.DockerClientConfig; - -public abstract class AbstrSyncDockerCmdExec, RES_T> extends AbstrDockerCmdExec - implements DockerCmdSyncExec { - - public AbstrSyncDockerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - public RES_T exec(CMD_T command) { - // this hack works because of ResponseStatusExceptionFilter - try (CMD_T cmd = command) { - try { - return execute(cmd); - } catch (ProcessingException e) { - if (e.getCause() instanceof DockerException) { - throw (DockerException) e.getCause(); - } else { - throw e; - } - } - } - } - - protected abstract RES_T execute(CMD_T command); -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/AttachContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/AttachContainerCmdExec.java deleted file mode 100644 index c21c305df..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/AttachContainerCmdExec.java +++ /dev/null @@ -1,46 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.AttachContainerCmd; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.async.FrameStreamProcessor; -import com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier; -import com.github.dockerjava.jaxrs.async.POSTCallbackNotifier; - -public class AttachContainerCmdExec extends AbstrAsyncDockerCmdExec implements - AttachContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(AttachContainerCmdExec.class); - - public AttachContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected AbstractCallbackNotifier callbackNotifier(AttachContainerCmd command, - ResultCallback resultCallback) { - - if (command.getStdin() != null) { - throw new UnsupportedOperationException( - "Passing stdin to the container is currently not supported. Try experimental netty engine!"); - } - - WebTarget webTarget = getBaseResource().path("/containers/{id}/attach").resolveTemplate("id", - command.getContainerId()); - - webTarget = booleanQueryParam(webTarget, "logs", command.hasLogsEnabled()); - webTarget = booleanQueryParam(webTarget, "stdout", command.hasStdoutEnabled()); - webTarget = booleanQueryParam(webTarget, "stderr", command.hasStderrEnabled()); - webTarget = booleanQueryParam(webTarget, "stream", command.hasFollowStreamEnabled()); - - LOGGER.trace("POST: {}", webTarget); - - return new POSTCallbackNotifier(new FrameStreamProcessor(), resultCallback, webTarget.request(), null); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/AuthCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/AuthCmdExec.java deleted file mode 100644 index 6812e397a..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/AuthCmdExec.java +++ /dev/null @@ -1,39 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; -import javax.ws.rs.core.Response; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.AuthCmd; -import com.github.dockerjava.api.exception.UnauthorizedException; -import com.github.dockerjava.api.model.AuthResponse; -import com.github.dockerjava.core.DockerClientConfig; - -public class AuthCmdExec extends AbstrSyncDockerCmdExec implements AuthCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(AuthCmdExec.class); - - public AuthCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected AuthResponse execute(AuthCmd command) { - WebTarget webResource = getBaseResource().path("/auth"); - LOGGER.trace("POST: {}", webResource); - Response response = webResource.request().accept(MediaType.APPLICATION_JSON) - .post(entity(command.getAuthConfig(), MediaType.APPLICATION_JSON)); - - if (response.getStatus() == 401) { - throw new UnauthorizedException("Unauthorized"); - } - - return response.readEntity(AuthResponse.class); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/BuildImageCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/BuildImageCmdExec.java deleted file mode 100644 index 43e7687eb..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/BuildImageCmdExec.java +++ /dev/null @@ -1,126 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.Invocation; -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import com.fasterxml.jackson.databind.ObjectMapper; - -import org.glassfish.jersey.client.ClientProperties; -import org.glassfish.jersey.client.RequestEntityProcessing; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.BuildImageCmd; -import com.github.dockerjava.api.model.AuthConfigurations; -import com.github.dockerjava.api.model.BuildResponseItem; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.async.JsonStreamProcessor; -import com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier; -import com.github.dockerjava.jaxrs.async.POSTCallbackNotifier; - -import java.io.IOException; -import java.net.URLEncoder; - -public class BuildImageCmdExec extends AbstrAsyncDockerCmdExec implements - BuildImageCmd.Exec { - private static final Logger LOGGER = LoggerFactory.getLogger(BuildImageCmdExec.class); - - private static final ObjectMapper MAPPER = new ObjectMapper(); - - public BuildImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - private Invocation.Builder resourceWithOptionalAuthConfig(BuildImageCmd command, Invocation.Builder request) { - final AuthConfigurations authConfigs = firstNonNull(command.getBuildAuthConfigs(), getBuildAuthConfigs()); - if (authConfigs != null && !authConfigs.getConfigs().isEmpty()) { - request = request.header("X-Registry-Config", registryConfigs(authConfigs)); - } - return request; - } - - private static AuthConfigurations firstNonNull(final AuthConfigurations fromCommand, - final AuthConfigurations fromConfig) { - if (fromCommand != null) { - return fromCommand; - } - if (fromConfig != null) { - return fromConfig; - } - return null; - } - - @Override - protected AbstractCallbackNotifier callbackNotifier(BuildImageCmd command, - ResultCallback resultCallback) { - - WebTarget webTarget = getBaseResource().path("/build"); - String dockerFilePath = command.getPathToDockerfile(); - - if (dockerFilePath != null && command.getRemote() == null && !"Dockerfile".equals(dockerFilePath)) { - webTarget = webTarget.queryParam("dockerfile", dockerFilePath); - } - if (command.getTag() != null) { - webTarget = webTarget.queryParam("t", command.getTag()); - } - if (command.getRemote() != null) { - webTarget = webTarget.queryParam("remote", command.getRemote().toString()); - } - - webTarget = booleanQueryParam(webTarget, "q", command.isQuiet()); - webTarget = booleanQueryParam(webTarget, "nocache", command.hasNoCacheEnabled()); - webTarget = booleanQueryParam(webTarget, "pull", command.hasPullEnabled()); - webTarget = booleanQueryParam(webTarget, "rm", command.hasRemoveEnabled()); - webTarget = booleanQueryParam(webTarget, "forcerm", command.isForcerm()); - - // this has to be handled differently as it should switch to 'false' - if (command.hasRemoveEnabled() == null || !command.hasRemoveEnabled()) { - webTarget = webTarget.queryParam("rm", "false"); - } - - if (command.getMemory() != null) { - webTarget = webTarget.queryParam("memory", command.getMemory()); - } - if (command.getMemswap() != null) { - webTarget = webTarget.queryParam("memswap", command.getMemswap()); - } - if (command.getCpushares() != null) { - webTarget = webTarget.queryParam("cpushares", command.getCpushares()); - } - if (command.getCpusetcpus() != null) { - webTarget = webTarget.queryParam("cpusetcpus", command.getCpusetcpus()); - } - - if (command.hasRemoveEnabled() == null || !command.hasRemoveEnabled()) { - webTarget = webTarget.queryParam("rm", "false"); - } - - if (command.getBuildArgs() != null && !command.getBuildArgs().isEmpty()) { - try { - webTarget = webTarget.queryParam("buildargs", - URLEncoder.encode(MAPPER.writeValueAsString(command.getBuildArgs()), "UTF-8")); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - if (command.getShmsize() != null) { - webTarget = webTarget.queryParam("shmsize", command.getShmsize()); - } - - webTarget.property(ClientProperties.REQUEST_ENTITY_PROCESSING, RequestEntityProcessing.CHUNKED); - webTarget.property(ClientProperties.CHUNKED_ENCODING_SIZE, 1024 * 1024); - - LOGGER.trace("POST: {}", webTarget); - - return new POSTCallbackNotifier<>(new JsonStreamProcessor<>(BuildResponseItem.class), - resultCallback, - resourceWithOptionalAuthConfig(command, webTarget.request()).accept(MediaType.TEXT_PLAIN), - entity(command.getTarInputStream(), "application/tar") - ); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/CommitCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/CommitCmdExec.java deleted file mode 100644 index 9acb4eff5..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/CommitCmdExec.java +++ /dev/null @@ -1,37 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.databind.node.ObjectNode; -import com.github.dockerjava.api.command.CommitCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class CommitCmdExec extends AbstrSyncDockerCmdExec implements CommitCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CommitCmdExec.class); - - public CommitCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected String execute(CommitCmd command) { - WebTarget webTarget = getBaseResource().path("/commit").queryParam("container", command.getContainerId()) - .queryParam("repo", command.getRepository()).queryParam("tag", command.getTag()) - .queryParam("m", command.getMessage()).queryParam("author", command.getAuthor()); - - webTarget = booleanQueryParam(webTarget, "pause", command.hasPauseEnabled()); - - LOGGER.trace("POST: {}", webTarget); - ObjectNode objectNode = webTarget.request().accept("application/vnd.docker.raw-stream") - .post(entity(command, MediaType.APPLICATION_JSON), ObjectNode.class); - return objectNode.get("Id").asText(); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/ConnectToNetworkCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/ConnectToNetworkCmdExec.java deleted file mode 100644 index 8a69b31fd..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/ConnectToNetworkCmdExec.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import com.github.dockerjava.api.command.ConnectToNetworkCmd; -import com.github.dockerjava.core.DockerClientConfig; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import static javax.ws.rs.client.Entity.entity; - -public class ConnectToNetworkCmdExec extends AbstrSyncDockerCmdExec - implements ConnectToNetworkCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ConnectToNetworkCmdExec.class); - - public ConnectToNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(ConnectToNetworkCmd command) { - - WebTarget webTarget = getBaseResource().path("/networks/" + command.getNetworkId() + "/connect"); - - LOGGER.trace("POST: {}", webTarget); - webTarget.request().post(entity(command, MediaType.APPLICATION_JSON)); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/ContainerDiffCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/ContainerDiffCmdExec.java deleted file mode 100644 index 29cbeebf7..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/ContainerDiffCmdExec.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import java.util.List; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.GenericType; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.ContainerDiffCmd; -import com.github.dockerjava.api.model.ChangeLog; -import com.github.dockerjava.core.DockerClientConfig; - -public class ContainerDiffCmdExec extends AbstrSyncDockerCmdExec> implements - ContainerDiffCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ContainerDiffCmdExec.class); - - public ContainerDiffCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected List execute(ContainerDiffCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/changes").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(new GenericType>() { - }); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/CopyArchiveFromContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/CopyArchiveFromContainerCmdExec.java deleted file mode 100644 index 7b6a6e2f4..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/CopyArchiveFromContainerCmdExec.java +++ /dev/null @@ -1,37 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import java.io.InputStream; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.Response; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.jaxrs.util.WrappedResponseInputStream; - -public class CopyArchiveFromContainerCmdExec extends AbstrSyncDockerCmdExec - implements CopyArchiveFromContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CopyArchiveFromContainerCmdExec.class); - - public CopyArchiveFromContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InputStream execute(CopyArchiveFromContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/archive").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("Get: " + webResource.toString()); - - Response response = webResource.queryParam("path", command.getResource()).request().accept("application/x-tar") - .get(); - - return new WrappedResponseInputStream(response); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/CopyArchiveToContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/CopyArchiveToContainerCmdExec.java deleted file mode 100644 index b38d337a8..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/CopyArchiveToContainerCmdExec.java +++ /dev/null @@ -1,37 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import java.io.InputStream; - -import javax.ws.rs.client.WebTarget; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class CopyArchiveToContainerCmdExec extends AbstrSyncDockerCmdExec implements - CopyArchiveToContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CopyArchiveFromContainerCmdExec.class); - - public CopyArchiveToContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(CopyArchiveToContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/archive").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("PUT: " + webResource.toString()); - InputStream streamToUpload = command.getTarInputStream(); - webResource.queryParam("path", command.getRemotePath()) - .queryParam("noOverwriteDirNonDir", command.isNoOverwriteDirNonDir()).request() - .put(entity(streamToUpload, "application/x-tar")).close(); - return null; - - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/CopyFileFromContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/CopyFileFromContainerCmdExec.java deleted file mode 100644 index 6afffe0c4..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/CopyFileFromContainerCmdExec.java +++ /dev/null @@ -1,40 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import java.io.InputStream; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; -import javax.ws.rs.core.Response; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.CopyFileFromContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.jaxrs.util.WrappedResponseInputStream; - -public class CopyFileFromContainerCmdExec extends AbstrSyncDockerCmdExec - implements CopyFileFromContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CopyFileFromContainerCmdExec.class); - - public CopyFileFromContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InputStream execute(CopyFileFromContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/copy").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: " + webResource.toString()); - - Response response = webResource.request().accept(MediaType.APPLICATION_OCTET_STREAM_TYPE) - .post(entity(command, MediaType.APPLICATION_JSON)); - - return new WrappedResponseInputStream(response); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/CreateContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/CreateContainerCmdExec.java deleted file mode 100644 index 651362b1c..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/CreateContainerCmdExec.java +++ /dev/null @@ -1,37 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.CreateContainerCmd; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.core.DockerClientConfig; - -public class CreateContainerCmdExec extends AbstrSyncDockerCmdExec - implements CreateContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CreateContainerCmdExec.class); - - public CreateContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected CreateContainerResponse execute(CreateContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/create"); - - if (command.getName() != null) { - webResource = webResource.queryParam("name", command.getName()); - } - - LOGGER.trace("POST: {} ", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(entity(command, MediaType.APPLICATION_JSON), CreateContainerResponse.class); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/CreateImageCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/CreateImageCmdExec.java deleted file mode 100644 index d5c40f83b..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/CreateImageCmdExec.java +++ /dev/null @@ -1,33 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.CreateImageCmd; -import com.github.dockerjava.api.command.CreateImageResponse; -import com.github.dockerjava.core.DockerClientConfig; - -public class CreateImageCmdExec extends AbstrSyncDockerCmdExec implements - CreateImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CreateImageCmdExec.class); - - public CreateImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected CreateImageResponse execute(CreateImageCmd command) { - WebTarget webResource = getBaseResource().path("/images/create").queryParam("repo", command.getRepository()) - .queryParam("tag", command.getTag()).queryParam("fromSrc", "-"); - - LOGGER.trace("POST: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_OCTET_STREAM_TYPE) - .post(entity(command.getImageStream(), MediaType.APPLICATION_OCTET_STREAM), CreateImageResponse.class); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/CreateNetworkCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/CreateNetworkCmdExec.java deleted file mode 100644 index 14f22b502..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/CreateNetworkCmdExec.java +++ /dev/null @@ -1,31 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import com.github.dockerjava.api.command.CreateNetworkCmd; -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.core.DockerClientConfig; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import static javax.ws.rs.client.Entity.entity; - -public class CreateNetworkCmdExec extends AbstrSyncDockerCmdExec implements - CreateNetworkCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CreateNetworkCmdExec.class); - - public CreateNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected CreateNetworkResponse execute(CreateNetworkCmd command) { - WebTarget webResource = getBaseResource().path("/networks/create"); - - LOGGER.trace("POST: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(entity(command, MediaType.APPLICATION_JSON), CreateNetworkResponse.class); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/CreateVolumeCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/CreateVolumeCmdExec.java deleted file mode 100644 index c71d2297e..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/CreateVolumeCmdExec.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.CreateVolumeCmd; -import com.github.dockerjava.api.command.CreateVolumeResponse; -import com.github.dockerjava.core.DockerClientConfig; - -public class CreateVolumeCmdExec extends AbstrSyncDockerCmdExec implements - CreateVolumeCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CreateVolumeCmdExec.class); - - public CreateVolumeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected CreateVolumeResponse execute(CreateVolumeCmd command) { - WebTarget webResource = getBaseResource().path("/volumes/create"); - - LOGGER.trace("POST: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(entity(command, MediaType.APPLICATION_JSON), CreateVolumeResponse.class); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/DisconnectFromNetworkCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/DisconnectFromNetworkCmdExec.java deleted file mode 100644 index 95c3f5c22..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/DisconnectFromNetworkCmdExec.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; -import com.github.dockerjava.core.DockerClientConfig; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import static javax.ws.rs.client.Entity.entity; - -public class DisconnectFromNetworkCmdExec extends AbstrSyncDockerCmdExec - implements DisconnectFromNetworkCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(DisconnectFromNetworkCmdExec.class); - - public DisconnectFromNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(DisconnectFromNetworkCmd command) { - - WebTarget webTarget = getBaseResource().path("/networks/" + command.getNetworkId() + "/disconnect"); - - LOGGER.trace("POST: {}", webTarget); - webTarget.request().post(entity(command, MediaType.APPLICATION_JSON)); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/DockerCmdExecFactoryImpl.java b/src/main/java/com/github/dockerjava/jaxrs/DockerCmdExecFactoryImpl.java deleted file mode 100644 index 1d407e233..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/DockerCmdExecFactoryImpl.java +++ /dev/null @@ -1,9 +0,0 @@ -package com.github.dockerjava.jaxrs; - -/** - * @author Kanstantsin Shautsou - * @deprecated clashes with netty impl. - */ -@Deprecated -public class DockerCmdExecFactoryImpl extends JerseyDockerCmdExecFactory { -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/EventsCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/EventsCmdExec.java deleted file mode 100644 index 3d4c8dc17..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/EventsCmdExec.java +++ /dev/null @@ -1,42 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static com.google.common.net.UrlEscapers.urlPathSegmentEscaper; - -import javax.ws.rs.client.WebTarget; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.EventsCmd; -import com.github.dockerjava.api.model.Event; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.async.JsonStreamProcessor; -import com.github.dockerjava.core.util.FiltersEncoder; -import com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier; -import com.github.dockerjava.jaxrs.async.GETCallbackNotifier; - -public class EventsCmdExec extends AbstrAsyncDockerCmdExec implements EventsCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(EventsCmdExec.class); - - public EventsCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected AbstractCallbackNotifier callbackNotifier(EventsCmd command, ResultCallback resultCallback) { - WebTarget webTarget = getBaseResource().path("/events").queryParam("since", command.getSince()) - .queryParam("until", command.getUntil()); - - if (command.getFilters() != null && !command.getFilters().isEmpty()) { - webTarget = webTarget - .queryParam("filters", urlPathSegmentEscaper().escape(FiltersEncoder.jsonEncode(command.getFilters()))); - } - - LOGGER.trace("GET: {}", webTarget); - - return new GETCallbackNotifier(new JsonStreamProcessor(Event.class), resultCallback, - webTarget.request()); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/ExecCreateCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/ExecCreateCmdExec.java deleted file mode 100644 index 4de6eedd2..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/ExecCreateCmdExec.java +++ /dev/null @@ -1,34 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.ExecCreateCmd; -import com.github.dockerjava.api.command.ExecCreateCmdResponse; -import com.github.dockerjava.core.DockerClientConfig; - -public class ExecCreateCmdExec extends AbstrSyncDockerCmdExec implements - ExecCreateCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(VersionCmdExec.class); - - public ExecCreateCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected ExecCreateCmdResponse execute(ExecCreateCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/exec").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(entity(command, MediaType.APPLICATION_JSON), ExecCreateCmdResponse.class); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/ExecStartCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/ExecStartCmdExec.java deleted file mode 100644 index dc7cf6598..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/ExecStartCmdExec.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.ExecStartCmd; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.async.FrameStreamProcessor; -import com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier; -import com.github.dockerjava.jaxrs.async.POSTCallbackNotifier; - -public class ExecStartCmdExec extends AbstrAsyncDockerCmdExec implements ExecStartCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ExecStartCmdExec.class); - - public ExecStartCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected AbstractCallbackNotifier callbackNotifier(ExecStartCmd command, - ResultCallback resultCallback) { - WebTarget webTarget = getBaseResource().path("/exec/{id}/start").resolveTemplate("id", command.getExecId()); - - LOGGER.trace("POST: {}", webTarget); - - return new POSTCallbackNotifier(new FrameStreamProcessor(), resultCallback, webTarget.request().accept( - MediaType.APPLICATION_JSON), entity(command, MediaType.APPLICATION_JSON)); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/InfoCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/InfoCmdExec.java deleted file mode 100644 index c544571ef..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/InfoCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.InfoCmd; -import com.github.dockerjava.api.model.Info; -import com.github.dockerjava.core.DockerClientConfig; - -public class InfoCmdExec extends AbstrSyncDockerCmdExec implements InfoCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(InfoCmdExec.class); - - public InfoCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Info execute(InfoCmd command) { - WebTarget webResource = getBaseResource().path("/info"); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(Info.class); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/InspectContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/InspectContainerCmdExec.java deleted file mode 100644 index 0599f04a1..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/InspectContainerCmdExec.java +++ /dev/null @@ -1,33 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.InspectContainerCmd; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.core.DockerClientConfig; - -public class InspectContainerCmdExec extends AbstrSyncDockerCmdExec - implements InspectContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(InspectContainerCmdExec.class); - - public InspectContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InspectContainerResponse execute(InspectContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/json") - .resolveTemplate("id", command.getContainerId()); - - webResource = booleanQueryParam(webResource, "size", command.getSize()); - - LOGGER.debug("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(InspectContainerResponse.class); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/InspectExecCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/InspectExecCmdExec.java deleted file mode 100644 index 4290d0371..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/InspectExecCmdExec.java +++ /dev/null @@ -1,27 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.InspectExecCmd; -import com.github.dockerjava.api.command.InspectExecResponse; -import com.github.dockerjava.core.DockerClientConfig; - -public class InspectExecCmdExec extends AbstrSyncDockerCmdExec implements - InspectExecCmd.Exec { - private static final Logger LOGGER = LoggerFactory.getLogger(InspectExecCmdExec.class); - - public InspectExecCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InspectExecResponse execute(InspectExecCmd command) { - WebTarget webResource = getBaseResource().path("/exec/{id}/json").resolveTemplate("id", command.getExecId()); - LOGGER.debug("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(InspectExecResponse.class); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/InspectImageCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/InspectImageCmdExec.java deleted file mode 100644 index c1ef42424..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/InspectImageCmdExec.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.InspectImageCmd; -import com.github.dockerjava.api.command.InspectImageResponse; -import com.github.dockerjava.core.DockerClientConfig; - -public class InspectImageCmdExec extends AbstrSyncDockerCmdExec implements - InspectImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(InspectImageCmdExec.class); - - public InspectImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InspectImageResponse execute(InspectImageCmd command) { - WebTarget webResource = getBaseResource().path("/images/{id}/json").resolveTemplate("id", command.getImageId()); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(InspectImageResponse.class); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/InspectNetworkCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/InspectNetworkCmdExec.java deleted file mode 100644 index c9b19f881..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/InspectNetworkCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import com.github.dockerjava.api.command.InspectNetworkCmd; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.core.DockerClientConfig; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -public class InspectNetworkCmdExec extends AbstrSyncDockerCmdExec implements InspectNetworkCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ListNetworksCmdExec.class); - - public InspectNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Network execute(InspectNetworkCmd command) { - - WebTarget webResource = getBaseResource().path("/networks/{id}").resolveTemplate("id", - command.getNetworkId()); - - LOGGER.debug("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(Network.class); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/InspectVolumeCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/InspectVolumeCmdExec.java deleted file mode 100644 index 9ca2ac02a..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/InspectVolumeCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.InspectVolumeCmd; -import com.github.dockerjava.api.command.InspectVolumeResponse; -import com.github.dockerjava.core.DockerClientConfig; - -public class InspectVolumeCmdExec extends AbstrSyncDockerCmdExec implements - InspectVolumeCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(InspectVolumeCmdExec.class); - - public InspectVolumeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InspectVolumeResponse execute(InspectVolumeCmd command) { - WebTarget webResource = getBaseResource().path("/volumes/{name}").resolveTemplate("name", command.getName()); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(InspectVolumeResponse.class); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/JerseyDockerCmdExecFactory.java b/src/main/java/com/github/dockerjava/jaxrs/JerseyDockerCmdExecFactory.java deleted file mode 100644 index 3339a2696..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/JerseyDockerCmdExecFactory.java +++ /dev/null @@ -1,583 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.io.IOException; -import java.net.InetSocketAddress; -import java.net.Proxy; -import java.net.ProxySelector; -import java.net.URI; -import java.net.URISyntaxException; -import java.util.List; - -import javax.net.ssl.SSLContext; -import javax.ws.rs.client.Client; -import javax.ws.rs.client.ClientBuilder; -import javax.ws.rs.client.ClientRequestFilter; -import javax.ws.rs.client.ClientResponseFilter; -import javax.ws.rs.client.WebTarget; - -import com.github.dockerjava.api.command.UpdateContainerCmd; -import com.github.dockerjava.core.SSLConfig; - -import org.apache.http.config.RegistryBuilder; -import org.apache.http.conn.socket.ConnectionSocketFactory; -import org.apache.http.conn.socket.PlainConnectionSocketFactory; -import org.apache.http.conn.ssl.SSLConnectionSocketFactory; -import org.apache.http.impl.conn.PoolingHttpClientConnectionManager; -import org.glassfish.jersey.CommonProperties; -import org.glassfish.jersey.apache.connector.ApacheClientProperties; -import org.glassfish.jersey.apache.connector.ApacheConnectorProvider; -import org.glassfish.jersey.client.ClientConfig; -import org.glassfish.jersey.client.ClientProperties; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.jaxrs.json.JacksonJsonProvider; -import com.github.dockerjava.api.command.AttachContainerCmd; -import com.github.dockerjava.api.command.AuthCmd; -import com.github.dockerjava.api.command.BuildImageCmd; -import com.github.dockerjava.api.command.CommitCmd; -import com.github.dockerjava.api.command.ConnectToNetworkCmd; -import com.github.dockerjava.api.command.ContainerDiffCmd; -import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; -import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; -import com.github.dockerjava.api.command.CopyFileFromContainerCmd; -import com.github.dockerjava.api.command.CreateContainerCmd; -import com.github.dockerjava.api.command.CreateImageCmd; -import com.github.dockerjava.api.command.CreateNetworkCmd; -import com.github.dockerjava.api.command.CreateVolumeCmd; -import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; -import com.github.dockerjava.api.command.DockerCmdExecFactory; -import com.github.dockerjava.api.command.EventsCmd; -import com.github.dockerjava.api.command.ExecCreateCmd; -import com.github.dockerjava.api.command.ExecStartCmd; -import com.github.dockerjava.api.command.InfoCmd; -import com.github.dockerjava.api.command.InspectContainerCmd; -import com.github.dockerjava.api.command.InspectExecCmd; -import com.github.dockerjava.api.command.InspectImageCmd; -import com.github.dockerjava.api.command.InspectNetworkCmd; -import com.github.dockerjava.api.command.InspectVolumeCmd; -import com.github.dockerjava.api.command.KillContainerCmd; -import com.github.dockerjava.api.command.ListContainersCmd; -import com.github.dockerjava.api.command.ListImagesCmd; -import com.github.dockerjava.api.command.ListNetworksCmd; -import com.github.dockerjava.api.command.ListVolumesCmd; -import com.github.dockerjava.api.command.LoadImageCmd; -import com.github.dockerjava.api.command.LogContainerCmd; -import com.github.dockerjava.api.command.PauseContainerCmd; -import com.github.dockerjava.api.command.PingCmd; -import com.github.dockerjava.api.command.PullImageCmd; -import com.github.dockerjava.api.command.PushImageCmd; -import com.github.dockerjava.api.command.RemoveContainerCmd; -import com.github.dockerjava.api.command.RemoveImageCmd; -import com.github.dockerjava.api.command.RemoveNetworkCmd; -import com.github.dockerjava.api.command.RemoveVolumeCmd; -import com.github.dockerjava.api.command.RestartContainerCmd; -import com.github.dockerjava.api.command.SaveImageCmd; -import com.github.dockerjava.api.command.SearchImagesCmd; -import com.github.dockerjava.api.command.StartContainerCmd; -import com.github.dockerjava.api.command.StatsCmd; -import com.github.dockerjava.api.command.StopContainerCmd; -import com.github.dockerjava.api.command.TagImageCmd; -import com.github.dockerjava.api.command.TopContainerCmd; -import com.github.dockerjava.api.command.UnpauseContainerCmd; -import com.github.dockerjava.api.command.VersionCmd; -import com.github.dockerjava.api.command.WaitContainerCmd; -import com.github.dockerjava.api.command.RenameContainerCmd; -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.jaxrs.filter.JsonClientFilter; -import com.github.dockerjava.jaxrs.filter.ResponseStatusExceptionFilter; -import com.github.dockerjava.jaxrs.filter.SelectiveLoggingFilter; - -//import org.glassfish.jersey.apache.connector.ApacheConnectorProvider; -// see https://github.com/docker-java/docker-java/issues/196 - -public class JerseyDockerCmdExecFactory implements DockerCmdExecFactory { - - private static final Logger LOGGER = LoggerFactory.getLogger(JerseyDockerCmdExecFactory.class.getName()); - - private Client client; - - private WebTarget baseResource; - - private Integer readTimeout = null; - - private Integer connectTimeout = null; - - private Integer maxTotalConnections = null; - - private Integer maxPerRouteConnections = null; - - private ClientRequestFilter[] clientRequestFilters = null; - - private ClientResponseFilter[] clientResponseFilters = null; - - private DockerClientConfig dockerClientConfig; - - private PoolingHttpClientConnectionManager connManager = null; - - @Override - public void init(DockerClientConfig dockerClientConfig) { - checkNotNull(dockerClientConfig, "config was not specified"); - this.dockerClientConfig = dockerClientConfig; - - ClientConfig clientConfig = new ClientConfig(); - clientConfig.connectorProvider(new ApacheConnectorProvider()); - clientConfig.property(CommonProperties.FEATURE_AUTO_DISCOVERY_DISABLE, true); - - clientConfig.register(ResponseStatusExceptionFilter.class); - clientConfig.register(JsonClientFilter.class); - clientConfig.register(JacksonJsonProvider.class); - - // logging may disabled via log level - clientConfig.register(new SelectiveLoggingFilter(LOGGER, true)); - - if (readTimeout != null) { - clientConfig.property(ClientProperties.READ_TIMEOUT, readTimeout); - } - - if (connectTimeout != null) { - clientConfig.property(ClientProperties.CONNECT_TIMEOUT, connectTimeout); - } - - if (clientResponseFilters != null) { - for (ClientResponseFilter clientResponseFilter : clientResponseFilters) { - if (clientResponseFilter != null) { - clientConfig.register(clientResponseFilter); - } - } - } - - if (clientRequestFilters != null) { - for (ClientRequestFilter clientRequestFilter : clientRequestFilters) { - if (clientRequestFilter != null) { - clientConfig.register(clientRequestFilter); - } - } - } - - URI originalUri = dockerClientConfig.getDockerHost(); - - String protocol = null; - - SSLContext sslContext = null; - - try { - final SSLConfig sslConfig = dockerClientConfig.getSSLConfig(); - if (sslConfig != null) { - sslContext = sslConfig.getSSLContext(); - } - } catch (Exception ex) { - throw new DockerClientException("Error in SSL Configuration", ex); - } - - if (sslContext != null) { - protocol = "https"; - } else { - protocol = "http"; - } - - if (!originalUri.getScheme().equals("unix")) { - - try { - originalUri = new URI(originalUri.toString().replaceFirst("tcp", protocol)); - } catch (URISyntaxException e) { - throw new RuntimeException(e); - } - - configureProxy(clientConfig, protocol); - } - - connManager = new PoolingHttpClientConnectionManager(getSchemeRegistry( - originalUri, sslContext)) { - - @Override - public void close() { - super.shutdown(); - } - - @Override - public void shutdown() { - // Disable shutdown of the pool. This will be done later, when this factory is closed - // This is a workaround for finalize method on jerseys ClientRuntime which - // closes the client and shuts down the connection pool when it is garbage collected - } - }; - - if (maxTotalConnections != null) { - connManager.setMaxTotal(maxTotalConnections); - } - if (maxPerRouteConnections != null) { - connManager.setDefaultMaxPerRoute(maxPerRouteConnections); - } - - clientConfig.property(ApacheClientProperties.CONNECTION_MANAGER, connManager); - - // Configure connection pool timeout - // clientConfig.property(ApacheClientProperties.REQUEST_CONFIG, RequestConfig.custom() - // .setConnectionRequestTimeout(1000).build()); - - ClientBuilder clientBuilder = ClientBuilder.newBuilder().withConfig(clientConfig); - - if (sslContext != null) { - clientBuilder.sslContext(sslContext); - } - - client = clientBuilder.build(); - - baseResource = client.target(sanitizeUrl(originalUri).toString()).path(dockerClientConfig.getApiVersion().asWebPathPart()); - } - - private URI sanitizeUrl(URI originalUri) { - if (originalUri.getScheme().equals("unix")) { - return UnixConnectionSocketFactory.sanitizeUri(originalUri); - } - return originalUri; - } - - private void configureProxy(ClientConfig clientConfig, String protocol) { - - List proxies = ProxySelector.getDefault().select(dockerClientConfig.getDockerHost()); - - for (Proxy proxy : proxies) { - InetSocketAddress address = (InetSocketAddress) proxy.address(); - if (address != null) { - String hostname = address.getHostName(); - int port = address.getPort(); - - clientConfig.property(ClientProperties.PROXY_URI, "http://" + hostname + ":" + port); - - String httpProxyUser = System.getProperty(protocol + ".proxyUser"); - if (httpProxyUser != null) { - clientConfig.property(ClientProperties.PROXY_USERNAME, httpProxyUser); - String httpProxyPassword = System.getProperty(protocol + ".proxyPassword"); - if (httpProxyPassword != null) { - clientConfig.property(ClientProperties.PROXY_PASSWORD, httpProxyPassword); - } - } - } - } - } - - private org.apache.http.config.Registry getSchemeRegistry(final URI originalUri, - SSLContext sslContext) { - RegistryBuilder registryBuilder = RegistryBuilder.create(); - registryBuilder.register("http", PlainConnectionSocketFactory.getSocketFactory()); - if (sslContext != null) { - registryBuilder.register("https", new SSLConnectionSocketFactory(sslContext)); - } - registryBuilder.register("unix", new UnixConnectionSocketFactory(originalUri)); - return registryBuilder.build(); - } - - protected WebTarget getBaseResource() { - checkNotNull(baseResource, "Factory not initialized, baseResource not set. You probably forgot to call init()!"); - return baseResource; - } - - protected DockerClientConfig getDockerClientConfig() { - checkNotNull(dockerClientConfig, - "Factor not initialized, dockerClientConfig not set. You probably forgot to call init()!"); - return dockerClientConfig; - } - - @Override - public AuthCmd.Exec createAuthCmdExec() { - return new AuthCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InfoCmd.Exec createInfoCmdExec() { - return new InfoCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public PingCmd.Exec createPingCmdExec() { - return new PingCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public VersionCmd.Exec createVersionCmdExec() { - return new VersionCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public PullImageCmd.Exec createPullImageCmdExec() { - return new PullImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public PushImageCmd.Exec createPushImageCmdExec() { - return new PushImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public SaveImageCmd.Exec createSaveImageCmdExec() { - return new SaveImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CreateImageCmd.Exec createCreateImageCmdExec() { - return new CreateImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public LoadImageCmd.Exec createLoadImageCmdExec() { - return new LoadImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public SearchImagesCmd.Exec createSearchImagesCmdExec() { - return new SearchImagesCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RemoveImageCmd.Exec createRemoveImageCmdExec() { - return new RemoveImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ListImagesCmd.Exec createListImagesCmdExec() { - return new ListImagesCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InspectImageCmd.Exec createInspectImageCmdExec() { - return new InspectImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ListContainersCmd.Exec createListContainersCmdExec() { - return new ListContainersCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CreateContainerCmd.Exec createCreateContainerCmdExec() { - return new CreateContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public StartContainerCmd.Exec createStartContainerCmdExec() { - return new StartContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InspectContainerCmd.Exec createInspectContainerCmdExec() { - return new InspectContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ExecCreateCmd.Exec createExecCmdExec() { - return new ExecCreateCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RemoveContainerCmd.Exec createRemoveContainerCmdExec() { - return new RemoveContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public WaitContainerCmd.Exec createWaitContainerCmdExec() { - return new WaitContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public AttachContainerCmd.Exec createAttachContainerCmdExec() { - return new AttachContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ExecStartCmd.Exec createExecStartCmdExec() { - return new ExecStartCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InspectExecCmd.Exec createInspectExecCmdExec() { - return new InspectExecCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public LogContainerCmd.Exec createLogContainerCmdExec() { - return new LogContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CopyArchiveFromContainerCmd.Exec createCopyArchiveFromContainerCmdExec() { - return new CopyArchiveFromContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CopyFileFromContainerCmd.Exec createCopyFileFromContainerCmdExec() { - return new CopyFileFromContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CopyArchiveToContainerCmd.Exec createCopyArchiveToContainerCmdExec() { - return new CopyArchiveToContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public StopContainerCmd.Exec createStopContainerCmdExec() { - return new StopContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ContainerDiffCmd.Exec createContainerDiffCmdExec() { - return new ContainerDiffCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public KillContainerCmd.Exec createKillContainerCmdExec() { - return new KillContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public UpdateContainerCmd.Exec createUpdateContainerCmdExec() { - return new UpdateContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RenameContainerCmd.Exec createRenameContainerCmdExec() { - return new RenameContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RestartContainerCmd.Exec createRestartContainerCmdExec() { - return new RestartContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CommitCmd.Exec createCommitCmdExec() { - return new CommitCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public BuildImageCmd.Exec createBuildImageCmdExec() { - return new BuildImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public TopContainerCmd.Exec createTopContainerCmdExec() { - return new TopContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public TagImageCmd.Exec createTagImageCmdExec() { - return new TagImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public PauseContainerCmd.Exec createPauseContainerCmdExec() { - return new PauseContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public UnpauseContainerCmd.Exec createUnpauseContainerCmdExec() { - return new UnpauseContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public EventsCmd.Exec createEventsCmdExec() { - return new EventsCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public StatsCmd.Exec createStatsCmdExec() { - return new StatsCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CreateVolumeCmd.Exec createCreateVolumeCmdExec() { - return new CreateVolumeCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InspectVolumeCmd.Exec createInspectVolumeCmdExec() { - return new InspectVolumeCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RemoveVolumeCmd.Exec createRemoveVolumeCmdExec() { - return new RemoveVolumeCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ListVolumesCmd.Exec createListVolumesCmdExec() { - return new ListVolumesCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ListNetworksCmd.Exec createListNetworksCmdExec() { - return new ListNetworksCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InspectNetworkCmd.Exec createInspectNetworkCmdExec() { - - return new InspectNetworkCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CreateNetworkCmd.Exec createCreateNetworkCmdExec() { - - return new CreateNetworkCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RemoveNetworkCmd.Exec createRemoveNetworkCmdExec() { - - return new RemoveNetworkCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ConnectToNetworkCmd.Exec createConnectToNetworkCmdExec() { - - return new ConnectToNetworkCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public DisconnectFromNetworkCmd.Exec createDisconnectFromNetworkCmdExec() { - - return new DisconnectFromNetworkCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public void close() throws IOException { - checkNotNull(client, "Factory not initialized. You probably forgot to call init()!"); - client.close(); - connManager.close(); - } - - public JerseyDockerCmdExecFactory withReadTimeout(Integer readTimeout) { - this.readTimeout = readTimeout; - return this; - } - - public JerseyDockerCmdExecFactory withConnectTimeout(Integer connectTimeout) { - this.connectTimeout = connectTimeout; - return this; - } - - public JerseyDockerCmdExecFactory withMaxTotalConnections(Integer maxTotalConnections) { - this.maxTotalConnections = maxTotalConnections; - return this; - } - - public JerseyDockerCmdExecFactory withMaxPerRouteConnections(Integer maxPerRouteConnections) { - this.maxPerRouteConnections = maxPerRouteConnections; - return this; - } - - public JerseyDockerCmdExecFactory withClientResponseFilters(ClientResponseFilter... clientResponseFilter) { - this.clientResponseFilters = clientResponseFilter; - return this; - } - - public JerseyDockerCmdExecFactory withClientRequestFilters(ClientRequestFilter... clientRequestFilters) { - this.clientRequestFilters = clientRequestFilters; - return this; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/KillContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/KillContainerCmdExec.java deleted file mode 100644 index 95c8ef83c..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/KillContainerCmdExec.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.KillContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class KillContainerCmdExec extends AbstrSyncDockerCmdExec implements - KillContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(KillContainerCmdExec.class); - - public KillContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(KillContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/kill").resolveTemplate("id", - command.getContainerId()); - - if (command.getSignal() != null) { - webResource = webResource.queryParam("signal", command.getSignal()); - } - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/ListContainersCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/ListContainersCmdExec.java deleted file mode 100644 index 50311ef54..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/ListContainersCmdExec.java +++ /dev/null @@ -1,54 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static com.google.common.net.UrlEscapers.urlPathSegmentEscaper; - -import java.util.List; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.GenericType; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.ListContainersCmd; -import com.github.dockerjava.api.model.Container; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.util.FiltersEncoder; - -public class ListContainersCmdExec extends AbstrSyncDockerCmdExec> implements - ListContainersCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ListContainersCmdExec.class); - - public ListContainersCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected List execute(ListContainersCmd command) { - WebTarget webTarget = getBaseResource().path("/containers/json").queryParam("since", command.getSinceId()) - .queryParam("before", command.getBeforeId()); - - webTarget = booleanQueryParam(webTarget, "all", command.hasShowAllEnabled()); - webTarget = booleanQueryParam(webTarget, "size", command.hasShowSizeEnabled()); - - if (command.getLimit() != null && command.getLimit() >= 0) { - webTarget = webTarget.queryParam("limit", String.valueOf(command.getLimit())); - } - - if (command.getFilters() != null && !command.getFilters().isEmpty()) { - webTarget = webTarget - .queryParam("filters", urlPathSegmentEscaper().escape(FiltersEncoder.jsonEncode(command.getFilters()))); - } - - LOGGER.trace("GET: {}", webTarget); - List containers = webTarget.request().accept(MediaType.APPLICATION_JSON) - .get(new GenericType>() { - }); - LOGGER.trace("Response: {}", containers); - - return containers; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/ListImagesCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/ListImagesCmdExec.java deleted file mode 100644 index 3d6320151..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/ListImagesCmdExec.java +++ /dev/null @@ -1,50 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static com.google.common.net.UrlEscapers.urlPathSegmentEscaper; - -import java.util.List; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.GenericType; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.ListImagesCmd; -import com.github.dockerjava.api.model.Image; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.util.FiltersEncoder; - -public class ListImagesCmdExec extends AbstrSyncDockerCmdExec> implements ListImagesCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ListImagesCmdExec.class); - - public ListImagesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected List execute(ListImagesCmd command) { - WebTarget webTarget = getBaseResource().path("/images/json"); - - webTarget = booleanQueryParam(webTarget, "all", command.hasShowAllEnabled()); - - if (command.getFilters() != null && !command.getFilters().isEmpty()) { - webTarget = webTarget.queryParam("filters", urlPathSegmentEscaper().escape(FiltersEncoder.jsonEncode(command.getFilters()))); - } - - if (command.getImageNameFilter() != null) { - webTarget = webTarget.queryParam("filter", urlPathSegmentEscaper().escape(command.getImageNameFilter())); - } - - LOGGER.trace("GET: {}", webTarget); - - List images = webTarget.request().accept(MediaType.APPLICATION_JSON).get(new GenericType>() { - }); - LOGGER.trace("Response: {}", images); - - return images; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/ListNetworksCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/ListNetworksCmdExec.java deleted file mode 100644 index 9137e28d2..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/ListNetworksCmdExec.java +++ /dev/null @@ -1,46 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import com.github.dockerjava.api.command.ListNetworksCmd; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.util.FiltersEncoder; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.GenericType; -import javax.ws.rs.core.MediaType; - -import java.util.List; - -import static com.google.common.net.UrlEscapers.urlPathSegmentEscaper; - -public class ListNetworksCmdExec extends AbstrSyncDockerCmdExec> implements - ListNetworksCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ListNetworksCmdExec.class); - - public ListNetworksCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected List execute(ListNetworksCmd command) { - WebTarget webTarget = getBaseResource().path("/networks"); - - if (command.getFilters() != null && !command.getFilters().isEmpty()) { - webTarget = webTarget - .queryParam("filters", urlPathSegmentEscaper().escape(FiltersEncoder.jsonEncode(command.getFilters()))); - } - - LOGGER.trace("GET: {}", webTarget); - List networks = webTarget.request().accept(MediaType.APPLICATION_JSON) - .get(new GenericType>() { - }); - LOGGER.trace("Response: {}", networks); - - return networks; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/ListVolumesCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/ListVolumesCmdExec.java deleted file mode 100644 index 4f2165b93..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/ListVolumesCmdExec.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static com.google.common.net.UrlEscapers.urlPathSegmentEscaper; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.ListVolumesCmd; -import com.github.dockerjava.api.command.ListVolumesResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.util.FiltersEncoder; - -public class ListVolumesCmdExec extends AbstrSyncDockerCmdExec implements ListVolumesCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ListVolumesCmdExec.class); - - public ListVolumesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected ListVolumesResponse execute(ListVolumesCmd command) { - WebTarget webTarget = getBaseResource().path("/volumes"); - - if (command.getFilters() != null && !command.getFilters().isEmpty()) { - webTarget = webTarget.queryParam("filters", urlPathSegmentEscaper().escape(FiltersEncoder.jsonEncode(command.getFilters()))); - } - - LOGGER.trace("GET: {}", webTarget); - - return webTarget.request().accept(MediaType.APPLICATION_JSON).get(ListVolumesResponse.class); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/LoadImageCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/LoadImageCmdExec.java deleted file mode 100644 index 23d6c493a..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/LoadImageCmdExec.java +++ /dev/null @@ -1,31 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.LoadImageCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class LoadImageCmdExec extends AbstrSyncDockerCmdExec implements LoadImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(LoadImageCmdExec.class); - - public LoadImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(LoadImageCmd command) { - WebTarget webTarget = getBaseResource().path("/images/load"); - - LOGGER.trace("POST: {}", webTarget); - webTarget.request().post(entity(command.getImageStream(), MediaType.APPLICATION_OCTET_STREAM)); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/LogContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/LogContainerCmdExec.java deleted file mode 100644 index 87bc83b98..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/LogContainerCmdExec.java +++ /dev/null @@ -1,49 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.LogContainerCmd; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.async.FrameStreamProcessor; -import com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier; -import com.github.dockerjava.jaxrs.async.GETCallbackNotifier; - -public class LogContainerCmdExec extends AbstrAsyncDockerCmdExec implements - LogContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(LogContainerCmdExec.class); - - public LogContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected AbstractCallbackNotifier callbackNotifier(LogContainerCmd command, - ResultCallback resultCallback) { - - WebTarget webTarget = getBaseResource().path("/containers/{id}/logs").resolveTemplate("id", - command.getContainerId()); - - if (command.getTail() != null) { - webTarget = webTarget.queryParam("tail", command.getTail()); - } - - if (command.getSince() != null) { - webTarget = webTarget.queryParam("since", command.getSince()); - } - - webTarget = booleanQueryParam(webTarget, "timestamps", command.hasTimestampsEnabled()); - webTarget = booleanQueryParam(webTarget, "stdout", command.hasStdoutEnabled()); - webTarget = booleanQueryParam(webTarget, "stderr", command.hasStderrEnabled()); - webTarget = booleanQueryParam(webTarget, "follow", command.hasFollowStreamEnabled()); - - LOGGER.trace("GET: {}", webTarget); - - return new GETCallbackNotifier(new FrameStreamProcessor(), resultCallback, webTarget.request()); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/PauseContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/PauseContainerCmdExec.java deleted file mode 100644 index 5e6beb835..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/PauseContainerCmdExec.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.PauseContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class PauseContainerCmdExec extends AbstrSyncDockerCmdExec implements - PauseContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(PauseContainerCmdExec.class); - - public PauseContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(PauseContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/pause").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/PingCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/PingCmdExec.java deleted file mode 100644 index ef6215358..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/PingCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.PingCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class PingCmdExec extends AbstrSyncDockerCmdExec implements PingCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(PingCmdExec.class); - - public PingCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(PingCmd command) { - WebTarget webResource = getBaseResource().path("/_ping"); - - LOGGER.trace("GET: {}", webResource); - webResource.request().get().close(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/PullImageCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/PullImageCmdExec.java deleted file mode 100644 index 1f0b21bf7..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/PullImageCmdExec.java +++ /dev/null @@ -1,53 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.Invocation; -import javax.ws.rs.client.Invocation.Builder; -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.PullImageCmd; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.PullResponseItem; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.async.JsonStreamProcessor; -import com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier; -import com.github.dockerjava.jaxrs.async.POSTCallbackNotifier; - -public class PullImageCmdExec extends AbstrAsyncDockerCmdExec implements - PullImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(PullImageCmdExec.class); - - public PullImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - private Invocation.Builder resourceWithOptionalAuthConfig(PullImageCmd command, Invocation.Builder request) { - AuthConfig authConfig = command.getAuthConfig(); - if (authConfig != null) { - request = request.header("X-Registry-Auth", registryAuth(authConfig)); - } - return request; - } - - @Override - protected AbstractCallbackNotifier callbackNotifier(PullImageCmd command, - ResultCallback resultCallback) { - - WebTarget webResource = getBaseResource().path("/images/create").queryParam("tag", command.getTag()) - .queryParam("fromImage", command.getRepository()).queryParam("registry", command.getRegistry()); - - LOGGER.trace("POST: {}", webResource); - Builder builder = resourceWithOptionalAuthConfig(command, webResource.request()).accept( - MediaType.APPLICATION_OCTET_STREAM_TYPE); - - return new POSTCallbackNotifier(new JsonStreamProcessor( - PullResponseItem.class), resultCallback, builder, entity(null, MediaType.APPLICATION_JSON)); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/PushImageCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/PushImageCmdExec.java deleted file mode 100644 index 2943ef7e8..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/PushImageCmdExec.java +++ /dev/null @@ -1,52 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.Invocation.Builder; -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.PushImageCmd; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.PushResponseItem; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.async.JsonStreamProcessor; -import com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier; -import com.github.dockerjava.jaxrs.async.POSTCallbackNotifier; - -public class PushImageCmdExec extends AbstrAsyncDockerCmdExec implements - PushImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(PushImageCmdExec.class); - - public PushImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - private String name(PushImageCmd command) { - String name = command.getName(); - AuthConfig authConfig = command.getAuthConfig(); - return (name.contains("/") || authConfig == null) ? name : authConfig.getUsername(); - } - - @Override - protected AbstractCallbackNotifier callbackNotifier(PushImageCmd command, - ResultCallback resultCallback) { - - WebTarget webResource = getBaseResource().path("/images/" + name(command) + "/push").queryParam("tag", - command.getTag()); - - final String registryAuth = registryAuth(command.getAuthConfig()); - LOGGER.trace("POST: {}", webResource); - - Builder builder = webResource.request().header("X-Registry-Auth", registryAuth) - .accept(MediaType.APPLICATION_JSON); - - return new POSTCallbackNotifier(new JsonStreamProcessor( - PushResponseItem.class), resultCallback, builder, entity(null, MediaType.APPLICATION_JSON)); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/RemoveContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/RemoveContainerCmdExec.java deleted file mode 100644 index d649b57a5..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/RemoveContainerCmdExec.java +++ /dev/null @@ -1,34 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.RemoveContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class RemoveContainerCmdExec extends AbstrSyncDockerCmdExec implements - RemoveContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(RemoveContainerCmdExec.class); - - public RemoveContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RemoveContainerCmd command) { - WebTarget webTarget = getBaseResource().path("/containers/" + command.getContainerId()); - - webTarget = booleanQueryParam(webTarget, "v", command.hasRemoveVolumesEnabled()); - webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); - - LOGGER.trace("DELETE: {}", webTarget); - webTarget.request().accept(MediaType.APPLICATION_JSON).delete().close(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/RemoveImageCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/RemoveImageCmdExec.java deleted file mode 100644 index 9b362df72..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/RemoveImageCmdExec.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.RemoveImageCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class RemoveImageCmdExec extends AbstrSyncDockerCmdExec implements RemoveImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(RemoveImageCmdExec.class); - - public RemoveImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RemoveImageCmd command) { - WebTarget webTarget = getBaseResource().path("/images/" + command.getImageId()); - - webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); - webTarget = booleanQueryParam(webTarget, "noprune", command.hasNoPruneEnabled()); - - LOGGER.trace("DELETE: {}", webTarget); - webTarget.request().delete().close(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/RemoveNetworkCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/RemoveNetworkCmdExec.java deleted file mode 100644 index acc2bdd27..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/RemoveNetworkCmdExec.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import com.github.dockerjava.api.command.RemoveNetworkCmd; -import com.github.dockerjava.core.DockerClientConfig; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -public class RemoveNetworkCmdExec extends AbstrSyncDockerCmdExec - implements RemoveNetworkCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(RemoveNetworkCmdExec.class); - - public RemoveNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RemoveNetworkCmd command) { - - WebTarget webTarget = getBaseResource().path("/networks/" + command.getNetworkId()); - - LOGGER.trace("DELETE: {}", webTarget); - webTarget.request().accept(MediaType.APPLICATION_JSON).delete().close(); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/RemoveVolumeCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/RemoveVolumeCmdExec.java deleted file mode 100644 index 618a85ed8..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/RemoveVolumeCmdExec.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.RemoveVolumeCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class RemoveVolumeCmdExec extends AbstrSyncDockerCmdExec implements - RemoveVolumeCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(RemoveVolumeCmdExec.class); - - public RemoveVolumeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RemoveVolumeCmd command) { - WebTarget webTarget = getBaseResource().path("/volumes/" + command.getName()); - - LOGGER.trace("DELETE: {}", webTarget); - webTarget.request().accept(MediaType.APPLICATION_JSON).delete().close(); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/RenameContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/RenameContainerCmdExec.java deleted file mode 100644 index c747ea9b0..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/RenameContainerCmdExec.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import com.github.dockerjava.api.command.RenameContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public class RenameContainerCmdExec extends AbstrSyncDockerCmdExec - implements RenameContainerCmd.Exec { - private static final Logger LOG = LoggerFactory.getLogger(RenameContainerCmdExec.class); - - public RenameContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RenameContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/rename") - .resolveTemplate("id", command.getContainerId()) - .queryParam("name", command.getName()); - - LOG.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/RestartContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/RestartContainerCmdExec.java deleted file mode 100644 index 765537f44..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/RestartContainerCmdExec.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.RestartContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class RestartContainerCmdExec extends AbstrSyncDockerCmdExec implements - RestartContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(RestartContainerCmdExec.class); - - public RestartContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RestartContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/restart").resolveTemplate("id", - command.getContainerId()); - - if (command.getTimeout() != null) { - webResource = webResource.queryParam("t", String.valueOf(command.getTimeout())); - } - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/SaveImageCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/SaveImageCmdExec.java deleted file mode 100644 index f3f9a0d35..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/SaveImageCmdExec.java +++ /dev/null @@ -1,33 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import java.io.InputStream; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; -import javax.ws.rs.core.Response; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.SaveImageCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.jaxrs.util.WrappedResponseInputStream; - -public class SaveImageCmdExec extends AbstrSyncDockerCmdExec implements SaveImageCmd.Exec { - private static final Logger LOGGER = LoggerFactory.getLogger(SaveImageCmdExec.class); - - public SaveImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InputStream execute(SaveImageCmd command) { - WebTarget webResource = getBaseResource().path("/images/" + command.getName() + "/get").queryParam("tag", - command.getTag()); - - LOGGER.trace("GET: {}", webResource); - Response response = webResource.request().accept(MediaType.APPLICATION_JSON).get(); - - return new WrappedResponseInputStream(response); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/SearchImagesCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/SearchImagesCmdExec.java deleted file mode 100644 index 90fdd80cd..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/SearchImagesCmdExec.java +++ /dev/null @@ -1,34 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import java.util.List; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.GenericType; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.SearchImagesCmd; -import com.github.dockerjava.api.model.SearchItem; -import com.github.dockerjava.core.DockerClientConfig; - -public class SearchImagesCmdExec extends AbstrSyncDockerCmdExec> implements - SearchImagesCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(SearchImagesCmdExec.class); - - public SearchImagesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected List execute(SearchImagesCmd command) { - WebTarget webResource = getBaseResource().path("/images/search").queryParam("term", command.getTerm()); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(new GenericType>() { - }); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/StartContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/StartContainerCmdExec.java deleted file mode 100644 index 0f75c4f2f..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/StartContainerCmdExec.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.StartContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class StartContainerCmdExec extends AbstrSyncDockerCmdExec implements - StartContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(StartContainerCmdExec.class); - - public StartContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(StartContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/start").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(entity(command, MediaType.APPLICATION_JSON)) - .close(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/StatsCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/StatsCmdExec.java deleted file mode 100644 index 8a604603a..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/StatsCmdExec.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.StatsCmd; -import com.github.dockerjava.api.model.Statistics; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.async.JsonStreamProcessor; -import com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier; -import com.github.dockerjava.jaxrs.async.GETCallbackNotifier; - -public class StatsCmdExec extends AbstrAsyncDockerCmdExec implements StatsCmd.Exec { - private static final Logger LOGGER = LoggerFactory.getLogger(StatsCmdExec.class); - - public StatsCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected AbstractCallbackNotifier callbackNotifier(StatsCmd command, - ResultCallback resultCallback) { - - WebTarget webTarget = getBaseResource().path("/containers/{id}/stats").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("GET: {}", webTarget); - - return new GETCallbackNotifier(new JsonStreamProcessor(Statistics.class), - resultCallback, webTarget.request()); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/StopContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/StopContainerCmdExec.java deleted file mode 100644 index 930ccbb7d..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/StopContainerCmdExec.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.StopContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class StopContainerCmdExec extends AbstrSyncDockerCmdExec implements - StopContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(StopContainerCmdExec.class); - - public StopContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(StopContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/stop").resolveTemplate("id", - command.getContainerId()); - - if (command.getTimeout() != null) { - webResource = webResource.queryParam("t", String.valueOf(command.getTimeout())); - } - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/TagImageCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/TagImageCmdExec.java deleted file mode 100644 index aac15f9dd..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/TagImageCmdExec.java +++ /dev/null @@ -1,31 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.TagImageCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class TagImageCmdExec extends AbstrSyncDockerCmdExec implements TagImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(TagImageCmdExec.class); - - public TagImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(TagImageCmd command) { - WebTarget webTarget = getBaseResource().path("/images/" + command.getImageId() + "/tag") - .queryParam("repo", command.getRepository()).queryParam("tag", command.getTag()); - - webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); - - LOGGER.trace("POST: {}", webTarget); - webTarget.request().post(null).close(); - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/TopContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/TopContainerCmdExec.java deleted file mode 100644 index 22841d8bf..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/TopContainerCmdExec.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.apache.commons.lang.StringUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.TopContainerCmd; -import com.github.dockerjava.api.command.TopContainerResponse; -import com.github.dockerjava.core.DockerClientConfig; - -public class TopContainerCmdExec extends AbstrSyncDockerCmdExec implements - TopContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(TopContainerCmdExec.class); - - public TopContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected TopContainerResponse execute(TopContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/top").resolveTemplate("id", - command.getContainerId()); - - if (!StringUtils.isEmpty(command.getPsArgs())) { - webResource = webResource.queryParam("ps_args", command.getPsArgs()); - } - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(TopContainerResponse.class); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/UnpauseContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/UnpauseContainerCmdExec.java deleted file mode 100644 index 5c3828358..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/UnpauseContainerCmdExec.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.UnpauseContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; - -public class UnpauseContainerCmdExec extends AbstrSyncDockerCmdExec implements - UnpauseContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(UnpauseContainerCmdExec.class); - - public UnpauseContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(UnpauseContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/unpause").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null).close(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/UpdateContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/UpdateContainerCmdExec.java deleted file mode 100644 index 474d7338e..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/UpdateContainerCmdExec.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import com.github.dockerjava.api.command.UpdateContainerCmd; -import com.github.dockerjava.api.model.UpdateContainerResponse; -import com.github.dockerjava.core.DockerClientConfig; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import static javax.ws.rs.client.Entity.entity; - -/** - * Update container settings. - * - * @author Kanstantsin Shautsou - */ -public class UpdateContainerCmdExec extends AbstrSyncDockerCmdExec - implements UpdateContainerCmd.Exec { - private static final Logger LOGGER = LoggerFactory.getLogger(UpdateContainerCmdExec.class); - - public UpdateContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected UpdateContainerResponse execute(UpdateContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/update") - .resolveTemplate("id", command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(entity(command, MediaType.APPLICATION_JSON), UpdateContainerResponse.class); - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/VersionCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/VersionCmdExec.java deleted file mode 100644 index 7fbc92784..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/VersionCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.VersionCmd; -import com.github.dockerjava.api.model.Version; -import com.github.dockerjava.core.DockerClientConfig; - -public class VersionCmdExec extends AbstrSyncDockerCmdExec implements VersionCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(VersionCmdExec.class); - - public VersionCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Version execute(VersionCmd command) { - WebTarget webResource = getBaseResource().path("/version"); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(Version.class); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/WaitContainerCmdExec.java b/src/main/java/com/github/dockerjava/jaxrs/WaitContainerCmdExec.java deleted file mode 100644 index 06bd40edc..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/WaitContainerCmdExec.java +++ /dev/null @@ -1,42 +0,0 @@ -package com.github.dockerjava.jaxrs; - -import static javax.ws.rs.client.Entity.entity; - -import javax.ws.rs.client.WebTarget; -import javax.ws.rs.core.MediaType; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.WaitContainerCmd; -import com.github.dockerjava.api.model.WaitResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.async.JsonStreamProcessor; -import com.github.dockerjava.jaxrs.async.AbstractCallbackNotifier; -import com.github.dockerjava.jaxrs.async.POSTCallbackNotifier; - -public class WaitContainerCmdExec extends AbstrAsyncDockerCmdExec implements - WaitContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(WaitContainerCmdExec.class); - - public WaitContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected AbstractCallbackNotifier callbackNotifier(WaitContainerCmd command, - ResultCallback resultCallback) { - - WebTarget webResource = getBaseResource().path("/containers/{id}/wait").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - - return new POSTCallbackNotifier(new JsonStreamProcessor(WaitResponse.class), - resultCallback, webResource.request().accept(MediaType.APPLICATION_JSON), entity(null, - MediaType.APPLICATION_JSON)); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/async/AbstractCallbackNotifier.java b/src/main/java/com/github/dockerjava/jaxrs/async/AbstractCallbackNotifier.java deleted file mode 100644 index cd7a7f809..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/async/AbstractCallbackNotifier.java +++ /dev/null @@ -1,88 +0,0 @@ -/* - * Created on 17.06.2015 - */ -package com.github.dockerjava.jaxrs.async; - -import static com.google.common.base.Preconditions.checkNotNull; - -import java.io.InputStream; -import java.util.concurrent.Callable; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.Future; -import java.util.concurrent.ThreadFactory; - -import javax.ws.rs.ProcessingException; -import javax.ws.rs.client.Invocation.Builder; -import javax.ws.rs.core.Response; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.core.async.ResponseStreamProcessor; -import com.github.dockerjava.jaxrs.util.WrappedResponseInputStream; -import com.google.common.util.concurrent.ThreadFactoryBuilder; - -public abstract class AbstractCallbackNotifier implements Callable { - - private final ResponseStreamProcessor responseStreamProcessor; - - private final ResultCallback resultCallback; - - private static final ThreadFactory FACTORY = - new ThreadFactoryBuilder().setDaemon(true).setNameFormat("dockerjava-jaxrs-async-%d").build(); - - protected final Builder requestBuilder; - - protected AbstractCallbackNotifier(ResponseStreamProcessor responseStreamProcessor, - ResultCallback resultCallback, Builder requestBuilder) { - checkNotNull(requestBuilder, "An WebTarget must be provided"); - checkNotNull(responseStreamProcessor, "A ResponseStreamProcessor must be provided"); - this.responseStreamProcessor = responseStreamProcessor; - this.resultCallback = resultCallback; - this.requestBuilder = requestBuilder; - } - - @Override - public Void call() throws Exception { - - Response response = null; - - try { - response = response(); - } catch (ProcessingException e) { - if (resultCallback != null) { - resultCallback.onError(e.getCause()); - } - return null; - } catch (Exception e) { - if (resultCallback != null) { - resultCallback.onError(e); - } - return null; - } - - try (InputStream inputStream = new WrappedResponseInputStream(response)) { - - if (resultCallback != null) { - responseStreamProcessor.processResponseStream(inputStream, resultCallback); - } - - return null; - } catch (Exception e) { - if (resultCallback != null) { - resultCallback.onError(e); - } - - return null; - } - } - - protected abstract Response response(); - - public static Future startAsyncProcessing(AbstractCallbackNotifier callbackNotifier) { - - ExecutorService executorService = Executors.newSingleThreadExecutor(FACTORY); - Future response = executorService.submit(callbackNotifier); - executorService.shutdown(); - return response; - } -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/async/GETCallbackNotifier.java b/src/main/java/com/github/dockerjava/jaxrs/async/GETCallbackNotifier.java deleted file mode 100644 index e6f4fd1d2..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/async/GETCallbackNotifier.java +++ /dev/null @@ -1,28 +0,0 @@ -/* - * Created on 23.06.2015 - */ -package com.github.dockerjava.jaxrs.async; - -import javax.ws.rs.client.Invocation.Builder; -import javax.ws.rs.core.Response; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.core.async.ResponseStreamProcessor; - -/** - * - * @author Marcus Linke - * - */ -public class GETCallbackNotifier extends AbstractCallbackNotifier { - - public GETCallbackNotifier(ResponseStreamProcessor responseStreamProcessor, ResultCallback resultCallback, - Builder requestBuilder) { - super(responseStreamProcessor, resultCallback, requestBuilder); - } - - protected Response response() { - return requestBuilder.get(Response.class); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/async/POSTCallbackNotifier.java b/src/main/java/com/github/dockerjava/jaxrs/async/POSTCallbackNotifier.java deleted file mode 100644 index bd96e8250..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/async/POSTCallbackNotifier.java +++ /dev/null @@ -1,32 +0,0 @@ -/* - * Created on 23.06.2015 - */ -package com.github.dockerjava.jaxrs.async; - -import javax.ws.rs.client.Entity; -import javax.ws.rs.client.Invocation.Builder; -import javax.ws.rs.core.Response; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.core.async.ResponseStreamProcessor; - -/** - * - * @author Marcus Linke - * - */ -public class POSTCallbackNotifier extends AbstractCallbackNotifier { - - Entity entity = null; - - public POSTCallbackNotifier(ResponseStreamProcessor responseStreamProcessor, ResultCallback resultCallback, - Builder requestBuilder, Entity entity) { - super(responseStreamProcessor, resultCallback, requestBuilder); - this.entity = entity; - } - - protected Response response() { - return requestBuilder.post(entity, Response.class); - } - -} diff --git a/src/main/java/com/github/dockerjava/jaxrs/filter/ResponseStatusExceptionFilter.java b/src/main/java/com/github/dockerjava/jaxrs/filter/ResponseStatusExceptionFilter.java deleted file mode 100644 index a88dccb0d..000000000 --- a/src/main/java/com/github/dockerjava/jaxrs/filter/ResponseStatusExceptionFilter.java +++ /dev/null @@ -1,91 +0,0 @@ -package com.github.dockerjava.jaxrs.filter; - -import java.io.EOFException; -import java.io.IOException; -import java.io.InputStream; -import java.nio.charset.Charset; - -import javax.ws.rs.client.ClientRequestContext; -import javax.ws.rs.client.ClientResponseContext; -import javax.ws.rs.client.ClientResponseFilter; -import javax.ws.rs.core.MediaType; - -import org.apache.commons.io.IOUtils; - -import com.github.dockerjava.api.exception.BadRequestException; -import com.github.dockerjava.api.exception.ConflictException; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.InternalServerErrorException; -import com.github.dockerjava.api.exception.NotAcceptableException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.exception.NotModifiedException; -import com.github.dockerjava.api.exception.UnauthorizedException; - -/** - * This {@link ClientResponseFilter} implementation detects http status codes and throws {@link DockerException}s - * - * @author Marcus Linke - * - */ -public class ResponseStatusExceptionFilter implements ClientResponseFilter { - - @Override - public void filter(ClientRequestContext requestContext, ClientResponseContext responseContext) throws IOException { - int status = responseContext.getStatus(); - switch (status) { - case 200: - case 201: - case 204: - return; - case 304: - throw new NotModifiedException(getBodyAsMessage(responseContext)); - case 400: - throw new BadRequestException(getBodyAsMessage(responseContext)); - case 401: - throw new UnauthorizedException(getBodyAsMessage(responseContext)); - case 404: - throw new NotFoundException(getBodyAsMessage(responseContext)); - case 406: - throw new NotAcceptableException(getBodyAsMessage(responseContext)); - case 409: - throw new ConflictException(getBodyAsMessage(responseContext)); - case 500: - throw new InternalServerErrorException(getBodyAsMessage(responseContext)); - default: - throw new DockerException(getBodyAsMessage(responseContext), status); - } - } - - private String getBodyAsMessage(ClientResponseContext responseContext) throws IOException { - if (responseContext.hasEntity()) { - int contentLength = responseContext.getLength(); - if (contentLength != -1) { - byte[] buffer = new byte[contentLength]; - try { - InputStream entityStream = responseContext.getEntityStream(); - IOUtils.readFully(entityStream, buffer); - entityStream.close(); - } catch (EOFException e) { - return null; - } - Charset charset = null; - MediaType mediaType = responseContext.getMediaType(); - if (mediaType != null) { - String charsetName = mediaType.getParameters().get("charset"); - if (charsetName != null) { - try { - charset = Charset.forName(charsetName); - } catch (Exception e) { - // Do noting... - } - } - } - if (charset == null) { - charset = Charset.defaultCharset(); - } - return new String(buffer, charset); - } - } - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/DockerCmdExecFactoryImpl.java b/src/main/java/com/github/dockerjava/netty/DockerCmdExecFactoryImpl.java deleted file mode 100644 index 25825d169..000000000 --- a/src/main/java/com/github/dockerjava/netty/DockerCmdExecFactoryImpl.java +++ /dev/null @@ -1,9 +0,0 @@ -package com.github.dockerjava.netty; - -/** - * @author Kanstantsin Shautsou - * @deprecated old clashing name - */ -@Deprecated -public class DockerCmdExecFactoryImpl extends NettyDockerCmdExecFactory { -} diff --git a/src/main/java/com/github/dockerjava/netty/InvocationBuilder.java b/src/main/java/com/github/dockerjava/netty/InvocationBuilder.java deleted file mode 100644 index 9669d6818..000000000 --- a/src/main/java/com/github/dockerjava/netty/InvocationBuilder.java +++ /dev/null @@ -1,469 +0,0 @@ -package com.github.dockerjava.netty; - -import io.netty.buffer.Unpooled; -import io.netty.channel.Channel; -import io.netty.channel.ChannelFuture; -import io.netty.channel.ChannelFutureListener; -import io.netty.channel.socket.DuplexChannel; -import io.netty.handler.codec.http.DefaultFullHttpRequest; -import io.netty.handler.codec.http.DefaultHttpRequest; -import io.netty.handler.codec.http.FullHttpRequest; -import io.netty.handler.codec.http.HttpClientCodec; -import io.netty.handler.codec.http.HttpClientUpgradeHandler; -import io.netty.handler.codec.http.HttpHeaderNames; -import io.netty.handler.codec.http.HttpHeaderValues; -import io.netty.handler.codec.http.HttpMethod; -import io.netty.handler.codec.http.HttpRequest; -import io.netty.handler.codec.http.HttpVersion; -import io.netty.handler.codec.http.LastHttpContent; -import io.netty.handler.codec.json.JsonObjectDecoder; -import io.netty.handler.stream.ChunkedStream; -import io.netty.handler.stream.ChunkedWriteHandler; -import io.netty.util.concurrent.Future; -import io.netty.util.concurrent.GenericFutureListener; - -import java.io.BufferedInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.util.HashMap; -import java.util.Map; - -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.core.type.TypeReference; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.core.async.ResultCallbackTemplate; -import com.github.dockerjava.netty.handler.FramedResponseStreamHandler; -import com.github.dockerjava.netty.handler.HttpConnectionHijackHandler; -import com.github.dockerjava.netty.handler.HttpRequestProvider; -import com.github.dockerjava.netty.handler.HttpResponseHandler; -import com.github.dockerjava.netty.handler.HttpResponseStreamHandler; -import com.github.dockerjava.netty.handler.JsonResponseCallbackHandler; - -/** - * This class is basically a replacement of javax.ws.rs.client.Invocation.Builder to allow simpler migration of JAX-RS code to a netty based - * implementation. - * - * @author Marcus Linke - */ -public class InvocationBuilder { - - public class ResponseCallback extends ResultCallbackTemplate, T> { - - private T result = null; - - public T awaitResult() { - try { - awaitCompletion(); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - return result; - } - - @Override - public void onNext(T object) { - result = object; - } - } - - private ChannelProvider channelProvider; - - private String resource; - - private Map headers = new HashMap(); - - public InvocationBuilder(ChannelProvider channelProvider, String resource) { - this.channelProvider = channelProvider; - this.resource = resource; - } - - public InvocationBuilder accept(MediaType mediaType) { - return header(HttpHeaderNames.ACCEPT.toString(), mediaType.getMediaType()); - } - - public InvocationBuilder header(String name, String value) { - headers.put(name, value); - return this; - } - - public void delete() { - - HttpRequestProvider requestProvider = httpDeleteRequestProvider(); - - ResponseCallback callback = new ResponseCallback(); - - HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, callback); - - Channel channel = getChannel(); - - channel.pipeline().addLast(responseHandler); - - sendRequest(requestProvider, channel); - - callback.awaitResult(); - } - - public void get(ResultCallback resultCallback) { - - HttpRequestProvider requestProvider = httpGetRequestProvider(); - - HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); - - FramedResponseStreamHandler streamHandler = new FramedResponseStreamHandler(resultCallback); - - Channel channel = getChannel(); - - channel.pipeline().addLast(responseHandler); - channel.pipeline().addLast(streamHandler); - - sendRequest(requestProvider, channel); - } - - public T get(TypeReference typeReference) { - - ResponseCallback callback = new ResponseCallback(); - - get(typeReference, callback); - - return callback.awaitResult(); - } - - public void get(TypeReference typeReference, ResultCallback resultCallback) { - - HttpRequestProvider requestProvider = httpGetRequestProvider(); - - Channel channel = getChannel(); - - JsonResponseCallbackHandler jsonResponseHandler = new JsonResponseCallbackHandler(typeReference, - resultCallback); - - HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); - - channel.pipeline().addLast(responseHandler); - channel.pipeline().addLast(new JsonObjectDecoder()); - channel.pipeline().addLast(jsonResponseHandler); - - sendRequest(requestProvider, channel); - - return; - } - - private DuplexChannel getChannel() { - return channelProvider.getChannel(); - } - - private HttpRequestProvider httpDeleteRequestProvider() { - return new HttpRequestProvider() { - @Override - public HttpRequest getHttpRequest(String uri) { - return prepareDeleteRequest(uri); - } - }; - } - - private HttpRequestProvider httpGetRequestProvider() { - return new HttpRequestProvider() { - @Override - public HttpRequest getHttpRequest(String uri) { - return prepareGetRequest(uri); - } - }; - } - - private HttpRequestProvider httpPostRequestProvider(final Object entity) { - return new HttpRequestProvider() { - @Override - public HttpRequest getHttpRequest(String uri) { - return preparePostRequest(uri, entity); - } - }; - } - - private HttpRequestProvider httpPutRequestProvider(final Object entity) { - return new HttpRequestProvider() { - @Override - public HttpRequest getHttpRequest(String uri) { - return preparePutRequest(uri, entity); - } - }; - } - - public InputStream post(final Object entity) { - - HttpRequestProvider requestProvider = httpPostRequestProvider(entity); - - Channel channel = getChannel(); - - ResponseCallback callback = new ResponseCallback(); - - HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, callback); - HttpResponseStreamHandler streamHandler = new HttpResponseStreamHandler(callback); - - channel.pipeline().addLast(responseHandler); - channel.pipeline().addLast(streamHandler); - - sendRequest(requestProvider, channel); - - return callback.awaitResult(); - } - - public void post(final Object entity, final InputStream stdin, final ResultCallback resultCallback) { - - HttpRequestProvider requestProvider = httpPostRequestProvider(entity); - - FramedResponseStreamHandler streamHandler = new FramedResponseStreamHandler(resultCallback); - - final DuplexChannel channel = getChannel(); - - // result callback's close() method must be called when the servers closes the connection - channel.closeFuture().addListener(new GenericFutureListener>() { - @Override - public void operationComplete(Future future) throws Exception { - resultCallback.onComplete(); - } - }); - - HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); - - HttpConnectionHijackHandler hijackHandler = new HttpConnectionHijackHandler(responseHandler); - - HttpClientCodec httpClientCodec = channel.pipeline().get(HttpClientCodec.class); - - channel.pipeline().addLast( - new HttpClientUpgradeHandler(httpClientCodec, hijackHandler, Integer.MAX_VALUE)); - channel.pipeline().addLast(streamHandler); - - sendRequest(requestProvider, channel); - - // wait for successful http upgrade procedure - hijackHandler.awaitUpgrade(); - - if (stdin != null) { - // now we can start a new thread that reads from stdin and writes to the channel - new Thread(new Runnable() { - - private int read(InputStream is, byte[] buf) { - try { - return is.read(buf); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - @Override - public void run() { - - byte[] buffer = new byte[1024]; - - int read; - while ((read = read(stdin, buffer)) != -1) { - channel.writeAndFlush(Unpooled.copiedBuffer(buffer, 0, read)); - } - - // we close the writing side of the socket, but keep the read side open to transfer stdout/stderr - channel.shutdownOutput(); - - } - }).start(); - } - } - - public T post(final Object entity, TypeReference typeReference) { - - ResponseCallback callback = new ResponseCallback(); - - post(entity, typeReference, callback); - - return callback.awaitResult(); - } - - public void post(final Object entity, TypeReference typeReference, final ResultCallback resultCallback) { - - HttpRequestProvider requestProvider = httpPostRequestProvider(entity); - - Channel channel = getChannel(); - - JsonResponseCallbackHandler jsonResponseHandler = new JsonResponseCallbackHandler(typeReference, - resultCallback); - - HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); - - channel.pipeline().addLast(responseHandler); - channel.pipeline().addLast(new JsonObjectDecoder()); - channel.pipeline().addLast(jsonResponseHandler); - - sendRequest(requestProvider, channel); - - return; - } - - private HttpRequest prepareDeleteRequest(String uri) { - - FullHttpRequest request = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, HttpMethod.DELETE, uri); - - setDefaultHeaders(request); - - return request; - } - - private FullHttpRequest prepareGetRequest(String uri) { - - FullHttpRequest request = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, HttpMethod.GET, uri); - - setDefaultHeaders(request); - - return request; - } - - private HttpRequest preparePostRequest(String uri, Object entity) { - return prepareEntityRequest(uri, entity, HttpMethod.POST); - } - - private HttpRequest preparePutRequest(String uri, Object entity) { - return prepareEntityRequest(uri, entity, HttpMethod.PUT); - } - - private HttpRequest prepareEntityRequest(String uri, Object entity, HttpMethod httpMethod) { - - HttpRequest request = null; - - if (entity != null) { - - FullHttpRequest fullRequest = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, httpMethod, uri); - - byte[] bytes; - try { - bytes = new ObjectMapper().writeValueAsBytes(entity); - } catch (JsonProcessingException e) { - throw new RuntimeException(e); - } - - fullRequest.headers().set(HttpHeaderNames.CONTENT_TYPE, "application/json"); - fullRequest.content().clear().writeBytes(Unpooled.copiedBuffer(bytes)); - fullRequest.headers().set(HttpHeaderNames.CONTENT_LENGTH, bytes.length); - - request = fullRequest; - } else { - request = new DefaultHttpRequest(HttpVersion.HTTP_1_1, httpMethod, uri); - request.headers().set(HttpHeaderNames.CONTENT_LENGTH, 0); - } - - setDefaultHeaders(request); - - return request; - } - - private void sendRequest(HttpRequestProvider requestProvider, Channel channel) { - - ChannelFuture channelFuture = channel.writeAndFlush(requestProvider.getHttpRequest(resource)); - - channelFuture.addListener(new ChannelFutureListener() { - @Override - public void operationComplete(ChannelFuture future) throws Exception { - } - }); - } - - private void setDefaultHeaders(HttpRequest request) { - request.headers().set(HttpHeaderNames.HOST, ""); - request.headers().set(HttpHeaderNames.CONNECTION, HttpHeaderValues.KEEP_ALIVE); - request.headers().set(HttpHeaderNames.ACCEPT_ENCODING, HttpHeaderValues.GZIP); - - for (Map.Entry entry : headers.entrySet()) { - request.headers().set((CharSequence) entry.getKey(), entry.getValue()); - } - } - - public T post(TypeReference typeReference, InputStream body) { - - ResponseCallback callback = new ResponseCallback(); - - post(typeReference, callback, body); - - return callback.awaitResult(); - } - - public void post(TypeReference typeReference, ResultCallback resultCallback, InputStream body) { - HttpRequestProvider requestProvider = httpPostRequestProvider(null); - - Channel channel = getChannel(); - - JsonResponseCallbackHandler jsonResponseHandler = new JsonResponseCallbackHandler(typeReference, - resultCallback); - - HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); - - channel.pipeline().addLast(new ChunkedWriteHandler()); - channel.pipeline().addLast(responseHandler); - channel.pipeline().addLast(new JsonObjectDecoder()); - channel.pipeline().addLast(jsonResponseHandler); - - HttpRequest request = requestProvider.getHttpRequest(resource); - - // don't accept FullHttpRequest here - if (request instanceof FullHttpRequest) { - throw new DockerClientException("fatal: request is instance of FullHttpRequest"); - } - - request.headers().set(HttpHeaderNames.TRANSFER_ENCODING, HttpHeaderValues.CHUNKED); - request.headers().remove(HttpHeaderNames.CONTENT_LENGTH); - - channel.write(request); - - channel.write(new ChunkedStream(new BufferedInputStream(body, 1024 * 1024), 1024 * 1024)); - channel.write(LastHttpContent.EMPTY_LAST_CONTENT); - channel.flush(); - } - - public InputStream get() { - HttpRequestProvider requestProvider = httpGetRequestProvider(); - - Channel channel = getChannel(); - - ResponseCallback resultCallback = new ResponseCallback(); - - HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); - - HttpResponseStreamHandler streamHandler = new HttpResponseStreamHandler(resultCallback); - - channel.pipeline().addLast(responseHandler); - channel.pipeline().addLast(streamHandler); - - sendRequest(requestProvider, channel); - - return resultCallback.awaitResult(); - } - - public void put(InputStream body, MediaType mediaType) { - HttpRequestProvider requestProvider = httpPutRequestProvider(null); - - Channel channel = getChannel(); - - ResponseCallback resultCallback = new ResponseCallback(); - - HttpResponseHandler responseHandler = new HttpResponseHandler(requestProvider, resultCallback); - - channel.pipeline().addLast(new ChunkedWriteHandler()); - channel.pipeline().addLast(responseHandler); - - HttpRequest request = requestProvider.getHttpRequest(resource); - - // don't accept FullHttpRequest here - if (request instanceof FullHttpRequest) { - throw new DockerClientException("fatal: request is instance of FullHttpRequest"); - } - - request.headers().set(HttpHeaderNames.TRANSFER_ENCODING, HttpHeaderValues.CHUNKED); - request.headers().remove(HttpHeaderNames.CONTENT_LENGTH); - request.headers().set(HttpHeaderNames.CONTENT_TYPE, mediaType.getMediaType()); - - channel.write(request); - channel.write(new ChunkedStream(new BufferedInputStream(body, 1024 * 1024))); - channel.writeAndFlush(LastHttpContent.EMPTY_LAST_CONTENT); - - resultCallback.awaitResult(); - }; -} diff --git a/src/main/java/com/github/dockerjava/netty/MediaType.java b/src/main/java/com/github/dockerjava/netty/MediaType.java deleted file mode 100644 index f65ca913f..000000000 --- a/src/main/java/com/github/dockerjava/netty/MediaType.java +++ /dev/null @@ -1,24 +0,0 @@ -package com.github.dockerjava.netty; - -/** - * This class is basically a replacement of javax.ws.rs.core.MediaType to allow simpler migration of JAX-RS code to a netty based - * implementation. - * - * @author Marcus Linke - */ -public enum MediaType { - - APPLICATION_JSON("application/json"), - APPLICATION_OCTET_STREAM("application/octet-stream"), - APPLICATION_X_TAR("application/x-tar"); - - private String mediaType; - - MediaType(String mediaType) { - this.mediaType = mediaType; - } - - public String getMediaType() { - return mediaType; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/NettyDockerCmdExecFactory.java b/src/main/java/com/github/dockerjava/netty/NettyDockerCmdExecFactory.java deleted file mode 100644 index cead92e92..000000000 --- a/src/main/java/com/github/dockerjava/netty/NettyDockerCmdExecFactory.java +++ /dev/null @@ -1,586 +0,0 @@ -package com.github.dockerjava.netty; - -import com.github.dockerjava.api.command.AttachContainerCmd; -import com.github.dockerjava.api.command.AuthCmd; -import com.github.dockerjava.api.command.BuildImageCmd; -import com.github.dockerjava.api.command.CommitCmd; -import com.github.dockerjava.api.command.ConnectToNetworkCmd; -import com.github.dockerjava.api.command.ContainerDiffCmd; -import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; -import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; -import com.github.dockerjava.api.command.CopyFileFromContainerCmd; -import com.github.dockerjava.api.command.CreateContainerCmd; -import com.github.dockerjava.api.command.CreateImageCmd; -import com.github.dockerjava.api.command.CreateNetworkCmd; -import com.github.dockerjava.api.command.CreateVolumeCmd; -import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; -import com.github.dockerjava.api.command.DockerCmdExecFactory; -import com.github.dockerjava.api.command.EventsCmd; -import com.github.dockerjava.api.command.ExecCreateCmd; -import com.github.dockerjava.api.command.ExecStartCmd; -import com.github.dockerjava.api.command.InfoCmd; -import com.github.dockerjava.api.command.InspectContainerCmd; -import com.github.dockerjava.api.command.InspectExecCmd; -import com.github.dockerjava.api.command.InspectImageCmd; -import com.github.dockerjava.api.command.InspectNetworkCmd; -import com.github.dockerjava.api.command.InspectVolumeCmd; -import com.github.dockerjava.api.command.KillContainerCmd; -import com.github.dockerjava.api.command.ListContainersCmd; -import com.github.dockerjava.api.command.ListImagesCmd; -import com.github.dockerjava.api.command.ListNetworksCmd; -import com.github.dockerjava.api.command.ListVolumesCmd; -import com.github.dockerjava.api.command.LoadImageCmd; -import com.github.dockerjava.api.command.LogContainerCmd; -import com.github.dockerjava.api.command.PauseContainerCmd; -import com.github.dockerjava.api.command.PingCmd; -import com.github.dockerjava.api.command.PullImageCmd; -import com.github.dockerjava.api.command.PushImageCmd; -import com.github.dockerjava.api.command.RemoveContainerCmd; -import com.github.dockerjava.api.command.RemoveImageCmd; -import com.github.dockerjava.api.command.RemoveNetworkCmd; -import com.github.dockerjava.api.command.RemoveVolumeCmd; -import com.github.dockerjava.api.command.RestartContainerCmd; -import com.github.dockerjava.api.command.SaveImageCmd; -import com.github.dockerjava.api.command.SearchImagesCmd; -import com.github.dockerjava.api.command.StartContainerCmd; -import com.github.dockerjava.api.command.StatsCmd; -import com.github.dockerjava.api.command.StopContainerCmd; -import com.github.dockerjava.api.command.TagImageCmd; -import com.github.dockerjava.api.command.TopContainerCmd; -import com.github.dockerjava.api.command.UnpauseContainerCmd; -import com.github.dockerjava.api.command.UpdateContainerCmd; -import com.github.dockerjava.api.command.VersionCmd; -import com.github.dockerjava.api.command.WaitContainerCmd; -import com.github.dockerjava.api.command.RenameContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.DockerClientImpl; -import com.github.dockerjava.core.SSLConfig; -import com.github.dockerjava.netty.exec.AttachContainerCmdExec; -import com.github.dockerjava.netty.exec.AuthCmdExec; -import com.github.dockerjava.netty.exec.BuildImageCmdExec; -import com.github.dockerjava.netty.exec.CommitCmdExec; -import com.github.dockerjava.netty.exec.ConnectToNetworkCmdExec; -import com.github.dockerjava.netty.exec.ContainerDiffCmdExec; -import com.github.dockerjava.netty.exec.CopyArchiveFromContainerCmdExec; -import com.github.dockerjava.netty.exec.CopyArchiveToContainerCmdExec; -import com.github.dockerjava.netty.exec.CopyFileFromContainerCmdExec; -import com.github.dockerjava.netty.exec.CreateContainerCmdExec; -import com.github.dockerjava.netty.exec.CreateImageCmdExec; -import com.github.dockerjava.netty.exec.CreateNetworkCmdExec; -import com.github.dockerjava.netty.exec.CreateVolumeCmdExec; -import com.github.dockerjava.netty.exec.DisconnectFromNetworkCmdExec; -import com.github.dockerjava.netty.exec.EventsCmdExec; -import com.github.dockerjava.netty.exec.ExecCreateCmdExec; -import com.github.dockerjava.netty.exec.ExecStartCmdExec; -import com.github.dockerjava.netty.exec.InfoCmdExec; -import com.github.dockerjava.netty.exec.InspectContainerCmdExec; -import com.github.dockerjava.netty.exec.InspectExecCmdExec; -import com.github.dockerjava.netty.exec.InspectImageCmdExec; -import com.github.dockerjava.netty.exec.InspectNetworkCmdExec; -import com.github.dockerjava.netty.exec.InspectVolumeCmdExec; -import com.github.dockerjava.netty.exec.KillContainerCmdExec; -import com.github.dockerjava.netty.exec.ListContainersCmdExec; -import com.github.dockerjava.netty.exec.ListImagesCmdExec; -import com.github.dockerjava.netty.exec.ListNetworksCmdExec; -import com.github.dockerjava.netty.exec.ListVolumesCmdExec; -import com.github.dockerjava.netty.exec.LoadImageCmdExec; -import com.github.dockerjava.netty.exec.LogContainerCmdExec; -import com.github.dockerjava.netty.exec.PauseContainerCmdExec; -import com.github.dockerjava.netty.exec.PingCmdExec; -import com.github.dockerjava.netty.exec.PullImageCmdExec; -import com.github.dockerjava.netty.exec.PushImageCmdExec; -import com.github.dockerjava.netty.exec.RemoveContainerCmdExec; -import com.github.dockerjava.netty.exec.RemoveImageCmdExec; -import com.github.dockerjava.netty.exec.RemoveNetworkCmdExec; -import com.github.dockerjava.netty.exec.RemoveVolumeCmdExec; -import com.github.dockerjava.netty.exec.RestartContainerCmdExec; -import com.github.dockerjava.netty.exec.SaveImageCmdExec; -import com.github.dockerjava.netty.exec.SearchImagesCmdExec; -import com.github.dockerjava.netty.exec.StartContainerCmdExec; -import com.github.dockerjava.netty.exec.StatsCmdExec; -import com.github.dockerjava.netty.exec.StopContainerCmdExec; -import com.github.dockerjava.netty.exec.TagImageCmdExec; -import com.github.dockerjava.netty.exec.TopContainerCmdExec; -import com.github.dockerjava.netty.exec.UnpauseContainerCmdExec; -import com.github.dockerjava.netty.exec.UpdateContainerCmdExec; -import com.github.dockerjava.netty.exec.VersionCmdExec; -import com.github.dockerjava.netty.exec.WaitContainerCmdExec; -import com.github.dockerjava.netty.exec.RenameContainerCmdExec; - -import io.netty.bootstrap.Bootstrap; -import io.netty.channel.ChannelInitializer; -import io.netty.channel.EventLoopGroup; -import io.netty.channel.epoll.EpollDomainSocketChannel; -import io.netty.channel.epoll.EpollEventLoopGroup; -import io.netty.channel.nio.NioEventLoopGroup; -import io.netty.channel.socket.DuplexChannel; -import io.netty.channel.socket.SocketChannel; -import io.netty.channel.socket.nio.NioSocketChannel; -import io.netty.channel.unix.DomainSocketAddress; -import io.netty.channel.unix.UnixChannel; -import io.netty.handler.codec.http.HttpClientCodec; -import io.netty.handler.logging.LoggingHandler; -import io.netty.handler.ssl.SslHandler; -import io.netty.util.concurrent.DefaultThreadFactory; - -import org.bouncycastle.jce.provider.BouncyCastleProvider; - -import javax.net.ssl.SSLEngine; -import javax.net.ssl.SSLParameters; - -import java.io.IOException; -import java.net.InetAddress; -import java.net.InetSocketAddress; -import java.net.SocketAddress; -import java.security.Security; - -import static com.google.common.base.Preconditions.checkNotNull; - -/** - * Experimental implementation of {@link DockerCmdExecFactory} that supports http connection hijacking that is needed to pass STDIN to the - * container. - * - * To use it just pass an instance via {@link DockerClientImpl#withDockerCmdExecFactory(DockerCmdExecFactory)} - * - * @see https://docs.docker.com/engine/reference/api/docker_remote_api_v1.21/#attach-to-a-container - * @see https://docs.docker.com/engine/reference/api/docker_remote_api_v1.21/#exec-start - * - * - * @author Marcus Linke - */ -public class NettyDockerCmdExecFactory implements DockerCmdExecFactory { - - private static String threadPrefix = "dockerjava-netty"; - - /* - * useful links: - * - * http://stackoverflow.com/questions/33296749/netty-connect-to-unix-domain-socket-failed - * http://netty.io/wiki/native-transports.html - * https://github.com/netty/netty/blob/master/example/src/main/java/io/netty/example/http/snoop/HttpSnoopClient.java - * https://github.com/slandelle/netty-request-chunking/blob/master/src/test/java/slandelle/ChunkingTest.java - */ - - private DockerClientConfig dockerClientConfig; - - private Bootstrap bootstrap; - - private EventLoopGroup eventLoopGroup; - - private NettyInitializer nettyInitializer; - - private ChannelProvider channelProvider = new ChannelProvider() { - @Override - public DuplexChannel getChannel() { - DuplexChannel channel = connect(); - channel.pipeline().addLast(new LoggingHandler(getClass())); - return channel; - } - }; - - @Override - public void init(DockerClientConfig dockerClientConfig) { - checkNotNull(dockerClientConfig, "config was not specified"); - this.dockerClientConfig = dockerClientConfig; - - bootstrap = new Bootstrap(); - - String scheme = dockerClientConfig.getDockerHost().getScheme(); - - if ("unix".equals(scheme)) { - nettyInitializer = new UnixDomainSocketInitializer(); - } else if ("tcp".equals(scheme)) { - nettyInitializer = new InetSocketInitializer(); - } - - eventLoopGroup = nettyInitializer.init(bootstrap, dockerClientConfig); - } - - private DuplexChannel connect() { - try { - return connect(bootstrap); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - - private DuplexChannel connect(final Bootstrap bootstrap) throws InterruptedException { - return nettyInitializer.connect(bootstrap); - } - - private interface NettyInitializer { - EventLoopGroup init(final Bootstrap bootstrap, DockerClientConfig dockerClientConfig); - - DuplexChannel connect(final Bootstrap bootstrap) throws InterruptedException; - } - - private class UnixDomainSocketInitializer implements NettyInitializer { - @Override - public EventLoopGroup init(Bootstrap bootstrap, DockerClientConfig dockerClientConfig) { - EventLoopGroup epollEventLoopGroup = new EpollEventLoopGroup(0, new DefaultThreadFactory(threadPrefix)); - bootstrap.group(epollEventLoopGroup).channel(EpollDomainSocketChannel.class) - .handler(new ChannelInitializer() { - @Override - protected void initChannel(final UnixChannel channel) throws Exception { - channel.pipeline().addLast(new HttpClientCodec()); - } - }); - return epollEventLoopGroup; - } - - @Override - public DuplexChannel connect(Bootstrap bootstrap) throws InterruptedException { - return (DuplexChannel) bootstrap.connect(new DomainSocketAddress("/var/run/docker.sock")).sync().channel(); - } - } - - private class InetSocketInitializer implements NettyInitializer { - @Override - public EventLoopGroup init(Bootstrap bootstrap, final DockerClientConfig dockerClientConfig) { - EventLoopGroup nioEventLoopGroup = new NioEventLoopGroup(0, new DefaultThreadFactory(threadPrefix)); - - InetAddress addr = InetAddress.getLoopbackAddress(); - - final SocketAddress proxyAddress = new InetSocketAddress(addr, 8008); - - Security.addProvider(new BouncyCastleProvider()); - - bootstrap.group(nioEventLoopGroup).channel(NioSocketChannel.class) - .handler(new ChannelInitializer() { - @Override - protected void initChannel(final SocketChannel channel) throws Exception { - // channel.pipeline().addLast(new - // HttpProxyHandler(proxyAddress)); - channel.pipeline().addLast(new HttpClientCodec()); - } - }); - - return nioEventLoopGroup; - } - - @Override - public DuplexChannel connect(Bootstrap bootstrap) throws InterruptedException { - String host = dockerClientConfig.getDockerHost().getHost(); - int port = dockerClientConfig.getDockerHost().getPort(); - - if (port == -1) { - throw new RuntimeException("no port configured for " + host); - } - - DuplexChannel channel = (DuplexChannel) bootstrap.connect(host, port).sync().channel(); - - final SslHandler ssl = initSsl(dockerClientConfig); - - if (ssl != null) { - channel.pipeline().addFirst(ssl); - } - - return channel; - } - - private SslHandler initSsl(DockerClientConfig dockerClientConfig) { - SslHandler ssl = null; - - try { - String host = dockerClientConfig.getDockerHost().getHost(); - int port = dockerClientConfig.getDockerHost().getPort(); - - final SSLConfig sslConfig = dockerClientConfig.getSSLConfig(); - - if (sslConfig != null && sslConfig.getSSLContext() != null) { - - SSLEngine engine = sslConfig.getSSLContext().createSSLEngine(host, port); - engine.setUseClientMode(true); - engine.setSSLParameters(enableHostNameVerification(engine.getSSLParameters())); - - // in the future we may use HostnameVerifier like here: - // https://github.com/AsyncHttpClient/async-http-client/blob/1.8.x/src/main/java/com/ning/http/client/providers/netty/NettyConnectListener.java#L76 - - ssl = new SslHandler(engine); - } - - } catch (Exception e) { - throw new RuntimeException(e); - } - - return ssl; - } - } - - protected DockerClientConfig getDockerClientConfig() { - checkNotNull(dockerClientConfig, - "Factor not initialized, dockerClientConfig not set. You probably forgot to call init()!"); - return dockerClientConfig; - } - - public SSLParameters enableHostNameVerification(SSLParameters sslParameters) { - sslParameters.setEndpointIdentificationAlgorithm("HTTPS"); - return sslParameters; - } - - @Override - public CopyArchiveFromContainerCmd.Exec createCopyArchiveFromContainerCmdExec() { - return new CopyArchiveFromContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CopyArchiveToContainerCmd.Exec createCopyArchiveToContainerCmdExec() { - return new CopyArchiveToContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public AuthCmd.Exec createAuthCmdExec() { - return new AuthCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InfoCmd.Exec createInfoCmdExec() { - return new InfoCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public PingCmd.Exec createPingCmdExec() { - return new PingCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public VersionCmd.Exec createVersionCmdExec() { - return new VersionCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public PullImageCmd.Exec createPullImageCmdExec() { - return new PullImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public PushImageCmd.Exec createPushImageCmdExec() { - return new PushImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public SaveImageCmd.Exec createSaveImageCmdExec() { - return new SaveImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CreateImageCmd.Exec createCreateImageCmdExec() { - return new CreateImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public LoadImageCmd.Exec createLoadImageCmdExec() { - return new LoadImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public SearchImagesCmd.Exec createSearchImagesCmdExec() { - return new SearchImagesCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RemoveImageCmd.Exec createRemoveImageCmdExec() { - return new RemoveImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ListImagesCmd.Exec createListImagesCmdExec() { - return new ListImagesCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InspectImageCmd.Exec createInspectImageCmdExec() { - return new InspectImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ListContainersCmd.Exec createListContainersCmdExec() { - return new ListContainersCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CreateContainerCmd.Exec createCreateContainerCmdExec() { - return new CreateContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public StartContainerCmd.Exec createStartContainerCmdExec() { - return new StartContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InspectContainerCmd.Exec createInspectContainerCmdExec() { - return new InspectContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ExecCreateCmd.Exec createExecCmdExec() { - return new ExecCreateCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RemoveContainerCmd.Exec createRemoveContainerCmdExec() { - return new RemoveContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public WaitContainerCmd.Exec createWaitContainerCmdExec() { - return new WaitContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public AttachContainerCmd.Exec createAttachContainerCmdExec() { - return new AttachContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ExecStartCmd.Exec createExecStartCmdExec() { - return new ExecStartCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InspectExecCmd.Exec createInspectExecCmdExec() { - return new InspectExecCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public LogContainerCmd.Exec createLogContainerCmdExec() { - return new LogContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CopyFileFromContainerCmd.Exec createCopyFileFromContainerCmdExec() { - return new CopyFileFromContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public StopContainerCmd.Exec createStopContainerCmdExec() { - return new StopContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ContainerDiffCmd.Exec createContainerDiffCmdExec() { - return new ContainerDiffCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public KillContainerCmd.Exec createKillContainerCmdExec() { - return new KillContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public UpdateContainerCmd.Exec createUpdateContainerCmdExec() { - return new UpdateContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RenameContainerCmd.Exec createRenameContainerCmdExec() { - return new RenameContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RestartContainerCmd.Exec createRestartContainerCmdExec() { - return new RestartContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CommitCmd.Exec createCommitCmdExec() { - return new CommitCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public BuildImageCmd.Exec createBuildImageCmdExec() { - return new BuildImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public TopContainerCmd.Exec createTopContainerCmdExec() { - return new TopContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public TagImageCmd.Exec createTagImageCmdExec() { - return new TagImageCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public PauseContainerCmd.Exec createPauseContainerCmdExec() { - return new PauseContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public UnpauseContainerCmd.Exec createUnpauseContainerCmdExec() { - return new UnpauseContainerCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public EventsCmd.Exec createEventsCmdExec() { - return new EventsCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public StatsCmd.Exec createStatsCmdExec() { - return new StatsCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CreateVolumeCmd.Exec createCreateVolumeCmdExec() { - return new CreateVolumeCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InspectVolumeCmd.Exec createInspectVolumeCmdExec() { - return new InspectVolumeCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RemoveVolumeCmd.Exec createRemoveVolumeCmdExec() { - return new RemoveVolumeCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ListVolumesCmd.Exec createListVolumesCmdExec() { - return new ListVolumesCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ListNetworksCmd.Exec createListNetworksCmdExec() { - return new ListNetworksCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public InspectNetworkCmd.Exec createInspectNetworkCmdExec() { - return new InspectNetworkCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public CreateNetworkCmd.Exec createCreateNetworkCmdExec() { - return new CreateNetworkCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public RemoveNetworkCmd.Exec createRemoveNetworkCmdExec() { - return new RemoveNetworkCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public ConnectToNetworkCmd.Exec createConnectToNetworkCmdExec() { - return new ConnectToNetworkCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public DisconnectFromNetworkCmd.Exec createDisconnectFromNetworkCmdExec() { - return new DisconnectFromNetworkCmdExec(getBaseResource(), getDockerClientConfig()); - } - - @Override - public void close() throws IOException { - checkNotNull(eventLoopGroup, "Factory not initialized. You probably forgot to call init()!"); - - eventLoopGroup.shutdownGracefully(); - } - - private WebTarget getBaseResource() { - return new WebTarget(channelProvider); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/WebTarget.java b/src/main/java/com/github/dockerjava/netty/WebTarget.java deleted file mode 100644 index ef1510f5c..000000000 --- a/src/main/java/com/github/dockerjava/netty/WebTarget.java +++ /dev/null @@ -1,110 +0,0 @@ -package com.github.dockerjava.netty; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.Map; - -import org.apache.commons.lang.StringUtils; - -import com.google.common.collect.ImmutableList; -import com.google.common.collect.ImmutableMap; - -/** - * This class is basically a replacement of javax.ws.rs.client.WebTarget to allow simpler migration of JAX-RS code to a netty based - * implementation. - * - * @author Marcus Linke - */ -public class WebTarget { - - private final ChannelProvider channelProvider; - - private final ImmutableList path; - - private final ImmutableMap queryParams; - - private static final String PATH_SEPARATOR = "/"; - - public WebTarget(ChannelProvider channelProvider) { - this(channelProvider, ImmutableList.of(), ImmutableMap.of()); - } - - private WebTarget(ChannelProvider channelProvider, - ImmutableList path, - ImmutableMap queryParams) { - this.channelProvider = channelProvider; - this.path = path; - this.queryParams = queryParams; - } - - public WebTarget path(String... components) { - ImmutableList.Builder newPath = ImmutableList.builder().addAll(this.path); - - for (String component : components) { - newPath.addAll(Arrays.asList(StringUtils.split(component, PATH_SEPARATOR))); - } - - return new WebTarget(channelProvider, newPath.build(), queryParams); - } - - public InvocationBuilder request() { - String resource = PATH_SEPARATOR + StringUtils.join(path, PATH_SEPARATOR); - - List params = new ArrayList<>(); - for (Map.Entry entry : queryParams.entrySet()) { - params.add(entry.getKey() + "=" + entry.getValue()); - } - - if (!params.isEmpty()) { - resource = resource + "?" + StringUtils.join(params, "&"); - } - - return new InvocationBuilder(channelProvider, resource); - } - - public WebTarget resolveTemplate(String name, Object value) { - ImmutableList.Builder newPath = ImmutableList.builder(); - for (String component : path) { - component = component.replaceAll("\\{" + name + "\\}", value.toString()); - newPath.add(component); - } - return new WebTarget(channelProvider, newPath.build(), queryParams); - } - - public WebTarget queryParam(String name, Object value) { - ImmutableMap.Builder builder = ImmutableMap.builder().putAll(queryParams); - if (value != null) { - builder.put(name, value.toString()); - } - return new WebTarget(channelProvider, path, builder.build()); - } - - @Override - public boolean equals(Object o) { - if (this == o) { - return true; - } - if (o == null || getClass() != o.getClass()) { - return false; - } - - WebTarget webTarget = (WebTarget) o; - - if (channelProvider != null ? !channelProvider.equals(webTarget.channelProvider) : webTarget.channelProvider != null) { - return false; - } - if (path != null ? !path.equals(webTarget.path) : webTarget.path != null) { - return false; - } - return queryParams != null ? queryParams.equals(webTarget.queryParams) : webTarget.queryParams == null; - } - - @Override - public int hashCode() { - int result = channelProvider != null ? channelProvider.hashCode() : 0; - result = 31 * result + (path != null ? path.hashCode() : 0); - result = 31 * result + (queryParams != null ? queryParams.hashCode() : 0); - return result; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/AbstrAsyncDockerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/AbstrAsyncDockerCmdExec.java deleted file mode 100644 index 3d1d4fac3..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/AbstrAsyncDockerCmdExec.java +++ /dev/null @@ -1,63 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.io.Closeable; -import java.io.IOException; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.AsyncDockerCmd; -import com.github.dockerjava.api.command.DockerCmdAsyncExec; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; - -public abstract class AbstrAsyncDockerCmdExec, A_RES_T> extends - AbstrDockerCmdExec implements DockerCmdAsyncExec { - - public AbstrAsyncDockerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - public Void exec(CMD_T command, ResultCallback resultCallback) { - return execute(command, resultCallback); - } - - protected final Void execute(final CMD_T command, final ResultCallback resultCallback) { - - ResultCallback delegatingResultCallback = new ResultCallback() { - - @Override - public void close() throws IOException { - resultCallback.close(); - command.close(); - } - - @Override - public void onStart(Closeable closeable) { - resultCallback.onStart(closeable); - } - - @Override - public void onNext(A_RES_T object) { - resultCallback.onNext(object); - } - - @Override - public void onError(Throwable throwable) { - resultCallback.onError(throwable); - } - - @Override - public void onComplete() { - resultCallback.onComplete(); - command.close(); - } - }; - - execute0(command, delegatingResultCallback); - - return null; - } - - protected abstract Void execute0(final CMD_T command, final ResultCallback resultCallback); - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/AbstrDockerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/AbstrDockerCmdExec.java deleted file mode 100644 index 85763b403..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/AbstrDockerCmdExec.java +++ /dev/null @@ -1,83 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static com.github.dockerjava.core.RemoteApiVersion.UNKNOWN_VERSION; -import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_19; -import static com.google.common.base.Preconditions.checkNotNull; - -import java.io.IOException; - -import com.fasterxml.jackson.databind.node.ObjectNode; -import org.apache.commons.codec.binary.Base64; - -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.AuthConfigurations; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.RemoteApiVersion; -import com.github.dockerjava.netty.WebTarget; - -public abstract class AbstrDockerCmdExec { - - private final DockerClientConfig dockerClientConfig; - - private final WebTarget baseResource; - - public AbstrDockerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - checkNotNull(baseResource, "baseResource was not specified"); - checkNotNull(dockerClientConfig, "dockerClientConfig was not specified"); - this.baseResource = baseResource; - this.dockerClientConfig = dockerClientConfig; - } - - protected WebTarget getBaseResource() { - return baseResource; - } - - protected AuthConfigurations getBuildAuthConfigs() { - return dockerClientConfig.getAuthConfigurations(); - } - - protected String registryAuth(AuthConfig authConfig) { - try { - return Base64.encodeBase64String(new ObjectMapper().writeValueAsString(authConfig).getBytes()); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - protected String registryConfigs(AuthConfigurations authConfigs) { - try { - final String json; - final ObjectMapper objectMapper = new ObjectMapper(); - final RemoteApiVersion apiVersion = dockerClientConfig.getApiVersion(); - - if (apiVersion.equals(UNKNOWN_VERSION)) { - ObjectNode rootNode = objectMapper.valueToTree(authConfigs.getConfigs()); // all registries - final ObjectNode authNodes = objectMapper.valueToTree(authConfigs); // wrapped in "configs":{} - rootNode.setAll(authNodes); // merge 2 variants - json = rootNode.toString(); - } else if (apiVersion.isGreaterOrEqual(VERSION_1_19)) { - json = objectMapper.writeValueAsString(authConfigs.getConfigs()); - } else { - json = objectMapper.writeValueAsString(authConfigs); - } - - return Base64.encodeBase64String(json.getBytes()); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - protected boolean bool(Boolean bool) { - return bool != null && bool; - } - - protected WebTarget booleanQueryParam(WebTarget webTarget, String name, Boolean value) { - if (bool(value)) { - webTarget = webTarget.queryParam(name, bool(value) + ""); - } - - return webTarget; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/AbstrSyncDockerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/AbstrSyncDockerCmdExec.java deleted file mode 100644 index 993799dab..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/AbstrSyncDockerCmdExec.java +++ /dev/null @@ -1,33 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.DockerCmd; -import com.github.dockerjava.api.command.DockerCmdSyncExec; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; - -public abstract class AbstrSyncDockerCmdExec, RES_T> extends AbstrDockerCmdExec - implements DockerCmdSyncExec { - - public AbstrSyncDockerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - public RES_T exec(CMD_T command) { - // this hack works because of ResponseStatusExceptionFilter - try (CMD_T cmd = command) { - try { - return execute(cmd); - } catch (RuntimeException e) { - if (e.getCause() instanceof DockerException) { - throw (DockerException) e.getCause(); - } else { - throw e; - } - } - } - } - - protected abstract RES_T execute(CMD_T command); -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/AttachContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/AttachContainerCmdExec.java deleted file mode 100644 index 2fdace48a..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/AttachContainerCmdExec.java +++ /dev/null @@ -1,39 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.AttachContainerCmd; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; - -public class AttachContainerCmdExec extends AbstrAsyncDockerCmdExec implements - AttachContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(AttachContainerCmdExec.class); - - public AttachContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute0(AttachContainerCmd command, ResultCallback resultCallback) { - - WebTarget webTarget = getBaseResource().path("/containers/{id}/attach").resolveTemplate("id", - command.getContainerId()); - - webTarget = booleanQueryParam(webTarget, "logs", command.hasLogsEnabled()); - webTarget = booleanQueryParam(webTarget, "stdout", command.hasStdoutEnabled()); - webTarget = booleanQueryParam(webTarget, "stderr", command.hasStderrEnabled()); - webTarget = booleanQueryParam(webTarget, "stdin", command.getStdin() != null); - webTarget = booleanQueryParam(webTarget, "stream", command.hasFollowStreamEnabled()); - - LOGGER.trace("POST: {}", webTarget); - - webTarget.request().post(null, command.getStdin(), resultCallback); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/AuthCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/AuthCmdExec.java deleted file mode 100644 index b6f004940..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/AuthCmdExec.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.AuthCmd; -import com.github.dockerjava.api.model.AuthResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class AuthCmdExec extends AbstrSyncDockerCmdExec implements AuthCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(AuthCmdExec.class); - - public AuthCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected AuthResponse execute(AuthCmd command) { - WebTarget webResource = getBaseResource().path("/auth"); - LOGGER.trace("POST: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(command.getAuthConfig(), new TypeReference() { - }); - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/BuildImageCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/BuildImageCmdExec.java deleted file mode 100644 index f4ff04576..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/BuildImageCmdExec.java +++ /dev/null @@ -1,112 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.fasterxml.jackson.databind.ObjectMapper; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.BuildImageCmd; -import com.github.dockerjava.api.model.AuthConfigurations; -import com.github.dockerjava.api.model.BuildResponseItem; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.InvocationBuilder; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -import java.io.IOException; - -public class BuildImageCmdExec extends AbstrAsyncDockerCmdExec implements - BuildImageCmd.Exec { - private static final Logger LOGGER = LoggerFactory.getLogger(BuildImageCmdExec.class); - - private static final ObjectMapper MAPPER = new ObjectMapper(); - - public BuildImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - private InvocationBuilder resourceWithOptionalAuthConfig(BuildImageCmd command, InvocationBuilder request) { - final AuthConfigurations authConfigs = firstNonNull(command.getBuildAuthConfigs(), getBuildAuthConfigs()); - if (authConfigs != null && !authConfigs.getConfigs().isEmpty()) { - request = request.header("X-Registry-Config", registryConfigs(authConfigs)); - } - return request; - } - - private static AuthConfigurations firstNonNull(final AuthConfigurations fromCommand, - final AuthConfigurations fromConfig) { - if (fromCommand != null) { - return fromCommand; - } - if (fromConfig != null) { - return fromConfig; - } - return null; - } - - @Override - protected Void execute0(BuildImageCmd command, ResultCallback resultCallback) { - - WebTarget webTarget = getBaseResource().path("/build"); - String dockerFilePath = command.getPathToDockerfile(); - - if (dockerFilePath != null && command.getRemote() == null && !"Dockerfile".equals(dockerFilePath)) { - webTarget = webTarget.queryParam("dockerfile", dockerFilePath); - } - if (command.getTag() != null) { - webTarget = webTarget.queryParam("t", command.getTag()); - } - if (command.getRemote() != null) { - webTarget = webTarget.queryParam("remote", command.getRemote().toString()); - } - - webTarget = booleanQueryParam(webTarget, "q", command.isQuiet()); - webTarget = booleanQueryParam(webTarget, "nocache", command.hasNoCacheEnabled()); - webTarget = booleanQueryParam(webTarget, "pull", command.hasPullEnabled()); - webTarget = booleanQueryParam(webTarget, "rm", command.hasRemoveEnabled()); - webTarget = booleanQueryParam(webTarget, "forcerm", command.isForcerm()); - - // this has to be handled differently as it should switch to 'false' - if (command.hasRemoveEnabled() == null || !command.hasRemoveEnabled()) { - webTarget = webTarget.queryParam("rm", "false"); - } - - if (command.getMemory() != null) { - webTarget = webTarget.queryParam("memory", command.getMemory()); - } - if (command.getMemswap() != null) { - webTarget = webTarget.queryParam("memswap", command.getMemswap()); - } - if (command.getCpushares() != null) { - webTarget = webTarget.queryParam("cpushares", command.getCpushares()); - } - if (command.getCpusetcpus() != null) { - webTarget = webTarget.queryParam("cpusetcpus", command.getCpusetcpus()); - } - - if (command.getBuildArgs() != null && !command.getBuildArgs().isEmpty()) { - try { - webTarget = webTarget.queryParam("buildargs", MAPPER.writeValueAsString(command.getBuildArgs())); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - if (command.getShmsize() != null) { - webTarget = webTarget.queryParam("shmsize", command.getShmsize()); - } - - LOGGER.trace("POST: {}", webTarget); - - InvocationBuilder builder = resourceWithOptionalAuthConfig(command, webTarget.request()) - .accept(MediaType.APPLICATION_JSON) - .header("Content-Type", "application/tar") - .header("encoding", "gzip"); - - builder.post(new TypeReference() { - }, resultCallback, command.getTarInputStream()); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/CommitCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/CommitCmdExec.java deleted file mode 100644 index e568345cd..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/CommitCmdExec.java +++ /dev/null @@ -1,37 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.fasterxml.jackson.databind.node.ObjectNode; -import com.github.dockerjava.api.command.CommitCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class CommitCmdExec extends AbstrSyncDockerCmdExec implements CommitCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CommitCmdExec.class); - - public CommitCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected String execute(CommitCmd command) { - WebTarget webTarget = getBaseResource().path("/commit").queryParam("container", command.getContainerId()) - .queryParam("repo", command.getRepository()).queryParam("tag", command.getTag()) - .queryParam("m", command.getMessage()).queryParam("author", command.getAuthor()); - - webTarget = booleanQueryParam(webTarget, "pause", command.hasPauseEnabled()); - - LOGGER.trace("POST: {}", webTarget); - ObjectNode objectNode = webTarget.request().accept(MediaType.APPLICATION_JSON) - .post(command, new TypeReference() { - }); - - return objectNode.get("Id").asText(); - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/ConnectToNetworkCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/ConnectToNetworkCmdExec.java deleted file mode 100644 index b7bff9d52..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/ConnectToNetworkCmdExec.java +++ /dev/null @@ -1,28 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.ConnectToNetworkCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public class ConnectToNetworkCmdExec extends AbstrSyncDockerCmdExec - implements ConnectToNetworkCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ConnectToNetworkCmdExec.class); - - public ConnectToNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(ConnectToNetworkCmd command) { - - WebTarget webTarget = getBaseResource().path("/networks/" + command.getNetworkId() + "/connect"); - - LOGGER.trace("POST: {}", webTarget); - webTarget.request().post(command); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/ContainerDiffCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/ContainerDiffCmdExec.java deleted file mode 100644 index 662b9c1d4..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/ContainerDiffCmdExec.java +++ /dev/null @@ -1,34 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.util.List; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.ContainerDiffCmd; -import com.github.dockerjava.api.model.ChangeLog; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class ContainerDiffCmdExec extends AbstrSyncDockerCmdExec> implements - ContainerDiffCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ContainerDiffCmdExec.class); - - public ContainerDiffCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected List execute(ContainerDiffCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/changes").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference>() { - }); - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/CopyArchiveFromContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/CopyArchiveFromContainerCmdExec.java deleted file mode 100644 index 6f4a93ba3..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/CopyArchiveFromContainerCmdExec.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.io.InputStream; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class CopyArchiveFromContainerCmdExec extends AbstrSyncDockerCmdExec - implements CopyArchiveFromContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CopyArchiveFromContainerCmdExec.class); - - public CopyArchiveFromContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InputStream execute(CopyArchiveFromContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/archive").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("Get: " + webResource.toString()); - - return webResource.queryParam("path", command.getResource()).request().accept(MediaType.APPLICATION_X_TAR) - .get(); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/CopyArchiveToContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/CopyArchiveToContainerCmdExec.java deleted file mode 100644 index c956a815d..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/CopyArchiveToContainerCmdExec.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.io.InputStream; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class CopyArchiveToContainerCmdExec extends AbstrSyncDockerCmdExec implements - CopyArchiveToContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CopyArchiveFromContainerCmdExec.class); - - public CopyArchiveToContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(CopyArchiveToContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/archive").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("PUT: " + webResource.toString()); - InputStream streamToUpload = command.getTarInputStream(); - - webResource.queryParam("path", command.getRemotePath()) - .queryParam("noOverwriteDirNonDir", command.isNoOverwriteDirNonDir()).request() - .put(streamToUpload, MediaType.APPLICATION_X_TAR); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/CopyFileFromContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/CopyFileFromContainerCmdExec.java deleted file mode 100644 index 338b9b973..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/CopyFileFromContainerCmdExec.java +++ /dev/null @@ -1,33 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.io.InputStream; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.CopyFileFromContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class CopyFileFromContainerCmdExec extends AbstrSyncDockerCmdExec - implements CopyFileFromContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CopyFileFromContainerCmdExec.class); - - public CopyFileFromContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InputStream execute(CopyFileFromContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/copy").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: " + webResource.toString()); - - return webResource.request().accept(MediaType.APPLICATION_OCTET_STREAM) - .post(command); - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/CreateContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/CreateContainerCmdExec.java deleted file mode 100644 index f7e4e39d9..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/CreateContainerCmdExec.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.CreateContainerCmd; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class CreateContainerCmdExec extends AbstrSyncDockerCmdExec - implements CreateContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CreateContainerCmdExec.class); - - public CreateContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected CreateContainerResponse execute(CreateContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/create"); - - if (command.getName() != null) { - webResource = webResource.queryParam("name", command.getName()); - } - - LOGGER.trace("POST: {} ", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(command, new TypeReference() { - }); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/CreateImageCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/CreateImageCmdExec.java deleted file mode 100644 index 8fd791f53..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/CreateImageCmdExec.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.CreateImageCmd; -import com.github.dockerjava.api.command.CreateImageResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class CreateImageCmdExec extends AbstrSyncDockerCmdExec implements - CreateImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CreateImageCmdExec.class); - - public CreateImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected CreateImageResponse execute(CreateImageCmd command) { - WebTarget webResource = getBaseResource().path("/images/create").queryParam("repo", command.getRepository()) - .queryParam("tag", command.getTag()).queryParam("fromSrc", "-"); - - LOGGER.trace("POST: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_OCTET_STREAM) - .post(new TypeReference() { - }, command.getImageStream()); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/CreateNetworkCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/CreateNetworkCmdExec.java deleted file mode 100644 index cfc825e80..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/CreateNetworkCmdExec.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.CreateNetworkCmd; -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public class CreateNetworkCmdExec extends AbstrSyncDockerCmdExec implements - CreateNetworkCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CreateNetworkCmdExec.class); - - public CreateNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected CreateNetworkResponse execute(CreateNetworkCmd command) { - WebTarget webResource = getBaseResource().path("/networks/create"); - - LOGGER.trace("POST: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(command, new TypeReference() { - }); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/CreateVolumeCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/CreateVolumeCmdExec.java deleted file mode 100644 index 746e46067..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/CreateVolumeCmdExec.java +++ /dev/null @@ -1,31 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.CreateVolumeCmd; -import com.github.dockerjava.api.command.CreateVolumeResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class CreateVolumeCmdExec extends AbstrSyncDockerCmdExec implements - CreateVolumeCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(CreateVolumeCmdExec.class); - - public CreateVolumeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected CreateVolumeResponse execute(CreateVolumeCmd command) { - WebTarget webResource = getBaseResource().path("/volumes/create"); - - LOGGER.trace("POST: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(command, new TypeReference() { - }); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/DisconnectFromNetworkCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/DisconnectFromNetworkCmdExec.java deleted file mode 100644 index ef139517f..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/DisconnectFromNetworkCmdExec.java +++ /dev/null @@ -1,28 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public class DisconnectFromNetworkCmdExec extends AbstrSyncDockerCmdExec - implements DisconnectFromNetworkCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(DisconnectFromNetworkCmdExec.class); - - public DisconnectFromNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(DisconnectFromNetworkCmd command) { - - WebTarget webTarget = getBaseResource().path("/networks/" + command.getNetworkId() + "/disconnect"); - - LOGGER.trace("POST: {}", webTarget); - webTarget.request().post(command); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/EventsCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/EventsCmdExec.java deleted file mode 100644 index f07230c07..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/EventsCmdExec.java +++ /dev/null @@ -1,42 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static com.google.common.net.UrlEscapers.urlPathSegmentEscaper; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.EventsCmd; -import com.github.dockerjava.api.model.Event; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.util.FiltersEncoder; -import com.github.dockerjava.netty.WebTarget; - -public class EventsCmdExec extends AbstrAsyncDockerCmdExec implements EventsCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(EventsCmdExec.class); - - public EventsCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute0(EventsCmd command, ResultCallback resultCallback) { - - WebTarget webTarget = getBaseResource().path("/events").queryParam("since", command.getSince()) - .queryParam("until", command.getUntil()); - - if (command.getFilters() != null && !command.getFilters().isEmpty()) { - webTarget = webTarget - .queryParam("filters", urlPathSegmentEscaper().escape(FiltersEncoder.jsonEncode(command.getFilters()))); - } - - LOGGER.trace("GET: {}", webTarget); - - webTarget.request().get(new TypeReference() { - }, resultCallback); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/ExecCreateCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/ExecCreateCmdExec.java deleted file mode 100644 index 1535cfe4f..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/ExecCreateCmdExec.java +++ /dev/null @@ -1,33 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.ExecCreateCmd; -import com.github.dockerjava.api.command.ExecCreateCmdResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class ExecCreateCmdExec extends AbstrSyncDockerCmdExec implements - ExecCreateCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ExecCreateCmdExec.class); - - public ExecCreateCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected ExecCreateCmdResponse execute(ExecCreateCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/exec").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(command, new TypeReference() { - }); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/ExecStartCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/ExecStartCmdExec.java deleted file mode 100644 index 374b2062d..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/ExecStartCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.ExecStartCmd; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class ExecStartCmdExec extends AbstrAsyncDockerCmdExec implements ExecStartCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ExecStartCmdExec.class); - - public ExecStartCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute0(ExecStartCmd command, ResultCallback resultCallback) { - WebTarget webTarget = getBaseResource().path("/exec/{id}/start").resolveTemplate("id", command.getExecId()); - - webTarget.request().accept(MediaType.APPLICATION_JSON).post(command, command.getStdin(), resultCallback); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/InfoCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/InfoCmdExec.java deleted file mode 100644 index 8fc211322..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/InfoCmdExec.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.InfoCmd; -import com.github.dockerjava.api.model.Info; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; - -/** - * http://stackoverflow.com/questions/33296749/netty-connect-to-unix-domain- socket-failed http://netty.io/wiki/native-transports.html - * https://github.com/netty/netty/blob/master/example/src/main/java/io/netty/ example/http/snoop/HttpSnoopClient.java - * - * @author Marcus Linke - * - */ -public class InfoCmdExec implements InfoCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(InfoCmdExec.class); - - private WebTarget webResource; - - public InfoCmdExec(WebTarget webResource, DockerClientConfig dockerClientConfig) { - this.webResource = webResource; - } - - @Override - public Info exec(InfoCmd command) { - return webResource.path("info").request().get(new TypeReference() { - }); - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/InspectContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/InspectContainerCmdExec.java deleted file mode 100644 index b6a4a06f2..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/InspectContainerCmdExec.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.InspectContainerCmd; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class InspectContainerCmdExec extends AbstrSyncDockerCmdExec - implements InspectContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(InspectContainerCmdExec.class); - - public InspectContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InspectContainerResponse execute(InspectContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/json").resolveTemplate("id", - command.getContainerId()); - - webResource = booleanQueryParam(webResource, "size", command.getSize()); - - LOGGER.debug("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON) - .get(new TypeReference() { - }); - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/InspectExecCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/InspectExecCmdExec.java deleted file mode 100644 index ee913f369..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/InspectExecCmdExec.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.InspectExecCmd; -import com.github.dockerjava.api.command.InspectExecResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class InspectExecCmdExec extends AbstrSyncDockerCmdExec implements - InspectExecCmd.Exec { - private static final Logger LOGGER = LoggerFactory.getLogger(InspectExecCmdExec.class); - - public InspectExecCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InspectExecResponse execute(InspectExecCmd command) { - WebTarget webResource = getBaseResource().path("/exec/{id}/json").resolveTemplate("id", command.getExecId()); - - LOGGER.debug("GET: {}", webResource); - - return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { - }); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/InspectImageCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/InspectImageCmdExec.java deleted file mode 100644 index b86072ecb..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/InspectImageCmdExec.java +++ /dev/null @@ -1,32 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.InspectImageCmd; -import com.github.dockerjava.api.command.InspectImageResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class InspectImageCmdExec extends AbstrSyncDockerCmdExec implements - InspectImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(InspectImageCmdExec.class); - - public InspectImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InspectImageResponse execute(InspectImageCmd command) { - WebTarget webResource = getBaseResource().path("/images/{id}/json").resolveTemplate("id", command.getImageId()); - - LOGGER.trace("GET: {}", webResource); - - return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { - }); - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/InspectNetworkCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/InspectNetworkCmdExec.java deleted file mode 100644 index 51777972f..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/InspectNetworkCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.InspectNetworkCmd; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public class InspectNetworkCmdExec extends AbstrSyncDockerCmdExec implements - InspectNetworkCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(InspectNetworkCmdExec.class); - - public InspectNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Network execute(InspectNetworkCmd command) { - WebTarget webResource = getBaseResource().path("/networks/{id}").resolveTemplate("id", command.getNetworkId()); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { - }); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/InspectVolumeCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/InspectVolumeCmdExec.java deleted file mode 100644 index 1c3adf47e..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/InspectVolumeCmdExec.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.InspectVolumeCmd; -import com.github.dockerjava.api.command.InspectVolumeResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class InspectVolumeCmdExec extends AbstrSyncDockerCmdExec implements - InspectVolumeCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(InspectVolumeCmdExec.class); - - public InspectVolumeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InspectVolumeResponse execute(InspectVolumeCmd command) { - WebTarget webResource = getBaseResource().path("/volumes/{name}").resolveTemplate("name", command.getName()); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { - }); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/KillContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/KillContainerCmdExec.java deleted file mode 100644 index b81f6449f..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/KillContainerCmdExec.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.KillContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class KillContainerCmdExec extends AbstrSyncDockerCmdExec implements - KillContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(KillContainerCmdExec.class); - - public KillContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(KillContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/kill").resolveTemplate("id", - command.getContainerId()); - - if (command.getSignal() != null) { - webResource = webResource.queryParam("signal", command.getSignal()); - } - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/ListContainersCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/ListContainersCmdExec.java deleted file mode 100644 index e5ab4b66b..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/ListContainersCmdExec.java +++ /dev/null @@ -1,55 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static com.google.common.net.UrlEscapers.urlPathSegmentEscaper; - -import java.util.List; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.ListContainersCmd; -import com.github.dockerjava.api.model.Container; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.util.FiltersEncoder; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class ListContainersCmdExec extends AbstrSyncDockerCmdExec> implements - ListContainersCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ListContainersCmdExec.class); - - public ListContainersCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected List execute(ListContainersCmd command) { - WebTarget webTarget = getBaseResource().path("/containers/json").queryParam("since", command.getSinceId()) - .queryParam("before", command.getBeforeId()); - - webTarget = booleanQueryParam(webTarget, "all", command.hasShowAllEnabled()); - webTarget = booleanQueryParam(webTarget, "size", command.hasShowSizeEnabled()); - - if (command.getLimit() != null && command.getLimit() >= 0) { - webTarget = webTarget.queryParam("limit", String.valueOf(command.getLimit())); - } - - if (command.getFilters() != null && !command.getFilters().isEmpty()) { - webTarget = webTarget - .queryParam("filters", urlPathSegmentEscaper().escape(FiltersEncoder.jsonEncode(command.getFilters()))); - } - - LOGGER.trace("GET: {}", webTarget); - - List containers = webTarget.request().accept(MediaType.APPLICATION_JSON) - .get(new TypeReference>() { - }); - - LOGGER.trace("Response: {}", containers); - - return containers; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/ListImagesCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/ListImagesCmdExec.java deleted file mode 100644 index a1a38add5..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/ListImagesCmdExec.java +++ /dev/null @@ -1,50 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static com.google.common.net.UrlEscapers.urlPathSegmentEscaper; - -import java.util.List; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.ListImagesCmd; -import com.github.dockerjava.api.model.Image; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.util.FiltersEncoder; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class ListImagesCmdExec extends AbstrSyncDockerCmdExec> implements ListImagesCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ListImagesCmdExec.class); - - public ListImagesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected List execute(ListImagesCmd command) { - WebTarget webTarget = getBaseResource().path("/images/json"); - - webTarget = booleanQueryParam(webTarget, "all", command.hasShowAllEnabled()); - - if (command.getFilters() != null && !command.getFilters().isEmpty()) { - webTarget = webTarget.queryParam("filters", urlPathSegmentEscaper().escape(FiltersEncoder.jsonEncode(command.getFilters()))); - } - - if (command.getImageNameFilter() != null) { - webTarget = webTarget.queryParam("filter", urlPathSegmentEscaper().escape(command.getImageNameFilter())); - } - - LOGGER.trace("GET: {}", webTarget); - - List images = webTarget.request().accept(MediaType.APPLICATION_JSON) - .get(new TypeReference>() { - }); - - LOGGER.trace("Response: {}", images); - - return images; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/ListNetworksCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/ListNetworksCmdExec.java deleted file mode 100644 index 11e5fe238..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/ListNetworksCmdExec.java +++ /dev/null @@ -1,40 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.ListNetworksCmd; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.util.FiltersEncoder; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import java.util.List; - -import static com.google.common.net.UrlEscapers.urlPathSegmentEscaper; - -public class ListNetworksCmdExec extends AbstrSyncDockerCmdExec> implements - ListNetworksCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ListNetworksCmdExec.class); - - public ListNetworksCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected List execute(ListNetworksCmd command) { - WebTarget webTarget = getBaseResource().path("/networks"); - - if (command.getFilters() != null && !command.getFilters().isEmpty()) { - webTarget = webTarget.queryParam("filters", urlPathSegmentEscaper().escape(FiltersEncoder.jsonEncode(command.getFilters()))); - } - - LOGGER.trace("GET: {}", webTarget); - - return webTarget.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference>() { - }); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/ListVolumesCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/ListVolumesCmdExec.java deleted file mode 100644 index 1db5684cb..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/ListVolumesCmdExec.java +++ /dev/null @@ -1,38 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static com.google.common.net.UrlEscapers.urlPathSegmentEscaper; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.ListVolumesCmd; -import com.github.dockerjava.api.command.ListVolumesResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.core.util.FiltersEncoder; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class ListVolumesCmdExec extends AbstrSyncDockerCmdExec implements - ListVolumesCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(ListVolumesCmdExec.class); - - public ListVolumesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected ListVolumesResponse execute(ListVolumesCmd command) { - WebTarget webTarget = getBaseResource().path("/volumes"); - - if (command.getFilters() != null && !command.getFilters().isEmpty()) { - webTarget = webTarget.queryParam("filters", urlPathSegmentEscaper().escape(FiltersEncoder.jsonEncode(command.getFilters()))); - } - - LOGGER.trace("GET: {}", webTarget); - - return webTarget.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { - }); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/LoadImageCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/LoadImageCmdExec.java deleted file mode 100644 index 23581119f..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/LoadImageCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.LoadImageCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; - -public class LoadImageCmdExec extends AbstrSyncDockerCmdExec implements - LoadImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(LoadImageCmdExec.class); - - public LoadImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(LoadImageCmd command) { - WebTarget webResource = getBaseResource().path("/images/load"); - - LOGGER.trace("POST: {}", webResource); - return webResource.request() - .post(new TypeReference() { - }, command.getImageStream()); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/LogContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/LogContainerCmdExec.java deleted file mode 100644 index a23bcb20d..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/LogContainerCmdExec.java +++ /dev/null @@ -1,46 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.LogContainerCmd; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; - -public class LogContainerCmdExec extends AbstrAsyncDockerCmdExec implements - LogContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(LogContainerCmdExec.class); - - public LogContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute0(LogContainerCmd command, ResultCallback resultCallback) { - - WebTarget webTarget = getBaseResource().path("/containers/{id}/logs").resolveTemplate("id", - command.getContainerId()); - - if (command.getTail() != null) { - webTarget = webTarget.queryParam("tail", command.getTail()); - } - - if (command.getSince() != null) { - webTarget = webTarget.queryParam("since", command.getSince()); - } - - webTarget = booleanQueryParam(webTarget, "timestamps", command.hasTimestampsEnabled()); - webTarget = booleanQueryParam(webTarget, "stdout", command.hasStdoutEnabled()); - webTarget = booleanQueryParam(webTarget, "stderr", command.hasStderrEnabled()); - webTarget = booleanQueryParam(webTarget, "follow", command.hasFollowStreamEnabled()); - - LOGGER.trace("GET: {}", webTarget); - - webTarget.request().get(resultCallback); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/PauseContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/PauseContainerCmdExec.java deleted file mode 100644 index 5d8443190..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/PauseContainerCmdExec.java +++ /dev/null @@ -1,31 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.PauseContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class PauseContainerCmdExec extends AbstrSyncDockerCmdExec implements - PauseContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(PauseContainerCmdExec.class); - - public PauseContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(PauseContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/pause").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/PingCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/PingCmdExec.java deleted file mode 100644 index 1302c9730..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/PingCmdExec.java +++ /dev/null @@ -1,28 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.PingCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; - -public class PingCmdExec extends AbstrSyncDockerCmdExec implements PingCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(PingCmdExec.class); - - public PingCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(PingCmd command) { - WebTarget webResource = getBaseResource().path("/_ping"); - - LOGGER.trace("GET: {}", webResource); - webResource.request().get(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/PullImageCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/PullImageCmdExec.java deleted file mode 100644 index 7f15b4e62..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/PullImageCmdExec.java +++ /dev/null @@ -1,46 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.PullImageCmd; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.PullResponseItem; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.InvocationBuilder; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class PullImageCmdExec extends AbstrAsyncDockerCmdExec implements - PullImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(PullImageCmdExec.class); - - public PullImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - private InvocationBuilder resourceWithOptionalAuthConfig(PullImageCmd command, InvocationBuilder request) { - AuthConfig authConfig = command.getAuthConfig(); - if (authConfig != null) { - request = request.header("X-Registry-Auth", registryAuth(authConfig)); - } - return request; - } - - @Override - protected Void execute0(PullImageCmd command, ResultCallback resultCallback) { - - WebTarget webResource = getBaseResource().path("/images/create").queryParam("tag", command.getTag()) - .queryParam("fromImage", command.getRepository()).queryParam("registry", command.getRegistry()); - - LOGGER.trace("POST: {}", webResource); - resourceWithOptionalAuthConfig(command, webResource.request()).accept(MediaType.APPLICATION_OCTET_STREAM).post( - null, new TypeReference() { - }, resultCallback); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/PushImageCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/PushImageCmdExec.java deleted file mode 100644 index d6dcdbd00..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/PushImageCmdExec.java +++ /dev/null @@ -1,48 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.PushImageCmd; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.PushResponseItem; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.InvocationBuilder; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class PushImageCmdExec extends AbstrAsyncDockerCmdExec implements - PushImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(PushImageCmdExec.class); - - public PushImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - private String name(PushImageCmd command) { - String name = command.getName(); - AuthConfig authConfig = command.getAuthConfig(); - return (name.contains("/") || authConfig == null) ? name : authConfig.getUsername(); - } - - @Override - protected Void execute0(PushImageCmd command, ResultCallback resultCallback) { - - WebTarget webResource = getBaseResource().path("/images/" + name(command) + "/push").queryParam("tag", - command.getTag()); - - final String registryAuth = registryAuth(command.getAuthConfig()); - LOGGER.trace("POST: {}", webResource); - - InvocationBuilder builder = webResource.request().header("X-Registry-Auth", registryAuth) - .accept(MediaType.APPLICATION_JSON); - - builder.post(null, new TypeReference() { - }, resultCallback); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/RemoveContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/RemoveContainerCmdExec.java deleted file mode 100644 index 63d16bda6..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/RemoveContainerCmdExec.java +++ /dev/null @@ -1,33 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.RemoveContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class RemoveContainerCmdExec extends AbstrSyncDockerCmdExec implements - RemoveContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(RemoveContainerCmdExec.class); - - public RemoveContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RemoveContainerCmd command) { - WebTarget webTarget = getBaseResource().path("/containers/" + command.getContainerId()); - - webTarget = booleanQueryParam(webTarget, "v", command.hasRemoveVolumesEnabled()); - webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); - - LOGGER.trace("DELETE: {}", webTarget); - webTarget.request().accept(MediaType.APPLICATION_JSON).delete(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/RemoveImageCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/RemoveImageCmdExec.java deleted file mode 100644 index 35ae053d3..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/RemoveImageCmdExec.java +++ /dev/null @@ -1,31 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.RemoveImageCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; - -public class RemoveImageCmdExec extends AbstrSyncDockerCmdExec implements RemoveImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(RemoveImageCmdExec.class); - - public RemoveImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RemoveImageCmd command) { - WebTarget webTarget = getBaseResource().path("/images/" + command.getImageId()); - - webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); - webTarget = booleanQueryParam(webTarget, "noprune", command.hasNoPruneEnabled()); - - LOGGER.trace("DELETE: {}", webTarget); - webTarget.request().delete(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/RemoveNetworkCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/RemoveNetworkCmdExec.java deleted file mode 100644 index deb52100d..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/RemoveNetworkCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.RemoveNetworkCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public class RemoveNetworkCmdExec extends AbstrSyncDockerCmdExec implements - RemoveNetworkCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(RemoveNetworkCmdExec.class); - - public RemoveNetworkCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RemoveNetworkCmd command) { - WebTarget webTarget = getBaseResource().path("/networks/" + command.getNetworkId()); - - LOGGER.trace("DELETE: {}", webTarget); - webTarget.request().accept(MediaType.APPLICATION_JSON).delete(); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/RemoveVolumeCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/RemoveVolumeCmdExec.java deleted file mode 100644 index baf6eabe1..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/RemoveVolumeCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.RemoveVolumeCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class RemoveVolumeCmdExec extends AbstrSyncDockerCmdExec implements - RemoveVolumeCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(RemoveVolumeCmdExec.class); - - public RemoveVolumeCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RemoveVolumeCmd command) { - WebTarget webTarget = getBaseResource().path("/volumes/" + command.getName()); - - LOGGER.trace("DELETE: {}", webTarget); - webTarget.request().accept(MediaType.APPLICATION_JSON).delete(); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/RenameContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/RenameContainerCmdExec.java deleted file mode 100644 index 43f97e961..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/RenameContainerCmdExec.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.RenameContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public class RenameContainerCmdExec extends AbstrSyncDockerCmdExec - implements RenameContainerCmd.Exec { - private static final Logger LOG = LoggerFactory.getLogger(RenameContainerCmdExec.class); - - public RenameContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RenameContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/rename") - .resolveTemplate("id", command.getContainerId()) - .queryParam("name", command.getName()); - - LOG.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/RestartContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/RestartContainerCmdExec.java deleted file mode 100644 index df9721054..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/RestartContainerCmdExec.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.RestartContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class RestartContainerCmdExec extends AbstrSyncDockerCmdExec implements - RestartContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(RestartContainerCmdExec.class); - - public RestartContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(RestartContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/restart").resolveTemplate("id", - command.getContainerId()); - - if (command.getTimeout() != null) { - webResource = webResource.queryParam("t", String.valueOf(command.getTimeout())); - } - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/SaveImageCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/SaveImageCmdExec.java deleted file mode 100644 index 629f49871..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/SaveImageCmdExec.java +++ /dev/null @@ -1,28 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.io.InputStream; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.SaveImageCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class SaveImageCmdExec extends AbstrSyncDockerCmdExec implements SaveImageCmd.Exec { - private static final Logger LOGGER = LoggerFactory.getLogger(SaveImageCmdExec.class); - - public SaveImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected InputStream execute(SaveImageCmd command) { - WebTarget webResource = getBaseResource().path("/images/" + command.getName() + "/get").queryParam("tag", - command.getTag()); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/SearchImagesCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/SearchImagesCmdExec.java deleted file mode 100644 index d31d00a9f..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/SearchImagesCmdExec.java +++ /dev/null @@ -1,33 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.util.List; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.SearchImagesCmd; -import com.github.dockerjava.api.model.SearchItem; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class SearchImagesCmdExec extends AbstrSyncDockerCmdExec> implements - SearchImagesCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(SearchImagesCmdExec.class); - - public SearchImagesCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected List execute(SearchImagesCmd command) { - WebTarget webResource = getBaseResource().path("/images/search").queryParam("term", command.getTerm()); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference>() { - }); - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/StartContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/StartContainerCmdExec.java deleted file mode 100644 index 12f2d8cbc..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/StartContainerCmdExec.java +++ /dev/null @@ -1,31 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.StartContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class StartContainerCmdExec extends AbstrSyncDockerCmdExec implements - StartContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(StartContainerCmdExec.class); - - public StartContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(StartContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/start").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(command); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/StatsCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/StatsCmdExec.java deleted file mode 100644 index d167d5686..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/StatsCmdExec.java +++ /dev/null @@ -1,33 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.StatsCmd; -import com.github.dockerjava.api.model.Statistics; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; - -public class StatsCmdExec extends AbstrAsyncDockerCmdExec implements StatsCmd.Exec { - private static final Logger LOGGER = LoggerFactory.getLogger(StatsCmdExec.class); - - public StatsCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute0(StatsCmd command, ResultCallback resultCallback) { - - WebTarget webTarget = getBaseResource().path("/containers/{id}/stats").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("GET: {}", webTarget); - - webTarget.request().get(new TypeReference() { - }, resultCallback); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/StopContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/StopContainerCmdExec.java deleted file mode 100644 index cd85d92c2..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/StopContainerCmdExec.java +++ /dev/null @@ -1,34 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.StopContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class StopContainerCmdExec extends AbstrSyncDockerCmdExec implements - StopContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(StopContainerCmdExec.class); - - public StopContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(StopContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/stop").resolveTemplate("id", - command.getContainerId()); - - if (command.getTimeout() != null) { - webResource = webResource.queryParam("t", String.valueOf(command.getTimeout())); - } - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null); - - return null; - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/TagImageCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/TagImageCmdExec.java deleted file mode 100644 index 6aa171f86..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/TagImageCmdExec.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.TagImageCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.WebTarget; - -public class TagImageCmdExec extends AbstrSyncDockerCmdExec implements TagImageCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(TagImageCmdExec.class); - - public TagImageCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(TagImageCmd command) { - WebTarget webTarget = getBaseResource().path("/images/" + command.getImageId() + "/tag") - .queryParam("repo", command.getRepository()).queryParam("tag", command.getTag()); - - webTarget = booleanQueryParam(webTarget, "force", command.hasForceEnabled()); - - LOGGER.trace("POST: {}", webTarget); - webTarget.request().post(null); - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/TopContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/TopContainerCmdExec.java deleted file mode 100644 index c4ae9249d..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/TopContainerCmdExec.java +++ /dev/null @@ -1,37 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.apache.commons.lang.StringUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.TopContainerCmd; -import com.github.dockerjava.api.command.TopContainerResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class TopContainerCmdExec extends AbstrSyncDockerCmdExec implements - TopContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(TopContainerCmdExec.class); - - public TopContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected TopContainerResponse execute(TopContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/top").resolveTemplate("id", - command.getContainerId()); - - if (!StringUtils.isEmpty(command.getPsArgs())) { - webResource = webResource.queryParam("ps_args", command.getPsArgs()); - } - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { - }); - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/UnpauseContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/UnpauseContainerCmdExec.java deleted file mode 100644 index c7cbfe017..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/UnpauseContainerCmdExec.java +++ /dev/null @@ -1,31 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.command.UnpauseContainerCmd; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class UnpauseContainerCmdExec extends AbstrSyncDockerCmdExec implements - UnpauseContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(UnpauseContainerCmdExec.class); - - public UnpauseContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute(UnpauseContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/unpause").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - webResource.request().accept(MediaType.APPLICATION_JSON).post(null); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/UpdateContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/UpdateContainerCmdExec.java deleted file mode 100644 index 3c88dead8..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/UpdateContainerCmdExec.java +++ /dev/null @@ -1,34 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.UpdateContainerCmd; -import com.github.dockerjava.api.model.UpdateContainerResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - - -/** - * @author Kanstantsin Shautsou - */ -public class UpdateContainerCmdExec extends AbstrSyncDockerCmdExec - implements UpdateContainerCmd.Exec { - private static final Logger LOGGER = LoggerFactory.getLogger(UpdateContainerCmdExec.class); - - public UpdateContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected UpdateContainerResponse execute(UpdateContainerCmd command) { - WebTarget webResource = getBaseResource().path("/containers/{id}/update") - .resolveTemplate("id", command.getContainerId()); - - LOGGER.trace("POST: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON) - .post(command, new TypeReference() { - }); - } -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/VersionCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/VersionCmdExec.java deleted file mode 100644 index 84857c208..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/VersionCmdExec.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.command.VersionCmd; -import com.github.dockerjava.api.model.Version; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class VersionCmdExec extends AbstrSyncDockerCmdExec implements VersionCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(VersionCmdExec.class); - - public VersionCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Version execute(VersionCmd command) { - WebTarget webResource = getBaseResource().path("/version"); - - LOGGER.trace("GET: {}", webResource); - return webResource.request().accept(MediaType.APPLICATION_JSON).get(new TypeReference() { - }); - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/exec/WaitContainerCmdExec.java b/src/main/java/com/github/dockerjava/netty/exec/WaitContainerCmdExec.java deleted file mode 100644 index 5883b7188..000000000 --- a/src/main/java/com/github/dockerjava/netty/exec/WaitContainerCmdExec.java +++ /dev/null @@ -1,36 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.WaitContainerCmd; -import com.github.dockerjava.api.model.WaitResponse; -import com.github.dockerjava.core.DockerClientConfig; -import com.github.dockerjava.netty.MediaType; -import com.github.dockerjava.netty.WebTarget; - -public class WaitContainerCmdExec extends AbstrAsyncDockerCmdExec implements - WaitContainerCmd.Exec { - - private static final Logger LOGGER = LoggerFactory.getLogger(WaitContainerCmdExec.class); - - public WaitContainerCmdExec(WebTarget baseResource, DockerClientConfig dockerClientConfig) { - super(baseResource, dockerClientConfig); - } - - @Override - protected Void execute0(WaitContainerCmd command, ResultCallback resultCallback) { - WebTarget webTarget = getBaseResource().path("/containers/{id}/wait").resolveTemplate("id", - command.getContainerId()); - - LOGGER.trace("POST: {}", webTarget); - - webTarget.request().accept(MediaType.APPLICATION_JSON).post((Object) null, new TypeReference() { - }, resultCallback); - - return null; - } - -} diff --git a/src/main/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandler.java b/src/main/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandler.java deleted file mode 100644 index 706228d04..000000000 --- a/src/main/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandler.java +++ /dev/null @@ -1,100 +0,0 @@ -package com.github.dockerjava.netty.handler; - -import io.netty.buffer.ByteBuf; -import io.netty.channel.ChannelHandlerContext; -import io.netty.channel.SimpleChannelInboundHandler; - -import java.io.IOException; -import java.io.InputStream; -import java.util.concurrent.LinkedTransferQueue; -import java.util.concurrent.TimeUnit; -import java.util.concurrent.atomic.AtomicBoolean; - -import com.github.dockerjava.api.async.ResultCallback; - -/** - * Handler that converts an incoming byte stream to an {@link InputStream}. - * - * @author marcus - */ -public class HttpResponseStreamHandler extends SimpleChannelInboundHandler { - - private HttpResponseInputStream stream = new HttpResponseInputStream(); - - public HttpResponseStreamHandler(ResultCallback resultCallback) { - resultCallback.onNext(stream); - } - - @Override - protected void channelRead0(ChannelHandlerContext ctx, ByteBuf msg) throws Exception { - stream.write(msg.copy()); - } - - @Override - public void channelReadComplete(ChannelHandlerContext ctx) throws Exception { - stream.close(); - super.channelReadComplete(ctx); - } - - public static class HttpResponseInputStream extends InputStream { - - private AtomicBoolean closed = new AtomicBoolean(false); - - private LinkedTransferQueue queue = new LinkedTransferQueue(); - - private ByteBuf current = null; - - public void write(ByteBuf byteBuf) { - queue.put(byteBuf); - } - - @Override - public void close() throws IOException { - closed.set(true); - super.close(); - } - - @Override - public int available() throws IOException { - poll(); - return readableBytes(); - } - - private int readableBytes() { - if (current != null) { - return current.readableBytes(); - } else { - return 0; - } - - } - - @Override - public int read() throws IOException { - - poll(); - - if (readableBytes() == 0) { - if (closed.get()) { - return -1; - } - } - - if (current != null && current.readableBytes() > 0) { - return current.readByte() & 0xff; - } else { - return read(); - } - } - - private void poll() { - if (readableBytes() == 0) { - try { - current = queue.poll(50, TimeUnit.MILLISECONDS); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - } - } -} diff --git a/src/main/resources/docker-java.properties b/src/main/resources/docker-java.properties deleted file mode 100644 index fc9209c97..000000000 --- a/src/main/resources/docker-java.properties +++ /dev/null @@ -1,11 +0,0 @@ -DOCKER_HOST=unix:///var/run/docker.sock -DOCKER_CONFIG=${user.home}/.docker -#DOCKER_TLS_VERIFY=1 -#DOCKER_CERT_PATH=${user.home}/.docker/certs - -api.version= -registry.url=https://index.docker.io/v1/ -registry.username=${user.name} -#registry.password= -#registry.email= - diff --git a/src/test/java/com/github/dockerjava/api/command/InspectContainerResponseTest.java b/src/test/java/com/github/dockerjava/api/command/InspectContainerResponseTest.java deleted file mode 100644 index f64cf4ead..000000000 --- a/src/test/java/com/github/dockerjava/api/command/InspectContainerResponseTest.java +++ /dev/null @@ -1,76 +0,0 @@ -/* - * Copyright 2015 CloudBees Inc., Oleg Nenashev. - * - * 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.github.dockerjava.api.command; - -import org.testng.annotations.Test; - -import java.io.IOException; - -import static com.github.dockerjava.test.serdes.JSONTestHelper.testRoundTrip; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.nullValue; -import static org.hamcrest.core.IsNot.not; -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertFalse; -import static org.testng.Assert.assertTrue; - -/** - * Tests for {@link InspectContainerResponse}. - * - * @author Oleg Nenashev - */ -public class InspectContainerResponseTest { - - @Test - public void roundTrip_full() throws IOException { - InspectContainerResponse[] responses = testRoundTrip(CommandJSONSamples.inspectContainerResponse_full, - InspectContainerResponse[].class); - assertEquals(1, responses.length); - final InspectContainerResponse response = responses[0]; - - // Check volumes: https://github.com/docker-java/docker-java/issues/211 - assertEquals(response.getVolumes().length, 2); - assertEquals(response.getVolumesRW().length, 2); - assertEquals(response.getVolumes()[1].getContainerPath(), "/bar/foo/myvol2"); - assertEquals(response.getVolumes()[1].getHostPath(), "/path2"); - assertEquals(response.getVolumesRW()[1].getVolume().getPath(), "/bar/foo/myvol2"); - assertFalse(response.getVolumesRW()[1].getAccessMode().toBoolean()); - assertTrue(response.getVolumesRW()[0].getAccessMode().toBoolean()); - } - - @Test - public void roundTrip_1_21_full() throws IOException { - InspectContainerResponse[] responses = testRoundTrip(CommandJSONSamples.inspectContainerResponse_full_1_21, - InspectContainerResponse[].class); - assertEquals(1, responses.length); - final InspectContainerResponse response = responses[0]; - final InspectContainerResponse.ContainerState state = response.getState(); - assertThat(state, not(nullValue())); - - assertFalse(state.getDead()); - assertThat(state.getStatus(), containsString("running")); - assertFalse(state.getRestarting()); - assertFalse(state.getOOMKilled()); - assertThat(state.getError(), isEmptyString()); - } - - @Test - public void roundTrip_empty() throws IOException { - testRoundTrip(CommandJSONSamples.inspectContainerResponse_empty, InspectContainerResponse[].class); - } -} diff --git a/src/test/java/com/github/dockerjava/api/command/InspectImageResponseTest.java b/src/test/java/com/github/dockerjava/api/command/InspectImageResponseTest.java deleted file mode 100644 index 4c604e6a0..000000000 --- a/src/test/java/com/github/dockerjava/api/command/InspectImageResponseTest.java +++ /dev/null @@ -1,209 +0,0 @@ -package com.github.dockerjava.api.command; - -import com.fasterxml.jackson.databind.JavaType; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.api.model.ContainerConfig; -import org.testng.annotations.Test; - -import java.io.IOException; -import java.util.Collections; - -import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_22; -import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.contains; -import static org.hamcrest.Matchers.containsInAnyOrder; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.hasSize; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.notNullValue; -import static org.hamcrest.Matchers.nullValue; - -/** - * @author Kanstantsin Shautsou - */ -public class InspectImageResponseTest { - @Test - public void serder1_22Json() throws IOException { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().constructType(InspectImageResponse.class); - - final InspectImageResponse inspectImage = testRoundTrip(VERSION_1_22, - "images/image1/inspect1.json", - type - ); - - final ContainerConfig config = new ContainerConfig() - .withAttachStderr(false) - .withAttachStdin(false) - .withAttachStdout(false) - .withCmd(null) - .withDomainName("") - .withEntrypoint(null) - .withEnv(new String[]{"HOME=/", "PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin"}) - .withExposedPorts(null) - .withHostName("aee9ba801acc") - .withImage("511136ea3c5a64f264b78b5433614aec563103b4d4702f3ba7d4d2698e22c158") - .withLabels(null) - .withMacAddress(null) - .withNetworkDisabled(null) - .withOnBuild(new String[]{}) - .withStdinOpen(false) - .withPortSpecs(null) - .withStdInOnce(false) - .withTty(false) - .withUser("") - .withVolumes(null) - .withWorkingDir(""); - - final ContainerConfig containerConfig = new ContainerConfig() - .withAttachStderr(false) - .withAttachStdin(false) - .withAttachStdout(false) - .withCmd(new String[]{"/bin/sh", "-c", "#(nop) MAINTAINER hack@worldticket.net"}) - .withDomainName("") - .withEntrypoint(null) - .withEnv(new String[]{"HOME=/", "PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin"}) - .withExposedPorts(null) - .withHostName("aee9ba801acc") - .withImage("511136ea3c5a64f264b78b5433614aec563103b4d4702f3ba7d4d2698e22c158") - .withLabels(null) - .withMacAddress(null) - .withNetworkDisabled(null) - .withOnBuild(new String[]{}) - .withStdinOpen(false) - .withPortSpecs(null) - .withStdInOnce(false) - .withTty(false) - .withUser("") - .withVolumes(null) - .withWorkingDir(""); - - assertThat(inspectImage, notNullValue()); - assertThat(inspectImage.getArch(), is("amd64")); - assertThat(inspectImage.getAuthor(), is("hack@worldticket.net")); - assertThat(inspectImage.getComment(), isEmptyString()); - - assertThat(inspectImage.getConfig(), notNullValue()); - assertThat(inspectImage.getConfig(), equalTo(config)); - - assertThat(inspectImage.getCreated(), is("2014-04-29T19:59:10.84997669Z")); - assertThat(inspectImage.getContainer(), is("aee9ba801acca0e648ffd91df204ba82ae85d97608a4864a019e2004d7e1b133")); - - assertThat(inspectImage.getContainerConfig(), notNullValue()); - assertThat(inspectImage.getContainerConfig(), equalTo(containerConfig)); - - assertThat(inspectImage.getDockerVersion(), is("0.8.1")); - assertThat(inspectImage.getId(), is("sha256:ee45fe0d1fcdf1a0f9c2d1e36c6f4b3202bbb2032f14d7c9312b27bfcf6aee24")); - assertThat(inspectImage.getOs(), is("linux")); - assertThat(inspectImage.getParent(), isEmptyString()); - assertThat(inspectImage.getSize(), is(0L)); - - assertThat(inspectImage.getRepoTags(), hasSize(1)); - assertThat(inspectImage.getRepoTags(), hasItem("hackmann/empty:latest")); - - final GraphDriver aufsGraphDriver = new GraphDriver().withName("aufs"); - final GraphDriver graphDriver = inspectImage.getGraphDriver(); - assertThat(graphDriver, notNullValue()); - assertThat(graphDriver, equalTo(aufsGraphDriver)); - assertThat(graphDriver.getName(), is("aufs")); - assertThat(graphDriver.getData(), nullValue()); - - assertThat(inspectImage.getVirtualSize(), is(0L)); - - - final InspectImageResponse inspectImageResponse = new InspectImageResponse().withArch("amd64") - .withAuthor("hack@worldticket.net") - .withComment("") - .withConfig(config) - .withContainer("aee9ba801acca0e648ffd91df204ba82ae85d97608a4864a019e2004d7e1b133") - .withContainerConfig(containerConfig) - .withCreated("2014-04-29T19:59:10.84997669Z") - .withDockerVersion("0.8.1") - .withId("sha256:ee45fe0d1fcdf1a0f9c2d1e36c6f4b3202bbb2032f14d7c9312b27bfcf6aee24") - .withOs("linux") - .withParent("") - .withSize(0L) - .withRepoTags(Collections.singletonList("hackmann/empty:latest")) - .withRepoDigests(Collections.emptyList()) - .withVirtualSize(0L) - .withGraphDriver(aufsGraphDriver); - - assertThat(inspectImage, equalTo(inspectImageResponse)); - } - - - @Test - public void serder1_22_doc() throws IOException { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().constructType(InspectImageResponse.class); - - final InspectImageResponse inspectImage = testRoundTrip(VERSION_1_22, - "images/docImage/doc.json", - type - ); - - assertThat(inspectImage, notNullValue()); - - assertThat(inspectImage.getRepoDigests(), hasSize(1)); - assertThat(inspectImage.getRepoDigests(), - contains("localhost:5000/test/busybox/example@" + - "sha256:cbbf2f9a99b47fc460d422812b6a5adff7dfee951d8fa2e4a98caa0382cfbdbf") - ); - - assertThat(inspectImage.getRepoTags(), hasSize(3)); - assertThat(inspectImage.getRepoTags(), containsInAnyOrder( - "example:1.0", - "example:latest", - "example:stable" - )); - } - - @Test - public void serder1_22_inspect_doc() throws IOException { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().constructType(InspectImageResponse.class); - - final InspectImageResponse inspectImage = testRoundTrip(VERSION_1_22, - "images/docImage/inspect_doc.json", - type - ); - - GraphData newGraphData = new GraphData() - .withDeviceId("5") - .withDeviceName("docker-253:1-2763198-d2cc496561d6d520cbc0236b4ba88c362c446a7619992123f11c809cded25b47") - .withDeviceSize("171798691840"); - - assertThat(inspectImage, notNullValue()); - GraphDriver graphDriver = inspectImage.getGraphDriver(); - assertThat(graphDriver, notNullValue()); - GraphData data = graphDriver.getData(); - - assertThat(data, is(newGraphData)); - - assertThat(data.getDeviceId(), is("5")); - assertThat(data.getDeviceName(), - is("docker-253:1-2763198-d2cc496561d6d520cbc0236b4ba88c362c446a7619992123f11c809cded25b47")); - assertThat(data.getDeviceSize(), - is("171798691840")); - } - - @Test - private void testOverlayNetworkRootDir() throws IOException { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().constructType(InspectImageResponse.class); - - final InspectImageResponse inspectImage = testRoundTrip(VERSION_1_22, "images/overlay/inspectOverlay.json", type); - - final GraphData overlayGraphData = new GraphData() - .withRootDir("/var/lib/docker/overlay/7e8d362d6b78d47eafe4863fd129cbcada35dbd419d7188cc1dbf1233d505576/root"); - final GraphDriver overlayGraphDriver = new GraphDriver().withName("overlay").withData(overlayGraphData); - final GraphDriver graphDriver = inspectImage.getGraphDriver(); - assertThat(graphDriver, notNullValue()); - assertThat(graphDriver, equalTo(overlayGraphDriver)); - assertThat(graphDriver.getName(), is("overlay")); - assertThat(graphDriver.getData(), equalTo(overlayGraphData)); - } -} diff --git a/src/test/java/com/github/dockerjava/api/model/AccessModeTest.java b/src/test/java/com/github/dockerjava/api/model/AccessModeTest.java deleted file mode 100644 index bdbc2e225..000000000 --- a/src/test/java/com/github/dockerjava/api/model/AccessModeTest.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.github.dockerjava.api.model; - -import static com.github.dockerjava.api.model.AccessMode.rw; -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -public class AccessModeTest { - - @Test - public void defaultAccessMode() { - assertEquals(AccessMode.DEFAULT, rw); - } - - @Test - public void stringify() { - assertEquals(AccessMode.rw.toString(), "rw"); - } - - @Test - public void fromString() { - assertEquals(AccessMode.valueOf("rw"), rw); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "No enum const.*") - public void fromIllegalString() { - AccessMode.valueOf("xx"); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/AuthConfigTest.java b/src/test/java/com/github/dockerjava/api/model/AuthConfigTest.java deleted file mode 100644 index db1553e33..000000000 --- a/src/test/java/com/github/dockerjava/api/model/AuthConfigTest.java +++ /dev/null @@ -1,64 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.databind.JavaType; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.core.RemoteApiVersion; -import org.testng.annotations.Test; - -import java.io.IOException; - -import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; -import static org.hamcrest.CoreMatchers.equalTo; -import static org.hamcrest.CoreMatchers.is; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.notNullValue; -import static org.testng.Assert.assertEquals; - -public class AuthConfigTest { - - @Test - public void defaultServerAddress() throws Exception { - assertEquals(new AuthConfig().getRegistryAddress(), "https://index.docker.io/v1/"); - } - - @Test - public void serderDocs1() throws IOException { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().uncheckedSimpleType(AuthConfig.class); - - final AuthConfig authConfig = testRoundTrip(RemoteApiVersion.VERSION_1_22, - "/other/AuthConfig/docs1.json", - type - ); - - assertThat(authConfig, notNullValue()); - assertThat(authConfig.getUsername(), is("jdoe")); - assertThat(authConfig.getPassword(), is("secret")); - assertThat(authConfig.getEmail(), is("jdoe@acme.com")); - - final AuthConfig authConfig1 = new AuthConfig().withUsername("jdoe") - .withPassword("secret") - .withEmail("jdoe@acme.com"); - - assertThat(authConfig1, equalTo(authConfig)); - } - - @Test - public void serderDocs2() throws IOException { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().uncheckedSimpleType(AuthConfig.class); - - final AuthConfig authConfig = testRoundTrip(RemoteApiVersion.VERSION_1_22, - "/other/AuthConfig/docs2.json", - type - ); - - assertThat(authConfig, notNullValue()); - assertThat(authConfig.getRegistrytoken(), is("9cbaf023786cd7...")); - - - final AuthConfig authConfig1 = new AuthConfig().withRegistrytoken("9cbaf023786cd7..."); - - assertThat(authConfig1, equalTo(authConfig)); - } -} diff --git a/src/test/java/com/github/dockerjava/api/model/BindTest.java b/src/test/java/com/github/dockerjava/api/model/BindTest.java deleted file mode 100644 index 3204ee716..000000000 --- a/src/test/java/com/github/dockerjava/api/model/BindTest.java +++ /dev/null @@ -1,117 +0,0 @@ -package com.github.dockerjava.api.model; - -import static com.github.dockerjava.api.model.AccessMode.ro; -import static com.github.dockerjava.api.model.AccessMode.rw; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.core.Is.is; - -import org.testng.annotations.Test; - -public class BindTest { - - @Test - public void parseUsingDefaultAccessMode() { - Bind bind = Bind.parse("/host:/container"); - assertThat(bind.getPath(), is("/host")); - assertThat(bind.getVolume().getPath(), is("/container")); - assertThat(bind.getAccessMode(), is(AccessMode.DEFAULT)); - assertThat(bind.getSecMode(), is(SELContext.none)); - } - - @Test - public void parseReadWrite() { - Bind bind = Bind.parse("/host:/container:rw"); - assertThat(bind.getPath(), is("/host")); - assertThat(bind.getVolume().getPath(), is("/container")); - assertThat(bind.getAccessMode(), is(rw)); - assertThat(bind.getSecMode(), is(SELContext.none)); - } - - @Test - public void parseReadOnly() { - Bind bind = Bind.parse("/host:/container:ro"); - assertThat(bind.getPath(), is("/host")); - assertThat(bind.getVolume().getPath(), is("/container")); - assertThat(bind.getAccessMode(), is(ro)); - assertThat(bind.getSecMode(), is(SELContext.none)); - } - - @Test - public void parseSELOnly() { - Bind bind = Bind.parse("/host:/container:Z"); - assertThat(bind.getPath(), is("/host")); - assertThat(bind.getVolume().getPath(), is("/container")); - assertThat(bind.getAccessMode(), is(AccessMode.DEFAULT)); - assertThat(bind.getSecMode(), is(SELContext.single)); - - bind = Bind.parse("/host:/container:z"); - assertThat(bind.getPath(), is("/host")); - assertThat(bind.getVolume().getPath(), is("/container")); - assertThat(bind.getAccessMode(), is(AccessMode.DEFAULT)); - assertThat(bind.getSecMode(), is(SELContext.shared)); - } - - @Test - public void parseReadWriteSEL() { - Bind bind = Bind.parse("/host:/container:rw,Z"); - assertThat(bind.getPath(), is("/host")); - assertThat(bind.getVolume().getPath(), is("/container")); - assertThat(bind.getAccessMode(), is(rw)); - assertThat(bind.getSecMode(), is(SELContext.single)); - } - - @Test - public void parseReadOnlySEL() { - Bind bind = Bind.parse("/host:/container:ro,z"); - assertThat(bind.getPath(), is("/host")); - assertThat(bind.getVolume().getPath(), is("/container")); - assertThat(bind.getAccessMode(), is(ro)); - assertThat(bind.getSecMode(), is(SELContext.shared)); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing Bind.*") - public void parseInvalidAccessMode() { - Bind.parse("/host:/container:xx"); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing Bind 'nonsense'") - public void parseInvalidInput() { - Bind.parse("nonsense"); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing Bind 'null'") - public void parseNull() { - Bind.parse(null); - } - - @Test - public void toStringReadOnly() { - assertThat(Bind.parse("/host:/container:ro").toString(), is("/host:/container:ro")); - } - - @Test - public void toStringReadWrite() { - assertThat(Bind.parse("/host:/container:rw").toString(), is("/host:/container:rw")); - } - - @Test - public void toStringDefaultAccessMode() { - assertThat(Bind.parse("/host:/container").toString(), is("/host:/container:rw")); - } - - @Test - public void toStringReadOnlySEL() { - assertThat(Bind.parse("/host:/container:ro,Z").toString(), is("/host:/container:ro,Z")); - } - - @Test - public void toStringReadWriteSEL() { - assertThat(Bind.parse("/host:/container:rw,z").toString(), is("/host:/container:rw,z")); - } - - @Test - public void toStringDefaultSEL() { - assertThat(Bind.parse("/host:/container:Z").toString(), is("/host:/container:rw,Z")); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/BindingTest.java b/src/test/java/com/github/dockerjava/api/model/BindingTest.java deleted file mode 100644 index 3b5ea2f6a..000000000 --- a/src/test/java/com/github/dockerjava/api/model/BindingTest.java +++ /dev/null @@ -1,57 +0,0 @@ -package com.github.dockerjava.api.model; - -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -import com.github.dockerjava.api.model.Ports.Binding; - -public class BindingTest { - - @Test - public void parseIpAndPort() { - assertEquals(Binding.parse("127.0.0.1:80"), Binding.bindIpAndPort("127.0.0.1", 80)); - } - - @Test - public void parsePortOnly() { - assertEquals(Binding.parse("80"), Binding.bindPort(80)); - } - - @Test - public void parseIPOnly() { - assertEquals(Binding.parse("127.0.0.1"), Binding.bindIp("127.0.0.1")); - } - - @Test - public void parseEmptyString() { - assertEquals(Binding.parse(""), Binding.empty()); - } - - // Strings can be used since it can be a range. Let the docker daemon do the validation. - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing Binding 'nonsense'", enabled = false) - public void parseInvalidInput() { - Binding.parse("nonsense"); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing Binding 'null'") - public void parseNull() { - Binding.parse(null); - } - - @Test - public void toStringIpAndHost() { - assertEquals(Binding.parse("127.0.0.1:80").toString(), "127.0.0.1:80"); - } - - @Test - public void toStringPortOnly() { - assertEquals(Binding.parse("80").toString(), "80"); - } - - @Test - public void toStringIpOnly() { - assertEquals(Binding.parse("127.0.0.1").toString(), "127.0.0.1"); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/CapabilityTest.java b/src/test/java/com/github/dockerjava/api/model/CapabilityTest.java deleted file mode 100644 index e45ad2e8e..000000000 --- a/src/test/java/com/github/dockerjava/api/model/CapabilityTest.java +++ /dev/null @@ -1,29 +0,0 @@ -package com.github.dockerjava.api.model; - -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -import com.fasterxml.jackson.databind.JsonMappingException; -import com.fasterxml.jackson.databind.ObjectMapper; - -public class CapabilityTest { - private final ObjectMapper objectMapper = new ObjectMapper(); - - @Test - public void serializeCapability() throws Exception { - String json = objectMapper.writeValueAsString(Capability.ALL); - assertEquals(json, "\"ALL\""); - } - - @Test - public void deserializeCapability() throws Exception { - Capability capability = objectMapper.readValue("\"ALL\"", Capability.class); - assertEquals(capability, Capability.ALL); - } - - @Test(expectedExceptions = JsonMappingException.class) - public void deserializeInvalidCapability() throws Exception { - objectMapper.readValue("\"nonsense\"", Capability.class); - } -} diff --git a/src/test/java/com/github/dockerjava/api/model/ExposedPortTest.java b/src/test/java/com/github/dockerjava/api/model/ExposedPortTest.java deleted file mode 100644 index 086613045..000000000 --- a/src/test/java/com/github/dockerjava/api/model/ExposedPortTest.java +++ /dev/null @@ -1,38 +0,0 @@ -package com.github.dockerjava.api.model; - -import static com.github.dockerjava.api.model.InternetProtocol.DEFAULT; -import static com.github.dockerjava.api.model.InternetProtocol.TCP; -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -public class ExposedPortTest { - - @Test - public void parsePortAndProtocol() { - ExposedPort exposedPort = ExposedPort.parse("80/tcp"); - assertEquals(exposedPort, new ExposedPort(80, TCP)); - } - - @Test - public void parsePortOnly() { - ExposedPort exposedPort = ExposedPort.parse("80"); - assertEquals(exposedPort, new ExposedPort(80, DEFAULT)); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing ExposedPort 'nonsense'") - public void parseInvalidInput() { - ExposedPort.parse("nonsense"); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing ExposedPort 'null'") - public void parseNull() { - ExposedPort.parse(null); - } - - @Test - public void stringify() { - assertEquals(ExposedPort.parse("80/tcp").toString(), "80/tcp"); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/IdentifierTest.java b/src/test/java/com/github/dockerjava/api/model/IdentifierTest.java deleted file mode 100644 index eb458ab4c..000000000 --- a/src/test/java/com/github/dockerjava/api/model/IdentifierTest.java +++ /dev/null @@ -1,42 +0,0 @@ -package com.github.dockerjava.api.model; - - -import org.testng.annotations.Test; - -import static org.testng.Assert.assertEquals; -import static org.testng.AssertJUnit.assertTrue; - -public class IdentifierTest { - - @Test - public void testFromCompoundString() throws Exception { - - Identifier i1 = Identifier.fromCompoundString("10.0.0.1/jim"); - Identifier i2 = Identifier.fromCompoundString("10.0.0.1/jim:123"); - Identifier i3 = Identifier.fromCompoundString("10.0.0.1:123/jim:124"); - Identifier i3A = Identifier.fromCompoundString("10.0.0.1:123/jim:latest"); - - assertTrue(!i1.tag.isPresent()); - assertEquals(i1.repository.name, "10.0.0.1/jim"); - - assertTrue(i2.tag.isPresent()); - assertEquals(i2.tag.get(), "123"); - assertEquals(i2.repository.name, "10.0.0.1/jim"); - - assertTrue(i3.tag.isPresent()); - assertEquals(i3.tag.get(), "124"); - assertEquals(i3.repository.name, "10.0.0.1:123/jim"); - assertEquals(i3.repository.getURL().getPort(), 123); - assertEquals(i3A.tag.get(), "latest"); - - Identifier i4 = Identifier.fromCompoundString("centos:latest"); - assertTrue(i4.tag.isPresent()); - assertEquals(i4.tag.get(), "latest"); - - Identifier i5 = Identifier.fromCompoundString("busybox"); - assertTrue(!i5.tag.isPresent()); - - Identifier i6 = Identifier.fromCompoundString("10.0.0.1:5000/my-test-image:1234"); - assertEquals(i6.repository.getPath(), "my-test-image"); - } -} diff --git a/src/test/java/com/github/dockerjava/api/model/InternetProtocolTest.java b/src/test/java/com/github/dockerjava/api/model/InternetProtocolTest.java deleted file mode 100644 index e3e8ce17a..000000000 --- a/src/test/java/com/github/dockerjava/api/model/InternetProtocolTest.java +++ /dev/null @@ -1,40 +0,0 @@ -package com.github.dockerjava.api.model; - -import static com.github.dockerjava.api.model.InternetProtocol.TCP; -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -public class InternetProtocolTest { - - @Test - public void defaultProtocol() { - assertEquals(InternetProtocol.DEFAULT, TCP); - } - - @Test - public void stringify() { - assertEquals(TCP.toString(), "tcp"); - } - - @Test - public void parseUpperCase() { - assertEquals(InternetProtocol.parse("TCP"), TCP); - } - - @Test - public void parseLowerCase() { - assertEquals(InternetProtocol.parse("tcp"), TCP); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing Protocol.*") - public void parseInvalidInput() { - InternetProtocol.parse("xx"); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing Protocol 'null'") - public void parseNull() { - InternetProtocol.parse(null); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/LinkTest.java b/src/test/java/com/github/dockerjava/api/model/LinkTest.java deleted file mode 100644 index 208e9f363..000000000 --- a/src/test/java/com/github/dockerjava/api/model/LinkTest.java +++ /dev/null @@ -1,38 +0,0 @@ -package com.github.dockerjava.api.model; - -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -public class LinkTest { - - @Test - public void parse() { - Link link = Link.parse("name:alias"); - assertEquals(link.getName(), "name"); - assertEquals(link.getAlias(), "alias"); - } - - @Test - public void parseWithContainerNames() { - Link link = Link.parse("/name:/conatiner/alias"); - assertEquals(link.getName(), "name"); - assertEquals(link.getAlias(), "alias"); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing Link 'nonsense'") - public void parseInvalidInput() { - Link.parse("nonsense"); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing Link 'null'") - public void parseNull() { - Link.parse(null); - } - - @Test - public void stringify() { - assertEquals(Link.parse("name:alias").toString(), "name:alias"); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/PortBindingTest.java b/src/test/java/com/github/dockerjava/api/model/PortBindingTest.java deleted file mode 100644 index 7603c3f1b..000000000 --- a/src/test/java/com/github/dockerjava/api/model/PortBindingTest.java +++ /dev/null @@ -1,54 +0,0 @@ -package com.github.dockerjava.api.model; - -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -import com.github.dockerjava.api.model.Ports.Binding; - -public class PortBindingTest { - - private static final ExposedPort TCP_8080 = ExposedPort.tcp(8080); - - @Test - public void fullDefinition() { - assertEquals(PortBinding.parse("127.0.0.1:80:8080/tcp"), - new PortBinding(Binding.bindIpAndPort("127.0.0.1", 80), TCP_8080)); - } - - @Test - public void noProtocol() { - assertEquals(PortBinding.parse("127.0.0.1:80:8080"), new PortBinding(Binding.bindIpAndPort("127.0.0.1", 80), TCP_8080)); - } - - @Test - public void noHostIp() { - assertEquals(PortBinding.parse("80:8080/tcp"), new PortBinding(Binding.bindPort(80), TCP_8080)); - } - - @Test - public void portsOnly() { - assertEquals(PortBinding.parse("80:8080"), new PortBinding(Binding.bindPort(80), TCP_8080)); - } - - @Test - public void exposedPortOnly() { - assertEquals(PortBinding.parse("8080"), new PortBinding(Binding.empty(), TCP_8080)); - } - - @Test - public void dynamicHostPort() { - assertEquals(PortBinding.parse("127.0.0.1::8080"), new PortBinding(Binding.bindIp("127.0.0.1"), TCP_8080)); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing PortBinding 'nonsense'", enabled = false) - public void parseInvalidInput() { - PortBinding.parse("nonsense"); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing PortBinding 'null'") - public void parseNull() { - PortBinding.parse(null); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/Ports_SerializingTest.java b/src/test/java/com/github/dockerjava/api/model/Ports_SerializingTest.java deleted file mode 100644 index fb89561e5..000000000 --- a/src/test/java/com/github/dockerjava/api/model/Ports_SerializingTest.java +++ /dev/null @@ -1,59 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.api.model.Ports.Binding; -import org.testng.annotations.Test; - -import java.util.Map; - -import static org.testng.Assert.assertEquals; - -public class Ports_SerializingTest { - private final ObjectMapper objectMapper = new ObjectMapper(); - - private final String jsonWithDoubleBindingForOnePort = "{\"80/tcp\":[{\"HostIp\":\"10.0.0.1\",\"HostPort\":\"80\"},{\"HostIp\":\"10.0.0.2\",\"HostPort\":\"80\"}]}"; - - private final String jsonWithNullBindingForOnePort = "{\"80/tcp\":null}"; - - @Test - public void deserializingPortWithMultipleBindings() throws Exception { - Ports ports = objectMapper.readValue(jsonWithDoubleBindingForOnePort, Ports.class); - Map map = ports.getBindings(); - assertEquals(map.size(), 1); - - Binding[] bindings = map.get(ExposedPort.tcp(80)); - assertEquals(bindings.length, 2); - assertEquals(bindings[0], new Binding("10.0.0.1", "80")); - assertEquals(bindings[1], new Binding("10.0.0.2", "80")); - } - - @Test - public void serializingPortWithMultipleBindings() throws Exception { - Ports ports = new Ports(); - ports.bind(ExposedPort.tcp(80), new Binding("10.0.0.1", "80")); - ports.bind(ExposedPort.tcp(80), new Binding("10.0.0.2", "80")); - assertEquals(objectMapper.writeValueAsString(ports), jsonWithDoubleBindingForOnePort); - } - - @Test - public void serializingEmptyBinding() throws Exception { - Ports ports = new Ports(ExposedPort.tcp(80), new Binding(null, null)); - assertEquals(objectMapper.writeValueAsString(ports), "{\"80/tcp\":[{\"HostIp\":\"\",\"HostPort\":\"\"}]}"); - } - - @Test - public void deserializingPortWithNullBindings() throws Exception { - Ports ports = objectMapper.readValue(jsonWithNullBindingForOnePort, Ports.class); - Map map = ports.getBindings(); - assertEquals(map.size(), 1); - - assertEquals(map.get(ExposedPort.tcp(80)), null); - } - - @Test - public void serializingWithNullBindings() throws Exception { - Ports ports = new Ports(); - ports.bind(ExposedPort.tcp(80), null); - assertEquals(objectMapper.writeValueAsString(ports), jsonWithNullBindingForOnePort); - } -} diff --git a/src/test/java/com/github/dockerjava/api/model/Ports_addBindingsTest.java b/src/test/java/com/github/dockerjava/api/model/Ports_addBindingsTest.java deleted file mode 100644 index 26e815f3b..000000000 --- a/src/test/java/com/github/dockerjava/api/model/Ports_addBindingsTest.java +++ /dev/null @@ -1,60 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.github.dockerjava.api.model.Ports.Binding; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.util.Map; - -import static org.testng.Assert.assertEquals; - -/** - * As there may be several {@link Binding}s per {@link ExposedPort}, it makes a difference if you add {@link PortBinding}s for the same or - * different {@link ExposedPort}s to {@link Ports}. This test verifies that the Map in {@link Ports} is populated correctly in both cases. - */ -public class Ports_addBindingsTest { - private static final ExposedPort TCP_80 = ExposedPort.tcp(80); - - private static final ExposedPort TCP_90 = ExposedPort.tcp(90); - - private static final Binding BINDING_8080 = Binding.bindPort(8080); - - private static final Binding BINDING_9090 = Binding.bindPort(9090); - - private Ports ports; - - @BeforeMethod - public void setup() { - ports = new Ports(); - } - - @Test - public void addTwoBindingsForDifferentExposedPorts() { - ports.add(new PortBinding(BINDING_8080, TCP_80), new PortBinding(BINDING_9090, TCP_90)); - - Map bindings = ports.getBindings(); - // two keys with one value each - assertEquals(bindings.size(), 2); - assertEquals(bindings.get(TCP_80), new Binding[] {BINDING_8080}); - assertEquals(bindings.get(TCP_90), new Binding[] {BINDING_9090}); - } - - @Test - public void addTwoBindingsForSameExposedPort() { - ports.add(new PortBinding(BINDING_8080, TCP_80), new PortBinding(BINDING_9090, TCP_80)); - - Map bindings = ports.getBindings(); - // one key with two values - assertEquals(bindings.size(), 1); - assertEquals(bindings.get(TCP_80), new Binding[] {BINDING_8080, BINDING_9090}); - } - - @Test - public void addNullBindings() { - ports.add(new PortBinding(null, TCP_80)); - Map bindings = ports.getBindings(); - // one key with two values - assertEquals(bindings.size(), 1); - assertEquals(bindings.get(TCP_80), null); - } -} diff --git a/src/test/java/com/github/dockerjava/api/model/RestartPolicy_ParsingTest.java b/src/test/java/com/github/dockerjava/api/model/RestartPolicy_ParsingTest.java deleted file mode 100644 index 3bc22c00e..000000000 --- a/src/test/java/com/github/dockerjava/api/model/RestartPolicy_ParsingTest.java +++ /dev/null @@ -1,38 +0,0 @@ -package com.github.dockerjava.api.model; - -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -public class RestartPolicy_ParsingTest { - - @Test - public void noRestart() throws Exception { - assertEquals(RestartPolicy.parse("no"), RestartPolicy.noRestart()); - } - - @Test - public void alwaysRestart() throws Exception { - assertEquals(RestartPolicy.parse("always"), RestartPolicy.alwaysRestart()); - } - - @Test - public void onFailureRestart() throws Exception { - assertEquals(RestartPolicy.parse("on-failure"), RestartPolicy.onFailureRestart(0)); - } - - @Test - public void onFailureRestartWithCount() throws Exception { - assertEquals(RestartPolicy.parse("on-failure:2"), RestartPolicy.onFailureRestart(2)); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing RestartPolicy 'nonsense'") - public void illegalSyntax() throws Exception { - RestartPolicy.parse("nonsense"); - } - - @Test(expectedExceptions = IllegalArgumentException.class, expectedExceptionsMessageRegExp = "Error parsing RestartPolicy 'on-failure:X'") - public void illegalRetryCount() throws Exception { - RestartPolicy.parse("on-failure:X"); - } -} diff --git a/src/test/java/com/github/dockerjava/api/model/RestartPolicy_SerializingTest.java b/src/test/java/com/github/dockerjava/api/model/RestartPolicy_SerializingTest.java deleted file mode 100644 index 74c97fcf6..000000000 --- a/src/test/java/com/github/dockerjava/api/model/RestartPolicy_SerializingTest.java +++ /dev/null @@ -1,44 +0,0 @@ -package com.github.dockerjava.api.model; - -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -import com.fasterxml.jackson.databind.ObjectMapper; - -/** - * Compares serialization results of various {@link RestartPolicy}s with what Docker (as of 1.3.3) actually sends when executing - * docker run --restart xxx. - */ -public class RestartPolicy_SerializingTest { - private final ObjectMapper objectMapper = new ObjectMapper(); - - @Test - // --restart no - public void noRestart() throws Exception { - String json = objectMapper.writeValueAsString(RestartPolicy.noRestart()); - assertEquals(json, "{\"MaximumRetryCount\":0,\"Name\":\"\"}"); - } - - @Test - // --restart always - public void alwaysRestart() throws Exception { - String json = objectMapper.writeValueAsString(RestartPolicy.alwaysRestart()); - assertEquals(json, "{\"MaximumRetryCount\":0,\"Name\":\"always\"}"); - } - - @Test - // --restart on-failure - public void onFailureRestart() throws Exception { - String json = objectMapper.writeValueAsString(RestartPolicy.onFailureRestart(0)); - assertEquals(json, "{\"MaximumRetryCount\":0,\"Name\":\"on-failure\"}"); - } - - @Test - // --restart on-failure:2 - public void onFailureRestartWithCount() throws Exception { - String json = objectMapper.writeValueAsString(RestartPolicy.onFailureRestart(2)); - assertEquals(json, "{\"MaximumRetryCount\":2,\"Name\":\"on-failure\"}"); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/RestartPolicy_toStringTest.java b/src/test/java/com/github/dockerjava/api/model/RestartPolicy_toStringTest.java deleted file mode 100644 index 69e1e88a2..000000000 --- a/src/test/java/com/github/dockerjava/api/model/RestartPolicy_toStringTest.java +++ /dev/null @@ -1,20 +0,0 @@ -package com.github.dockerjava.api.model; - -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.DataProvider; -import org.testng.annotations.Test; - -public class RestartPolicy_toStringTest { - - @DataProvider(name = "input") - public Object[][] restartPolicies() { - return new Object[][] { {"no"}, {"always"}, {"on-failure"}, {"on-failure:2"}}; - } - - @Test(dataProvider = "input") - public void serializationWithoutCount(String policy) throws Exception { - assertEquals(RestartPolicy.parse(policy).toString(), policy); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/VersionTest.java b/src/test/java/com/github/dockerjava/api/model/VersionTest.java deleted file mode 100644 index c32d93069..000000000 --- a/src/test/java/com/github/dockerjava/api/model/VersionTest.java +++ /dev/null @@ -1,40 +0,0 @@ -package com.github.dockerjava.api.model; - -import com.fasterxml.jackson.databind.JavaType; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.core.RemoteApiVersion; -import org.testng.annotations.Test; - -import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; -import static org.hamcrest.CoreMatchers.is; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.notNullValue; - -/** - * @author Kanstantsin Shautsou - */ -public class VersionTest { - - @Test - public void testSerDer1() throws Exception { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().uncheckedSimpleType(Version.class); - - final Version version = testRoundTrip(RemoteApiVersion.VERSION_1_22, - "/version/1.json", - type - ); - - assertThat(version, notNullValue()); - - assertThat(version.getVersion(), is("1.10.1")); - assertThat(version.getApiVersion(), is("1.22")); - assertThat(version.getGitCommit(), is("9e83765")); - assertThat(version.getGoVersion(), is("go1.5.3")); - assertThat(version.getOperatingSystem(), is("linux")); - assertThat(version.getArch(), is("amd64")); - assertThat(version.getKernelVersion(), is("4.1.17-boot2docker")); - assertThat(version.getBuildTime(), is("2016-02-11T20:39:58.688092588+00:00")); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/VolumeBindsTest.java b/src/test/java/com/github/dockerjava/api/model/VolumeBindsTest.java deleted file mode 100644 index 7d35c25a3..000000000 --- a/src/test/java/com/github/dockerjava/api/model/VolumeBindsTest.java +++ /dev/null @@ -1,31 +0,0 @@ -package com.github.dockerjava.api.model; - -import static org.testng.Assert.assertEquals; - -import java.io.IOException; - -import org.testng.annotations.Test; - -import com.fasterxml.jackson.databind.JsonMappingException; -import com.fasterxml.jackson.databind.ObjectMapper; - -public class VolumeBindsTest { - @Test - public void t() throws IOException { - String s = "{\"/data\":\"/some/path\"}"; - ObjectMapper objectMapper = new ObjectMapper(); - VolumeBinds volumeBinds = objectMapper.readValue(s, VolumeBinds.class); - VolumeBind[] binds = volumeBinds.getBinds(); - assertEquals(binds.length, 1); - assertEquals(binds[0].getHostPath(), "/some/path"); - assertEquals(binds[0].getContainerPath(), "/data"); - } - - @Test(expectedExceptions = JsonMappingException.class) - public void t1() throws IOException { - String s = "{\"/data\": {} }"; - ObjectMapper objectMapper = new ObjectMapper(); - objectMapper.readValue(s, VolumeBinds.class); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/VolumeFrom_SerializingTest.java b/src/test/java/com/github/dockerjava/api/model/VolumeFrom_SerializingTest.java deleted file mode 100644 index 55c7088c4..000000000 --- a/src/test/java/com/github/dockerjava/api/model/VolumeFrom_SerializingTest.java +++ /dev/null @@ -1,26 +0,0 @@ -package com.github.dockerjava.api.model; - -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -import com.fasterxml.jackson.databind.ObjectMapper; - -public class VolumeFrom_SerializingTest { - private final ObjectMapper objectMapper = new ObjectMapper(); - - private final String json = "\"container1:ro\""; - - @Test - public void deserializing() throws Exception { - VolumesFrom volumeFrom = objectMapper.readValue(json, VolumesFrom.class); - assertEquals(volumeFrom, new VolumesFrom("container1", AccessMode.ro)); - } - - @Test - public void serializing() throws Exception { - VolumesFrom volumeFrom = new VolumesFrom("container1", AccessMode.ro); - assertEquals(objectMapper.writeValueAsString(volumeFrom), json); - } - -} diff --git a/src/test/java/com/github/dockerjava/api/model/VolumeTest.java b/src/test/java/com/github/dockerjava/api/model/VolumeTest.java deleted file mode 100644 index 5b8311425..000000000 --- a/src/test/java/com/github/dockerjava/api/model/VolumeTest.java +++ /dev/null @@ -1,12 +0,0 @@ -package com.github.dockerjava.api.model; - -import static org.testng.Assert.assertEquals; - -import org.testng.annotations.Test; - -public class VolumeTest { - @Test - public void getPath() { - assertEquals(new Volume("/path").getPath(), "/path"); - } -} diff --git a/src/test/java/com/github/dockerjava/client/AbstractDockerClientTest.java b/src/test/java/com/github/dockerjava/client/AbstractDockerClientTest.java deleted file mode 100644 index e6cc00ac0..000000000 --- a/src/test/java/com/github/dockerjava/client/AbstractDockerClientTest.java +++ /dev/null @@ -1,275 +0,0 @@ -package com.github.dockerjava.client; - -import java.io.File; -import java.io.IOException; -import java.io.InputStream; -import java.io.StringWriter; -import java.lang.reflect.Method; -import java.net.DatagramSocket; -import java.net.ServerSocket; -import java.util.ArrayList; -import java.util.List; - -import org.apache.commons.io.IOUtils; -import org.apache.commons.io.LineIterator; -import org.apache.commons.lang.StringUtils; -import org.hamcrest.FeatureMatcher; -import org.hamcrest.Matcher; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.Assert; -import org.testng.ITestResult; - -import com.github.dockerjava.api.DockerClient; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse.Mount; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.api.model.Volume; -import com.github.dockerjava.core.DockerClientBuilder; -import com.github.dockerjava.core.DefaultDockerClientConfig; -import com.github.dockerjava.core.TestDockerCmdExecFactory; -import com.github.dockerjava.core.command.BuildImageResultCallback; -import com.github.dockerjava.core.command.LogContainerResultCallback; -import com.github.dockerjava.core.command.PullImageResultCallback; - -public abstract class AbstractDockerClientTest extends Assert { - - public static final Logger LOG = LoggerFactory.getLogger(AbstractDockerClientTest.class); - - protected DockerClient dockerClient; - - protected TestDockerCmdExecFactory dockerCmdExecFactory = initTestDockerCmdExecFactory(); - - protected TestDockerCmdExecFactory initTestDockerCmdExecFactory() { - return new TestDockerCmdExecFactory( - DockerClientBuilder.getDefaultDockerCmdExecFactory()); - } - - public void beforeTest() throws Exception { - - LOG.info("======================= BEFORETEST ======================="); - LOG.info("Connecting to Docker server"); - dockerClient = DockerClientBuilder.getInstance(config()) - .withDockerCmdExecFactory(dockerCmdExecFactory) - .build(); - - try { - dockerClient.inspectImageCmd("busybox").exec(); - } catch (NotFoundException e) { - LOG.info("Pulling image 'busybox'"); - // need to block until image is pulled completely - dockerClient.pullImageCmd("busybox").withTag("latest").exec(new PullImageResultCallback()).awaitSuccess(); - } - - assertNotNull(dockerClient); - LOG.info("======================= END OF BEFORETEST =======================\n\n"); - } - - private DefaultDockerClientConfig config() { - return config(null); - } - - protected DefaultDockerClientConfig config(String password) { - DefaultDockerClientConfig.Builder builder = DefaultDockerClientConfig.createDefaultConfigBuilder() - .withRegistryUrl("https://index.docker.io/v1/"); - if (password != null) { - builder = builder.withRegistryPassword(password); - } - - return builder.build(); - } - - public void afterTest() { - LOG.info("======================= END OF AFTERTEST ======================="); - } - - public void beforeMethod(Method method) { - LOG.info(String.format("################################## STARTING %s ##################################", - method.getName())); - } - - public void afterMethod(ITestResult result) { - - for (String container : dockerCmdExecFactory.getContainerNames()) { - LOG.info("Cleaning up temporary container {}", container); - - try { - dockerClient.removeContainerCmd(container).withForce(true).exec(); - } catch (DockerException ignore) { - // ignore.printStackTrace(); - } - } - - for (String image : dockerCmdExecFactory.getImageNames()) { - LOG.info("Cleaning up temporary image with {}", image); - try { - dockerClient.removeImageCmd(image).withForce(true).exec(); - } catch (DockerException ignore) { - // ignore.printStackTrace(); - } - } - - for (String volume : dockerCmdExecFactory.getVolumeNames()) { - LOG.info("Cleaning up temporary volume with {}", volume); - try { - dockerClient.removeVolumeCmd(volume).exec(); - } catch (DockerException ignore) { - // ignore.printStackTrace(); - } - } - - for (String networkId : dockerCmdExecFactory.getNetworkIds()) { - LOG.info("Cleaning up temporary network with {}", networkId); - try { - dockerClient.removeNetworkCmd(networkId).exec(); - } catch (DockerException ignore) { - // ignore.printStackTrace(); - } - } - - LOG.info("################################## END OF {} ##################################\n", result.getName()); - } - - protected String asString(InputStream response) { - return consumeAsString(response); - } - - public static String consumeAsString(InputStream response) { - - StringWriter logwriter = new StringWriter(); - - try { - LineIterator itr = IOUtils.lineIterator(response, "UTF-8"); - - while (itr.hasNext()) { - String line = itr.next(); - logwriter.write(line + (itr.hasNext() ? "\n" : "")); - LOG.info("line: " + line); - } - response.close(); - - return logwriter.toString(); - } catch (IOException e) { - throw new RuntimeException(e); - } finally { - IOUtils.closeQuietly(response); - } - } - - // UTIL - - /** - * Checks to see if a specific port is available. - * - * @param port - * the port to check for availability - */ - public static Boolean available(int port) { - if (port < 1100 || port > 60000) { - throw new IllegalArgumentException("Invalid start port: " + port); - } - - ServerSocket ss = null; - DatagramSocket ds = null; - try { - ss = new ServerSocket(port); - ss.setReuseAddress(true); - ds = new DatagramSocket(port); - ds.setReuseAddress(true); - return true; - } catch (IOException ignored) { - } finally { - if (ds != null) { - ds.close(); - } - - if (ss != null) { - try { - ss.close(); - } catch (IOException e) { - /* should not be thrown */ - } - } - } - - return false; - } - - protected MountedVolumes mountedVolumes(Matcher> subMatcher) { - return new MountedVolumes(subMatcher, "Mounted volumes", "mountedVolumes"); - } - - private static class MountedVolumes extends FeatureMatcher> { - public MountedVolumes(Matcher> subMatcher, String featureDescription, String featureName) { - super(subMatcher, featureDescription, featureName); - } - - @Override - public List featureValueOf(InspectContainerResponse item) { - List volumes = new ArrayList(); - for (Mount mount : item.getMounts()) { - volumes.add(mount.getDestination()); - } - return volumes; - } - } - - protected String containerLog(String containerId) throws Exception { - return dockerClient.logContainerCmd(containerId).withStdOut(true).exec(new LogContainerTestCallback()) - .awaitCompletion().toString(); - } - - public static class LogContainerTestCallback extends LogContainerResultCallback { - protected final StringBuffer log = new StringBuffer(); - - List collectedFrames = new ArrayList(); - - boolean collectFrames = false; - - public LogContainerTestCallback() { - this(false); - } - - public LogContainerTestCallback(boolean collectFrames) { - this.collectFrames = collectFrames; - } - - @Override - public void onNext(Frame frame) { - if(collectFrames) collectedFrames.add(frame); - log.append(new String(frame.getPayload())); - } - - @Override - public String toString() { - return log.toString(); - } - - - public List getCollectedFrames() { - return collectedFrames; - } - } - - protected String buildImage(File baseDir) throws Exception { - - return dockerClient.buildImageCmd(baseDir).withNoCache(true).exec(new BuildImageResultCallback()) - .awaitImageId(); - } - - protected Network findNetwork(List networks, String name) { - - for (Network network : networks) { - if (StringUtils.equals(network.getName(), name)) { - return network; - } - } - - fail("No network found."); - return null; - } - -} diff --git a/src/test/java/com/github/dockerjava/client/DockerClientTest.java b/src/test/java/com/github/dockerjava/client/DockerClientTest.java deleted file mode 100644 index b124fdc9a..000000000 --- a/src/test/java/com/github/dockerjava/client/DockerClientTest.java +++ /dev/null @@ -1,77 +0,0 @@ -package com.github.dockerjava.client; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; - -import java.lang.reflect.Method; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.core.command.WaitContainerResultCallback; - -/** - * Unit test for DockerClient. - * - * @author Konstantin Pelykh (kpelykh@gmail.com) - */ -@Test(groups = "integration") -public class DockerClientTest extends AbstractDockerClientTest { - public static final Logger LOG = LoggerFactory.getLogger(DockerClientTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void testRunShlex() throws DockerException { - - String[] commands = new String[] { - "true", - "echo \"The Young Descendant of Tepes & Septette for the Dead Princess\"", - "echo -n 'The Young Descendant of Tepes & Septette for the Dead Princess'", - "/bin/sh -c echo Hello World", "/bin/sh -c echo 'Hello World'", "echo 'Night of Nights'", - "true && echo 'Night of Nights'" - }; - - for (String command : commands) { - LOG.info("Running command: [{}]", command); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd(command) - .exec(); - dockerClient.startContainerCmd(container.getId()); - - int exitcode = dockerClient.waitContainerCmd(container.getId()) - .exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - - assertThat(exitcode, equalTo(0)); - } - } - -} diff --git a/src/test/java/com/github/dockerjava/core/AuthConfigFileTest.java b/src/test/java/com/github/dockerjava/core/AuthConfigFileTest.java deleted file mode 100644 index 221a926b5..000000000 --- a/src/test/java/com/github/dockerjava/core/AuthConfigFileTest.java +++ /dev/null @@ -1,95 +0,0 @@ -/** - * Copyright (C) 2014 SignalFuse, Inc. - */ -package com.github.dockerjava.core; - -import java.io.File; -import java.io.IOException; - -import org.testng.Assert; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.model.AuthConfig; - -public class AuthConfigFileTest { - - private final File FILESROOT = new File(Thread.currentThread().getContextClassLoader() - .getResource("testAuthConfigFile").getFile()); - - @Test(expectedExceptions = IOException.class, expectedExceptionsMessageRegExp = "The Auth Config file is empty") - public void emptyFile() throws IOException { - runTest("emptyFile"); - } - - @Test(expectedExceptions = IOException.class, expectedExceptionsMessageRegExp = "The Auth Config file is empty") - public void tooSmallFile() throws IOException { - runTest("tooSmallFile"); - } - - @Test(expectedExceptions = IOException.class, expectedExceptionsMessageRegExp = "Invalid auth configuration file") - public void invalidJsonInvalidAuth() throws IOException { - runTest("invalidJsonInvalidAuth"); - } - - @Test(expectedExceptions = IOException.class, expectedExceptionsMessageRegExp = "Invalid Auth config file") - public void invalidLegacyAuthLine() throws IOException { - runTest("invalidLegacyAuthLine"); - } - - @Test(expectedExceptions = IOException.class, expectedExceptionsMessageRegExp = "Invalid auth configuration file") - public void invalidLegacyInvalidAuth() throws IOException { - runTest("invalidLegacyInvalidAuth"); - } - - @Test(expectedExceptions = IOException.class, expectedExceptionsMessageRegExp = "Invalid Auth config file") - public void invalidLegacyEmailLine() throws IOException { - runTest("invalidLegacyEmailLine"); - } - - @Test - public void validJson() throws IOException { - AuthConfig authConfig1 = new AuthConfig() - .withEmail("foo@example.com") - .withUsername("foo") - .withPassword("bar") - .withRegistryAddress("quay.io"); - - AuthConfig authConfig2 = new AuthConfig() - .withEmail("moo@example.com") - .withUsername("foo1") - .withPassword("bar1") - .withRegistryAddress(AuthConfig.DEFAULT_SERVER_ADDRESS); - - AuthConfigFile expected = new AuthConfigFile(); - expected.addConfig(authConfig1); - expected.addConfig(authConfig2); - - Assert.assertEquals(runTest("validJson"), expected); - - } - - @Test - public void validLegacy() throws IOException { - AuthConfig authConfig = new AuthConfig() - .withEmail("foo@example.com") - .withUsername("foo") - .withPassword("bar") - .withRegistryAddress(AuthConfig.DEFAULT_SERVER_ADDRESS); - - AuthConfigFile expected = new AuthConfigFile(); - expected.addConfig(authConfig); - - Assert.assertEquals(runTest("validLegacy"), expected); - } - - @Test - public void nonExistent() throws IOException { - AuthConfigFile expected = new AuthConfigFile(); - Assert.assertEquals(runTest("idontexist"), expected); - } - - private AuthConfigFile runTest(String testFileName) throws IOException { - return AuthConfigFile.loadConfig(new File(FILESROOT, testFileName)); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/CompressArchiveUtilTest.java b/src/test/java/com/github/dockerjava/core/CompressArchiveUtilTest.java deleted file mode 100644 index fa4164b94..000000000 --- a/src/test/java/com/github/dockerjava/core/CompressArchiveUtilTest.java +++ /dev/null @@ -1,64 +0,0 @@ -package com.github.dockerjava.core; - -import static java.util.Arrays.asList; -import static org.hamcrest.CoreMatchers.equalTo; -import static org.hamcrest.CoreMatchers.is; -import static org.hamcrest.MatcherAssert.assertThat; - -import java.io.BufferedInputStream; -import java.io.File; -import java.io.FileInputStream; -import java.io.FileOutputStream; -import java.io.IOException; -import java.util.zip.GZIPInputStream; - -import com.github.dockerjava.core.util.CompressArchiveUtil; -import org.apache.commons.compress.archivers.tar.TarArchiveEntry; -import org.apache.commons.compress.archivers.tar.TarArchiveInputStream; -import org.apache.commons.io.FileUtils; -import org.apache.commons.io.IOUtils; -import org.testng.annotations.Test; - -public class CompressArchiveUtilTest { - - @Test - public void testExecutableFlagIsPreserved() throws Exception { - File executableFile = createExecutableFile(); - File archive = CompressArchiveUtil.archiveTARFiles(executableFile.getParentFile(), asList(executableFile), - "archive"); - File expectedFile = extractFileByName(archive, "executableFile.sh.result"); - - assertThat("should be executable", expectedFile.canExecute()); - } - - private File createExecutableFile() throws IOException { - File baseDir = new File(FileUtils.getTempDirectoryPath()); - File executableFile = new File(baseDir, "executableFile.sh"); - executableFile.createNewFile(); - executableFile.setExecutable(true); - assertThat(executableFile.canExecute(), is(true)); - return executableFile; - } - - private File extractFileByName(File archive, String filenameToExtract) throws IOException { - File baseDir = new File(FileUtils.getTempDirectoryPath()); - File expectedFile = new File(baseDir, filenameToExtract); - expectedFile.delete(); - assertThat(expectedFile.exists(), is(false)); - - TarArchiveInputStream tarArchiveInputStream = new TarArchiveInputStream(new GZIPInputStream( - new BufferedInputStream(new FileInputStream(archive)))); - TarArchiveEntry entry; - while ((entry = tarArchiveInputStream.getNextTarEntry()) != null) { - String individualFiles = entry.getName(); - // there should be only one file in this archive - assertThat(individualFiles, equalTo("executableFile.sh")); - IOUtils.copy(tarArchiveInputStream, new FileOutputStream(expectedFile)); - if ((entry.getMode() & 0755) == 0755) { - expectedFile.setExecutable(true); - } - } - tarArchiveInputStream.close(); - return expectedFile; - } -} diff --git a/src/test/java/com/github/dockerjava/core/DefaultDockerClientConfigTest.java b/src/test/java/com/github/dockerjava/core/DefaultDockerClientConfigTest.java deleted file mode 100644 index 54e70c511..000000000 --- a/src/test/java/com/github/dockerjava/core/DefaultDockerClientConfigTest.java +++ /dev/null @@ -1,204 +0,0 @@ -package com.github.dockerjava.core; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.core.Is.is; -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertNull; - -import java.lang.reflect.Field; -import java.net.URI; -import java.util.Collections; -import java.util.HashMap; -import java.util.Map; -import java.util.Properties; - -import org.apache.commons.lang.SerializationUtils; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.model.AuthConfig; - -public class DefaultDockerClientConfigTest { - - public static final DefaultDockerClientConfig EXAMPLE_CONFIG = newExampleConfig(); - - private static DefaultDockerClientConfig newExampleConfig() { - - String dockerCertPath = dockerCertPath(); - - return new DefaultDockerClientConfig(URI.create("tcp://foo"), "dockerConfig", "apiVersion", "registryUrl", "registryUsername", "registryPassword", "registryEmail", - new LocalDirectorySSLConfig(dockerCertPath)); - } - - private static String homeDir() { - return "target/test-classes/someHomeDir"; - } - - private static String dockerCertPath() { - return homeDir() + "/.docker"; - } - - @Test - public void equals() throws Exception { - assertEquals(EXAMPLE_CONFIG, newExampleConfig()); - } - - @Test - public void environmentDockerHost() throws Exception { - - // given docker host in env - Map env = new HashMap(); - env.put(DefaultDockerClientConfig.DOCKER_HOST, "tcp://baz:8768"); - // and it looks to be SSL disabled - env.remove("DOCKER_CERT_PATH"); - - // given default cert path - Properties systemProperties = new Properties(); - systemProperties.setProperty("user.name", "someUserName"); - systemProperties.setProperty("user.home", homeDir()); - - // when you build a config - DefaultDockerClientConfig config = buildConfig(env, systemProperties); - - assertEquals(config.getDockerHost(), URI.create("tcp://baz:8768")); - } - - @Test - public void environment() throws Exception { - - // given a default config in env properties - Map env = new HashMap(); - env.put(DefaultDockerClientConfig.DOCKER_HOST, "tcp://foo"); - env.put(DefaultDockerClientConfig.API_VERSION, "apiVersion"); - env.put(DefaultDockerClientConfig.REGISTRY_USERNAME, "registryUsername"); - env.put(DefaultDockerClientConfig.REGISTRY_PASSWORD, "registryPassword"); - env.put(DefaultDockerClientConfig.REGISTRY_EMAIL, "registryEmail"); - env.put(DefaultDockerClientConfig.REGISTRY_URL, "registryUrl"); - env.put(DefaultDockerClientConfig.DOCKER_CONFIG, "dockerConfig"); - env.put(DefaultDockerClientConfig.DOCKER_CERT_PATH, dockerCertPath()); - env.put(DefaultDockerClientConfig.DOCKER_TLS_VERIFY, "1"); - - // when you build a config - DefaultDockerClientConfig config = buildConfig(env, new Properties()); - - // then we get the example object - assertEquals(config, EXAMPLE_CONFIG); - } - - private DefaultDockerClientConfig buildConfig(Map env, Properties systemProperties) { - return DefaultDockerClientConfig.createDefaultConfigBuilder(env, systemProperties).build(); - } - - @Test - public void defaults() throws Exception { - - // given default cert path - Properties systemProperties = new Properties(); - systemProperties.setProperty("user.name", "someUserName"); - systemProperties.setProperty("user.home", homeDir()); - - // when you build config - DefaultDockerClientConfig config = buildConfig(Collections. emptyMap(), systemProperties); - - // then the cert path is as expected - assertEquals(config.getDockerHost(), URI.create("unix:///var/run/docker.sock")); - assertEquals(config.getRegistryUsername(), "someUserName"); - assertEquals(config.getRegistryUrl(), AuthConfig.DEFAULT_SERVER_ADDRESS); - assertEquals(config.getApiVersion(), RemoteApiVersion.unknown()); - assertEquals(config.getDockerConfig(), homeDir() + "/.docker"); - assertNull(config.getSSLConfig()); - } - - @Test - public void systemProperties() throws Exception { - - // given system properties based on the example - Properties systemProperties = new Properties(); - systemProperties.put(DefaultDockerClientConfig.DOCKER_HOST, "tcp://foo"); - systemProperties.put(DefaultDockerClientConfig.API_VERSION, "apiVersion"); - systemProperties.put(DefaultDockerClientConfig.REGISTRY_USERNAME, "registryUsername"); - systemProperties.put(DefaultDockerClientConfig.REGISTRY_PASSWORD, "registryPassword"); - systemProperties.put(DefaultDockerClientConfig.REGISTRY_EMAIL, "registryEmail"); - systemProperties.put(DefaultDockerClientConfig.REGISTRY_URL, "registryUrl"); - systemProperties.put(DefaultDockerClientConfig.DOCKER_CONFIG, "dockerConfig"); - systemProperties.put(DefaultDockerClientConfig.DOCKER_CERT_PATH, dockerCertPath()); - systemProperties.put(DefaultDockerClientConfig.DOCKER_TLS_VERIFY, "1"); - - // when you build new config - DefaultDockerClientConfig config = buildConfig(Collections. emptyMap(), systemProperties); - - // then it is the same as the example - assertEquals(config, EXAMPLE_CONFIG); - - } - - @Test - public void serializableTest() { - final byte[] serialized = SerializationUtils.serialize(EXAMPLE_CONFIG); - final DefaultDockerClientConfig deserialized = (DefaultDockerClientConfig) SerializationUtils.deserialize(serialized); - - assertThat("Deserialized object mush match source object", deserialized, equalTo(EXAMPLE_CONFIG)); - } - - @Test() - public void testSslContextEmpty() throws Exception { - new DefaultDockerClientConfig(URI.create("tcp://foo"), "dockerConfig", "apiVersion", "registryUrl", "registryUsername", "registryPassword", "registryEmail", - null); - } - - - - @Test() - public void testTlsVerifyAndCertPath() throws Exception { - new DefaultDockerClientConfig(URI.create("tcp://foo"), "dockerConfig", "apiVersion", "registryUrl", "registryUsername", "registryPassword", "registryEmail", - new LocalDirectorySSLConfig(dockerCertPath())); - } - - @Test(expectedExceptions = DockerClientException.class) - public void testWrongHostScheme() throws Exception { - new DefaultDockerClientConfig(URI.create("http://foo"), "dockerConfig", "apiVersion", "registryUrl", "registryUsername", "registryPassword", "registryEmail", - null); - } - - @Test() - public void testTcpHostScheme() throws Exception { - new DefaultDockerClientConfig(URI.create("tcp://foo"), "dockerConfig", "apiVersion", "registryUrl", "registryUsername", "registryPassword", "registryEmail", - null); - } - - @Test() - public void testUnixHostScheme() throws Exception { - new DefaultDockerClientConfig(URI.create("unix://foo"), "dockerConfig", "apiVersion", "registryUrl", "registryUsername", "registryPassword", "registryEmail", - null); - } - - @Test - public void withDockerTlsVerify() throws Exception { - DefaultDockerClientConfig.Builder builder = new DefaultDockerClientConfig.Builder(); - Field field = builder.getClass().getDeclaredField("dockerTlsVerify"); - field.setAccessible(true); - - builder.withDockerTlsVerify(""); - assertThat((Boolean) field.get(builder), is(false)); - - builder.withDockerTlsVerify("false"); - assertThat((Boolean) field.get(builder), is(false)); - - builder.withDockerTlsVerify("FALSE"); - assertThat((Boolean) field.get(builder), is(false)); - - builder.withDockerTlsVerify("true"); - assertThat((Boolean) field.get(builder), is(true)); - - builder.withDockerTlsVerify("TRUE"); - assertThat((Boolean) field.get(builder), is(true)); - - builder.withDockerTlsVerify("0"); - assertThat((Boolean) field.get(builder), is(false)); - - builder.withDockerTlsVerify("1"); - assertThat((Boolean) field.get(builder), is(true)); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/DockerClientBuilderTest.java b/src/test/java/com/github/dockerjava/core/DockerClientBuilderTest.java deleted file mode 100644 index 45623c5b5..000000000 --- a/src/test/java/com/github/dockerjava/core/DockerClientBuilderTest.java +++ /dev/null @@ -1,82 +0,0 @@ -package com.github.dockerjava.core; - -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertFalse; -import static org.testng.Assert.assertNotNull; - -import java.util.ArrayList; -import java.util.Collections; -import java.util.HashSet; -import java.util.List; -import java.util.Set; - -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.DockerCmdExecFactory; - -public class DockerClientBuilderTest { - // Amount of instances created in test - private static final int AMOUNT = 100; - - @Test - public void testConcurrentClientBuilding() throws Exception { - // we use it to check instance uniqueness - final Set instances = Collections.synchronizedSet(new HashSet()); - - Runnable runnable = new Runnable() { - @Override - public void run() { - DockerCmdExecFactory factory = DockerClientBuilder.getDefaultDockerCmdExecFactory(); - // factory created - assertNotNull(factory); - // and is unique - assertFalse(instances.contains(factory)); - instances.add(factory); - } - }; - - parallel(AMOUNT, runnable); - // set contains all required unique instances - assertEquals(instances.size(), AMOUNT); - } - - public static void parallel(int threads, final Runnable task) throws Exception { - final ExceptionListener exceptionListener = new ExceptionListener(); - Runnable runnable = new Runnable() { - @Override - public void run() { - try { - task.run(); - } catch (Throwable e) { - exceptionListener.onException(e); - } - } - }; - - List threadList = new ArrayList<>(threads); - for (int i = 0; i < threads; i++) { - Thread thread = new Thread(runnable); - thread.start(); - threadList.add(thread); - } - for (Thread thread : threadList) { - thread.join(); - } - Throwable exception = exceptionListener.getException(); - if (exception != null) { - throw new RuntimeException(exception); - } - } - - private static class ExceptionListener { - private Throwable exception; - - private synchronized void onException(Throwable e) { - exception = e; - } - - private synchronized Throwable getException() { - return exception; - } - } -} diff --git a/src/test/java/com/github/dockerjava/core/DockerClientImplTest.java b/src/test/java/com/github/dockerjava/core/DockerClientImplTest.java deleted file mode 100644 index 1df4636ba..000000000 --- a/src/test/java/com/github/dockerjava/core/DockerClientImplTest.java +++ /dev/null @@ -1,41 +0,0 @@ -package com.github.dockerjava.core; - -import static org.testng.Assert.assertEquals; -import static org.testng.AssertJUnit.fail; - -import java.net.URI; - -import org.testng.annotations.Test; - -public class DockerClientImplTest { - - @Test - public void configuredInstanceAuthConfig() throws Exception { - // given a config with null serverAddress - DefaultDockerClientConfig dockerClientConfig = new DefaultDockerClientConfig(URI.create("tcp://foo"), null, null, null, "", "", "", null); - DockerClientImpl dockerClient = DockerClientImpl.getInstance(dockerClientConfig); - - // when we get the auth config - try { - dockerClient.authConfig(); - fail(); - } catch (NullPointerException e) { - // then we get a NPE with expected message - assertEquals(e.getMessage(), "Configured serverAddress is null."); - } - } - - @Test - public void defaultInstanceAuthConfig() throws Exception { - - System.setProperty("user.home", "target/test-classes/someHomeDir"); - - // given a default client - DockerClientImpl dockerClient = DockerClientImpl.getInstance(); - - // when we get the auth config - dockerClient.authConfig(); - - // then we do not get an exception - } -} diff --git a/src/test/java/com/github/dockerjava/core/NameParserTest.java b/src/test/java/com/github/dockerjava/core/NameParserTest.java deleted file mode 100644 index 865f9a401..000000000 --- a/src/test/java/com/github/dockerjava/core/NameParserTest.java +++ /dev/null @@ -1,127 +0,0 @@ -/* - * Created on 17.08.2015 - */ -package com.github.dockerjava.core; - -import com.github.dockerjava.core.exception.InvalidRepositoryNameException; -import org.apache.commons.lang.StringUtils; -import org.testng.annotations.Test; - -import static org.testng.Assert.assertEquals; - -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.core.NameParser.HostnameReposName; -import com.github.dockerjava.core.NameParser.ReposTag; - -/** - * - * - * @author Marcus Linke - * - */ -public class NameParserTest { - - @Test - public void testValidateRepoName() throws Exception { - NameParser.validateRepoName("repository"); - NameParser.validateRepoName("namespace/repository"); - NameParser.validateRepoName("namespace-with-dashes/repository"); - NameParser.validateRepoName("namespace/repository-with-dashes"); - NameParser.validateRepoName("namespace.with.dots/repository"); - NameParser.validateRepoName("namespace/repository.with.dots"); - NameParser.validateRepoName("namespace_with_underscores/repository"); - NameParser.validateRepoName("namespace/repository_with_underscores"); - } - - @Test(expectedExceptions = InvalidRepositoryNameException.class) - public void testValidateRepoNameEmpty() throws Exception { - NameParser.validateRepoName(""); - } - - @Test(expectedExceptions = InvalidRepositoryNameException.class) - public void testValidateRepoNameExceedsMaxLength() throws Exception { - NameParser.validateRepoName(StringUtils.repeat("repository", 255)); - } - - @Test(expectedExceptions = InvalidRepositoryNameException.class) - public void testValidateRepoNameEndWithDash() throws Exception { - NameParser.validateRepoName("repository-"); - } - - @Test(expectedExceptions = InvalidRepositoryNameException.class) - public void testValidateRepoNameStartWithDash() throws Exception { - NameParser.validateRepoName("-repository"); - } - - @Test(expectedExceptions = InvalidRepositoryNameException.class) - public void testValidateRepoNameEndWithDot() throws Exception { - NameParser.validateRepoName("repository."); - } - - @Test(expectedExceptions = InvalidRepositoryNameException.class) - public void testValidateRepoNameStartWithDot() throws Exception { - NameParser.validateRepoName(".repository"); - } - - @Test(expectedExceptions = InvalidRepositoryNameException.class) - public void testValidateRepoNameEndWithUnderscore() throws Exception { - NameParser.validateRepoName("repository_"); - } - - @Test(expectedExceptions = InvalidRepositoryNameException.class) - public void testValidateRepoNameStartWithUnderscore() throws Exception { - NameParser.validateRepoName("_repository"); - } - - @Test(expectedExceptions = InvalidRepositoryNameException.class) - public void testValidateRepoNameWithColon() throws Exception { - NameParser.validateRepoName("repository:with:colon"); - } - - @Test - public void testResolveSimpleRepositoryName() throws Exception { - HostnameReposName resolved = NameParser.resolveRepositoryName("repository"); - assertEquals(resolved, new HostnameReposName(AuthConfig.DEFAULT_SERVER_ADDRESS, "repository")); - } - - @Test - public void testResolveRepositoryNameWithNamespace() throws Exception { - HostnameReposName resolved = NameParser.resolveRepositoryName("namespace/repository"); - assertEquals(resolved, new HostnameReposName(AuthConfig.DEFAULT_SERVER_ADDRESS, "namespace/repository")); - } - - @Test - public void testResolveRepositoryNameWithNamespaceAndHostname() throws Exception { - HostnameReposName resolved = NameParser.resolveRepositoryName("localhost:5000/namespace/repository"); - assertEquals(resolved, new HostnameReposName("localhost:5000", "namespace/repository")); - } - - @Test(expectedExceptions = InvalidRepositoryNameException.class) - public void testResolveRepositoryNameWithIndex() throws Exception { - NameParser.resolveRepositoryName("index.docker.io/repository"); - } - - @Test - public void testResolveReposTagWithoutTagSimple() throws Exception { - ReposTag resolved = NameParser.parseRepositoryTag("repository"); - assertEquals(resolved, new ReposTag("repository", "")); - - resolved = NameParser.parseRepositoryTag("namespace/repository"); - assertEquals(resolved, new ReposTag("namespace/repository", "")); - - resolved = NameParser.parseRepositoryTag("localhost:5000/namespace/repository"); - assertEquals(resolved, new ReposTag("localhost:5000/namespace/repository", "")); - } - - @Test - public void testResolveReposTagWithTag() throws Exception { - ReposTag resolved = NameParser.parseRepositoryTag("repository:tag"); - assertEquals(resolved, new ReposTag("repository", "tag")); - - resolved = NameParser.parseRepositoryTag("namespace/repository:tag"); - assertEquals(resolved, new ReposTag("namespace/repository", "tag")); - - resolved = NameParser.parseRepositoryTag("localhost:5000/namespace/repository:tag"); - assertEquals(resolved, new ReposTag("localhost:5000/namespace/repository", "tag")); - } -} diff --git a/src/test/java/com/github/dockerjava/core/TestDockerCmdExecFactory.java b/src/test/java/com/github/dockerjava/core/TestDockerCmdExecFactory.java deleted file mode 100644 index 46d6d4dcc..000000000 --- a/src/test/java/com/github/dockerjava/core/TestDockerCmdExecFactory.java +++ /dev/null @@ -1,445 +0,0 @@ -package com.github.dockerjava.core; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.AttachContainerCmd; -import com.github.dockerjava.api.command.AuthCmd.Exec; -import com.github.dockerjava.api.command.BuildImageCmd; -import com.github.dockerjava.api.command.CommitCmd; -import com.github.dockerjava.api.command.ConnectToNetworkCmd; -import com.github.dockerjava.api.command.ContainerDiffCmd; -import com.github.dockerjava.api.command.CopyArchiveFromContainerCmd; -import com.github.dockerjava.api.command.CopyArchiveToContainerCmd; -import com.github.dockerjava.api.command.CopyFileFromContainerCmd; -import com.github.dockerjava.api.command.CreateContainerCmd; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.CreateImageCmd; -import com.github.dockerjava.api.command.CreateImageResponse; -import com.github.dockerjava.api.command.CreateNetworkCmd; -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.command.CreateVolumeCmd; -import com.github.dockerjava.api.command.CreateVolumeResponse; -import com.github.dockerjava.api.command.DisconnectFromNetworkCmd; -import com.github.dockerjava.api.command.DockerCmdExecFactory; -import com.github.dockerjava.api.command.EventsCmd; -import com.github.dockerjava.api.command.ExecCreateCmd; -import com.github.dockerjava.api.command.ExecStartCmd; -import com.github.dockerjava.api.command.InfoCmd; -import com.github.dockerjava.api.command.InspectContainerCmd; -import com.github.dockerjava.api.command.InspectExecCmd; -import com.github.dockerjava.api.command.InspectImageCmd; -import com.github.dockerjava.api.command.InspectNetworkCmd; -import com.github.dockerjava.api.command.InspectVolumeCmd; -import com.github.dockerjava.api.command.KillContainerCmd; -import com.github.dockerjava.api.command.ListContainersCmd; -import com.github.dockerjava.api.command.ListImagesCmd; -import com.github.dockerjava.api.command.ListNetworksCmd; -import com.github.dockerjava.api.command.ListVolumesCmd; -import com.github.dockerjava.api.command.LoadImageCmd; -import com.github.dockerjava.api.command.LogContainerCmd; -import com.github.dockerjava.api.command.PauseContainerCmd; -import com.github.dockerjava.api.command.PingCmd; -import com.github.dockerjava.api.command.PullImageCmd; -import com.github.dockerjava.api.command.PushImageCmd; -import com.github.dockerjava.api.command.RemoveContainerCmd; -import com.github.dockerjava.api.command.RemoveImageCmd; -import com.github.dockerjava.api.command.RemoveNetworkCmd; -import com.github.dockerjava.api.command.RemoveVolumeCmd; -import com.github.dockerjava.api.command.RenameContainerCmd; -import com.github.dockerjava.api.command.RestartContainerCmd; -import com.github.dockerjava.api.command.SaveImageCmd; -import com.github.dockerjava.api.command.SearchImagesCmd; -import com.github.dockerjava.api.command.StartContainerCmd; -import com.github.dockerjava.api.command.StatsCmd; -import com.github.dockerjava.api.command.StopContainerCmd; -import com.github.dockerjava.api.command.TagImageCmd; -import com.github.dockerjava.api.command.TopContainerCmd; -import com.github.dockerjava.api.command.UnpauseContainerCmd; -import com.github.dockerjava.api.command.UpdateContainerCmd; -import com.github.dockerjava.api.command.VersionCmd; -import com.github.dockerjava.api.command.WaitContainerCmd; -import com.github.dockerjava.api.model.BuildResponseItem; - -import java.io.IOException; -import java.security.SecureRandom; -import java.util.ArrayList; -import java.util.List; - -/** - * Special {@link DockerCmdExecFactory} implementation that collects container and image creations while test execution for the purpose of - * automatically cleanup. - * - * @author Marcus Linke - */ -public class TestDockerCmdExecFactory implements DockerCmdExecFactory { - - private List containerNames = new ArrayList(); - - private List imageNames = new ArrayList(); - - private List volumeNames = new ArrayList(); - - private List networkIds = new ArrayList<>(); - - private DockerCmdExecFactory delegate; - - public TestDockerCmdExecFactory(DockerCmdExecFactory delegate) { - this.delegate = delegate; - } - - @Override - public void init(DockerClientConfig dockerClientConfig) { - delegate.init(dockerClientConfig); - } - - @Override - public void close() throws IOException { - delegate.close(); - } - - @Override - public CreateContainerCmd.Exec createCreateContainerCmdExec() { - return new CreateContainerCmd.Exec() { - @Override - public CreateContainerResponse exec(CreateContainerCmd command) { - CreateContainerResponse createContainerResponse = delegate.createCreateContainerCmdExec().exec(command); - containerNames.add(createContainerResponse.getId()); - return createContainerResponse; - } - }; - } - - @Override - public RemoveContainerCmd.Exec createRemoveContainerCmdExec() { - return new RemoveContainerCmd.Exec() { - @Override - public Void exec(RemoveContainerCmd command) { - delegate.createRemoveContainerCmdExec().exec(command); - containerNames.remove(command.getContainerId()); - return null; - } - }; - } - - @Override - public CreateImageCmd.Exec createCreateImageCmdExec() { - return new CreateImageCmd.Exec() { - @Override - public CreateImageResponse exec(CreateImageCmd command) { - CreateImageResponse createImageResponse = delegate.createCreateImageCmdExec().exec(command); - imageNames.add(createImageResponse.getId()); - return createImageResponse; - } - }; - } - - @Override - public LoadImageCmd.Exec createLoadImageCmdExec() { - return new LoadImageCmd.Exec() { - @Override - public Void exec(LoadImageCmd command) { - delegate.createLoadImageCmdExec().exec(command); - return null; - } - }; - } - - @Override - public RemoveImageCmd.Exec createRemoveImageCmdExec() { - return new RemoveImageCmd.Exec() { - @Override - public Void exec(RemoveImageCmd command) { - delegate.createRemoveImageCmdExec().exec(command); - imageNames.remove(command.getImageId()); - return null; - } - }; - } - - @Override - public BuildImageCmd.Exec createBuildImageCmdExec() { - return new BuildImageCmd.Exec() { - @Override - public Void exec(BuildImageCmd command, ResultCallback resultCallback) { - // can't detect image id here so tagging it - String tag = command.getTag(); - if (tag == null || "".equals(tag.trim())) { - tag = "" + new SecureRandom().nextInt(Integer.MAX_VALUE); - command.withTag(tag); - } - delegate.createBuildImageCmdExec().exec(command, resultCallback); - imageNames.add(tag); - return null; - } - }; - } - - @Override - public Exec createAuthCmdExec() { - return delegate.createAuthCmdExec(); - } - - @Override - public InfoCmd.Exec createInfoCmdExec() { - return delegate.createInfoCmdExec(); - } - - @Override - public PingCmd.Exec createPingCmdExec() { - return delegate.createPingCmdExec(); - } - - @Override - public ExecCreateCmd.Exec createExecCmdExec() { - return delegate.createExecCmdExec(); - } - - @Override - public VersionCmd.Exec createVersionCmdExec() { - return delegate.createVersionCmdExec(); - } - - @Override - public PullImageCmd.Exec createPullImageCmdExec() { - return delegate.createPullImageCmdExec(); - } - - @Override - public PushImageCmd.Exec createPushImageCmdExec() { - return delegate.createPushImageCmdExec(); - } - - @Override - public SaveImageCmd.Exec createSaveImageCmdExec() { - return delegate.createSaveImageCmdExec(); - } - - @Override - public SearchImagesCmd.Exec createSearchImagesCmdExec() { - return delegate.createSearchImagesCmdExec(); - } - - @Override - public ListImagesCmd.Exec createListImagesCmdExec() { - return delegate.createListImagesCmdExec(); - } - - @Override - public InspectImageCmd.Exec createInspectImageCmdExec() { - return delegate.createInspectImageCmdExec(); - } - - @Override - public ListContainersCmd.Exec createListContainersCmdExec() { - return delegate.createListContainersCmdExec(); - } - - @Override - public StartContainerCmd.Exec createStartContainerCmdExec() { - return delegate.createStartContainerCmdExec(); - } - - @Override - public InspectContainerCmd.Exec createInspectContainerCmdExec() { - return delegate.createInspectContainerCmdExec(); - } - - @Override - public WaitContainerCmd.Exec createWaitContainerCmdExec() { - return delegate.createWaitContainerCmdExec(); - } - - @Override - public AttachContainerCmd.Exec createAttachContainerCmdExec() { - return delegate.createAttachContainerCmdExec(); - } - - @Override - public ExecStartCmd.Exec createExecStartCmdExec() { - return delegate.createExecStartCmdExec(); - } - - @Override - public InspectExecCmd.Exec createInspectExecCmdExec() { - return delegate.createInspectExecCmdExec(); - } - - @Override - public LogContainerCmd.Exec createLogContainerCmdExec() { - return delegate.createLogContainerCmdExec(); - } - - @Override - public CopyFileFromContainerCmd.Exec createCopyFileFromContainerCmdExec() { - return delegate.createCopyFileFromContainerCmdExec(); - } - - @Override - public CopyArchiveFromContainerCmd.Exec createCopyArchiveFromContainerCmdExec() { - return delegate.createCopyArchiveFromContainerCmdExec(); - } - - @Override - public CopyArchiveToContainerCmd.Exec createCopyArchiveToContainerCmdExec() { - return delegate.createCopyArchiveToContainerCmdExec(); - } - - @Override - public StopContainerCmd.Exec createStopContainerCmdExec() { - return delegate.createStopContainerCmdExec(); - } - - @Override - public ContainerDiffCmd.Exec createContainerDiffCmdExec() { - return delegate.createContainerDiffCmdExec(); - } - - @Override - public KillContainerCmd.Exec createKillContainerCmdExec() { - return delegate.createKillContainerCmdExec(); - } - - @Override - public UpdateContainerCmd.Exec createUpdateContainerCmdExec() { - return delegate.createUpdateContainerCmdExec(); - } - - @Override - public RenameContainerCmd.Exec createRenameContainerCmdExec(){ - return delegate.createRenameContainerCmdExec(); - } - - @Override - public RestartContainerCmd.Exec createRestartContainerCmdExec() { - return delegate.createRestartContainerCmdExec(); - } - - @Override - public CommitCmd.Exec createCommitCmdExec() { - return delegate.createCommitCmdExec(); - } - - @Override - public TopContainerCmd.Exec createTopContainerCmdExec() { - return delegate.createTopContainerCmdExec(); - } - - @Override - public TagImageCmd.Exec createTagImageCmdExec() { - return delegate.createTagImageCmdExec(); - } - - @Override - public PauseContainerCmd.Exec createPauseContainerCmdExec() { - return delegate.createPauseContainerCmdExec(); - } - - @Override - public UnpauseContainerCmd.Exec createUnpauseContainerCmdExec() { - return delegate.createUnpauseContainerCmdExec(); - } - - @Override - public EventsCmd.Exec createEventsCmdExec() { - return delegate.createEventsCmdExec(); - } - - @Override - public StatsCmd.Exec createStatsCmdExec() { - return delegate.createStatsCmdExec(); - } - - @Override - public CreateVolumeCmd.Exec createCreateVolumeCmdExec() { - return new CreateVolumeCmd.Exec() { - @Override - public CreateVolumeResponse exec(CreateVolumeCmd command) { - CreateVolumeResponse result = delegate.createCreateVolumeCmdExec().exec(command); - volumeNames.add(command.getName()); - return result; - } - }; - } - - @Override - public InspectVolumeCmd.Exec createInspectVolumeCmdExec() { - return delegate.createInspectVolumeCmdExec(); - } - - @Override - public RemoveVolumeCmd.Exec createRemoveVolumeCmdExec() { - return new RemoveVolumeCmd.Exec() { - @Override - public Void exec(RemoveVolumeCmd command) { - delegate.createRemoveVolumeCmdExec().exec(command); - volumeNames.remove(command.getName()); - return null; - } - }; - } - - @Override - public ListVolumesCmd.Exec createListVolumesCmdExec() { - return delegate.createListVolumesCmdExec(); - } - - @Override - public ListNetworksCmd.Exec createListNetworksCmdExec() { - return delegate.createListNetworksCmdExec(); - } - - @Override - public InspectNetworkCmd.Exec createInspectNetworkCmdExec() { - return delegate.createInspectNetworkCmdExec(); - } - - @Override - public CreateNetworkCmd.Exec createCreateNetworkCmdExec() { - - return new CreateNetworkCmd.Exec() { - @Override - public CreateNetworkResponse exec(CreateNetworkCmd command) { - CreateNetworkResponse result = delegate.createCreateNetworkCmdExec().exec(command); - networkIds.add(result.getId()); - return result; - } - }; - } - - @Override - public RemoveNetworkCmd.Exec createRemoveNetworkCmdExec() { - return new RemoveNetworkCmd.Exec() { - @Override - public Void exec(RemoveNetworkCmd command) { - delegate.createRemoveNetworkCmdExec().exec(command); - networkIds.remove(command.getNetworkId()); - return null; - } - }; - } - - @Override - public ConnectToNetworkCmd.Exec createConnectToNetworkCmdExec() { - return delegate.createConnectToNetworkCmdExec(); - } - - @Override - public DisconnectFromNetworkCmd.Exec createDisconnectFromNetworkCmdExec() { - return delegate.createDisconnectFromNetworkCmdExec(); - } - - public List getContainerNames() { - return new ArrayList(containerNames); - } - - public List getImageNames() { - return new ArrayList(imageNames); - } - - public List getVolumeNames() { - return new ArrayList(volumeNames); - } - - public List getNetworkIds() { - return new ArrayList<>(networkIds); - } -} diff --git a/src/test/java/com/github/dockerjava/core/async/JsonStreamProcessorTest.java b/src/test/java/com/github/dockerjava/core/async/JsonStreamProcessorTest.java deleted file mode 100644 index 6c9ee8a15..000000000 --- a/src/test/java/com/github/dockerjava/core/async/JsonStreamProcessorTest.java +++ /dev/null @@ -1,65 +0,0 @@ -/* - * Created on 16.02.2016 - */ -package com.github.dockerjava.core.async; - -import static org.testng.Assert.assertFalse; - -import java.io.ByteArrayInputStream; -import java.io.Closeable; -import java.io.IOException; -import java.io.InputStream; -import java.util.ArrayList; -import java.util.List; - -import org.testng.annotations.Test; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.model.PullResponseItem; - - -/** - * - * @author Marcus Linke - * - */ -public class JsonStreamProcessorTest { - - @Test - public void processEmptyJson() throws Exception { - - InputStream response = new ByteArrayInputStream("{}".getBytes()); - - JsonStreamProcessor jsonStreamProcessor = new JsonStreamProcessor(PullResponseItem.class); - - final List completed = new ArrayList(); - - jsonStreamProcessor.processResponseStream(response, new ResultCallback() { - - @Override - public void close() throws IOException { - } - - @Override - public void onStart(Closeable closeable) { - } - - @Override - public void onNext(PullResponseItem object) { - assertFalse(true, "onNext called for empty json"); - } - - @Override - public void onError(Throwable throwable) { - } - - @Override - public void onComplete() { - completed.add(true); - } - }); - - assertFalse(completed.isEmpty(), "Stream processing not completed"); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/AttachContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/AttachContainerCmdImplTest.java deleted file mode 100644 index ef69984f3..000000000 --- a/src/test/java/com/github/dockerjava/core/command/AttachContainerCmdImplTest.java +++ /dev/null @@ -1,164 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.apache.commons.lang.StringUtils.isEmpty; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.io.ByteArrayInputStream; -import java.io.File; -import java.io.InputStream; -import java.lang.reflect.Method; -import java.util.concurrent.TimeUnit; - -import org.apache.commons.codec.binary.StringUtils; -import org.testng.ITestResult; -import org.testng.SkipException; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.api.model.StreamType; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class AttachContainerCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void attachContainerWithoutTTY() throws Exception { - - String snippet = "hello world"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("echo", snippet) - .withTty(false).exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - AttachContainerTestCallback callback = new AttachContainerTestCallback() { - @Override - public void onNext(Frame frame) { - assertEquals(frame.getStreamType(), StreamType.STDOUT); - super.onNext(frame); - }; - }; - - dockerClient.attachContainerCmd(container.getId()).withStdErr(true).withStdOut(true).withFollowStream(true) - .withLogs(true).exec(callback).awaitCompletion(30, TimeUnit.SECONDS); - callback.close(); - - assertThat(callback.toString(), containsString(snippet)); - } - - @Test - public void attachContainerWithTTY() throws Exception { - - File baseDir = new File(Thread.currentThread().getContextClassLoader() - .getResource("attachContainerTestDockerfile").getFile()); - - String imageId = buildImage(baseDir); - - CreateContainerResponse container = dockerClient.createContainerCmd(imageId).withTty(true).exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - AttachContainerTestCallback callback = new AttachContainerTestCallback() { - @Override - public void onNext(Frame frame) { - assertEquals(frame.getStreamType(), StreamType.RAW); - super.onNext(frame); - }; - }; - - dockerClient.attachContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .withFollowStream(true) - .exec(callback) - .awaitCompletion(15, TimeUnit.SECONDS); - callback.close(); - - System.out.println("log: " + callback.toString()); - - // HexDump.dump(collectFramesCallback.toString().getBytes(), 0, System.out, 0); - assertThat(callback.toString(), containsString("stdout\r\nstderr")); - } - - @Test(expectedExceptions = UnsupportedOperationException.class) - public void attachContainerStdinUnsupported() throws Exception { - - String snippet = "hello world"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("echo", snippet) - .withTty(false).exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - AttachContainerTestCallback callback = new AttachContainerTestCallback() { - @Override - public void onNext(Frame frame) { - assertEquals(frame.getStreamType(), StreamType.STDOUT); - super.onNext(frame); - }; - }; - - InputStream stdin = new ByteArrayInputStream("".getBytes()); - - dockerClient.attachContainerCmd(container.getId()).withStdErr(true).withStdOut(true).withFollowStream(true) - .withLogs(true).withStdIn(stdin).exec(callback).awaitCompletion(30, TimeUnit.SECONDS); - callback.close(); - } - - public static class AttachContainerTestCallback extends AttachContainerResultCallback { - private StringBuffer log = new StringBuffer(); - - @Override - public void onNext(Frame item) { - log.append(new String(item.getPayload())); - super.onNext(item); - } - - @Override - public RuntimeException getFirstError() { - return super.getFirstError(); - } - - @Override - public String toString() { - return log.toString(); - } - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/AuthCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/AuthCmdImplTest.java deleted file mode 100644 index 887c99130..000000000 --- a/src/test/java/com/github/dockerjava/core/command/AuthCmdImplTest.java +++ /dev/null @@ -1,53 +0,0 @@ -package com.github.dockerjava.core.command; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.UnauthorizedException; -import com.github.dockerjava.api.model.AuthResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.core.DockerClientBuilder; - -@Test(groups = "integration") -public class AuthCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void testAuth() throws Exception { - AuthResponse response = dockerClient.authCmd().exec(); - - assertEquals(response.getStatus(), "Login Succeeded"); - } - - // Disabled because of 500/InternalServerException - @Test(enabled = false, expectedExceptions = UnauthorizedException.class, expectedExceptionsMessageRegExp = "Wrong login/password, please try again") - public void testAuthInvalid() throws Exception { - - DockerClientBuilder.getInstance(config("garbage")).build().authCmd().exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/BuildImageCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/BuildImageCmdImplTest.java deleted file mode 100644 index 31f3c3045..000000000 --- a/src/test/java/com/github/dockerjava/core/command/BuildImageCmdImplTest.java +++ /dev/null @@ -1,269 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.nullValue; - -import java.io.File; -import java.io.FileInputStream; -import java.io.InputStream; -import java.lang.reflect.Method; -import java.util.Collection; -import java.util.UUID; - -import org.apache.commons.io.FileUtils; -import org.apache.commons.io.filefilter.TrueFileFilter; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.BuildImageCmd; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectImageResponse; -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.AuthConfigurations; -import com.github.dockerjava.api.model.ExposedPort; -import com.github.dockerjava.api.model.PortBinding; -import com.github.dockerjava.api.model.Ports.Binding; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.core.util.CompressArchiveUtil; - -@Test(groups = "integration") -public class BuildImageCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void author() throws Exception { - - String imageId = buildImage(fileFromBuildTestResource("AUTHOR")); - - InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(imageId).exec(); - assertThat(inspectImageResponse, not(nullValue())); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - - assertThat(inspectImageResponse.getAuthor(), equalTo("Guillaume J. Charmes \"guillaume@dotcloud.com\"")); - } - - @Test - public void buildImageFromTar() throws Exception { - File baseDir = fileFromBuildTestResource("ADD/file"); - Collection files = FileUtils.listFiles(baseDir, TrueFileFilter.INSTANCE, TrueFileFilter.INSTANCE); - File tarFile = CompressArchiveUtil.archiveTARFiles(baseDir, files, UUID.randomUUID().toString()); - String response = dockerfileBuild(new FileInputStream(tarFile)); - assertThat(response, containsString("Successfully executed testrun.sh")); - } - - @Test - public void onBuild() throws Exception { - File baseDir = fileFromBuildTestResource("ONBUILD/parent"); - - dockerClient.buildImageCmd(baseDir).withNoCache(true).withTag("docker-java-onbuild") - .exec(new BuildImageResultCallback()).awaitImageId(); - baseDir = fileFromBuildTestResource("ONBUILD/child"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("Successfully executed testrun.sh")); - } - - @Test - public void addUrl() throws Exception { - File baseDir = fileFromBuildTestResource("ADD/url"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("Example Domain")); - } - - @Test - public void addFileInSubfolder() throws Exception { - File baseDir = fileFromBuildTestResource("ADD/fileInSubfolder"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("Successfully executed testrun.sh")); - } - - @Test - public void addFilesViaWildcard() throws Exception { - File baseDir = fileFromBuildTestResource("ADD/filesViaWildcard"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("Successfully executed testinclude1.sh")); - assertThat(response, not(containsString("Successfully executed testinclude2.sh"))); - } - - @Test - public void addFolder() throws Exception { - File baseDir = fileFromBuildTestResource("ADD/folder"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("Successfully executed testAddFolder.sh")); - } - - private String dockerfileBuild(InputStream tarInputStream) throws Exception { - - return execBuild(dockerClient.buildImageCmd().withTarInputStream(tarInputStream)); - } - - private String dockerfileBuild(File baseDir) throws Exception { - - return execBuild(dockerClient.buildImageCmd(baseDir)); - } - - private String execBuild(BuildImageCmd buildImageCmd) throws Exception { - String imageId = buildImageCmd.withNoCache(true).exec(new BuildImageResultCallback()).awaitImageId(); - - // Create container based on image - CreateContainerResponse container = dockerClient.createContainerCmd(imageId).exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - dockerClient.waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()).awaitStatusCode(); - - return containerLog(container.getId()); - } - - @Test(expectedExceptions = {DockerClientException.class}) - public void dockerignoreDockerfileIgnored() throws Exception { - File baseDir = fileFromBuildTestResource("dockerignore/DockerfileIgnored"); - - dockerClient.buildImageCmd(baseDir).withNoCache(true).exec(new BuildImageResultCallback()).awaitImageId(); - } - - @Test - public void dockerignoreDockerfileNotIgnored() throws Exception { - File baseDir = fileFromBuildTestResource("dockerignore/DockerfileNotIgnored"); - - dockerClient.buildImageCmd(baseDir).withNoCache(true).exec(new BuildImageResultCallback()).awaitImageId(); - } - - @Test(expectedExceptions = {DockerClientException.class}) - public void dockerignoreInvalidDockerIgnorePattern() throws Exception { - File baseDir = fileFromBuildTestResource("dockerignore/InvalidDockerignorePattern"); - - dockerClient.buildImageCmd(baseDir).withNoCache(true).exec(new BuildImageResultCallback()).awaitImageId(); - } - - @Test() - public void dockerignoreValidDockerIgnorePattern() throws Exception { - File baseDir = fileFromBuildTestResource("dockerignore/ValidDockerignorePattern"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("/tmp/a/a /tmp/a/c /tmp/a/d")); - } - - @Test - public void env() throws Exception { - File baseDir = fileFromBuildTestResource("ENV"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("testENVSubstitution successfully completed")); - } - - @Test - public void fromPrivateRegistry() throws Exception { - File baseDir = new File(Thread.currentThread().getContextClassLoader().getResource("privateRegistry").getFile()); - - String imageId = buildImage(baseDir); - - InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(imageId).exec(); - assertThat(inspectImageResponse, not(nullValue())); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - - dockerClient.tagImageCmd(imageId, "testregistry", "2").withForce().exec(); - - // see https://github.com/docker/distribution/blob/master/docs/deploying.md#native-basic-auth - CreateContainerResponse testregistry = dockerClient - .createContainerCmd("testregistry:2") - .withName("registry") - .withPortBindings(new PortBinding(Binding.bindPort(5000), ExposedPort.tcp(5000))) - .withEnv("REGISTRY_AUTH=htpasswd", "REGISTRY_AUTH_HTPASSWD_REALM=Registry Realm", - "REGISTRY_AUTH_HTPASSWD_PATH=/auth/htpasswd", "REGISTRY_LOG_LEVEL=debug", - "REGISTRY_HTTP_TLS_CERTIFICATE=/certs/domain.crt", "REGISTRY_HTTP_TLS_KEY=/certs/domain.key") - .exec(); - - dockerClient.startContainerCmd(testregistry.getId()).exec(); - - // wait for registry to boot - Thread.sleep(3000); - - // credentials as configured in /auth/htpasswd - AuthConfig authConfig = new AuthConfig() - .withUsername("testuser") - .withPassword("testpassword") - .withEmail("foo@bar.de") - .withRegistryAddress("localhost:5000"); - - dockerClient.authCmd().withAuthConfig(authConfig).exec(); - dockerClient.tagImageCmd("busybox:latest", "localhost:5000/testuser/busybox", "latest").withForce().exec(); - - dockerClient.pushImageCmd("localhost:5000/testuser/busybox").withTag("latest").withAuthConfig(authConfig) - .exec(new PushImageResultCallback()).awaitSuccess(); - - dockerClient.removeImageCmd("localhost:5000/testuser/busybox").withForce(true).exec(); - - baseDir = fileFromBuildTestResource("FROM/privateRegistry"); - - AuthConfigurations authConfigurations = new AuthConfigurations(); - authConfigurations.addConfig(authConfig); - - imageId = dockerClient.buildImageCmd(baseDir).withNoCache(true).withBuildAuthConfigs(authConfigurations) - .exec(new BuildImageResultCallback()).awaitImageId(); - - inspectImageResponse = dockerClient.inspectImageCmd(imageId).exec(); - assertThat(inspectImageResponse, not(nullValue())); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - - } - - @Test - public void buildArgs() throws Exception { - File baseDir = fileFromBuildTestResource("buildArgs"); - - String imageId = dockerClient.buildImageCmd(baseDir).withNoCache(true).withBuildArg("testArg", "abc") - .exec(new BuildImageResultCallback()) - .awaitImageId(); - - InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(imageId).exec(); - assertThat(inspectImageResponse, not(nullValue())); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - - assertThat(inspectImageResponse.getConfig().getLabels().get("test"), equalTo("abc")); - } - - public void dockerfileNotInBaseDirectory() throws Exception { - File baseDirectory = fileFromBuildTestResource("dockerfileNotInBaseDirectory"); - File dockerfile = fileFromBuildTestResource("dockerfileNotInBaseDirectory/dockerfileFolder/Dockerfile"); - BuildImageCmd command = dockerClient.buildImageCmd() - .withBaseDirectory(baseDirectory) - .withDockerfile(dockerfile); - - String response = execBuild(command); - - assertThat(response, containsString("Successfully executed testrun.sh")); - } - - private File fileFromBuildTestResource(String resource) { - return new File(Thread.currentThread().getContextClassLoader() - .getResource("buildTests/" + resource).getFile()); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/CommitCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/CommitCmdImplTest.java deleted file mode 100644 index 64e1d7ddf..000000000 --- a/src/test/java/com/github/dockerjava/core/command/CommitCmdImplTest.java +++ /dev/null @@ -1,76 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.startsWith; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectImageResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class CommitCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void commit() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("touch", "/test").exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - LOG.info("Commiting container: {}", container.toString()); - String imageId = dockerClient.commitCmd(container.getId()).exec(); - - InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(imageId).exec(); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - - assertThat(inspectImageResponse, hasField("container", startsWith(container.getId()))); - assertThat(inspectImageResponse.getContainerConfig().getImage(), equalTo("busybox")); - - InspectImageResponse busyboxImg = dockerClient.inspectImageCmd("busybox").exec(); - - assertThat(inspectImageResponse.getParent(), equalTo(busyboxImg.getId())); - } - - @Test(expectedExceptions = NotFoundException.class) - public void commitNonExistingContainer() throws DockerException { - - dockerClient.commitCmd("non-existent").exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/ConnectToNetworkCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/ConnectToNetworkCmdImplTest.java deleted file mode 100644 index 84bb59a81..000000000 --- a/src/test/java/com/github/dockerjava/core/command/ConnectToNetworkCmdImplTest.java +++ /dev/null @@ -1,97 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.model.ContainerNetwork; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.client.AbstractDockerClientTest; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; -import java.util.Collections; - -@Test(groups = "integration") -public class ConnectToNetworkCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void connectToNetwork() throws InterruptedException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - dockerClient.startContainerCmd(container.getId()).exec(); - - CreateNetworkResponse network = dockerClient.createNetworkCmd().withName("testNetwork").exec(); - - dockerClient.connectToNetworkCmd().withNetworkId(network.getId()).withContainerId(container.getId()).exec(); - - Network updatedNetwork = dockerClient.inspectNetworkCmd().withNetworkId(network.getId()).exec(); - - assertTrue(updatedNetwork.getContainers().containsKey(container.getId())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertNotNull(inspectContainerResponse.getNetworkSettings().getNetworks().get("testNetwork")); - } - - @Test - public void connectToNetworkWithContainerNetwork() throws InterruptedException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - dockerClient.startContainerCmd(container.getId()).exec(); - - CreateNetworkResponse network = dockerClient.createNetworkCmd() - .withName("testNetwork") - .withIpam(new Network.Ipam() - .withConfig(new Network.Ipam.Config() - .withSubnet("10.100.100.0/24"))) - .exec(); - - dockerClient.connectToNetworkCmd() - .withNetworkId(network.getId()) - .withContainerId(container.getId()) - .withContainerNetwork(new ContainerNetwork() - .withAliases("testing") - .withIpamConfig(new ContainerNetwork.Ipam() - .withIpv4Address("10.100.100.100"))) - .exec(); - - Network updatedNetwork = dockerClient.inspectNetworkCmd().withNetworkId(network.getId()).exec(); - - Network.ContainerNetworkConfig containerNetworkConfig = updatedNetwork.getContainers().get(container.getId()); - assertNotNull(containerNetworkConfig); - assertEquals(containerNetworkConfig.getIpv4Address(), "10.100.100.100/24"); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - ContainerNetwork testNetwork = inspectContainerResponse.getNetworkSettings().getNetworks().get("testNetwork"); - assertNotNull(testNetwork); - assertEquals(testNetwork.getAliases(), Collections.singletonList("testing")); - assertEquals(testNetwork.getGateway(), "10.100.100.1"); - assertEquals(testNetwork.getIpAddress(), "10.100.100.100"); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/ContainerDiffCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/ContainerDiffCmdImplTest.java deleted file mode 100644 index e51db21f4..000000000 --- a/src/test/java/com/github/dockerjava/core/command/ContainerDiffCmdImplTest.java +++ /dev/null @@ -1,75 +0,0 @@ -package com.github.dockerjava.core.command; - -import static ch.lambdaj.Lambda.selectUnique; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; -import java.util.List; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.model.ChangeLog; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class ContainerDiffCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void testContainerDiff() throws DockerException { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("touch", "/test").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - int exitCode = dockerClient.waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - assertThat(exitCode, equalTo(0)); - - List filesystemDiff = dockerClient.containerDiffCmd(container.getId()).exec(); - LOG.info("Container DIFF: {}", filesystemDiff.toString()); - - assertThat(filesystemDiff.size(), equalTo(1)); - ChangeLog testChangeLog = selectUnique(filesystemDiff, hasField("path", equalTo("/test"))); - - assertThat(testChangeLog, hasField("path", equalTo("/test"))); - assertThat(testChangeLog, hasField("kind", equalTo(1))); - } - - @Test(expectedExceptions = NotFoundException.class) - public void testContainerDiffWithNonExistingContainer() throws DockerException { - - dockerClient.containerDiffCmd("non-existing").exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/CopyArchiveFromContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/CopyArchiveFromContainerCmdImplTest.java deleted file mode 100644 index 59c459382..000000000 --- a/src/test/java/com/github/dockerjava/core/command/CopyArchiveFromContainerCmdImplTest.java +++ /dev/null @@ -1,110 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyOrNullString; -import static org.hamcrest.Matchers.not; - -import java.io.InputStream; -import java.lang.reflect.Method; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.nio.file.StandardOpenOption; - -import org.apache.commons.compress.archivers.tar.TarArchiveEntry; -import org.apache.commons.compress.archivers.tar.TarArchiveInputStream; -import org.apache.commons.io.IOUtils; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.core.util.CompressArchiveUtil; - -@Test(groups = "integration") -public class CopyArchiveFromContainerCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void copyFromContainer() throws Exception { - // TODO extract this into a shared method - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("docker-java-itest-copyFromContainer").withCmd("touch", "/copyFromContainer").exec(); - - LOG.info("Created container: {}", container); - assertThat(container.getId(), not(isEmptyOrNullString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InputStream response = dockerClient.copyArchiveFromContainerCmd(container.getId(), "/copyFromContainer").exec(); - Boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied from the container."); - - // read the stream fully. Otherwise, the underlying stream will not be closed. - String responseAsString = asString(response); - assertNotNull(responseAsString); - assertTrue(responseAsString.length() > 0); - } - - @Test(expectedExceptions = NotFoundException.class) - public void copyFromNonExistingContainer() throws Exception { - - dockerClient.copyArchiveFromContainerCmd("non-existing", "/test").exec(); - } - - @Test - public void copyFromContainerBinaryFile() throws Exception { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("docker-java-itest-copyFromContainerBinaryFile").exec(); - - LOG.info("Created container: {}", container); - assertThat(container.getId(), not(isEmptyOrNullString())); - - Path temp = Files.createTempFile("", ".tar.gz"); - Path binaryFile = Paths.get("src/test/resources/testCopyFromArchive/binary.dat"); - CompressArchiveUtil.tar(binaryFile, temp, true, false); - - try (InputStream uploadStream = Files.newInputStream(temp)) { - dockerClient.copyArchiveToContainerCmd(container.getId()).withTarInputStream(uploadStream).exec(); - } - - InputStream response = dockerClient.copyArchiveFromContainerCmd(container.getId(), "/binary.dat").exec(); - Boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied from the container."); - - try (TarArchiveInputStream tarInputStream = new TarArchiveInputStream(response)) { - TarArchiveEntry nextTarEntry = tarInputStream.getNextTarEntry(); - - assertEquals(nextTarEntry.getName(), "binary.dat"); - try (InputStream binaryFileInputStream = Files.newInputStream(binaryFile, StandardOpenOption.READ)) { - assertTrue(IOUtils.contentEquals(binaryFileInputStream, tarInputStream)); - } - - assertNull(tarInputStream.getNextTarEntry(), "Nothing except binary.dat is expected to be copied."); - } - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/CopyArchiveToContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/CopyArchiveToContainerCmdImplTest.java deleted file mode 100644 index 658c1f090..000000000 --- a/src/test/java/com/github/dockerjava/core/command/CopyArchiveToContainerCmdImplTest.java +++ /dev/null @@ -1,153 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyOrNullString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.equalTo; - -import java.io.IOException; -import java.io.InputStream; -import java.lang.reflect.Method; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; - -import org.apache.commons.io.FileUtils; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.core.util.CompressArchiveUtil; - -@Test(groups = "integration") -public class CopyArchiveToContainerCmdImplTest extends AbstractDockerClientTest { - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void copyFileToContainer() throws Exception { - CreateContainerResponse container = prepareContainerForCopy(); - Path temp = Files.createTempFile("", ".tar.gz"); - CompressArchiveUtil.tar(Paths.get("src/test/resources/testReadFile"), temp, true, false); - try (InputStream uploadStream = Files.newInputStream(temp)) { - dockerClient.copyArchiveToContainerCmd(container.getId()).withTarInputStream(uploadStream).exec(); - assertFileCopied(container); - } - } - - @Test - public void copyStreamToContainer() throws Exception { - CreateContainerResponse container = prepareContainerForCopy(); - dockerClient.copyArchiveToContainerCmd(container.getId()).withHostResource("src/test/resources/testReadFile") - .exec(); - assertFileCopied(container); - } - - private CreateContainerResponse prepareContainerForCopy() { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("docker-java-itest-copyToContainer").exec(); - LOG.info("Created container: {}", container); - assertThat(container.getId(), not(isEmptyOrNullString())); - dockerClient.startContainerCmd(container.getId()).exec(); - // Copy a folder to the container - return container; - } - - private void assertFileCopied(CreateContainerResponse container) throws IOException { - try (InputStream response = dockerClient.copyArchiveFromContainerCmd(container.getId(), "testReadFile").exec()) { - boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied to the container."); - } - } - - @Test(expectedExceptions = NotFoundException.class) - public void copyToNonExistingContainer() throws Exception { - - dockerClient.copyArchiveToContainerCmd("non-existing").withHostResource("src/test/resources/testReadFile").exec(); - } - - @Test - public void copyDirWithLastAddedTarEntryEmptyDir() throws Exception{ - // create a temp dir - Path localDir = Files.createTempDirectory(null); - localDir.toFile().deleteOnExit(); - // create empty sub-dir with name b - Files.createDirectory(localDir.resolve("b")); - // create sub-dir with name a - Path dirWithFile = Files.createDirectory(localDir.resolve("a")); - // create file in sub-dir b, name or conter are irrelevant - Files.createFile(dirWithFile.resolve("file")); - - // create a test container - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("sleep", "9999") - .exec(); - // start the container - dockerClient.startContainerCmd(container.getId()).exec(); - // copy data from local dir to container - dockerClient.copyArchiveToContainerCmd(container.getId()) - .withHostResource(localDir.toString()) - .exec(); - - // cleanup dir - FileUtils.deleteDirectory(localDir.toFile()); - } - - @Test - public void copyFileWithExecutePermission() throws Exception { - // create script file, add permission to execute - Path scriptPath = Files.createTempFile("run", ".sh"); - boolean executable = scriptPath.toFile().setExecutable(true, false); - if (!executable){ - throw new Exception("Execute permission on file not set!"); - } - String snippet = "Running script with execute permission."; - String scriptTextStr = "#!/bin/sh\necho \"" + snippet + "\""; - // write content for created script - Files.write(scriptPath, scriptTextStr.getBytes()); - // create a test container which starts and waits 3 seconds for the - // script to be copied to the container's home dir and then executes it - String containerCmd = "sleep 3; /home/" + scriptPath.getFileName().toString(); - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("test") - .withCmd("/bin/sh", "-c", containerCmd) - .exec(); - // start the container - dockerClient.startContainerCmd(container.getId()).exec(); - // copy script to container home dir - dockerClient.copyArchiveToContainerCmd(container.getId()) - .withRemotePath("/home") - .withHostResource(scriptPath.toString()) - .exec(); - // await exid code - int exitCode = dockerClient.waitContainerCmd(container.getId()) - .exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - // check result - assertThat(exitCode, equalTo(0)); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/CopyFileFromContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/CopyFileFromContainerCmdImplTest.java deleted file mode 100644 index 8bd2c44fe..000000000 --- a/src/test/java/com/github/dockerjava/core/command/CopyFileFromContainerCmdImplTest.java +++ /dev/null @@ -1,70 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyOrNullString; -import static org.hamcrest.Matchers.not; - -import java.io.InputStream; -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class CopyFileFromContainerCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void copyFromContainer() throws Exception { - // TODO extract this into a shared method - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("docker-java-itest-copyFromContainer").withCmd("touch", "/copyFromContainer").exec(); - - LOG.info("Created container: {}", container); - assertThat(container.getId(), not(isEmptyOrNullString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InputStream response = dockerClient.copyFileFromContainerCmd(container.getId(), "/copyFromContainer").exec(); - Boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied from the container."); - - // read the stream fully. Otherwise, the underlying stream will not be closed. - String responseAsString = asString(response); - assertNotNull(responseAsString); - assertTrue(responseAsString.length() > 0); - } - - @Test(expectedExceptions = NotFoundException.class) - public void copyFromNonExistingContainer() throws Exception { - - dockerClient.copyFileFromContainerCmd("non-existing", "/test").exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/CreateContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/CreateContainerCmdImplTest.java deleted file mode 100644 index 7babafc91..000000000 --- a/src/test/java/com/github/dockerjava/core/command/CreateContainerCmdImplTest.java +++ /dev/null @@ -1,756 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.ConflictException; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Bind; -import com.github.dockerjava.api.model.ContainerNetwork; -import com.github.dockerjava.api.model.Device; -import com.github.dockerjava.api.model.ExposedPort; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.api.model.HostConfig; -import com.github.dockerjava.api.model.Link; -import com.github.dockerjava.api.model.LogConfig; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.api.model.Ports; -import com.github.dockerjava.api.model.RestartPolicy; -import com.github.dockerjava.api.model.Ulimit; -import com.github.dockerjava.api.model.Volume; -import com.github.dockerjava.api.model.VolumesFrom; -import com.github.dockerjava.api.model.Ports.Binding; -import com.github.dockerjava.client.AbstractDockerClientTest; - -import org.apache.commons.io.FileUtils; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; -import org.testng.internal.junit.ArrayAsserts; - -import java.lang.reflect.Method; -import java.security.SecureRandom; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.Map; -import java.util.UUID; -import java.util.concurrent.TimeUnit; - -import static com.github.dockerjava.api.model.Capability.MKNOD; -import static com.github.dockerjava.api.model.Capability.NET_ADMIN; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.contains; -import static org.hamcrest.Matchers.containsInAnyOrder; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.hasItemInArray; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.notNullValue; -import static org.hamcrest.Matchers.startsWith; - -@Test(groups = "integration") -public class CreateContainerCmdImplTest extends AbstractDockerClientTest { - public static final String BUSYBOX_IMAGE = "busybox"; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(expectedExceptions = ConflictException.class) - public void createContainerWithExistingName() throws DockerException { - - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("env") - .withName(containerName).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("env").withName(containerName).exec(); - } - - @Test - public void createContainerWithVolume() throws DockerException { - - Volume volume = new Volume("/var/log"); - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withVolumes(volume) - .withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - LOG.info("Inspect container {}", inspectContainerResponse.getConfig().getVolumes()); - - assertThat(inspectContainerResponse.getConfig().getVolumes().keySet(), contains("/var/log")); - - assertThat(inspectContainerResponse.getMounts().get(0).getDestination(), equalTo(volume)); - assertThat(inspectContainerResponse.getMounts().get(0).getMode(), equalTo("")); - assertThat(inspectContainerResponse.getMounts().get(0).getRW(), equalTo(true)); - } - - @Test - public void createContainerWithReadOnlyVolume() throws DockerException { - - Volume volume = new Volume("/srv/test"); - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withVolumes(volume) - .withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - LOG.info("Inspect container {}", inspectContainerResponse.getConfig().getVolumes()); - - assertThat(inspectContainerResponse.getConfig().getVolumes().keySet(), contains("/srv/test")); - - assertThat(inspectContainerResponse.getMounts().get(0).getDestination(), equalTo(volume)); - // TODO: Create a read-only volume and test like this - // assertFalse(inspectContainerResponse.getMounts().get(0).getRW()); - } - - @Test - public void createContainerWithVolumesFrom() throws DockerException { - - Volume volume1 = new Volume("/opt/webapp1"); - Volume volume2 = new Volume("/opt/webapp2"); - - String container1Name = UUID.randomUUID().toString(); - - Bind bind1 = new Bind("/src/webapp1", volume1); - Bind bind2 = new Bind("/src/webapp2", volume2); - - // create a running container with bind mounts - CreateContainerResponse container1 = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("sleep", "9999") - .withName(container1Name) - .withBinds(bind1, bind2).exec(); - LOG.info("Created container1 {}", container1.toString()); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - - assertThat(Arrays.asList(inspectContainerResponse1.getHostConfig().getBinds()), containsInAnyOrder(bind1, bind2)); - - assertThat(inspectContainerResponse1, mountedVolumes(containsInAnyOrder(volume1, volume2))); - - // create a second container with volumes from first container - CreateContainerResponse container2 = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("sleep", "9999") - .withVolumesFrom(new VolumesFrom(container1Name)).exec(); - LOG.info("Created container2 {}", container2.toString()); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - - // No volumes are created, the information is just stored in .HostConfig.VolumesFrom - assertThat(inspectContainerResponse2.getHostConfig().getVolumesFrom(), hasItemInArray(new VolumesFrom( - container1Name))); - - // To ensure that the information stored in VolumesFrom really is considered - // when starting the container, we start it and verify that it has the same - // bind mounts as the first container. - // This is somehow out of scope here, but it helped me to understand how the - // VolumesFrom feature really works. - dockerClient.startContainerCmd(container2.getId()).exec(); - LOG.info("Started container2 {}", container2.toString()); - - inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()).exec(); - - assertThat(inspectContainerResponse2.getHostConfig().getVolumesFrom(), hasItemInArray(new VolumesFrom( - container1Name))); - - assertThat(inspectContainerResponse2, mountedVolumes(containsInAnyOrder(volume1, volume2))); - } - - @Test - public void createContainerWithEnv() throws Exception { - final String testVariable = "VARIABLE=success"; - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withEnv(testVariable) - .withCmd("env") - .exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), hasItem(testVariable)); - - dockerClient.startContainerCmd(container.getId()).exec(); - - assertThat(containerLog(container.getId()), containsString(testVariable)); - } - - @Test - public void createContainerWithHostname() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withHostName("docker-java") - .withCmd("env").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getConfig().getHostName(), equalTo("docker-java")); - - dockerClient.startContainerCmd(container.getId()).exec(); - - assertThat(containerLog(container.getId()), containsString("HOSTNAME=docker-java")); - } - - @Test(expectedExceptions = ConflictException.class) - public void createContainerWithName() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withName("container") - .withCmd("env").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getName(), equalTo("/container")); - - - dockerClient.createContainerCmd(BUSYBOX_IMAGE).withName("container").withCmd("env").exec(); - } - - @Test - public void createContainerWithLink() throws DockerException { - - CreateContainerResponse container1 = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("sleep", "9999") - .withName("container1").exec(); - LOG.info("Created container1 {}", container1.toString()); - assertThat(container1.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); - assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); - - CreateContainerResponse container2 = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withName("container2") - .withCmd("env").withLinks(new Link("container1", "container1Link")).exec(); - LOG.info("Created container {}", container2.toString()); - assertThat(container2.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[]{new Link("container1", - "container1Link")})); - } - - @Test - public void createContainerWithLinkInCustomNetwork() throws DockerException { - - CreateNetworkResponse createNetworkResponse = dockerClient.createNetworkCmd() - .withName("linkNet") - .exec(); - - assertNotNull(createNetworkResponse.getId()); - - CreateContainerResponse container1 = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withNetworkMode("linkNet") - .withCmd("sleep", "9999") - .withName("container1") - .exec(); - - assertThat(container1.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); - assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); - - CreateContainerResponse container2 = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withNetworkMode("linkNet") - .withName("container2") - .withCmd("env") - .withLinks(new Link("container1", "container1Link")) - .exec(); - - LOG.info("Created container {}", container2.toString()); - assertThat(container2.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - - ContainerNetwork linkNet = inspectContainerResponse2.getNetworkSettings().getNetworks().get("linkNet"); - assertNotNull(linkNet); - ArrayAsserts.assertArrayEquals(new Link[]{ new Link("container1", "container1Link")}, linkNet.getLinks()); - } - - @Test - public void createContainerWithCustomIp() throws DockerException { - - CreateNetworkResponse createNetworkResponse = dockerClient.createNetworkCmd() - .withIpam(new Network.Ipam() - .withConfig(new Network.Ipam.Config() - .withSubnet("10.100.101.0/24"))) - .withName("customIpNet") - .exec(); - - assertNotNull(createNetworkResponse.getId()); - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withNetworkMode("customIpNet") - .withCmd("sleep", "9999") - .withName("container") - .withIpv4Address("10.100.101.100") - .exec(); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()) - .exec(); - - ContainerNetwork customIpNet = inspectContainerResponse.getNetworkSettings().getNetworks().get("customIpNet"); - assertNotNull(customIpNet); - assertEquals(customIpNet.getGateway(), "10.100.101.1"); - assertEquals(customIpNet.getIpAddress(), "10.100.101.100"); - } - - @Test - public void createContainerWithAlias() throws DockerException { - - CreateNetworkResponse createNetworkResponse = dockerClient.createNetworkCmd() - .withName("aliasNet") - .exec(); - - assertNotNull(createNetworkResponse.getId()); - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withNetworkMode("aliasNet") - .withCmd("sleep", "9999") - .withName("container") - .withAliases("server") - .exec(); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()) - .exec(); - - ContainerNetwork aliasNet = inspectContainerResponse.getNetworkSettings().getNetworks().get("aliasNet"); - assertEquals(aliasNet.getAliases(), Collections.singletonList("server")); - } - - @Test - public void createContainerWithCapAddAndCapDrop() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCapAdd(NET_ADMIN) - .withCapDrop(MKNOD).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapAdd()), contains(NET_ADMIN)); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapDrop()), contains(MKNOD)); - } - - @Test - public void createContainerWithDns() throws DockerException { - - String aDnsServer = "8.8.8.8"; - String anotherDnsServer = "8.8.4.4"; - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("true") - .withDns(aDnsServer, anotherDnsServer).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDns()), - contains(aDnsServer, anotherDnsServer)); - } - - @Test - public void createContainerWithEntrypoint() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withName("container") - .withEntrypoint("sleep", "9999").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEntrypoint()), contains("sleep", "9999")); - - } - - @Test - public void createContainerWithExtraHosts() throws DockerException { - - String[] extraHosts = {"dockerhost:127.0.0.1", "otherhost:10.0.0.1"}; - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withName("container") - .withExtraHosts(extraHosts).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getExtraHosts()), - containsInAnyOrder("dockerhost:127.0.0.1", "otherhost:10.0.0.1")); - } - - @Test - public void createContainerWithDevices() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("sleep", "9999") - .withDevices(new Device("rwm", "/dev/nulo", "/dev/zero")).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDevices()), contains(new Device("rwm", - "/dev/nulo", "/dev/zero"))); - } - - @Test - public void createContainerWithPortBindings() throws DockerException { - - ExposedPort tcp22 = ExposedPort.tcp(22); - ExposedPort tcp23 = ExposedPort.tcp(23); - - Ports portBindings = new Ports(); - portBindings.bind(tcp22, Binding.bindPort(11022)); - portBindings.bind(tcp23, Binding.bindPort(11023)); - portBindings.bind(tcp23, Binding.bindPort(11024)); - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("true") - .withExposedPorts(tcp22, tcp23).withPortBindings(portBindings).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getConfig().getExposedPorts()), contains(tcp22, tcp23)); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp22)[0], - is(equalTo(Binding.bindPort(11022)))); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[0], - is(equalTo(Binding.bindPort(11023)))); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[1], - is(equalTo(Binding.bindPort(11024)))); - - } - - @Test - public void createContainerWithLinking() throws DockerException { - - CreateContainerResponse container1 = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("sleep", "9999") - .withName("container1").exec(); - - LOG.info("Created container1 {}", container1.toString()); - assertThat(container1.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); - - assertThat(inspectContainerResponse1.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse1.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getId(), startsWith(container1.getId())); - assertThat(inspectContainerResponse1.getName(), equalTo("/container1")); - assertThat(inspectContainerResponse1.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getState(), is(notNullValue())); - assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); - - if (!inspectContainerResponse1.getState().getRunning()) { - assertThat(inspectContainerResponse1.getState().getExitCode(), is(equalTo(0))); - } - - CreateContainerResponse container2 = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("sleep", "9999") - .withName("container2").withLinks(new Link("container1", "container1Link")).exec(); - - LOG.info("Created container2 {}", container2.toString()); - assertThat(container2.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - LOG.info("Container2 Inspect: {}", inspectContainerResponse2.toString()); - - assertThat(inspectContainerResponse2.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse2.getHostConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[]{new Link("container1", - "container1Link")})); - assertThat(inspectContainerResponse2.getId(), startsWith(container2.getId())); - assertThat(inspectContainerResponse2.getName(), equalTo("/container2")); - assertThat(inspectContainerResponse2.getImageId(), not(isEmptyString())); - - } - - @Test - public void createContainerWithRestartPolicy() throws DockerException { - - RestartPolicy restartPolicy = RestartPolicy.onFailureRestart(5); - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("sleep", "9999") - .withRestartPolicy(restartPolicy).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getHostConfig().getRestartPolicy(), is(equalTo(restartPolicy))); - } - - @Test - public void createContainerWithPidMode() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("true") - .withPidMode("host").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getHostConfig().getPidMode(), is(equalTo("host"))); - } - - /** - * This tests support for --net option for the docker run command: --net="bridge" Set the Network mode for the container 'bridge': - * creates a new network stack for the container on the docker bridge 'none': no networking for this container 'container:': reuses - * another container network stack 'host': use the host network stack inside the container. Note: the host mode gives the container full - * access to local system services such as D-bus and is therefore considered insecure. - */ - @Test - public void createContainerWithNetworkMode() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("true") - .withNetworkMode("host").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getHostConfig().getNetworkMode(), is(equalTo("host"))); - } - - @Test - public void createContainerWithMacAddress() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withMacAddress("00:80:41:ae:fd:7e").withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertEquals(inspectContainerResponse.getConfig().getMacAddress(), "00:80:41:ae:fd:7e"); - } - - @Test(groups = "ignoreInCircleCi") - public void createContainerWithULimits() throws DockerException { - - Ulimit[] ulimits = {new Ulimit("nproc", 709, 1026), new Ulimit("nofile", 1024, 4096)}; - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withName("container") - .withUlimits(ulimits).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getUlimits()), - containsInAnyOrder(new Ulimit("nproc", 709, 1026), new Ulimit("nofile", 1024, 4096))); - - } - - @Test(groups = "ignoreInCircleCi") - public void createContainerWithLabels() throws DockerException { - - Map labels = new HashMap(); - labels.put("com.github.dockerjava.null", null); - labels.put("com.github.dockerjava.Boolean", "true"); - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withCmd("sleep", "9999") - .withLabels(labels).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - // null becomes empty string - labels.put("com.github.dockerjava.null", ""); - assertThat(inspectContainerResponse.getConfig().getLabels(), is(equalTo(labels))); - } - - @Test(groups = "ignoreInCircleCi") - public void createContainerWithLogConfig() throws DockerException { - - LogConfig logConfig = new LogConfig(LogConfig.LoggingType.NONE, null); - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE).withLogConfig(logConfig).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - // null becomes empty string - assertEquals(inspectContainerResponse.getHostConfig().getLogConfig().type, logConfig.type); - } - - /** - * https://github.com/calavera/docker/blob/3781cde61ff10b1d9114ae5b4c5c1d1b2c20a1ee/integration-cli/docker_cli_run_unix_test.go#L319-L333 - */ - @Test - public void testWithStopSignal() throws Exception { - Integer signal = 10; // SIGUSR1 in busybox - - CreateContainerResponse resp = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withCmd("/bin/sh", "-c", "trap 'echo \"exit trapped 10\"; exit 10' USR1; while true; do sleep 1; done") - .withAttachStdin(true) - .withTty(true) - .withStopSignal(signal.toString()) - .exec(); - final String containerId = resp.getId(); - assertThat(containerId, not(isEmptyString())); - dockerClient.startContainerCmd(containerId).exec(); - - InspectContainerResponse inspect = dockerClient.inspectContainerCmd(containerId).exec(); - assertThat(inspect.getState().getRunning(), is(true)); - - dockerClient.stopContainerCmd(containerId).exec(); - Thread.sleep(TimeUnit.SECONDS.toMillis(3)); - - inspect = dockerClient.inspectContainerCmd(containerId).exec(); - assertThat(inspect.getState().getRunning(), is(false)); - assertThat(inspect.getState().getExitCode(), is(signal)); - - StringBuilder stringBuilder = new StringBuilder(); - final StringBuilderLogReader callback = new StringBuilderLogReader(stringBuilder); - dockerClient.logContainerCmd(containerId) - .withStdErr(true) - .withStdOut(true) - .withTailAll() - .exec(callback) - .awaitCompletion(); - - String log = callback.builder.toString(); - assertThat(log, is("exit trapped 10")); - } - - private static class StringBuilderLogReader extends LogContainerResultCallback { - public StringBuilder builder; - - public StringBuilderLogReader(StringBuilder builder) { - this.builder = builder; - } - - @Override - public void onNext(Frame item) { - builder.append(new String(item.getPayload()).trim()); - super.onNext(item); - } - } - - @Test(groups = "ignoreInCircleCi") - public void createContainerWithCgroupParent() throws DockerException { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCgroupParent("/parent").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainer = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainer.getHostConfig().getCgroupParent(), is("/parent")); - } - - @SuppressWarnings("Duplicates") - @Test - public void createContainerWithShmSize() throws DockerException { - HostConfig hostConfig = new HostConfig().withShmSize(96 * FileUtils.ONE_MB); - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withHostConfig(hostConfig).withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertEquals(inspectContainerResponse.getHostConfig().getShmSize(), hostConfig.getShmSize()); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/CreateNetworkCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/CreateNetworkCmdImplTest.java deleted file mode 100644 index cc70331f2..000000000 --- a/src/test/java/com/github/dockerjava/core/command/CreateNetworkCmdImplTest.java +++ /dev/null @@ -1,67 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.client.AbstractDockerClientTest; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; - -@Test(groups = "integration") -public class CreateNetworkCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void createNetwork() throws DockerException { - - String networkName = "testNetwork"; - - CreateNetworkResponse createNetworkResponse = dockerClient.createNetworkCmd().withName(networkName).exec(); - - assertNotNull(createNetworkResponse.getId()); - - Network network = dockerClient.inspectNetworkCmd().withNetworkId(createNetworkResponse.getId()).exec(); - assertEquals(network.getName(), networkName); - assertEquals(network.getDriver(), "bridge"); - } - - @Test - public void createNetworkWithIpamConfig() throws DockerException { - - String networkName = "testNetwork"; - Network.Ipam ipam = new Network.Ipam().withConfig(new Network.Ipam.Config().withSubnet("10.67.79.0/24")); - CreateNetworkResponse createNetworkResponse = dockerClient.createNetworkCmd().withName(networkName).withIpam(ipam).exec(); - - assertNotNull(createNetworkResponse.getId()); - - Network network = dockerClient.inspectNetworkCmd().withNetworkId(createNetworkResponse.getId()).exec(); - assertEquals(network.getName(), networkName); - assertEquals(network.getDriver(), "bridge"); - assertEquals("10.67.79.0/24", network.getIpam().getConfig().iterator().next().getSubnet()); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/CreateVolumeCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/CreateVolumeCmdImplTest.java deleted file mode 100644 index 39688b424..000000000 --- a/src/test/java/com/github/dockerjava/core/command/CreateVolumeCmdImplTest.java +++ /dev/null @@ -1,75 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateVolumeResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class CreateVolumeCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void createVolume() throws DockerException { - - String volumeName = "volume1"; - - CreateVolumeResponse createVolumeResponse = dockerClient.createVolumeCmd().withName(volumeName) - .withDriver("local").exec(); - - assertThat(createVolumeResponse.getName(), equalTo(volumeName)); - assertThat(createVolumeResponse.getDriver(), equalTo("local")); - assertThat(createVolumeResponse.getMountpoint(), containsString("/volume1/")); - } - - @Test - public void createVolumeWithExistingName() throws DockerException { - - String volumeName = "volume1"; - - CreateVolumeResponse createVolumeResponse1 = dockerClient.createVolumeCmd().withName(volumeName) - .withDriver("local").exec(); - - assertThat(createVolumeResponse1.getName(), equalTo(volumeName)); - assertThat(createVolumeResponse1.getDriver(), equalTo("local")); - assertThat(createVolumeResponse1.getMountpoint(), containsString("/volume1/")); - - CreateVolumeResponse createVolumeResponse2 = dockerClient.createVolumeCmd().withName(volumeName) - .withDriver("local").exec(); - - assertThat(createVolumeResponse2.getName(), equalTo(volumeName)); - assertThat(createVolumeResponse2.getDriver(), equalTo("local")); - assertThat(createVolumeResponse2.getMountpoint(), equalTo(createVolumeResponse1.getMountpoint())); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/DisconnectFromNetworkCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/DisconnectFromNetworkCmdImplTest.java deleted file mode 100644 index c4219b54e..000000000 --- a/src/test/java/com/github/dockerjava/core/command/DisconnectFromNetworkCmdImplTest.java +++ /dev/null @@ -1,81 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.client.AbstractDockerClientTest; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; - -@Test(groups = "integration") -public class DisconnectFromNetworkCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void disconnectFromNetwork() throws InterruptedException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - dockerClient.startContainerCmd(container.getId()).exec(); - - CreateNetworkResponse network = dockerClient.createNetworkCmd().withName("testNetwork").exec(); - - dockerClient.connectToNetworkCmd().withNetworkId(network.getId()).withContainerId(container.getId()).exec(); - - Network updatedNetwork = dockerClient.inspectNetworkCmd().withNetworkId(network.getId()).exec(); - - assertTrue(updatedNetwork.getContainers().containsKey(container.getId())); - - dockerClient.disconnectFromNetworkCmd().withNetworkId(network.getId()).withContainerId(container.getId()).exec(); - - updatedNetwork = dockerClient.inspectNetworkCmd().withNetworkId(network.getId()).exec(); - - assertFalse(updatedNetwork.getContainers().containsKey(container.getId())); - } - - @Test - public void forceDisconnectFromNetwork() throws InterruptedException { - - CreateNetworkResponse network = dockerClient.createNetworkCmd().withName("testNetwork").exec(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withNetworkMode("testNetwork") - .withCmd("sleep", "9999") - .exec(); - - dockerClient.startContainerCmd(container.getId()).exec(); - - dockerClient.disconnectFromNetworkCmd() - .withNetworkId(network.getId()) - .withContainerId(container.getId()) - .withForce(true) - .exec(); - - Network updatedNetwork = dockerClient.inspectNetworkCmd().withNetworkId(network.getId()).exec(); - assertFalse(updatedNetwork.getContainers().containsKey(container.getId())); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/DockerfileFixture.java b/src/test/java/com/github/dockerjava/core/command/DockerfileFixture.java deleted file mode 100644 index acb8b8a5d..000000000 --- a/src/test/java/com/github/dockerjava/core/command/DockerfileFixture.java +++ /dev/null @@ -1,81 +0,0 @@ -package com.github.dockerjava.core.command; - -import java.io.File; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.api.DockerClient; -import com.github.dockerjava.api.exception.InternalServerErrorException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Image; - -/** - * Start and stop a single container for testing. - */ -public class DockerfileFixture implements AutoCloseable { - - private static final Logger LOGGER = LoggerFactory.getLogger(DockerfileFixture.class); - - private final DockerClient dockerClient; - - private String directory; - - private String repository; - - private String containerId; - - public DockerfileFixture(DockerClient dockerClient, String directory) { - this.dockerClient = dockerClient; - this.directory = directory; - } - - public void open() throws Exception { - - LOGGER.info("building {}", directory); - - dockerClient.buildImageCmd(new File("src/test/resources", directory)).withNoCache(true) - .exec(new BuildImageResultCallback()).awaitImageId(); - - Image lastCreatedImage = dockerClient.listImagesCmd().exec().get(0); - - repository = lastCreatedImage.getRepoTags()[0]; - - LOGGER.info("created {} {}", lastCreatedImage.getId(), repository); - - containerId = dockerClient.createContainerCmd(lastCreatedImage.getId()).exec().getId(); - - LOGGER.info("starting {}", containerId); - - dockerClient.startContainerCmd(containerId).exec(); - } - - @Override - public void close() throws Exception { - - if (containerId != null) { - LOGGER.info("removing container {}", containerId); - try { - dockerClient.removeContainerCmd(containerId).withForce(true) // stop too - .exec(); - } catch (NotFoundException | InternalServerErrorException ignored) { - LOGGER.info("ignoring {}", ignored.getMessage()); - } - containerId = null; - } - - if (repository != null) { - LOGGER.info("removing repository {}", repository); - try { - dockerClient.removeImageCmd(repository).withForce(true).exec(); - } catch (NotFoundException | InternalServerErrorException e) { - LOGGER.info("ignoring {}", e.getMessage()); - } - repository = null; - } - } - - public String getContainerId() { - return containerId; - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/EventsCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/EventsCmdImplTest.java deleted file mode 100644 index 939f95087..000000000 --- a/src/test/java/com/github/dockerjava/core/command/EventsCmdImplTest.java +++ /dev/null @@ -1,159 +0,0 @@ -package com.github.dockerjava.core.command; - -import java.lang.reflect.Method; -import java.util.ArrayList; -import java.util.List; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.TimeUnit; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.model.Event; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class EventsCmdImplTest extends AbstractDockerClientTest { - - private static int KNOWN_NUM_EVENTS = 4; - - private static String getEpochTime() { - return String.valueOf(System.currentTimeMillis() / 1000); - } - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - /* - * This specific test may fail with boot2docker as time may not in sync with host system - */ - @Test - public void testEventStreamTimeBound() throws Exception { - // Don't include other tests events - TimeUnit.SECONDS.sleep(1); - - String startTime = getEpochTime(); - int expectedEvents = generateEvents(); - String endTime = getEpochTime(); - - CountDownLatch countDownLatch = new CountDownLatch(expectedEvents); - EventsTestCallback eventCallback = new EventsTestCallback(countDownLatch); - - dockerClient.eventsCmd().withSince(startTime).withUntil(endTime).exec(eventCallback); - - Boolean zeroCount = countDownLatch.await(10, TimeUnit.SECONDS); - - eventCallback.close(); - - assertTrue(zeroCount, "Received only: " + eventCallback.getEvents()); - } - - @Test - public void testEventStreaming1() throws Exception { - // Don't include other tests events - TimeUnit.SECONDS.sleep(1); - - CountDownLatch countDownLatch = new CountDownLatch(KNOWN_NUM_EVENTS); - EventsTestCallback eventCallback = new EventsTestCallback(countDownLatch); - - dockerClient.eventsCmd().withSince(getEpochTime()).exec(eventCallback); - - generateEvents(); - - Boolean zeroCount = countDownLatch.await(10, TimeUnit.SECONDS); - - eventCallback.close(); - assertTrue(zeroCount, "Received only: " + eventCallback.getEvents()); - } - - @Test - public void testEventStreaming2() throws Exception { - // Don't include other tests events - TimeUnit.SECONDS.sleep(1); - - CountDownLatch countDownLatch = new CountDownLatch(KNOWN_NUM_EVENTS); - EventsTestCallback eventCallback = new EventsTestCallback(countDownLatch); - - dockerClient.eventsCmd().withSince(getEpochTime()).exec(eventCallback); - - generateEvents(); - - Boolean zeroCount = countDownLatch.await(10, TimeUnit.SECONDS); - - eventCallback.close(); - assertTrue(zeroCount, "Received only: " + eventCallback.getEvents()); - } - - public void testEventStreamingWithFilter() throws Exception { - // Don't include other tests events - TimeUnit.SECONDS.sleep(1); - - CountDownLatch countDownLatch = new CountDownLatch(1); - EventsTestCallback eventCallback = dockerClient.eventsCmd().withEventFilter("start") - .exec(new EventsTestCallback(countDownLatch)); - - generateEvents(); - - Boolean zeroCount = countDownLatch.await(10, TimeUnit.SECONDS); - - eventCallback.close(); - assertTrue(zeroCount, "Received only: " + eventCallback.getEvents()); - } - - /** - * This method generates {#link KNOWN_NUM_EVENTS} events - */ - private int generateEvents() throws Exception { - String testImage = "busybox"; - - dockerClient.pullImageCmd(testImage).exec(new PullImageResultCallback()).awaitSuccess(); - - CreateContainerResponse container = dockerClient.createContainerCmd(testImage).withCmd("sleep", "9999").exec(); - dockerClient.startContainerCmd(container.getId()).exec(); - dockerClient.stopContainerCmd(container.getId()).exec(); - return KNOWN_NUM_EVENTS; - } - - private class EventsTestCallback extends EventsResultCallback { - - private final CountDownLatch countDownLatch; - - private final List events = new ArrayList(); - - public EventsTestCallback(CountDownLatch countDownLatch) { - this.countDownLatch = countDownLatch; - } - - public void onNext(Event event) { - LOG.info("Received event #{}: {}", countDownLatch.getCount(), event); - countDownLatch.countDown(); - events.add(event); - } - - public List getEvents() { - return new ArrayList(events); - } - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/ExecCreateCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/ExecCreateCmdImplTest.java deleted file mode 100644 index 139c81f56..000000000 --- a/src/test/java/com/github/dockerjava/core/command/ExecCreateCmdImplTest.java +++ /dev/null @@ -1,61 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; -import java.security.SecureRandom; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.ExecCreateCmdResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class ExecCreateCmdImplTest extends AbstractDockerClientTest { - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void execCreateTest() { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("top") - .withName(containerName).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - ExecCreateCmdResponse execCreateCmdResponse = dockerClient.execCreateCmd(container.getId()) - .withCmd("touch", "file.log").exec(); - - assertThat(execCreateCmdResponse.getId(), not(isEmptyString())); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/ExecStartCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/ExecStartCmdImplTest.java deleted file mode 100644 index c4f0b9189..000000000 --- a/src/test/java/com/github/dockerjava/core/command/ExecStartCmdImplTest.java +++ /dev/null @@ -1,95 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.io.InputStream; -import java.lang.reflect.Method; -import java.security.SecureRandom; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.ExecCreateCmdResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class ExecStartCmdImplTest extends AbstractDockerClientTest { - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void execStart() throws Exception { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("top") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - ExecCreateCmdResponse execCreateCmdResponse = dockerClient.execCreateCmd(container.getId()) - .withAttachStdout(true).withCmd("touch", "/execStartTest.log").exec(); - dockerClient.execStartCmd(execCreateCmdResponse.getId()).exec( - new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); - - InputStream response = dockerClient.copyArchiveFromContainerCmd(container.getId(), "/execStartTest.log").exec(); - Boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied from the container."); - - // read the stream fully. Otherwise, the underlying stream will not be closed. - String responseAsString = asString(response); - assertNotNull(responseAsString); - assertTrue(responseAsString.length() > 0); - } - - @Test(groups = "ignoreInCircleCi") - public void execStartAttached() throws Exception { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - ExecCreateCmdResponse execCreateCmdResponse = dockerClient.execCreateCmd(container.getId()) - .withAttachStdout(true).withCmd("touch", "/execStartTest.log").exec(); - dockerClient.execStartCmd(execCreateCmdResponse.getId()).withDetach(false).withTty(true) - .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); - - InputStream response = dockerClient.copyArchiveFromContainerCmd(container.getId(), "/execStartTest.log").exec(); - Boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied from the container."); - - // read the stream fully. Otherwise, the underlying stream will not be closed. - String responseAsString = asString(response); - assertNotNull(responseAsString); - assertTrue(responseAsString.length() > 0); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/FrameReaderITest.java b/src/test/java/com/github/dockerjava/core/command/FrameReaderITest.java deleted file mode 100644 index 06d07667c..000000000 --- a/src/test/java/com/github/dockerjava/core/command/FrameReaderITest.java +++ /dev/null @@ -1,117 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.contains; -import static org.hamcrest.Matchers.containsInAnyOrder; -import static org.hamcrest.Matchers.hasEntry; -import static org.hamcrest.Matchers.hasProperty; -import static org.hamcrest.Matchers.hasSize; -import static org.hamcrest.Matchers.is; -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertFalse; - -import java.util.ArrayList; -import java.util.Iterator; -import java.util.LinkedHashSet; -import java.util.List; - -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.DockerClient; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.api.model.StreamType; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.core.DockerClientBuilder; - -@Test(groups = "integration") -public class FrameReaderITest { - - private DockerClient dockerClient; - - private DockerfileFixture dockerfileFixture; - - @BeforeTest - public void beforeTest() throws Exception { - dockerClient = DockerClientBuilder.getInstance().build(); - dockerfileFixture = new DockerfileFixture(dockerClient, "frameReaderDockerfile"); - dockerfileFixture.open(); - } - - @AfterTest - public void deleteDockerContainerImage() throws Exception { - dockerfileFixture.close(); - dockerClient.close(); - } - - @Test - public void canCloseFrameReaderAndReadExpectedLines() throws Exception { - // wait for the container to be successfully executed - int exitCode = dockerClient.waitContainerCmd(dockerfileFixture.getContainerId()) - .exec(new WaitContainerResultCallback()).awaitStatusCode(); - assertEquals(0, exitCode); - - final List loggingFrames = getLoggingFrames(); - final Frame outFrame = new Frame(StreamType.STDOUT, "to stdout\n".getBytes()); - final Frame errFrame = new Frame(StreamType.STDERR, "to stderr\n".getBytes()); - - assertThat(loggingFrames, containsInAnyOrder(outFrame, errFrame)); - assertThat(loggingFrames, hasSize(2)); - } - - private List getLoggingFrames() throws Exception { - - FrameReaderITestCallback collectFramesCallback = new FrameReaderITestCallback(); - - dockerClient.logContainerCmd(dockerfileFixture.getContainerId()).withStdOut(true).withStdErr(true) - .withTailAll() - // we can't follow stream here as it blocks reading from resulting InputStream infinitely - // .withFollowStream() - .exec(collectFramesCallback).awaitCompletion(); - - return collectFramesCallback.frames; - } - - @Test - public void canLogInOneThreadAndExecuteCommandsInAnother() throws Exception { - - Thread thread = new Thread(new Runnable() { - @Override - public void run() { - try { - - Iterator frames = getLoggingFrames().iterator(); - - while (frames.hasNext()) { - frames.next(); - } - - } catch (Exception e) { - throw new RuntimeException(e); - } - } - }); - - thread.start(); - - try (DockerfileFixture busyboxDockerfile = new DockerfileFixture(dockerClient, "busyboxDockerfile")) { - busyboxDockerfile.open(); - } - - thread.join(); - - } - - public static class FrameReaderITestCallback extends LogContainerResultCallback { - - public List frames = new ArrayList(); - - @Override - public void onNext(Frame item) { - frames.add(item); - super.onNext(item); - } - - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/InfoCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/InfoCmdImplTest.java deleted file mode 100644 index d234efe4c..000000000 --- a/src/test/java/com/github/dockerjava/core/command/InfoCmdImplTest.java +++ /dev/null @@ -1,71 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyOrNullString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.model.Info; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class InfoCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void info() throws DockerException { - // Make sure that there is at least one container for the assertion - // TODO extract this into a shared method - if (dockerClient.listContainersCmd().withShowAll(true).exec().size() == 0) { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("docker-java-itest-info").withCmd("touch", "/test").exec(); - - LOG.info("Created container: {}", container); - assertThat(container.getId(), not(isEmptyOrNullString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - } - - Info dockerInfo = dockerClient.infoCmd().exec(); - LOG.info(dockerInfo.toString()); - - assertTrue(dockerInfo.toString().contains("containers")); - assertTrue(dockerInfo.toString().contains("images")); - assertTrue(dockerInfo.toString().contains("debug")); - - assertTrue(dockerInfo.getContainers() > 0); - assertTrue(dockerInfo.getImages() > 0); - assertTrue(dockerInfo.getNFd() > 0); - assertTrue(dockerInfo.getNGoroutines() > 0); - assertTrue(dockerInfo.getNCPU() > 0); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/InspectContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/InspectContainerCmdImplTest.java deleted file mode 100644 index c0eaab3a6..000000000 --- a/src/test/java/com/github/dockerjava/core/command/InspectContainerCmdImplTest.java +++ /dev/null @@ -1,119 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; -import java.security.SecureRandom; - -import com.github.dockerjava.api.command.InspectContainerCmd; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class InspectContainerCmdImplTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(InspectContainerCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test() - public void inspectContainer() throws DockerException { - - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("top") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse containerInfo = dockerClient.inspectContainerCmd(container.getId()).exec(); - assertEquals(containerInfo.getId(), container.getId()); - - } - - @Test() - public void inspectContainerWithSize() throws DockerException { - - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("top") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerCmd command = dockerClient.inspectContainerCmd(container.getId()).withSize(true); - assertTrue(command.getSize()); - InspectContainerResponse containerInfo = command.exec(); - assertEquals(containerInfo.getId(), container.getId()); - assertNotNull(containerInfo.getSizeRootFs()); - assertTrue(containerInfo.getSizeRootFs().intValue() > 0 ); - } - - @Test(expectedExceptions = NotFoundException.class) - public void inspectNonExistingContainer() throws DockerException { - dockerClient.inspectContainerCmd("non-existing").exec(); - } - - @Test - public void inspectContainerRestartCount() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("env").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getRestartCount(), equalTo(0)); - } - - @Test - public void inspectContainerNetworkSettings() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("env").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertFalse(inspectContainerResponse.getNetworkSettings().getHairpinMode()); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/InspectExecCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/InspectExecCmdImplTest.java deleted file mode 100644 index d86bcc39e..000000000 --- a/src/test/java/com/github/dockerjava/core/command/InspectExecCmdImplTest.java +++ /dev/null @@ -1,144 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_22; -import static com.github.dockerjava.utils.TestUtils.getVersion; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.notNullValue; -import static org.hamcrest.Matchers.nullValue; - -import java.io.IOException; -import java.lang.reflect.Method; -import java.security.SecureRandom; - -import com.github.dockerjava.core.RemoteApiVersion; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.ExecCreateCmdResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.command.InspectExecResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.test.serdes.JSONTestHelper; - -@Test(groups = "integration") -public class InspectExecCmdImplTest extends AbstractDockerClientTest { - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void inspectExec() throws Exception { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - // Check that file does not exist - ExecCreateCmdResponse checkFileExec1 = dockerClient.execCreateCmd(container.getId()).withAttachStdout(true) - .withAttachStderr(true).withCmd("test", "-e", "/marker").exec(); - LOG.info("Created exec {}", checkFileExec1.toString()); - assertThat(checkFileExec1.getId(), not(isEmptyString())); - dockerClient.execStartCmd(checkFileExec1.getId()).withDetach(false) - .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); - InspectExecResponse first = dockerClient.inspectExecCmd(checkFileExec1.getId()).exec(); - assertThat(first.isRunning(), is(false)); - assertThat(first.getExitCode(), is(1)); - - // Create the file - ExecCreateCmdResponse touchFileExec = dockerClient.execCreateCmd(container.getId()).withAttachStdout(true) - .withAttachStderr(true).withCmd("touch", "/marker").exec(); - LOG.info("Created exec {}", touchFileExec.toString()); - assertThat(touchFileExec.getId(), not(isEmptyString())); - dockerClient.execStartCmd(touchFileExec.getId()).withDetach(false) - .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); - InspectExecResponse second = dockerClient.inspectExecCmd(touchFileExec.getId()).exec(); - assertThat(second.isRunning(), is(false)); - assertThat(second.getExitCode(), is(0)); - - // Check that file does exist now - ExecCreateCmdResponse checkFileExec2 = dockerClient.execCreateCmd(container.getId()).withAttachStdout(true) - .withAttachStderr(true).withCmd("test", "-e", "/marker").exec(); - LOG.info("Created exec {}", checkFileExec2.toString()); - assertThat(checkFileExec2.getId(), not(isEmptyString())); - dockerClient.execStartCmd(checkFileExec2.getId()).withDetach(false) - .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); - InspectExecResponse third = dockerClient.inspectExecCmd(checkFileExec2.getId()).exec(); - assertThat(third.isRunning(), is(false)); - assertThat(third.getExitCode(), is(0)); - - // Get container info and check its roundtrip to ensure the consistency - InspectContainerResponse containerInfo = dockerClient.inspectContainerCmd(container.getId()).exec(); - assertEquals(containerInfo.getId(), container.getId()); - JSONTestHelper.testRoundTrip(containerInfo); - } - - @Test(groups = "ignoreInCircleCi") - public void inspectExecNetworkSettings() throws IOException { - final RemoteApiVersion apiVersion = getVersion(dockerClient); - - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - ExecCreateCmdResponse exec = dockerClient.execCreateCmd(container.getId()).withAttachStdout(true) - .withAttachStderr(true).withCmd("/bin/bash").exec(); - LOG.info("Created exec {}", exec.toString()); - assertThat(exec.getId(), not(isEmptyString())); - - InspectExecResponse inspectExecResponse = dockerClient.inspectExecCmd(exec.getId()).exec(); - - if (apiVersion.isGreaterOrEqual(RemoteApiVersion.VERSION_1_22)) { - assertThat(inspectExecResponse.getExitCode(), is(nullValue())); - assertThat(inspectExecResponse.getCanRemove(), is(false)); - assertThat(inspectExecResponse.getContainerID(), is(container.getId())); - } else { - assertThat(inspectExecResponse.getExitCode(), is(0)); - assertNotNull(inspectExecResponse.getContainer().getNetworkSettings().getNetworks().get("bridge")); - } - - assertThat(inspectExecResponse.isOpenStdin(), is(false)); - assertThat(inspectExecResponse.isOpenStdout(), is(true)); - assertThat(inspectExecResponse.isRunning(), is(false)); - - final InspectExecResponse.Container inspectContainer = inspectExecResponse.getContainer(); - if (apiVersion.isGreaterOrEqual(VERSION_1_22)) { - assertThat(inspectContainer, nullValue()); - } else { - assertThat(inspectContainer, notNullValue()); - assertNotNull(inspectContainer.getNetworkSettings().getNetworks().get("bridge")); - } - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/InspectNetworkCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/InspectNetworkCmdImplTest.java deleted file mode 100644 index 9d992eb15..000000000 --- a/src/test/java/com/github/dockerjava/core/command/InspectNetworkCmdImplTest.java +++ /dev/null @@ -1,57 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.client.AbstractDockerClientTest; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; -import java.util.List; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; - -@Test(groups = "integration") -public class InspectNetworkCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void inspectNetwork() throws DockerException { - - List networks = dockerClient.listNetworksCmd().exec(); - - Network expected = findNetwork(networks, "bridge"); - - Network network = dockerClient.inspectNetworkCmd().withNetworkId(expected.getId()).exec(); - - assertThat(network.getName(), equalTo(expected.getName())); - assertThat(network.getScope(), equalTo(expected.getScope())); - assertThat(network.getDriver(), equalTo(expected.getDriver())); - assertThat(network.getIpam().getConfig().get(0).getSubnet(), equalTo(expected.getIpam().getConfig().get(0).getSubnet())); - assertThat(network.getIpam().getDriver(), equalTo(expected.getIpam().getDriver())); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/InspectVolumeCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/InspectVolumeCmdImplTest.java deleted file mode 100644 index c24f173df..000000000 --- a/src/test/java/com/github/dockerjava/core/command/InspectVolumeCmdImplTest.java +++ /dev/null @@ -1,65 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.InspectVolumeResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class InspectVolumeCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void inspectVolume() throws DockerException { - - String volumeName = "volume1"; - - dockerClient.createVolumeCmd().withName(volumeName).withDriver("local").exec(); - - InspectVolumeResponse inspectVolumeResponse = dockerClient.inspectVolumeCmd(volumeName).exec(); - - assertThat(inspectVolumeResponse.getName(), equalTo("volume1")); - assertThat(inspectVolumeResponse.getDriver(), equalTo("local")); - assertThat(inspectVolumeResponse.getMountpoint(), containsString("/volume1/")); - } - - @Test(expectedExceptions = NotFoundException.class) - public void inspectNonExistentVolume() throws DockerException { - - String volumeName = "non-existing"; - - dockerClient.inspectVolumeCmd(volumeName).exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/KillContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/KillContainerCmdImplTest.java deleted file mode 100644 index c8589271e..000000000 --- a/src/test/java/com/github/dockerjava/core/command/KillContainerCmdImplTest.java +++ /dev/null @@ -1,76 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class KillContainerCmdImplTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(KillContainerCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void killContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - LOG.info("Killing container: {}", container.getId()); - dockerClient.killContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); - assertThat(inspectContainerResponse.getState().getExitCode(), not(equalTo(0))); - - } - - @Test(expectedExceptions = NotFoundException.class) - public void killNonExistingContainer() throws DockerException { - - dockerClient.killContainerCmd("non-existing").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/ListContainersCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/ListContainersCmdImplTest.java deleted file mode 100644 index 7bbfb3945..000000000 --- a/src/test/java/com/github/dockerjava/core/command/ListContainersCmdImplTest.java +++ /dev/null @@ -1,170 +0,0 @@ -package com.github.dockerjava.core.command; - -import static ch.lambdaj.Lambda.filter; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.notNullValue; -import static org.hamcrest.Matchers.startsWith; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; -import java.util.List; -import java.util.Map; - -import org.hamcrest.Matcher; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.model.Container; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.google.common.collect.ImmutableMap; - -@Test(groups = "integration") -public class ListContainersCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void testListContainers() throws Exception { - - String testImage = "busybox"; - - // need to block until image is pulled completely - dockerClient.pullImageCmd(testImage).exec(new PullImageResultCallback()).awaitSuccess(); - - List containers = dockerClient.listContainersCmd().withShowAll(true).exec(); - assertThat(containers, notNullValue()); - LOG.info("Container List: {}", containers); - - int size = containers.size(); - - CreateContainerResponse container1 = dockerClient.createContainerCmd(testImage).withCmd("echo").exec(); - - assertThat(container1.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container1.getId()).exec(); - - assertThat(inspectContainerResponse.getConfig().getImage(), is(equalTo(testImage))); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - LOG.info("container id: " + container1.getId()); - - List containers2 = dockerClient.listContainersCmd().withShowAll(true).exec(); - - for (Container container : containers2) { - LOG.info("listContainer: id=" + container.getId() + " image=" + container.getImage()); - } - - assertThat(size + 1, is(equalTo(containers2.size()))); - Matcher matcher = hasItem(hasField("id", startsWith(container1.getId()))); - assertThat(containers2, matcher); - - List filteredContainers = filter(hasField("id", startsWith(container1.getId())), containers2); - assertThat(filteredContainers.size(), is(equalTo(1))); - - for (Container container : filteredContainers) { - LOG.info("filteredContainer: " + container); - } - - Container container2 = filteredContainers.get(0); - assertThat(container2.getCommand(), not(isEmptyString())); - assertThat(container2.getImage(), startsWith(testImage)); - } - - @Test - public void testListContainersWithLabelsFilter() throws Exception { - - String testImage = "busybox"; - - // need to block until image is pulled completely - dockerClient.pullImageCmd(testImage).exec(new PullImageResultCallback()).awaitCompletion(); - - List containers = dockerClient.listContainersCmd().withShowAll(true).exec(); - assertThat(containers, notNullValue()); - LOG.info("Container List: {}", containers); - - int size = containers.size(); - - CreateContainerResponse container1 = dockerClient.createContainerCmd(testImage).withCmd("echo").exec(); - - assertThat(container1.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container1.getId()).exec(); - - assertThat(inspectContainerResponse.getConfig().getImage(), is(equalTo(testImage))); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - LOG.info("container id: " + container1.getId()); - - List containers2 = dockerClient.listContainersCmd().withShowAll(true).exec(); - - for (Container container : containers2) { - LOG.info("listContainer: id=" + container.getId() + " image=" + container.getImage()); - } - - assertThat(size + 1, is(equalTo(containers2.size()))); - Matcher matcher = hasItem(hasField("id", startsWith(container1.getId()))); - assertThat(containers2, matcher); - - List filteredContainers = filter(hasField("id", startsWith(container1.getId())), containers2); - assertThat(filteredContainers.size(), is(equalTo(1))); - - for (Container container : filteredContainers) { - LOG.info("filteredContainer: " + container); - } - - Container container2 = filteredContainers.get(0); - assertThat(container2.getCommand(), not(isEmptyString())); - assertThat(container2.getImage(), startsWith(testImage)); - - Map labels = ImmutableMap.of("test", "docker-java"); - - // list with filter by label - dockerClient.createContainerCmd(testImage).withCmd("echo").withLabels(labels).exec(); - filteredContainers = dockerClient.listContainersCmd().withShowAll(true) - .withLabelFilter(labels).exec(); - assertThat(filteredContainers.size(), is(equalTo(1))); - Container container3 = filteredContainers.get(0); - assertThat(container3.getCommand(), not(isEmptyString())); - assertThat(container3.getImage(), startsWith(testImage)); - - filteredContainers = dockerClient.listContainersCmd().withShowAll(true) - .withLabelFilter("test").exec(); - assertThat(filteredContainers.size(), is(equalTo(1))); - container3 = filteredContainers.get(0); - assertThat(container3.getCommand(), not(isEmptyString())); - assertThat(container3.getImage(), startsWith(testImage)); - assertEquals(container3.getLabels(), labels); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/ListImagesCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/ListImagesCmdImplTest.java deleted file mode 100644 index 1b811867a..000000000 --- a/src/test/java/com/github/dockerjava/core/command/ListImagesCmdImplTest.java +++ /dev/null @@ -1,101 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.emptyArray; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.greaterThan; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.notNullValue; - -import java.lang.reflect.Method; -import java.util.List; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.model.Image; -import com.github.dockerjava.api.model.Info; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class ListImagesCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void listImages() throws DockerException { - List images = dockerClient.listImagesCmd().withShowAll(true).exec(); - assertThat(images, notNullValue()); - LOG.info("Images List: {}", images); - Info info = dockerClient.infoCmd().exec(); - - assertThat(images.size(), equalTo(info.getImages())); - - Image img = images.get(0); - assertThat(img.getCreated(), is(greaterThan(0L))); - assertThat(img.getVirtualSize(), is(greaterThan(0L))); - assertThat(img.getId(), not(isEmptyString())); - assertThat(img.getRepoTags(), not(emptyArray())); - } - - @Test(groups = "ignoreInCircleCi") - public void listImagesWithDanglingFilter() throws DockerException { - String imageId = createDanglingImage(); - List images = dockerClient.listImagesCmd().withDanglingFilter(true).withShowAll(true) - .exec(); - assertThat(images, notNullValue()); - LOG.info("Images List: {}", images); - assertThat(images.size(), is(greaterThan(0))); - Boolean imageInFilteredList = isImageInFilteredList(images, imageId); - assertTrue(imageInFilteredList); - } - - private boolean isImageInFilteredList(List images, String expectedImageId) { - for (Image image : images) { - if (expectedImageId.equals(image.getId())) { - return true; - } - } - return false; - } - - private String createDanglingImage() { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - LOG.info("Committing container {}", container.toString()); - String imageId = dockerClient.commitCmd(container.getId()).exec(); - - dockerClient.stopContainerCmd(container.getId()).exec(); - dockerClient.removeContainerCmd(container.getId()).exec(); - return imageId; - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/ListNetworksCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/ListNetworksCmdImplTest.java deleted file mode 100644 index 192adb4bc..000000000 --- a/src/test/java/com/github/dockerjava/core/command/ListNetworksCmdImplTest.java +++ /dev/null @@ -1,55 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; - -import java.lang.reflect.Method; -import java.util.List; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class ListNetworksCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void listNetworks() throws DockerException { - - List networks = dockerClient.listNetworksCmd().exec(); - - Network network = findNetwork(networks, "bridge"); - - assertThat(network.getName(), equalTo("bridge")); - assertThat(network.getScope(), equalTo("local")); - assertThat(network.getDriver(), equalTo("bridge")); - assertThat(network.getIpam().getDriver(), equalTo("default")); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/ListVolumesCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/ListVolumesCmdImplTest.java deleted file mode 100644 index 646f502c8..000000000 --- a/src/test/java/com/github/dockerjava/core/command/ListVolumesCmdImplTest.java +++ /dev/null @@ -1,59 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.greaterThanOrEqualTo; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateVolumeResponse; -import com.github.dockerjava.api.command.ListVolumesResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class ListVolumesCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void listVolumes() throws DockerException { - - CreateVolumeResponse createVolumeResponse = dockerClient.createVolumeCmd().withName("volume1") - .withDriver("local").exec(); - - assertThat(createVolumeResponse.getName(), equalTo("volume1")); - assertThat(createVolumeResponse.getDriver(), equalTo("local")); - assertThat(createVolumeResponse.getMountpoint(), containsString("/volume1/")); - - ListVolumesResponse listVolumesResponse = dockerClient.listVolumesCmd().exec(); - - assertThat(listVolumesResponse.getVolumes().size(), greaterThanOrEqualTo(1)); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/LoadImageCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/LoadImageCmdImplTest.java deleted file mode 100644 index 7cecf7334..000000000 --- a/src/test/java/com/github/dockerjava/core/command/LoadImageCmdImplTest.java +++ /dev/null @@ -1,76 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.model.Image; -import com.github.dockerjava.client.AbstractDockerClientTest; - -import com.github.dockerjava.utils.TestResources; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.io.InputStream; -import java.lang.reflect.Method; -import java.nio.file.Files; -import java.util.List; - -import static java.util.Arrays.asList; -import static java.util.Collections.singletonList; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.core.IsNull.notNullValue; -import static org.hamcrest.core.IsEqual.equalTo; - -@Test(groups = "integration") -public class LoadImageCmdImplTest extends AbstractDockerClientTest { - - private String expectedImageId; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - expectedImageId = "sha256:56031f66eb0cef2e2e5cb2d1dabafaa0ebcd0a18a507d313b5bdb8c0472c5eba"; - if (findImageWithId(expectedImageId, dockerClient.listImagesCmd().exec()) != null) { - dockerClient.removeImageCmd(expectedImageId).exec(); - } - } - - @AfterMethod - public void afterMethod(ITestResult result) { - dockerClient.removeImageCmd(expectedImageId).exec(); - super.afterMethod(result); - } - - @Test - public void loadImageFromTar() throws Exception { - try (InputStream uploadStream = Files.newInputStream(TestResources.getApiImagesLoadTestTarball())) { - dockerClient.loadImageCmd(uploadStream).exec(); - } - - final Image image = findImageWithId(expectedImageId, dockerClient.listImagesCmd().exec()); - - assertThat("Can't find expected image after loading from a tar archive!", image, notNullValue()); - assertThat("Image after loading from a tar archive has wrong tags!", - asList(image.getRepoTags()), equalTo(singletonList("docker-java/load:1.0"))); - } - - private Image findImageWithId(final String id, final List images) { - for (Image image : images) { - if (id.equals(image.getId())) { - return image; - } - } - return null; - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/LogContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/LogContainerCmdImplTest.java deleted file mode 100644 index 869bcbb1a..000000000 --- a/src/test/java/com/github/dockerjava/core/command/LogContainerCmdImplTest.java +++ /dev/null @@ -1,222 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.containsString; - -import java.io.IOException; -import java.lang.reflect.Method; -import java.util.concurrent.TimeUnit; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.StreamType; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class LogContainerCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void asyncLogContainerWithTtyEnabled() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("/bin/sh", "-c", "while true; do echo hello; sleep 1; done") - .withTty(true) - .exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()) - .exec(); - - LogContainerTestCallback loggingCallback = new LogContainerTestCallback(true); - - // this essentially test the since=0 case - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .withFollowStream(true) - .withTailAll() - .exec(loggingCallback); - - loggingCallback.awaitCompletion(3, TimeUnit.SECONDS); - - assertTrue(loggingCallback.toString().contains("hello")); - - assertEquals(loggingCallback.getCollectedFrames().get(0).getStreamType(), StreamType.RAW); - } - - @Test - public void asyncLogContainerWithTtyDisabled() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("/bin/sh", "-c", "while true; do echo hello; sleep 1; done") - .withTty(false) - .exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()) - .exec(); - - LogContainerTestCallback loggingCallback = new LogContainerTestCallback(true); - - // this essentially test the since=0 case - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .withFollowStream(true) - .withTailAll() - .exec(loggingCallback); - - loggingCallback.awaitCompletion(3, TimeUnit.SECONDS); - - assertTrue(loggingCallback.toString().contains("hello")); - - assertEquals(loggingCallback.getCollectedFrames().get(0).getStreamType(), StreamType.STDOUT); - } - - @Test - public void asyncLogNonExistingContainer() throws Exception { - - LogContainerTestCallback loggingCallback = new LogContainerTestCallback() { - @Override - public void onError(Throwable throwable) { - - assertEquals(throwable.getClass().getName(), NotFoundException.class.getName()); - - try { - // close the callback to prevent the call to onComplete - close(); - } catch (IOException e) { - throw new RuntimeException(); - } - - super.onError(throwable); - } - - public void onComplete() { - super.onComplete(); - fail("expected NotFoundException"); - }; - }; - - dockerClient.logContainerCmd("non-existing").withStdErr(true).withStdOut(true).exec(loggingCallback) - .awaitCompletion(); - } - - @Test - public void asyncMultipleLogContainer() throws Exception { - - String snippet = "hello world"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("/bin/echo", snippet) - .exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - int exitCode = dockerClient.waitContainerCmd(container.getId()) - .exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - - assertThat(exitCode, equalTo(0)); - - LogContainerTestCallback loggingCallback = new LogContainerTestCallback(); - - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .exec(loggingCallback); - - loggingCallback.close(); - - loggingCallback = new LogContainerTestCallback(); - - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .exec(loggingCallback); - - loggingCallback.close(); - - loggingCallback = new LogContainerTestCallback(); - - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .exec(loggingCallback); - - loggingCallback.awaitCompletion(); - - assertTrue(loggingCallback.toString().contains(snippet)); - } - - @Test - public void asyncLogContainerWithSince() throws Exception { - String snippet = "hello world"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("/bin/echo", snippet) - .exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - int timestamp = (int) (System.currentTimeMillis() / 1000); - - dockerClient.startContainerCmd(container.getId()).exec(); - - int exitCode = dockerClient.waitContainerCmd(container.getId()) - .exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - - assertThat(exitCode, equalTo(0)); - - LogContainerTestCallback loggingCallback = new LogContainerTestCallback(); - - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .withSince(timestamp) - .exec(loggingCallback); - - loggingCallback.awaitCompletion(); - - assertThat(loggingCallback.toString(), containsString(snippet)); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/PauseCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/PauseCmdImplTest.java deleted file mode 100644 index f100c4f89..000000000 --- a/src/test/java/com/github/dockerjava/core/command/PauseCmdImplTest.java +++ /dev/null @@ -1,103 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.InternalServerErrorException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.utils.ContainerUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -@Test(groups = {"integration", "ignoreInCircleCi"}) -public class PauseCmdImplTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(PauseCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void pauseRunningContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.pauseContainer(dockerClient, container); - } - - @Test(expectedExceptions = NotFoundException.class) - public void pauseNonExistingContainer() { - - dockerClient.pauseContainerCmd("non-existing").exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void pauseStoppedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.stopContainer(dockerClient, container); - - dockerClient.pauseContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void pausePausedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.pauseContainer(dockerClient, container); - - dockerClient.pauseContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void pauseCreatedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.pauseContainerCmd(container.getId()).exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/PullImageCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/PullImageCmdImplTest.java deleted file mode 100644 index d5e6d2c48..000000000 --- a/src/test/java/com/github/dockerjava/core/command/PullImageCmdImplTest.java +++ /dev/null @@ -1,117 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.lessThanOrEqualTo; -import static org.hamcrest.Matchers.notNullValue; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.InspectImageResponse; -import com.github.dockerjava.api.command.PullImageCmd; -import com.github.dockerjava.api.model.Info; -import com.github.dockerjava.api.model.PullResponseItem; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class PullImageCmdImplTest extends AbstractDockerClientTest { - - private static final PullImageCmd.Exec NOP_EXEC = new PullImageCmd.Exec() { - public Void exec(PullImageCmd command, ResultCallback resultCallback) { - return null; - }; - }; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void nullAuthConfig() throws Exception { - PullImageCmdImpl pullImageCmd = new PullImageCmdImpl(NOP_EXEC, null, ""); - try { - pullImageCmd.withAuthConfig(null); - fail(); - } catch (Exception e) { - assertEquals(e.getMessage(), "authConfig was not specified"); - } finally { - pullImageCmd.close(); - } - } - - @Test - public void testPullImage() throws Exception { - Info info = dockerClient.infoCmd().exec(); - LOG.info("Client info: {}", info.toString()); - - int imgCount = info.getImages(); - LOG.info("imgCount1: {}", imgCount); - - // This should be an image that is not used by other repositories - // already - // pulled down, preferably small in size. If tag is not used pull will - // download all images in that repository but tmpImgs will only - // deleted 'latest' image but not images with other tags - String testImage = "hackmann/empty"; - - LOG.info("Removing image: {}", testImage); - - try { - dockerClient.removeImageCmd(testImage).withForce(true).exec(); - } catch (NotFoundException e) { - // just ignore if not exist - } - - info = dockerClient.infoCmd().exec(); - LOG.info("Client info: {}", info.toString()); - - imgCount = info.getImages(); - LOG.info("imgCount2: {}", imgCount); - - LOG.info("Pulling image: {}", testImage); - - dockerClient.pullImageCmd(testImage).exec(new PullImageResultCallback()).awaitSuccess(); - - info = dockerClient.infoCmd().exec(); - LOG.info("Client info after pull, {}", info.toString()); - - assertThat(imgCount, lessThanOrEqualTo(info.getImages())); - - InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(testImage).exec(); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - assertThat(inspectImageResponse, notNullValue()); - } - - @Test - public void testPullNonExistingImage() throws Exception { - - // does not throw an exception - // stream needs to be fully read in order to close the underlying connection - dockerClient.pullImageCmd("xvxcv/foo").exec(new PullImageResultCallback()).awaitCompletion(); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/PushImageCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/PushImageCmdImplTest.java deleted file mode 100644 index 0fb32c311..000000000 --- a/src/test/java/com/github/dockerjava/core/command/PushImageCmdImplTest.java +++ /dev/null @@ -1,78 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = {"integration-auth", "integration"}) -public class PushImageCmdImplTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(PushImageCmdImplTest.class); - - String username; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - username = dockerClient.authConfig().getUsername(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void pushLatest() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - LOG.info("Committing container: {}", container.toString()); - String imageId = dockerClient.commitCmd(container.getId()).withRepository(username + "/busybox").exec(); - - // we have to block until image is pushed - dockerClient.pushImageCmd(username + "/busybox").exec(new PushImageResultCallback()).awaitSuccess(); - - LOG.info("Removing image: {}", imageId); - dockerClient.removeImageCmd(imageId).exec(); - - dockerClient.pullImageCmd(username + "/busybox").exec(new PullImageResultCallback()).awaitSuccess(); - } - - @Test(expectedExceptions = DockerClientException.class) - public void pushNonExistentImage() throws Exception { - - dockerClient.pushImageCmd(username + "/xxx").exec(new PushImageResultCallback()).awaitSuccess(); - - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/RemoveContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/RemoveContainerCmdImplTest.java deleted file mode 100644 index 26c8e0e19..000000000 --- a/src/test/java/com/github/dockerjava/core/command/RemoveContainerCmdImplTest.java +++ /dev/null @@ -1,77 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.startsWith; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; -import java.util.List; - -import org.hamcrest.Matcher; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.model.Container; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class RemoveContainerCmdImplTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(RemoveContainerCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void removeContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true").exec(); - - dockerClient.startContainerCmd(container.getId()).exec(); - dockerClient.waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()).awaitStatusCode(); - - LOG.info("Removing container: {}", container.getId()); - dockerClient.removeContainerCmd(container.getId()).exec(); - - List containers2 = dockerClient.listContainersCmd().withShowAll(true).exec(); - - Matcher matcher = not(hasItem(hasField("id", startsWith(container.getId())))); - assertThat(containers2, matcher); - - } - - @Test(expectedExceptions = NotFoundException.class) - public void removeNonExistingContainer() throws DockerException { - - dockerClient.removeContainerCmd("non-existing").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/RemoveImageCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/RemoveImageCmdImplTest.java deleted file mode 100644 index 4427da997..000000000 --- a/src/test/java/com/github/dockerjava/core/command/RemoveImageCmdImplTest.java +++ /dev/null @@ -1,83 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.startsWith; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; -import java.util.List; - -import org.hamcrest.Matcher; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.model.Container; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class RemoveImageCmdImplTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(RemoveImageCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void removeImage() throws DockerException, InterruptedException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - LOG.info("Committing container {}", container.toString()); - String imageId = dockerClient.commitCmd(container.getId()).exec(); - - dockerClient.stopContainerCmd(container.getId()).exec(); - dockerClient.removeContainerCmd(container.getId()).exec(); - - LOG.info("Removing image: {}", imageId); - dockerClient.removeImageCmd(imageId).exec(); - - List containers = dockerClient.listContainersCmd().withShowAll(true).exec(); - - Matcher matcher = not(hasItem(hasField("id", startsWith(imageId)))); - assertThat(containers, matcher); - } - - @Test(expectedExceptions = NotFoundException.class) - public void removeNonExistingImage() throws DockerException, InterruptedException { - - dockerClient.removeImageCmd("non-existing").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/RemoveNetworkCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/RemoveNetworkCmdImplTest.java deleted file mode 100644 index 228e48c4d..000000000 --- a/src/test/java/com/github/dockerjava/core/command/RemoveNetworkCmdImplTest.java +++ /dev/null @@ -1,73 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.client.AbstractDockerClientTest; -import org.hamcrest.Matcher; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; -import java.util.List; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.startsWith; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -@Test(groups = "integration") -public class RemoveNetworkCmdImplTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(RemoveNetworkCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void removeNetwork() throws DockerException { - - CreateNetworkResponse network = dockerClient.createNetworkCmd().withName("test-network").exec(); - - LOG.info("Removing network: {}", network.getId()); - dockerClient.removeNetworkCmd(network.getId()).exec(); - - List networks = dockerClient.listNetworksCmd().exec(); - - Matcher matcher = not(hasItem(hasField("id", startsWith(network.getId())))); - assertThat(networks, matcher); - - } - - @Test(expectedExceptions = NotFoundException.class) - public void removeNonExistingContainer() throws DockerException { - - dockerClient.removeNetworkCmd("non-existing").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/RemoveVolumeCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/RemoveVolumeCmdImplTest.java deleted file mode 100644 index 9ed690b65..000000000 --- a/src/test/java/com/github/dockerjava/core/command/RemoveVolumeCmdImplTest.java +++ /dev/null @@ -1,60 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateVolumeResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class RemoveVolumeCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(expectedExceptions = NotFoundException.class) - public void removeVolume() throws DockerException { - - String volumeName = "volume1"; - - CreateVolumeResponse createVolumeResponse = dockerClient.createVolumeCmd().withName(volumeName) - .withDriver("local").exec(); - - assertThat(createVolumeResponse.getName(), equalTo(volumeName)); - assertThat(createVolumeResponse.getDriver(), equalTo("local")); - assertThat(createVolumeResponse.getMountpoint(), containsString("/volume1/")); - - dockerClient.removeVolumeCmd(volumeName).exec(); - - dockerClient.inspectVolumeCmd(volumeName).exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/RenameContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/RenameContainerCmdImplTest.java deleted file mode 100644 index 582ab2556..000000000 --- a/src/test/java/com/github/dockerjava/core/command/RenameContainerCmdImplTest.java +++ /dev/null @@ -1,76 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.client.AbstractDockerClientTest; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -@Test(groups = "integration") -public class RenameContainerCmdImplTest extends AbstractDockerClientTest { - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void renameContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - String name1 = inspectContainerResponse.getName(); - - dockerClient.renameContainerCmd(container.getId()) - .withName(getClass().getCanonicalName() + "renameContainer") - .exec(); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect After Rename: {}", inspectContainerResponse2.toString()); - - String name2 = inspectContainerResponse2.getName(); - - assertNotEquals(name1, name2); - - dockerClient.killContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = NotFoundException.class) - public void renameExistingContainer() throws DockerException, InterruptedException { - dockerClient.renameContainerCmd("non-existing") - .withName(getClass().getCanonicalName() + "renameExistingContainer") - .exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/RestartContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/RestartContainerCmdImplTest.java deleted file mode 100644 index 84916f082..000000000 --- a/src/test/java/com/github/dockerjava/core/command/RestartContainerCmdImplTest.java +++ /dev/null @@ -1,79 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class RestartContainerCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void restartContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - String startTime = inspectContainerResponse.getState().getStartedAt(); - - dockerClient.restartContainerCmd(container.getId()).withtTimeout(2).exec(); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect After Restart: {}", inspectContainerResponse2.toString()); - - String startTime2 = inspectContainerResponse2.getState().getStartedAt(); - - assertThat(startTime, not(equalTo(startTime2))); - - assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(true))); - - dockerClient.killContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = NotFoundException.class) - public void restartNonExistingContainer() throws DockerException, InterruptedException { - - dockerClient.restartContainerCmd("non-existing").exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/SaveImageCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/SaveImageCmdImplTest.java deleted file mode 100644 index ffc81d4e0..000000000 --- a/src/test/java/com/github/dockerjava/core/command/SaveImageCmdImplTest.java +++ /dev/null @@ -1,55 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.greaterThan; - -import java.io.InputStream; -import java.lang.reflect.Method; - -import org.apache.commons.io.IOUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class SaveImageCmdImplTest extends AbstractDockerClientTest { - public static final Logger LOG = LoggerFactory.getLogger(SaveImageCmdImplTest.class); - - String username; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void saveImage() throws Exception { - - InputStream image = IOUtils.toBufferedInputStream(dockerClient.saveImageCmd("busybox").exec()); - assertThat(image.available(), greaterThan(0)); - - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/SearchImagesCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/SearchImagesCmdImplTest.java deleted file mode 100644 index 78e018427..000000000 --- a/src/test/java/com/github/dockerjava/core/command/SearchImagesCmdImplTest.java +++ /dev/null @@ -1,59 +0,0 @@ -package com.github.dockerjava.core.command; - -import static ch.lambdaj.Lambda.filter; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.is; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; -import java.util.List; - -import org.hamcrest.Matcher; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.SearchItem; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class SearchImagesCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void searchImages() throws DockerException { - List dockerSearch = dockerClient.searchImagesCmd("busybox").exec(); - LOG.info("Search returned {}", dockerSearch.toString()); - - Matcher matcher = hasItem(hasField("name", equalTo("busybox"))); - assertThat(dockerSearch, matcher); - - assertThat(filter(hasField("name", is("busybox")), dockerSearch).size(), equalTo(1)); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/StartContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/StartContainerCmdImplTest.java deleted file mode 100644 index 4bb8f2b99..000000000 --- a/src/test/java/com/github/dockerjava/core/command/StartContainerCmdImplTest.java +++ /dev/null @@ -1,551 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.command.StartContainerCmd; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.InternalServerErrorException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Bind; -import com.github.dockerjava.api.model.Device; -import com.github.dockerjava.api.model.ExposedPort; -import com.github.dockerjava.api.model.Link; -import com.github.dockerjava.api.model.Ports; -import com.github.dockerjava.api.model.RestartPolicy; -import com.github.dockerjava.api.model.Volume; -import com.github.dockerjava.api.model.VolumesFrom; -import com.github.dockerjava.api.model.Ports.Binding; -import com.github.dockerjava.client.AbstractDockerClientTest; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; -import java.util.Arrays; -import java.util.List; -import java.util.UUID; - -import static com.github.dockerjava.api.model.AccessMode.ro; -import static com.github.dockerjava.api.model.Capability.MKNOD; -import static com.github.dockerjava.api.model.Capability.NET_ADMIN; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.contains; -import static org.hamcrest.Matchers.containsInAnyOrder; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.hasSize; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.notNullValue; -import static org.hamcrest.Matchers.startsWith; - -@Test(groups = "integration") -public class StartContainerCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void startContainerWithVolumes() throws DockerException { - - // see http://docs.docker.io/use/working_with_volumes/ - Volume volume1 = new Volume("/opt/webapp1"); - - Volume volume2 = new Volume("/opt/webapp2"); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withVolumes(volume1, volume2) - .withCmd("true").withBinds(new Bind("/src/webapp1", volume1, ro), new Bind("/src/webapp2", volume2)) - .exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getConfig().getVolumes().keySet(), contains("/opt/webapp1", "/opt/webapp2")); - - dockerClient.startContainerCmd(container.getId()).exec(); - - dockerClient.waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()).awaitStatusCode(); - - inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse, mountedVolumes(containsInAnyOrder(volume1, volume2))); - - final List mounts = inspectContainerResponse.getMounts(); - - assertThat(mounts, hasSize(2)); - - final InspectContainerResponse.Mount mount1 = new InspectContainerResponse.Mount() - .withRw(false).withMode("ro").withDestination(volume1).withSource("/src/webapp1"); - final InspectContainerResponse.Mount mount2 = new InspectContainerResponse.Mount() - .withRw(true).withMode("rw").withDestination(volume2).withSource("/src/webapp2"); - - assertThat(mounts, containsInAnyOrder(mount1, mount2)); - } - - @Test - public void startContainerWithVolumesFrom() throws DockerException { - - Volume volume1 = new Volume("/opt/webapp1"); - Volume volume2 = new Volume("/opt/webapp2"); - - String container1Name = UUID.randomUUID().toString(); - - CreateContainerResponse container1 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(container1Name) - .withBinds(new Bind("/src/webapp1", volume1), new Bind("/src/webapp2", volume2)).exec(); - LOG.info("Created container1 {}", container1.toString()); - - dockerClient.startContainerCmd(container1.getId()).exec(); - LOG.info("Started container1 {}", container1.toString()); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - - assertThat(inspectContainerResponse1, mountedVolumes(containsInAnyOrder(volume1, volume2))); - - CreateContainerResponse container2 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withVolumesFrom(new VolumesFrom(container1Name)).exec(); - LOG.info("Created container2 {}", container2.toString()); - - dockerClient.startContainerCmd(container2.getId()).exec(); - LOG.info("Started container2 {}", container2.toString()); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - - assertThat(inspectContainerResponse2, mountedVolumes(containsInAnyOrder(volume1, volume2))); - } - - @Test - public void startContainerWithDns() throws DockerException { - - String aDnsServer = "8.8.8.8"; - String anotherDnsServer = "8.8.4.4"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withDns(aDnsServer, anotherDnsServer).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDns()), - contains(aDnsServer, anotherDnsServer)); - } - - @Test - public void startContainerWithDnsSearch() throws DockerException { - - String dnsSearch = "example.com"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withDnsSearch(dnsSearch).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - dockerClient.startContainerCmd(container.getId()).exec(); - - inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDnsSearch()), contains(dnsSearch)); - } - - @Test - public void startContainerWithPortBindings() throws DockerException { - - ExposedPort tcp22 = ExposedPort.tcp(22); - ExposedPort tcp23 = ExposedPort.tcp(23); - - Ports portBindings = new Ports(); - portBindings.bind(tcp22, Binding.bindPort(11022)); - portBindings.bind(tcp23, Binding.bindPort(11023)); - portBindings.bind(tcp23, Binding.bindPort(11024)); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withExposedPorts(tcp22, tcp23).withPortBindings(portBindings).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - dockerClient.startContainerCmd(container.getId()).exec(); - - inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getConfig().getExposedPorts()), contains(tcp22, tcp23)); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp22)[0], - is(equalTo(Binding.bindPort(11022)))); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[0], - is(equalTo(Binding.bindPort(11023)))); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[1], - is(equalTo(Binding.bindPort(11024)))); - - } - - @Test - public void startContainerWithRandomPortBindings() throws DockerException { - - ExposedPort tcp22 = ExposedPort.tcp(22); - ExposedPort tcp23 = ExposedPort.tcp(23); - - Ports portBindings = new Ports(); - portBindings.bind(tcp22, Binding.empty()); - portBindings.bind(tcp23, Binding.empty()); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withExposedPorts(tcp22, tcp23).withPortBindings(portBindings).withPublishAllPorts(true).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getConfig().getExposedPorts()), contains(tcp22, tcp23)); - - assertThat(inspectContainerResponse.getNetworkSettings().getPorts().getBindings().get(tcp22)[0].getHostPortSpec(), - is(not(equalTo(String.valueOf(tcp22.getPort()))))); - - assertThat(inspectContainerResponse.getNetworkSettings().getPorts().getBindings().get(tcp23)[0].getHostPortSpec(), - is(not(equalTo(String.valueOf(tcp23.getPort()))))); - - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void startContainerWithConflictingPortBindings() throws DockerException { - - ExposedPort tcp22 = ExposedPort.tcp(22); - ExposedPort tcp23 = ExposedPort.tcp(23); - - Ports portBindings = new Ports(); - portBindings.bind(tcp22, Binding.bindPort(11022)); - portBindings.bind(tcp23, Binding.bindPort(11022)); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withExposedPorts(tcp22, tcp23).withPortBindings(portBindings).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - } - - @Test - public void startContainerWithLinkingDeprecated() throws DockerException { - - CreateContainerResponse container1 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container1").exec(); - - LOG.info("Created container1 {}", container1.toString()); - assertThat(container1.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); - - assertThat(inspectContainerResponse1.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse1.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getId(), startsWith(container1.getId())); - assertThat(inspectContainerResponse1.getName(), equalTo("/container1")); - assertThat(inspectContainerResponse1.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getState(), is(notNullValue())); - assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); - - if (!inspectContainerResponse1.getState().getRunning()) { - assertThat(inspectContainerResponse1.getState().getExitCode(), is(equalTo(0))); - } - - CreateContainerResponse container2 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container2").withLinks(new Link("container1", "container1Link")).exec(); - - LOG.info("Created container2 {}", container2.toString()); - assertThat(container2.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container2.getId()).exec(); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - LOG.info("Container2 Inspect: {}", inspectContainerResponse2.toString()); - - assertThat(inspectContainerResponse2.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse2.getHostConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[] {new Link("container1", - "container1Link")})); - assertThat(inspectContainerResponse2.getId(), startsWith(container2.getId())); - assertThat(inspectContainerResponse2.getName(), equalTo("/container2")); - assertThat(inspectContainerResponse2.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse2.getState(), is(notNullValue())); - assertThat(inspectContainerResponse2.getState().getRunning(), is(true)); - - } - - @Test - public void startContainerWithLinking() throws DockerException { - - CreateContainerResponse container1 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container1").exec(); - - LOG.info("Created container1 {}", container1.toString()); - assertThat(container1.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); - - assertThat(inspectContainerResponse1.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse1.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getId(), startsWith(container1.getId())); - assertThat(inspectContainerResponse1.getName(), equalTo("/container1")); - assertThat(inspectContainerResponse1.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getState(), is(notNullValue())); - assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); - - if (!inspectContainerResponse1.getState().getRunning()) { - assertThat(inspectContainerResponse1.getState().getExitCode(), is(equalTo(0))); - } - - CreateContainerResponse container2 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container2").withLinks(new Link("container1", "container1Link")).exec(); - - LOG.info("Created container2 {}", container2.toString()); - assertThat(container2.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container2.getId()).exec(); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - LOG.info("Container2 Inspect: {}", inspectContainerResponse2.toString()); - - assertThat(inspectContainerResponse2.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse2.getHostConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[] {new Link("container1", - "container1Link")})); - assertThat(inspectContainerResponse2.getId(), startsWith(container2.getId())); - assertThat(inspectContainerResponse2.getName(), equalTo("/container2")); - assertThat(inspectContainerResponse2.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse2.getState(), is(notNullValue())); - assertThat(inspectContainerResponse2.getState().getRunning(), is(true)); - - } - - @Test - public void startContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd(new String[] {"top"}) - .exec(); - - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - assertThat(inspectContainerResponse.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse.getId(), not(isEmptyString())); - - assertThat(inspectContainerResponse.getId(), startsWith(container.getId())); - - assertThat(inspectContainerResponse.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse.getState(), is(notNullValue())); - - assertThat(inspectContainerResponse.getState().getRunning(), is(true)); - - if (!inspectContainerResponse.getState().getRunning()) { - assertThat(inspectContainerResponse.getState().getExitCode(), is(equalTo(0))); - } - } - - @Test(expectedExceptions = NotFoundException.class) - public void testStartNonExistingContainer() throws DockerException { - - dockerClient.startContainerCmd("non-existing").exec(); - } - - /** - * This tests support for --net option for the docker run command: --net="bridge" Set the Network mode for the container 'bridge': - * creates a new network stack for the container on the docker bridge 'none': no networking for this container 'container:': reuses - * another container network stack 'host': use the host network stack inside the container. Note: the host mode gives the container full - * access to local system services such as D-bus and is therefore considered insecure. - */ - @Test - public void startContainerWithNetworkMode() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withNetworkMode("host").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - dockerClient.startContainerCmd(container.getId()).exec(); - - inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getHostConfig().getNetworkMode(), is(equalTo("host"))); - } - - @Test - public void startContainerWithCapAddAndCapDrop() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withCapAdd(NET_ADMIN).withCapDrop(MKNOD).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getState().getRunning(), is(true)); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapAdd()), contains(NET_ADMIN)); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapDrop()), contains(MKNOD)); - } - - @Test - public void startContainerWithDevices() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withDevices(new Device("rwm", "/dev/nulo", "/dev/zero")).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getState().getRunning(), is(true)); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDevices()), contains(new Device("rwm", - "/dev/nulo", "/dev/zero"))); - } - - @Test - public void startContainerWithExtraHosts() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withExtraHosts("dockerhost:127.0.0.1").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getState().getRunning(), is(true)); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getExtraHosts()), - contains("dockerhost:127.0.0.1")); - } - - @Test - public void startContainerWithRestartPolicy() throws DockerException { - - RestartPolicy restartPolicy = RestartPolicy.onFailureRestart(5); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withRestartPolicy(restartPolicy).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getState().getRunning(), is(true)); - - assertThat(inspectContainerResponse.getHostConfig().getRestartPolicy(), is(equalTo(restartPolicy))); - } - - @Test - public void existingHostConfigIsPreservedByBlankStartCmd() throws DockerException { - - String dnsServer = "8.8.8.8"; - - // prepare a container with custom DNS - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withDns(dnsServer) - .withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - // start container _without_any_customization_ (important!) - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - // The DNS setting survived. - assertThat(inspectContainerResponse.getHostConfig().getDns(), is(notNullValue())); - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDns()), contains(dnsServer)); - } - - @Test - public void anUnconfiguredCommandSerializesToEmptyJson() throws Exception { - ObjectMapper objectMapper = new ObjectMapper(); - StartContainerCmd command = dockerClient.startContainerCmd(""); - assertThat(objectMapper.writeValueAsString(command), is("{}")); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/StatsCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/StatsCmdImplTest.java deleted file mode 100644 index d7f9ec1c5..000000000 --- a/src/test/java/com/github/dockerjava/core/command/StatsCmdImplTest.java +++ /dev/null @@ -1,106 +0,0 @@ -package com.github.dockerjava.core.command; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.io.IOException; -import java.lang.reflect.Method; -import java.security.SecureRandom; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.TimeUnit; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.model.Statistics; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.core.async.ResultCallbackTemplate; - -@Test(groups = "integration") -public class StatsCmdImplTest extends AbstractDockerClientTest { - - private static int NUM_STATS = 5; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void testStatsStreaming() throws InterruptedException, IOException { - TimeUnit.SECONDS.sleep(1); - - CountDownLatch countDownLatch = new CountDownLatch(NUM_STATS); - - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("top") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - StatsCallbackTest statsCallback = dockerClient.statsCmd(container.getId()).exec( - new StatsCallbackTest(countDownLatch)); - - countDownLatch.await(3, TimeUnit.SECONDS); - Boolean gotStats = statsCallback.gotStats(); - - LOG.info("Stop stats collection"); - - statsCallback.close(); - - LOG.info("Stopping container"); - dockerClient.stopContainerCmd(container.getId()).exec(); - dockerClient.removeContainerCmd(container.getId()).exec(); - - LOG.info("Completed test"); - assertTrue(gotStats, "Expected true"); - - } - - private class StatsCallbackTest extends ResultCallbackTemplate { - private final CountDownLatch countDownLatch; - - private Boolean gotStats = false; - - public StatsCallbackTest(CountDownLatch countDownLatch) { - this.countDownLatch = countDownLatch; - } - - @Override - public void onNext(Statistics stats) { - LOG.info("Received stats #{}: {}", countDownLatch.getCount(), stats); - if (stats != null) { - gotStats = true; - } - countDownLatch.countDown(); - } - - public Boolean gotStats() { - return gotStats; - } - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/StopContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/StopContainerCmdImplTest.java deleted file mode 100644 index 382e5b29c..000000000 --- a/src/test/java/com/github/dockerjava/core/command/StopContainerCmdImplTest.java +++ /dev/null @@ -1,85 +0,0 @@ -package com.github.dockerjava.core.command; - -import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_22; -import static com.github.dockerjava.utils.TestUtils.getVersion; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; - -import com.github.dockerjava.core.RemoteApiVersion; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class StopContainerCmdImplTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(StopContainerCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void testStopContainer() throws DockerException { - final RemoteApiVersion apiVersion = getVersion(dockerClient); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - LOG.info("Stopping container: {}", container.getId()); - dockerClient.stopContainerCmd(container.getId()).withTimeout(2).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); - - final Integer exitCode = inspectContainerResponse.getState().getExitCode(); - if (apiVersion.equals(VERSION_1_22)) { - assertThat(exitCode, is(0)); - } else { - assertThat(exitCode, not(0)); - } - } - - @Test(expectedExceptions = NotFoundException.class) - public void testStopNonExistingContainer() throws DockerException { - - dockerClient.stopContainerCmd("non-existing").withTimeout(2).exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/TagImageCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/TagImageCmdImplTest.java deleted file mode 100644 index 3bf033212..000000000 --- a/src/test/java/com/github/dockerjava/core/command/TagImageCmdImplTest.java +++ /dev/null @@ -1,59 +0,0 @@ -package com.github.dockerjava.core.command; - -import java.lang.reflect.Method; - -import org.apache.commons.lang.math.RandomUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class TagImageCmdImplTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(TagImageCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void tagImage() throws Exception { - String tag = "" + RandomUtils.nextInt(Integer.MAX_VALUE); - - dockerClient.tagImageCmd("busybox:latest", "docker-java/busybox", tag).exec(); - - dockerClient.removeImageCmd("docker-java/busybox:" + tag).exec(); - } - - @Test(expectedExceptions = NotFoundException.class) - public void tagNonExistingImage() throws Exception { - - String tag = "" + RandomUtils.nextInt(Integer.MAX_VALUE); - dockerClient.tagImageCmd("non-existing", "docker-java/busybox", tag).exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/UnpauseCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/UnpauseCmdImplTest.java deleted file mode 100644 index f640c4182..000000000 --- a/src/test/java/com/github/dockerjava/core/command/UnpauseCmdImplTest.java +++ /dev/null @@ -1,118 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.InternalServerErrorException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.utils.ContainerUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -@Test(groups = {"integration", "ignoreInCircleCi"}) -public class UnpauseCmdImplTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(UnpauseCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void unpausePausedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.pauseContainer(dockerClient, container); - - ContainerUtils.unpaseContainer(dockerClient, container); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void unpauseRunningContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - dockerClient.unpauseContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void unpauseStoppedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.stopContainer(dockerClient, container); - - dockerClient.unpauseContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = NotFoundException.class) - public void unpauseNonExistingContainer() { - - dockerClient.unpauseContainerCmd("non-existing").exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void unpauseCreatedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.unpauseContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void unpauseUnpausedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.pauseContainer(dockerClient, container); - - dockerClient.unpauseContainerCmd(container.getId()).exec(); - dockerClient.unpauseContainerCmd(container.getId()).exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/UpdateContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/UpdateContainerCmdImplTest.java deleted file mode 100644 index 3c7a9fd47..000000000 --- a/src/test/java/com/github/dockerjava/core/command/UpdateContainerCmdImplTest.java +++ /dev/null @@ -1,121 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.fasterxml.jackson.databind.JavaType; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.HostConfig; -import com.github.dockerjava.api.model.UpdateContainerResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.core.RemoteApiVersion; -import org.testng.ITestResult; -import org.testng.SkipException; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.io.IOException; -import java.lang.reflect.Method; - -import static com.github.dockerjava.test.serdes.JSONSamples.testRoundTrip; -import static com.github.dockerjava.utils.TestUtils.getVersion; -import static org.hamcrest.CoreMatchers.is; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.collection.IsCollectionWithSize.hasSize; -import static org.hamcrest.core.IsNull.notNullValue; - -/** - * @author Kanstantsin Shautsou - */ -@Test(groups = "integration") -public class UpdateContainerCmdImplTest extends AbstractDockerClientTest { - - public static final String BUSYBOX_IMAGE = "busybox"; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void updateContainer() throws DockerException, IOException { - final RemoteApiVersion apiVersion = getVersion(dockerClient); - - if (!apiVersion.isGreaterOrEqual(RemoteApiVersion.VERSION_1_22)) { - throw new SkipException("API version should be >= 1.22"); - } - - CreateContainerResponse response = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withCmd("sleep", "9999") - .exec(); - String containerId = response.getId(); - dockerClient.startContainerCmd(containerId).exec(); - - InspectContainerResponse inspectBefore = dockerClient.inspectContainerCmd(containerId).exec(); - - final UpdateContainerResponse updateResponse = dockerClient.updateContainerCmd(containerId) - .withBlkioWeight(300) - .withCpuShares(512) - .withCpuPeriod(100000) - .withCpuQuota(50000) -// .withCpusetCpus("0") // depends on env - .withCpusetMems("0") - .withMemory(314572800L) - .withMemorySwap(514288000L) - .withMemoryReservation(209715200L) -// .withKernelMemory(52428800) Can not update kernel memory to a running container, please stop it first. - .exec(); - - // true only on docker toolbox (1.10.1) -// assertThat(updateResponse.getWarnings(), hasSize(1)); -// assertThat(updateResponse.getWarnings().get(0), -// is("Your kernel does not support Block I/O weight. Weight discarded.")); - - InspectContainerResponse inspectAfter = dockerClient.inspectContainerCmd(containerId).exec(); - final HostConfig afterHostConfig = inspectAfter.getHostConfig(); - - assertThat(afterHostConfig.getMemory(), - is(314572800L)); - -// assertThat(afterHostConfig.getBlkioWeight(), is(300)); - assertThat(afterHostConfig.getCpuShares(), is(512)); - assertThat(afterHostConfig.getCpuPeriod(), is(100000)); - assertThat(afterHostConfig.getCpuQuota(), is(50000)); - assertThat(afterHostConfig.getCpusetMems(), is("0")); - - assertThat(afterHostConfig.getMemoryReservation(), is(209715200L)); - assertThat(afterHostConfig.getMemorySwap(), is(514288000L)); - - } - - @Test(enabled = false, description = "impossible to serder because model bundled in cmd") - public void serDerDocs1() throws IOException { - final ObjectMapper mapper = new ObjectMapper(); - final JavaType type = mapper.getTypeFactory().uncheckedSimpleType(UpdateContainerCmdImpl.class); - - final UpdateContainerCmdImpl upd = testRoundTrip(RemoteApiVersion.VERSION_1_22, - "/containers/container/update/docs.json", - type - ); - - assertThat(upd, notNullValue()); - } -} diff --git a/src/test/java/com/github/dockerjava/core/command/VersionCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/VersionCmdImplTest.java deleted file mode 100644 index d8bfb3f23..000000000 --- a/src/test/java/com/github/dockerjava/core/command/VersionCmdImplTest.java +++ /dev/null @@ -1,52 +0,0 @@ -package com.github.dockerjava.core.command; - -import java.lang.reflect.Method; - -import org.apache.commons.lang.StringUtils; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Version; -import com.github.dockerjava.client.AbstractDockerClientTest; - -@Test(groups = "integration") -public class VersionCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void version() throws DockerException { - Version version = dockerClient.versionCmd().exec(); - LOG.info(version.toString()); - - assertTrue(version.getGoVersion().length() > 0); - assertTrue(version.getVersion().length() > 0); - - assertEquals(StringUtils.split(version.getVersion(), ".").length, 3); - - } - -} diff --git a/src/test/java/com/github/dockerjava/core/command/WaitContainerCmdImplTest.java b/src/test/java/com/github/dockerjava/core/command/WaitContainerCmdImplTest.java deleted file mode 100644 index b92dcc657..000000000 --- a/src/test/java/com/github/dockerjava/core/command/WaitContainerCmdImplTest.java +++ /dev/null @@ -1,125 +0,0 @@ -package com.github.dockerjava.core.command; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.WaitResponse; -import com.github.dockerjava.client.AbstractDockerClientTest; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; -import java.util.concurrent.TimeUnit; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.*; - -@Test(groups = "integration") -public class WaitContainerCmdImplTest extends AbstractDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void testWaitContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true").exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - int exitCode = dockerClient.waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - LOG.info("Container exit code: {}", exitCode); - - assertThat(exitCode, equalTo(0)); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); - assertThat(inspectContainerResponse.getState().getExitCode(), is(equalTo(exitCode))); - } - - @Test(expectedExceptions = NotFoundException.class) - public void testWaitNonExistingContainer() throws DockerException { - - WaitContainerResultCallback callback = new WaitContainerResultCallback() { - public void onNext(WaitResponse waitResponse) { - fail("expected NotFoundException"); - } - }; - - dockerClient.waitContainerCmd("non-existing").exec(callback).awaitStatusCode(); - } - - @Test - public void testWaitContainerAbort() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - WaitContainerResultCallback callback = dockerClient.waitContainerCmd(container.getId()).exec( - new WaitContainerResultCallback()); - - Thread.sleep(5000); - - callback.close(); - - dockerClient.killContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); - } - - @Test - public void testWaitContainerTimeout() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "10").exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - WaitContainerResultCallback callback = dockerClient.waitContainerCmd(container.getId()).exec( - new WaitContainerResultCallback()); - try { - callback.awaitStatusCode(100, TimeUnit.MILLISECONDS); - fail("Should throw exception on timeout."); - } catch(DockerClientException e){ - LOG.info(e.getMessage()); - } - } -} diff --git a/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileStatementAddTest.java b/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileStatementAddTest.java deleted file mode 100644 index 102b113e3..000000000 --- a/src/test/java/com/github/dockerjava/core/dockerfile/DockerfileStatementAddTest.java +++ /dev/null @@ -1,44 +0,0 @@ -package com.github.dockerjava.core.dockerfile; - -import com.github.dockerjava.api.exception.DockerClientException; -import com.google.common.base.Optional; -import org.hamcrest.Matcher; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.annotations.DataProvider; -import org.testng.annotations.Test; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.contains; -import static org.hamcrest.Matchers.containsInAnyOrder; -import static org.hamcrest.Matchers.is; - -public class DockerfileStatementAddTest { - - private static final Logger LOG = LoggerFactory.getLogger(DockerfileStatementAddTest.class); - - @DataProvider(name = "valid scenarios") - public Object[][] validScenarios() { - return new Object[][] { {"ADD src dest", contains("src"), "dest"}, - {"ADD \"src file\" \"dest\"", contains("src file"), "dest"}, - {"ADD src\"file dest", contains("src\"file"), "dest"}, - {"ADD src1 src2 dest", containsInAnyOrder("src1", "src2"), "dest"}, - {"COPY src dest", contains("src"), "dest"}, - {"COPY \"src file\" \"dest\"", contains("src file"), "dest"}, - {"COPY src\"file dest", contains("src\"file"), "dest"}, - {"COPY src1 src2 dest", containsInAnyOrder("src1", "src2"), "dest"}}; - } - - @Test(dataProvider = "valid scenarios") - public void testAddOrCopyPattern(String command, Matcher matchesExpectation, String expectedDest) { - Optional optionalAdd = DockerfileStatement.Add.create(command); - assertThat(optionalAdd.isPresent(), is(true)); - assertThat(optionalAdd.get().sources, matchesExpectation); - assertThat(optionalAdd.get().destination, is(expectedDest)); - } - - @Test(expectedExceptions = {DockerClientException.class}) - public void shouldThrowExceptionIfDestNotSpecified() { - DockerfileStatement.Add.create("ADD src"); - } -} diff --git a/src/test/java/com/github/dockerjava/core/util/CertificateUtilsTest.java b/src/test/java/com/github/dockerjava/core/util/CertificateUtilsTest.java deleted file mode 100644 index 36c00c076..000000000 --- a/src/test/java/com/github/dockerjava/core/util/CertificateUtilsTest.java +++ /dev/null @@ -1,51 +0,0 @@ -package com.github.dockerjava.core.util; - -import org.testng.annotations.Test; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.core.Is.is; - -public class CertificateUtilsTest { - private static final String baseDir = CertificateUtilsTest.class.getResource( - CertificateUtilsTest.class.getSimpleName() + "/").getFile(); - - @Test - public void allFilesExist() { - assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "allFilesExist"), is(true)); - } - - @Test - public void caAndCertAndKeyMissing() { - assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "caAndCertAndKeyMissing"), is(false)); - } - - @Test - public void caAndCertMissing() { - assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "caAndCertMissing"), is(false)); - } - - @Test - public void caAndKeyMissing() { - assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "caAndKeyMissing"), is(false)); - } - - @Test - public void caMissing() { - assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "caMissing"), is(false)); - } - - @Test - public void certAndKeyMissing() { - assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "certAndKeyMissing"), is(false)); - } - - @Test - public void certMissing() { - assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "certMissing"), is(false)); - } - - @Test - public void keyMissing() { - assertThat(CertificateUtils.verifyCertificatesExist(baseDir + "keyMissing"), is(false)); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/AbstractNettyDockerClientTest.java b/src/test/java/com/github/dockerjava/netty/AbstractNettyDockerClientTest.java deleted file mode 100644 index 57f7aa421..000000000 --- a/src/test/java/com/github/dockerjava/netty/AbstractNettyDockerClientTest.java +++ /dev/null @@ -1,17 +0,0 @@ -package com.github.dockerjava.netty; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.github.dockerjava.client.AbstractDockerClientTest; -import com.github.dockerjava.core.TestDockerCmdExecFactory; - -public abstract class AbstractNettyDockerClientTest extends AbstractDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(AbstractNettyDockerClientTest.class); - - @Override - protected TestDockerCmdExecFactory initTestDockerCmdExecFactory() { - return new TestDockerCmdExecFactory(new NettyDockerCmdExecFactory()); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/WebTargetTest.java b/src/test/java/com/github/dockerjava/netty/WebTargetTest.java deleted file mode 100644 index a462eb8b5..000000000 --- a/src/test/java/com/github/dockerjava/netty/WebTargetTest.java +++ /dev/null @@ -1,39 +0,0 @@ -package com.github.dockerjava.netty; - -import static org.testng.Assert.assertEquals; - -import org.mockito.Mock; -import org.mockito.MockitoAnnotations; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -/** - * @author Alexander Koshevoy - */ -public class WebTargetTest { - @Mock private ChannelProvider channelProvider; - - @BeforeMethod - public void setUp() throws Exception { - MockitoAnnotations.initMocks(this); - } - - @Test - public void verifyImmutability() throws Exception { - WebTarget emptyWebTarget = new WebTarget(channelProvider); - - WebTarget initWebTarget = emptyWebTarget.path("/containers/{id}/attach").resolveTemplate("id", "d03da378b592") - .queryParam("logs", "true"); - - WebTarget anotherWebTarget = emptyWebTarget.path("/containers/{id}/attach") - .resolveTemplate("id", "2cfada4e3c07").queryParam("stdin", "true"); - - assertEquals(new WebTarget(channelProvider), emptyWebTarget); - - assertEquals(new WebTarget(channelProvider).path("/containers/d03da378b592/attach") - .queryParam("logs", "true"), initWebTarget); - - assertEquals(new WebTarget(channelProvider).path("/containers/2cfada4e3c07/attach") - .queryParam("stdin", "true"), anotherWebTarget); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/AttachContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/AttachContainerCmdExecTest.java deleted file mode 100644 index 7daa59b25..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/AttachContainerCmdExecTest.java +++ /dev/null @@ -1,174 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static java.util.concurrent.TimeUnit.SECONDS; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.io.ByteArrayInputStream; -import java.io.File; -import java.io.InputStream; -import java.lang.reflect.Method; -import java.util.concurrent.TimeUnit; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.model.Frame; -import com.github.dockerjava.api.model.StreamType; -import com.github.dockerjava.core.command.AttachContainerResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class AttachContainerCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void attachContainerWithoutTTY() throws Exception { - - String snippet = "hello world"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("echo", snippet) - .withTty(false).exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - AttachContainerTestCallback callback = new AttachContainerTestCallback() { - @Override - public void onNext(Frame frame) { - assertEquals(frame.getStreamType(), StreamType.STDOUT); - super.onNext(frame); - }; - }; - - dockerClient.attachContainerCmd(container.getId()).withStdErr(true).withStdOut(true).withFollowStream(true) - .withLogs(true).exec(callback).awaitCompletion(10, SECONDS); - callback.close(); - - assertThat(callback.toString(), containsString(snippet)); - } - - @Test - public void attachContainerWithStdin() throws Exception { - - String snippet = "hello world"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("/bin/sh", "-c", "sleep 1 && read line && echo $line") - .withTty(false) - .withStdinOpen(true) - .exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - Thread.sleep(SECONDS.toMillis(3)); //wait bash initialisation - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertTrue(inspectContainerResponse.getState().getRunning()); - - AttachContainerTestCallback callback = new AttachContainerTestCallback() { - @Override - public void onNext(Frame frame) { - assertEquals(frame.getStreamType(), StreamType.STDOUT); - super.onNext(frame); - } - }; - - InputStream stdin = new ByteArrayInputStream((snippet + "\n").getBytes()); - - dockerClient.attachContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .withFollowStream(true) - .withStdIn(stdin) - .exec(callback) - .awaitCompletion(15, SECONDS); - callback.close(); - - assertThat(callback.toString(), containsString(snippet)); - } - - @Test - public void attachContainerWithTTY() throws Exception { - - File baseDir = new File(Thread.currentThread().getContextClassLoader() - .getResource("attachContainerTestDockerfile").getFile()); - - String imageId = buildImage(baseDir); - - CreateContainerResponse container = dockerClient.createContainerCmd(imageId).withTty(true).exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - AttachContainerTestCallback callback = new AttachContainerTestCallback() { - @Override - public void onNext(Frame frame) { - assertEquals(frame.getStreamType(), StreamType.RAW); - super.onNext(frame); - }; - }; - - dockerClient.attachContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .withFollowStream(true) - .exec(callback) - .awaitCompletion(10, SECONDS); - callback.close(); - - // HexDump.dump(collectFramesCallback.toString().getBytes(), 0, System.out, 0); - - assertThat(callback.toString(), containsString("stdout\r\nstderr")); - } - - public static class AttachContainerTestCallback extends AttachContainerResultCallback { - private StringBuffer log = new StringBuffer(); - - @Override - public void onNext(Frame item) { - log.append(new String(item.getPayload())); - super.onNext(item); - } - - @Override - public String toString() { - return log.toString(); - } - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/AuthCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/AuthCmdExecTest.java deleted file mode 100644 index 690d478f4..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/AuthCmdExecTest.java +++ /dev/null @@ -1,51 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.UnauthorizedException; -import com.github.dockerjava.api.model.AuthResponse; -import com.github.dockerjava.core.DockerClientBuilder; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class AuthCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void testAuth() throws Exception { - AuthResponse response = dockerClient.authCmd().exec(); - - assertEquals(response.getStatus(), "Login Succeeded"); - } - - @Test(expectedExceptions = UnauthorizedException.class) - public void testAuthInvalid() throws Exception { - DockerClientBuilder.getInstance(config("garbage")).build().authCmd().exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/BuildImageCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/BuildImageCmdExecTest.java deleted file mode 100644 index 1474ef25f..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/BuildImageCmdExecTest.java +++ /dev/null @@ -1,274 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.nullValue; - -import java.io.File; -import java.io.FileInputStream; -import java.io.InputStream; -import java.lang.reflect.Method; -import java.util.Collection; -import java.util.UUID; - -import org.apache.commons.io.FileUtils; -import org.apache.commons.io.filefilter.TrueFileFilter; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.BuildImageCmd; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.command.InspectImageResponse; -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.model.AuthConfig; -import com.github.dockerjava.api.model.AuthConfigurations; -import com.github.dockerjava.api.model.ExposedPort; -import com.github.dockerjava.api.model.PortBinding; -import com.github.dockerjava.api.model.Ports; -import com.github.dockerjava.api.model.Ports.Binding; -import com.github.dockerjava.core.command.BuildImageResultCallback; -import com.github.dockerjava.core.command.PushImageResultCallback; -import com.github.dockerjava.core.command.WaitContainerResultCallback; -import com.github.dockerjava.core.util.CompressArchiveUtil; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class BuildImageCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void author() throws Exception { - - String imageId = buildImage(fileFromBuildTestResource("AUTHOR")); - - InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(imageId).exec(); - assertThat(inspectImageResponse, not(nullValue())); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - - assertThat(inspectImageResponse.getAuthor(), equalTo("Guillaume J. Charmes \"guillaume@dotcloud.com\"")); - } - - @Test - public void buildImageFromTar() throws Exception { - File baseDir = fileFromBuildTestResource("ADD/file"); - Collection files = FileUtils.listFiles(baseDir, TrueFileFilter.INSTANCE, TrueFileFilter.INSTANCE); - File tarFile = CompressArchiveUtil.archiveTARFiles(baseDir, files, UUID.randomUUID().toString()); - String response = dockerfileBuild(new FileInputStream(tarFile)); - assertThat(response, containsString("Successfully executed testrun.sh")); - } - - @Test - public void onBuild() throws Exception { - File baseDir = fileFromBuildTestResource("ONBUILD/parent"); - - dockerClient.buildImageCmd(baseDir).withNoCache(true).withTag("docker-java-onbuild") - .exec(new BuildImageResultCallback()).awaitImageId(); - baseDir = fileFromBuildTestResource("ONBUILD/child"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("Successfully executed testrun.sh")); - } - - @Test - public void addUrl() throws Exception { - File baseDir = fileFromBuildTestResource("ADD/url"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("Example Domain")); - } - - @Test - public void addFileInSubfolder() throws Exception { - File baseDir = fileFromBuildTestResource("ADD/fileInSubfolder"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("Successfully executed testrun.sh")); - } - - @Test - public void addFilesViaWildcard() throws Exception { - File baseDir = fileFromBuildTestResource("ADD/filesViaWildcard"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("Successfully executed testinclude1.sh")); - assertThat(response, not(containsString("Successfully executed testinclude2.sh"))); - } - - @Test - public void addFolder() throws Exception { - File baseDir = fileFromBuildTestResource("ADD/folder"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("Successfully executed testAddFolder.sh")); - } - - private String dockerfileBuild(InputStream tarInputStream) throws Exception { - - return execBuild(dockerClient.buildImageCmd().withTarInputStream(tarInputStream)); - } - - private String dockerfileBuild(File baseDir) throws Exception { - - return execBuild(dockerClient.buildImageCmd(baseDir)); - } - - private String execBuild(BuildImageCmd buildImageCmd) throws Exception { - String imageId = buildImageCmd.withNoCache(true).exec(new BuildImageResultCallback()).awaitImageId(); - - // Create container based on image - CreateContainerResponse container = dockerClient.createContainerCmd(imageId).exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - dockerClient.waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()).awaitStatusCode(); - - return containerLog(container.getId()); - } - - @Test(expectedExceptions = {DockerClientException.class}) - public void dockerignoreDockerfileIgnored() throws Exception { - File baseDir = fileFromBuildTestResource("dockerignore/DockerfileIgnored"); - - dockerClient.buildImageCmd(baseDir).withNoCache(true).exec(new BuildImageResultCallback()).awaitImageId(); - } - - @Test - public void dockerignoreDockerfileNotIgnored() throws Exception { - File baseDir = fileFromBuildTestResource("dockerignore/DockerfileNotIgnored"); - - dockerClient.buildImageCmd(baseDir).withNoCache(true).exec(new BuildImageResultCallback()).awaitImageId(); - } - - @Test(expectedExceptions = {DockerClientException.class}) - public void dockerignoreInvalidDockerIgnorePattern() throws Exception { - File baseDir = fileFromBuildTestResource("dockerignore/InvalidDockerignorePattern"); - - dockerClient.buildImageCmd(baseDir).withNoCache(true).exec(new BuildImageResultCallback()).awaitImageId(); - } - - @Test() - public void dockerignoreValidDockerIgnorePattern() throws Exception { - File baseDir = fileFromBuildTestResource("dockerignore/ValidDockerignorePattern"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("/tmp/a/a /tmp/a/c /tmp/a/d")); - } - - @Test - public void env() throws Exception { - File baseDir = fileFromBuildTestResource("ENV"); - String response = dockerfileBuild(baseDir); - assertThat(response, containsString("testENVSubstitution successfully completed")); - } - - @Test - public void fromPrivateRegistry() throws Exception { - File baseDir = new File(Thread.currentThread().getContextClassLoader().getResource("privateRegistry").getFile()); - - String imageId = buildImage(baseDir); - - InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(imageId).exec(); - assertThat(inspectImageResponse, not(nullValue())); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - - dockerClient.tagImageCmd(imageId, "testregistry", "2").withForce().exec(); - - // see https://github.com/docker/distribution/blob/master/docs/deploying.md#native-basic-auth - CreateContainerResponse testregistry = dockerClient - .createContainerCmd("testregistry:2") - .withName("registry") - .withPortBindings(new PortBinding(Binding.bindPort(5000), ExposedPort.tcp(5000))) - .withEnv("REGISTRY_AUTH=htpasswd", "REGISTRY_AUTH_HTPASSWD_REALM=Registry Realm", - "REGISTRY_AUTH_HTPASSWD_PATH=/auth/htpasswd", "REGISTRY_LOG_LEVEL=debug", - "REGISTRY_HTTP_TLS_CERTIFICATE=/certs/domain.crt", "REGISTRY_HTTP_TLS_KEY=/certs/domain.key") - .exec(); - - dockerClient.startContainerCmd(testregistry.getId()).exec(); - - // wait for registry to boot - Thread.sleep(3000); - - // credentials as configured in /auth/htpasswd - AuthConfig authConfig = new AuthConfig() - .withUsername("testuser") - .withPassword("testpassword") - .withEmail("foo@bar.de") - .withRegistryAddress("localhost:5000"); - - dockerClient.authCmd().withAuthConfig(authConfig).exec(); - dockerClient.tagImageCmd("busybox:latest", "localhost:5000/testuser/busybox", "latest").withForce().exec(); - - dockerClient.pushImageCmd("localhost:5000/testuser/busybox").withTag("latest").withAuthConfig(authConfig) - .exec(new PushImageResultCallback()).awaitSuccess(); - - dockerClient.removeImageCmd("localhost:5000/testuser/busybox").withForce(true).exec(); - - baseDir = fileFromBuildTestResource("FROM/privateRegistry"); - - AuthConfigurations authConfigurations = new AuthConfigurations(); - authConfigurations.addConfig(authConfig); - - imageId = dockerClient.buildImageCmd(baseDir).withNoCache(true).withBuildAuthConfigs(authConfigurations) - .exec(new BuildImageResultCallback()).awaitImageId(); - - inspectImageResponse = dockerClient.inspectImageCmd(imageId).exec(); - assertThat(inspectImageResponse, not(nullValue())); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - - } - - @Test - public void buildArgs() throws Exception { - File baseDir = fileFromBuildTestResource("buildArgs"); - - String imageId = dockerClient.buildImageCmd(baseDir).withNoCache(true).withBuildArg("testArg", "abc") - .exec(new BuildImageResultCallback()) - .awaitImageId(); - - InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(imageId).exec(); - assertThat(inspectImageResponse, not(nullValue())); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - - assertThat(inspectImageResponse.getConfig().getLabels().get("test"), equalTo("abc")); - } - - public void dockerfileNotInBaseDirectory() throws Exception { - File baseDirectory = fileFromBuildTestResource("dockerfileNotInBaseDirectory"); - File dockerfile = fileFromBuildTestResource("dockerfileNotInBaseDirectory/dockerfileFolder/Dockerfile"); - BuildImageCmd command = dockerClient.buildImageCmd() - .withBaseDirectory(baseDirectory) - .withDockerfile(dockerfile); - - String response = execBuild(command); - - assertThat(response, containsString("Successfully executed testrun.sh")); - } - - private File fileFromBuildTestResource(String resource) { - return new File(Thread.currentThread().getContextClassLoader() - .getResource("buildTests/" + resource).getFile()); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/CommitCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/CommitCmdExecTest.java deleted file mode 100644 index 3d1a66333..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/CommitCmdExecTest.java +++ /dev/null @@ -1,77 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.startsWith; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectImageResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class CommitCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void commit() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("touch", "/test").exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - LOG.info("Commiting container: {}", container.toString()); - String imageId = dockerClient.commitCmd(container.getId()).exec(); - - InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(imageId).exec(); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - - assertThat(inspectImageResponse, hasField("container", startsWith(container.getId()))); - assertThat(inspectImageResponse.getContainerConfig().getImage(), equalTo("busybox")); - - InspectImageResponse busyboxImg = dockerClient.inspectImageCmd("busybox").exec(); - - assertThat(inspectImageResponse.getParent(), equalTo(busyboxImg.getId())); - } - - @Test(expectedExceptions = NotFoundException.class) - public void commitNonExistingContainer() throws DockerException { - - dockerClient.commitCmd("non-existent").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/ConnectToNetworkCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/ConnectToNetworkCmdExecTest.java deleted file mode 100644 index ee1a681b6..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/ConnectToNetworkCmdExecTest.java +++ /dev/null @@ -1,98 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.model.ContainerNetwork; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; -import java.util.Collections; - -@Test(groups = "integration") -public class ConnectToNetworkCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void connectToNetwork() throws InterruptedException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - dockerClient.startContainerCmd(container.getId()).exec(); - - CreateNetworkResponse network = dockerClient.createNetworkCmd().withName("testNetwork").exec(); - - dockerClient.connectToNetworkCmd().withNetworkId(network.getId()).withContainerId(container.getId()).exec(); - - Network updatedNetwork = dockerClient.inspectNetworkCmd().withNetworkId(network.getId()).exec(); - - assertTrue(updatedNetwork.getContainers().containsKey(container.getId())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertNotNull(inspectContainerResponse.getNetworkSettings().getNetworks().get("testNetwork")); - } - - @Test - public void connectToNetworkWithContainerNetwork() throws InterruptedException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - dockerClient.startContainerCmd(container.getId()).exec(); - - CreateNetworkResponse network = dockerClient.createNetworkCmd() - .withName("testNetwork") - .withIpam(new Network.Ipam() - .withConfig(new Network.Ipam.Config() - .withSubnet("10.100.100.0/24"))) - .exec(); - - dockerClient.connectToNetworkCmd() - .withNetworkId(network.getId()) - .withContainerId(container.getId()) - .withContainerNetwork(new ContainerNetwork() - .withAliases("testing") - .withIpamConfig(new ContainerNetwork.Ipam() - .withIpv4Address("10.100.100.100"))) - .exec(); - - Network updatedNetwork = dockerClient.inspectNetworkCmd().withNetworkId(network.getId()).exec(); - - Network.ContainerNetworkConfig containerNetworkConfig = updatedNetwork.getContainers().get(container.getId()); - assertNotNull(containerNetworkConfig); - assertEquals(containerNetworkConfig.getIpv4Address(), "10.100.100.100/24"); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - ContainerNetwork testNetwork = inspectContainerResponse.getNetworkSettings().getNetworks().get("testNetwork"); - assertNotNull(testNetwork); - assertEquals(testNetwork.getAliases(), Collections.singletonList("testing")); - assertEquals(testNetwork.getGateway(), "10.100.100.1"); - assertEquals(testNetwork.getIpAddress(), "10.100.100.100"); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/ContainerDiffCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/ContainerDiffCmdExecTest.java deleted file mode 100644 index c21c426b5..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/ContainerDiffCmdExecTest.java +++ /dev/null @@ -1,77 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static ch.lambdaj.Lambda.selectUnique; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; -import java.util.List; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.ChangeLog; -import com.github.dockerjava.core.command.WaitContainerResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class ContainerDiffCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void testContainerDiff() throws DockerException { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("touch", "/test").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - int exitCode = dockerClient.waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - assertThat(exitCode, equalTo(0)); - - List filesystemDiff = dockerClient.containerDiffCmd(container.getId()).exec(); - LOG.info("Container DIFF: {}", filesystemDiff.toString()); - - assertThat(filesystemDiff.size(), equalTo(1)); - ChangeLog testChangeLog = selectUnique(filesystemDiff, hasField("path", equalTo("/test"))); - - assertThat(testChangeLog, hasField("path", equalTo("/test"))); - assertThat(testChangeLog, hasField("kind", equalTo(1))); - } - - @Test(expectedExceptions = NotFoundException.class) - public void testContainerDiffWithNonExistingContainer() throws DockerException { - - dockerClient.containerDiffCmd("non-existing").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/CopyArchiveFromContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/CopyArchiveFromContainerCmdExecTest.java deleted file mode 100644 index 02b56156d..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/CopyArchiveFromContainerCmdExecTest.java +++ /dev/null @@ -1,110 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyOrNullString; -import static org.hamcrest.Matchers.not; - -import java.io.InputStream; -import java.lang.reflect.Method; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.nio.file.StandardOpenOption; - -import org.apache.commons.compress.archivers.tar.TarArchiveEntry; -import org.apache.commons.compress.archivers.tar.TarArchiveInputStream; -import org.apache.commons.io.IOUtils; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.core.util.CompressArchiveUtil; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class CopyArchiveFromContainerCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void copyFromContainer() throws Exception { - // TODO extract this into a shared method - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("docker-java-itest-copyFromContainer").withCmd("touch", "/copyFromContainer").exec(); - - LOG.info("Created container: {}", container); - assertThat(container.getId(), not(isEmptyOrNullString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InputStream response = dockerClient.copyArchiveFromContainerCmd(container.getId(), "/copyFromContainer").exec(); - Boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied from the container."); - - // read the stream fully. Otherwise, the underlying stream will not be closed. - String responseAsString = asString(response); - assertNotNull(responseAsString); - assertTrue(responseAsString.length() > 0); - } - - @Test(expectedExceptions = NotFoundException.class) - public void copyFromNonExistingContainer() throws Exception { - - dockerClient.copyArchiveFromContainerCmd("non-existing", "/test").exec(); - } - - @Test - public void copyFromContainerBinaryFile() throws Exception { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("docker-java-itest-copyFromContainerBinaryFile").exec(); - - LOG.info("Created container: {}", container); - assertThat(container.getId(), not(isEmptyOrNullString())); - - Path temp = Files.createTempFile("", ".tar.gz"); - Path binaryFile = Paths.get("src/test/resources/testCopyFromArchive/binary.dat"); - CompressArchiveUtil.tar(binaryFile, temp, true, false); - - try (InputStream uploadStream = Files.newInputStream(temp)) { - dockerClient.copyArchiveToContainerCmd(container.getId()).withTarInputStream(uploadStream).exec(); - } - - InputStream response = dockerClient.copyArchiveFromContainerCmd(container.getId(), "/binary.dat").exec(); - Boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied from the container."); - - try (TarArchiveInputStream tarInputStream = new TarArchiveInputStream(response)) { - TarArchiveEntry nextTarEntry = tarInputStream.getNextTarEntry(); - - assertEquals(nextTarEntry.getName(), "binary.dat"); - try (InputStream binaryFileInputStream = Files.newInputStream(binaryFile, StandardOpenOption.READ)) { - assertTrue(IOUtils.contentEquals(binaryFileInputStream, tarInputStream)); - } - - assertNull(tarInputStream.getNextTarEntry(), "Nothing except binary.dat is expected to be copied."); - } - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/CopyArchiveToContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/CopyArchiveToContainerCmdExecTest.java deleted file mode 100644 index d75d36ea8..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/CopyArchiveToContainerCmdExecTest.java +++ /dev/null @@ -1,154 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyOrNullString; -import static org.hamcrest.Matchers.not; - -import java.io.IOException; -import java.io.InputStream; -import java.lang.reflect.Method; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; - -import org.apache.commons.io.FileUtils; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.core.command.WaitContainerResultCallback; -import com.github.dockerjava.core.util.CompressArchiveUtil; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class CopyArchiveToContainerCmdExecTest extends AbstractNettyDockerClientTest { - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void copyFileToContainer() throws Exception { - CreateContainerResponse container = prepareContainerForCopy(); - Path temp = Files.createTempFile("", ".tar.gz"); - CompressArchiveUtil.tar(Paths.get("src/test/resources/testReadFile"), temp, true, false); - try (InputStream uploadStream = Files.newInputStream(temp)) { - dockerClient.copyArchiveToContainerCmd(container.getId()).withTarInputStream(uploadStream).exec(); - assertFileCopied(container); - } - } - - @Test - public void copyStreamToContainer() throws Exception { - CreateContainerResponse container = prepareContainerForCopy(); - dockerClient.copyArchiveToContainerCmd(container.getId()).withHostResource("src/test/resources/testReadFile") - .exec(); - assertFileCopied(container); - } - - private CreateContainerResponse prepareContainerForCopy() { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("docker-java-itest-copyToContainer").exec(); - LOG.info("Created container: {}", container); - assertThat(container.getId(), not(isEmptyOrNullString())); - dockerClient.startContainerCmd(container.getId()).exec(); - // Copy a folder to the container - return container; - } - - private void assertFileCopied(CreateContainerResponse container) throws IOException { - try (InputStream response = dockerClient.copyArchiveFromContainerCmd(container.getId(), "testReadFile").exec()) { - boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied to the container."); - } - } - - @Test(expectedExceptions = NotFoundException.class) - public void copyToNonExistingContainer() throws Exception { - - dockerClient.copyArchiveToContainerCmd("non-existing").withHostResource("src/test/resources/testReadFile").exec(); - } - - @Test - public void copyDirWithLastAddedTarEntryEmptyDir() throws Exception{ - // create a temp dir - Path localDir = Files.createTempDirectory(null); - localDir.toFile().deleteOnExit(); - // create empty sub-dir with name b - Files.createDirectory(localDir.resolve("b")); - // create sub-dir with name a - Path dirWithFile = Files.createDirectory(localDir.resolve("a")); - // create file in sub-dir b, name or conter are irrelevant - Files.createFile(dirWithFile.resolve("file")); - - // create a test container - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("sleep", "9999") - .exec(); - // start the container - dockerClient.startContainerCmd(container.getId()).exec(); - // copy data from local dir to container - dockerClient.copyArchiveToContainerCmd(container.getId()) - .withHostResource(localDir.toString()) - .exec(); - - // cleanup dir - FileUtils.deleteDirectory(localDir.toFile()); - } - - @Test - public void copyFileWithExecutePermission() throws Exception { - // create script file, add permission to execute - Path scriptPath = Files.createTempFile("run", ".sh"); - boolean executable = scriptPath.toFile().setExecutable(true, false); - if (!executable){ - throw new Exception("Execute permission on file not set!"); - } - String snippet = "Running script with execute permission."; - String scriptTextStr = "#!/bin/sh\necho \"" + snippet + "\""; - // write content for created script - Files.write(scriptPath, scriptTextStr.getBytes()); - // create a test container which starts and waits 3 seconds for the - // script to be copied to the container's home dir and then executes it - String containerCmd = "sleep 3; /home/" + scriptPath.getFileName().toString(); - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("test") - .withCmd("/bin/sh", "-c", containerCmd) - .exec(); - // start the container - dockerClient.startContainerCmd(container.getId()).exec(); - // copy script to container home dir - dockerClient.copyArchiveToContainerCmd(container.getId()) - .withRemotePath("/home") - .withHostResource(scriptPath.toString()) - .exec(); - // await exid code - int exitCode = dockerClient.waitContainerCmd(container.getId()) - .exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - // check result - assertThat(exitCode, equalTo(0)); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/CopyFileFromContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/CopyFileFromContainerCmdExecTest.java deleted file mode 100644 index b512cfa69..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/CopyFileFromContainerCmdExecTest.java +++ /dev/null @@ -1,70 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyOrNullString; -import static org.hamcrest.Matchers.not; - -import java.io.InputStream; -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class CopyFileFromContainerCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void copyFromContainer() throws Exception { - // TODO extract this into a shared method - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("docker-java-itest-copyFromContainer").withCmd("touch", "/copyFromContainer").exec(); - - LOG.info("Created container: {}", container); - assertThat(container.getId(), not(isEmptyOrNullString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InputStream response = dockerClient.copyFileFromContainerCmd(container.getId(), "/copyFromContainer").exec(); - Boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied from the container."); - - // read the stream fully. Otherwise, the underlying stream will not be closed. - String responseAsString = asString(response); - assertNotNull(responseAsString); - assertTrue(responseAsString.length() > 0); - } - - @Test(expectedExceptions = NotFoundException.class) - public void copyFromNonExistingContainer() throws Exception { - - dockerClient.copyFileFromContainerCmd("non-existing", "/test").exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/CreateContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/CreateContainerCmdExecTest.java deleted file mode 100644 index 2213ca7b7..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/CreateContainerCmdExecTest.java +++ /dev/null @@ -1,697 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.ConflictException; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Bind; -import com.github.dockerjava.api.model.ContainerNetwork; -import com.github.dockerjava.api.model.Device; -import com.github.dockerjava.api.model.ExposedPort; -import com.github.dockerjava.api.model.HostConfig; -import com.github.dockerjava.api.model.Link; -import com.github.dockerjava.api.model.LogConfig; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.api.model.Ports; -import com.github.dockerjava.api.model.Ports.Binding; -import com.github.dockerjava.api.model.RestartPolicy; -import com.github.dockerjava.api.model.Ulimit; -import com.github.dockerjava.api.model.Volume; -import com.github.dockerjava.api.model.VolumesFrom; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -import org.apache.commons.io.FileUtils; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; -import org.testng.internal.junit.ArrayAsserts; - -import java.lang.reflect.Method; -import java.security.SecureRandom; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.Map; -import java.util.UUID; - -import static com.github.dockerjava.api.model.Capability.MKNOD; -import static com.github.dockerjava.api.model.Capability.NET_ADMIN; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.contains; -import static org.hamcrest.Matchers.containsInAnyOrder; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.hasItemInArray; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.notNullValue; -import static org.hamcrest.Matchers.startsWith; - -@Test(groups = "integration") -public class CreateContainerCmdExecTest extends AbstractNettyDockerClientTest { - public static final String BUSYBOX_IMAGE = "busybox"; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(expectedExceptions = ConflictException.class) - public void createContainerWithExistingName() throws DockerException { - - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("env") - .withName(containerName).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.createContainerCmd("busybox").withCmd("env").withName(containerName).exec(); - } - - @Test - public void createContainerWithVolume() throws DockerException { - - Volume volume = new Volume("/var/log"); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withVolumes(volume) - .withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - LOG.info("Inspect container {}", inspectContainerResponse.getConfig().getVolumes()); - - assertThat(inspectContainerResponse.getConfig().getVolumes().keySet(), contains("/var/log")); - - assertThat(inspectContainerResponse.getMounts().get(0).getDestination(), equalTo(volume)); - assertThat(inspectContainerResponse.getMounts().get(0).getMode(), equalTo("")); - assertThat(inspectContainerResponse.getMounts().get(0).getRW(), equalTo(true)); - } - - @Test - public void createContainerWithReadOnlyVolume() throws DockerException { - - Volume volume = new Volume("/srv/test"); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withVolumes(volume) - .withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - LOG.info("Inspect container {}", inspectContainerResponse.getConfig().getVolumes()); - - assertThat(inspectContainerResponse.getConfig().getVolumes().keySet(), contains("/srv/test")); - - assertThat(inspectContainerResponse.getMounts().get(0).getDestination(), equalTo(volume)); - // TODO: Create a read-only volume and test like this - // assertFalse(inspectContainerResponse.getMounts().get(0).getRW()); - } - - @Test - public void createContainerWithVolumesFrom() throws DockerException { - - Volume volume1 = new Volume("/opt/webapp1"); - Volume volume2 = new Volume("/opt/webapp2"); - - String container1Name = UUID.randomUUID().toString(); - - Bind bind1 = new Bind("/src/webapp1", volume1); - Bind bind2 = new Bind("/src/webapp2", volume2); - - // create a running container with bind mounts - CreateContainerResponse container1 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(container1Name) - .withBinds(bind1, bind2).exec(); - LOG.info("Created container1 {}", container1.toString()); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - - assertThat(Arrays.asList(inspectContainerResponse1.getHostConfig().getBinds()), containsInAnyOrder(bind1, bind2)); - - assertThat(inspectContainerResponse1, mountedVolumes(containsInAnyOrder(volume1, volume2))); - - // create a second container with volumes from first container - CreateContainerResponse container2 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withVolumesFrom(new VolumesFrom(container1Name)).exec(); - LOG.info("Created container2 {}", container2.toString()); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - - // No volumes are created, the information is just stored in .HostConfig.VolumesFrom - assertThat(inspectContainerResponse2.getHostConfig().getVolumesFrom(), hasItemInArray(new VolumesFrom( - container1Name))); - - // To ensure that the information stored in VolumesFrom really is considered - // when starting the container, we start it and verify that it has the same - // bind mounts as the first container. - // This is somehow out of scope here, but it helped me to understand how the - // VolumesFrom feature really works. - dockerClient.startContainerCmd(container2.getId()).exec(); - LOG.info("Started container2 {}", container2.toString()); - - inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()).exec(); - - assertThat(inspectContainerResponse2.getHostConfig().getVolumesFrom(), hasItemInArray(new VolumesFrom( - container1Name))); - - assertThat(inspectContainerResponse2, mountedVolumes(containsInAnyOrder(volume1, volume2))); - } - - @Test - public void createContainerWithEnv() throws Exception { - final String testVariable = "VARIABLE=success"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withEnv(testVariable) - .withCmd("env") - .exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEnv()), hasItem(testVariable)); - - dockerClient.startContainerCmd(container.getId()).exec(); - - assertThat(containerLog(container.getId()), containsString(testVariable)); - } - - @Test - public void createContainerWithHostname() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withHostName("docker-java") - .withCmd("env").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getConfig().getHostName(), equalTo("docker-java")); - - dockerClient.startContainerCmd(container.getId()).exec(); - - assertThat(containerLog(container.getId()), containsString("HOSTNAME=docker-java")); - } - - @Test - public void createContainerWithName() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withName("container") - .withCmd("env").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getName(), equalTo("/container")); - } - - @Test - public void createContainerWithLink() throws DockerException { - - CreateContainerResponse container1 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container1").exec(); - LOG.info("Created container1 {}", container1.toString()); - assertThat(container1.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); - assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); - - CreateContainerResponse container2 = dockerClient.createContainerCmd("busybox").withName("container2") - .withCmd("env").withLinks(new Link("container1", "container1Link")).exec(); - LOG.info("Created container {}", container2.toString()); - assertThat(container2.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[] {new Link("container1", - "container1Link")})); - } - - @Test - public void createContainerWithLinkInCustomNetwork() throws DockerException { - - CreateNetworkResponse createNetworkResponse = dockerClient.createNetworkCmd() - .withName("linkNet") - .exec(); - - assertNotNull(createNetworkResponse.getId()); - - CreateContainerResponse container1 = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withNetworkMode("linkNet") - .withCmd("sleep", "9999") - .withName("container1") - .exec(); - - assertThat(container1.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); - assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); - - CreateContainerResponse container2 = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withNetworkMode("linkNet") - .withName("container2") - .withCmd("env") - .withLinks(new Link("container1", "container1Link")) - .exec(); - - LOG.info("Created container {}", container2.toString()); - assertThat(container2.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - - ContainerNetwork linkNet = inspectContainerResponse2.getNetworkSettings().getNetworks().get("linkNet"); - assertNotNull(linkNet); - ArrayAsserts.assertArrayEquals(new Link[]{ new Link("container1", "container1Link")}, linkNet.getLinks()); - } - - @Test - public void createContainerWithCustomIp() throws DockerException { - - CreateNetworkResponse createNetworkResponse = dockerClient.createNetworkCmd() - .withIpam(new Network.Ipam() - .withConfig(new Network.Ipam.Config() - .withSubnet("10.100.101.0/24"))) - .withName("customIpNet") - .exec(); - - assertNotNull(createNetworkResponse.getId()); - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withNetworkMode("customIpNet") - .withCmd("sleep", "9999") - .withName("container") - .withIpv4Address("10.100.101.100") - .exec(); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()) - .exec(); - - ContainerNetwork customIpNet = inspectContainerResponse.getNetworkSettings().getNetworks().get("customIpNet"); - assertNotNull(customIpNet); - assertEquals(customIpNet.getGateway(), "10.100.101.1"); - assertEquals(customIpNet.getIpAddress(), "10.100.101.100"); - } - - @Test - public void createContainerWithAlias() throws DockerException { - - CreateNetworkResponse createNetworkResponse = dockerClient.createNetworkCmd() - .withName("aliasNet") - .exec(); - - assertNotNull(createNetworkResponse.getId()); - - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withNetworkMode("aliasNet") - .withCmd("sleep", "9999") - .withName("container") - .withAliases("server") - .exec(); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()) - .exec(); - - ContainerNetwork aliasNet = inspectContainerResponse.getNetworkSettings().getNetworks().get("aliasNet"); - assertEquals(aliasNet.getAliases(), Collections.singletonList("server")); - } - - @Test - public void createContainerWithCapAddAndCapDrop() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCapAdd(NET_ADMIN) - .withCapDrop(MKNOD).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapAdd()), contains(NET_ADMIN)); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapDrop()), contains(MKNOD)); - } - - @Test - public void createContainerWithDns() throws DockerException { - - String aDnsServer = "8.8.8.8"; - String anotherDnsServer = "8.8.4.4"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withDns(aDnsServer, anotherDnsServer).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDns()), - contains(aDnsServer, anotherDnsServer)); - } - - @Test - public void createContainerWithEntrypoint() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withName("container") - .withEntrypoint("sleep", "9999").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getConfig().getEntrypoint()), contains("sleep", "9999")); - - } - - @Test - public void createContainerWithExtraHosts() throws DockerException { - - String[] extraHosts = {"dockerhost:127.0.0.1", "otherhost:10.0.0.1"}; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withName("container") - .withExtraHosts(extraHosts).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getExtraHosts()), - containsInAnyOrder("dockerhost:127.0.0.1", "otherhost:10.0.0.1")); - } - - @Test - public void createContainerWithDevices() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withDevices(new Device("rwm", "/dev/nulo", "/dev/zero")).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDevices()), contains(new Device("rwm", - "/dev/nulo", "/dev/zero"))); - } - - @Test - public void createContainerWithPortBindings() throws DockerException { - - ExposedPort tcp22 = ExposedPort.tcp(22); - ExposedPort tcp23 = ExposedPort.tcp(23); - - Ports portBindings = new Ports(); - portBindings.bind(tcp22, Binding.bindPort(11022)); - portBindings.bind(tcp23, Binding.bindPort(11023)); - portBindings.bind(tcp23, Binding.bindPort(11024)); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withExposedPorts(tcp22, tcp23).withPortBindings(portBindings).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getConfig().getExposedPorts()), contains(tcp22, tcp23)); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp22)[0], - is(equalTo(Binding.bindPort(11022)))); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[0], - is(equalTo(Binding.bindPort(11023)))); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[1], - is(equalTo(Binding.bindPort(11024)))); - - } - - @Test - public void createContainerWithLinking() throws DockerException { - - CreateContainerResponse container1 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container1").exec(); - - LOG.info("Created container1 {}", container1.toString()); - assertThat(container1.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); - - assertThat(inspectContainerResponse1.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse1.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getId(), startsWith(container1.getId())); - assertThat(inspectContainerResponse1.getName(), equalTo("/container1")); - assertThat(inspectContainerResponse1.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getState(), is(notNullValue())); - assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); - - if (!inspectContainerResponse1.getState().getRunning()) { - assertThat(inspectContainerResponse1.getState().getExitCode(), is(equalTo(0))); - } - - CreateContainerResponse container2 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container2").withLinks(new Link("container1", "container1Link")).exec(); - - LOG.info("Created container2 {}", container2.toString()); - assertThat(container2.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - LOG.info("Container2 Inspect: {}", inspectContainerResponse2.toString()); - - assertThat(inspectContainerResponse2.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse2.getHostConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[] {new Link("container1", - "container1Link")})); - assertThat(inspectContainerResponse2.getId(), startsWith(container2.getId())); - assertThat(inspectContainerResponse2.getName(), equalTo("/container2")); - assertThat(inspectContainerResponse2.getImageId(), not(isEmptyString())); - - } - - @Test - public void createContainerWithRestartPolicy() throws DockerException { - - RestartPolicy restartPolicy = RestartPolicy.onFailureRestart(5); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withRestartPolicy(restartPolicy).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getHostConfig().getRestartPolicy(), is(equalTo(restartPolicy))); - } - - @Test - public void createContainerWithPidMode() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withPidMode("host").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getHostConfig().getPidMode(), is(equalTo("host"))); - } - - /** - * This tests support for --net option for the docker run command: --net="bridge" Set the Network mode for the container 'bridge': - * creates a new network stack for the container on the docker bridge 'none': no networking for this container 'container:': reuses - * another container network stack 'host': use the host network stack inside the container. Note: the host mode gives the container full - * access to local system services such as D-bus and is therefore considered insecure. - */ - @Test - public void createContainerWithNetworkMode() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withNetworkMode("host").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getHostConfig().getNetworkMode(), is(equalTo("host"))); - } - - @Test - public void createContainerWithMacAddress() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withMacAddress("00:80:41:ae:fd:7e").withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertEquals(inspectContainerResponse.getConfig().getMacAddress(), "00:80:41:ae:fd:7e"); - } - - @Test(groups = "ignoreInCircleCi") - public void createContainerWithULimits() throws DockerException { - - Ulimit[] ulimits = {new Ulimit("nproc", 709, 1026), new Ulimit("nofile", 1024, 4096)}; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withName("container") - .withUlimits(ulimits).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getUlimits()), - containsInAnyOrder(new Ulimit("nproc", 709, 1026), new Ulimit("nofile", 1024, 4096))); - - } - - @Test(groups = "ignoreInCircleCi") - public void createContainerWithLabels() throws DockerException { - - Map labels = new HashMap(); - labels.put("com.github.dockerjava.null", null); - labels.put("com.github.dockerjava.Boolean", "true"); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withLabels(labels).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - // null becomes empty string - labels.put("com.github.dockerjava.null", ""); - assertThat(inspectContainerResponse.getConfig().getLabels(), is(equalTo(labels))); - } - - @Test(groups = "ignoreInCircleCi") - public void createContainerWithLogConfig() throws DockerException { - - LogConfig logConfig = new LogConfig(LogConfig.LoggingType.NONE, null); - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withLogConfig(logConfig).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - // null becomes empty string - assertEquals(inspectContainerResponse.getHostConfig().getLogConfig().type, logConfig.type); - } - - @Test(groups = "ignoreInCircleCi") - public void createContainerWithCgroupParent() throws DockerException { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCgroupParent("/parent").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainer = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainer.getHostConfig().getCgroupParent(), is("/parent")); - } - - @SuppressWarnings("Duplicates") - @Test - public void createContainerWithShmSize() throws DockerException { - HostConfig hostConfig = new HostConfig().withShmSize(96 * FileUtils.ONE_MB); - CreateContainerResponse container = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withHostConfig(hostConfig).withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertEquals(inspectContainerResponse.getHostConfig().getShmSize(), hostConfig.getShmSize()); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/CreateNetworkCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/CreateNetworkCmdExecTest.java deleted file mode 100644 index 1e8c339b1..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/CreateNetworkCmdExecTest.java +++ /dev/null @@ -1,68 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; - -@Test(groups = "integration") -public class CreateNetworkCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void createNetwork() throws DockerException { - - String networkName = "testNetwork"; - - CreateNetworkResponse createNetworkResponse = dockerClient.createNetworkCmd().withName(networkName).exec(); - - assertNotNull(createNetworkResponse.getId()); - - Network network = dockerClient.inspectNetworkCmd().withNetworkId(createNetworkResponse.getId()).exec(); - assertEquals(network.getName(), networkName); - assertEquals(network.getDriver(), "bridge"); - } - - @Test - public void createNetworkWithIpamConfig() throws DockerException { - - String networkName = "testNetwork"; - Network.Ipam ipam = new Network.Ipam().withConfig(new Network.Ipam.Config().withSubnet("10.67.79.0/24")); - CreateNetworkResponse createNetworkResponse = dockerClient.createNetworkCmd().withName(networkName).withIpam(ipam).exec(); - - assertNotNull(createNetworkResponse.getId()); - - Network network = dockerClient.inspectNetworkCmd().withNetworkId(createNetworkResponse.getId()).exec(); - assertEquals(network.getName(), networkName); - assertEquals(network.getDriver(), "bridge"); - assertEquals("10.67.79.0/24", network.getIpam().getConfig().iterator().next().getSubnet()); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/CreateVolumeCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/CreateVolumeCmdExecTest.java deleted file mode 100644 index 1e1a30b73..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/CreateVolumeCmdExecTest.java +++ /dev/null @@ -1,73 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateVolumeResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class CreateVolumeCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void createVolume() throws DockerException { - - CreateVolumeResponse createVolumeResponse = dockerClient.createVolumeCmd().withName("volume1") - .withDriver("local").exec(); - - assertThat(createVolumeResponse.getName(), equalTo("volume1")); - assertThat(createVolumeResponse.getDriver(), equalTo("local")); - assertThat(createVolumeResponse.getMountpoint(), containsString("/volume1/")); - } - - @Test - public void createVolumeWithExistingName() throws DockerException { - - String volumeName = "volume1"; - - CreateVolumeResponse createVolumeResponse1 = dockerClient.createVolumeCmd().withName(volumeName) - .withDriver("local").exec(); - - assertThat(createVolumeResponse1.getName(), equalTo(volumeName)); - assertThat(createVolumeResponse1.getDriver(), equalTo("local")); - assertThat(createVolumeResponse1.getMountpoint(), containsString("/volume1/")); - - CreateVolumeResponse createVolumeResponse2 = dockerClient.createVolumeCmd().withName(volumeName) - .withDriver("local").exec(); - - assertThat(createVolumeResponse2.getName(), equalTo(volumeName)); - assertThat(createVolumeResponse2.getDriver(), equalTo("local")); - assertThat(createVolumeResponse2.getMountpoint(), equalTo(createVolumeResponse1.getMountpoint())); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/DisconnectFromNetworkCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/DisconnectFromNetworkCmdExecTest.java deleted file mode 100644 index 1f2563dec..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/DisconnectFromNetworkCmdExecTest.java +++ /dev/null @@ -1,82 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; - -@Test(groups = "integration") -public class DisconnectFromNetworkCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void disconnectFromNetwork() throws InterruptedException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - dockerClient.startContainerCmd(container.getId()).exec(); - - CreateNetworkResponse network = dockerClient.createNetworkCmd().withName("testNetwork").exec(); - - dockerClient.connectToNetworkCmd().withNetworkId(network.getId()).withContainerId(container.getId()).exec(); - - Network updatedNetwork = dockerClient.inspectNetworkCmd().withNetworkId(network.getId()).exec(); - - assertTrue(updatedNetwork.getContainers().containsKey(container.getId())); - - dockerClient.disconnectFromNetworkCmd().withNetworkId(network.getId()).withContainerId(container.getId()).exec(); - - updatedNetwork = dockerClient.inspectNetworkCmd().withNetworkId(network.getId()).exec(); - - assertFalse(updatedNetwork.getContainers().containsKey(container.getId())); - } - - @Test - public void forceDisconnectFromNetwork() throws InterruptedException { - - CreateNetworkResponse network = dockerClient.createNetworkCmd().withName("testNetwork").exec(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withNetworkMode("testNetwork") - .withCmd("sleep", "9999") - .exec(); - - dockerClient.startContainerCmd(container.getId()).exec(); - - dockerClient.disconnectFromNetworkCmd() - .withNetworkId(network.getId()) - .withContainerId(container.getId()) - .withForce(true) - .exec(); - - Network updatedNetwork = dockerClient.inspectNetworkCmd().withNetworkId(network.getId()).exec(); - assertFalse(updatedNetwork.getContainers().containsKey(container.getId())); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/EventsCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/EventsCmdExecTest.java deleted file mode 100644 index a634e2562..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/EventsCmdExecTest.java +++ /dev/null @@ -1,161 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.lang.reflect.Method; -import java.util.ArrayList; -import java.util.List; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.TimeUnit; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.model.Event; -import com.github.dockerjava.core.command.EventsResultCallback; -import com.github.dockerjava.core.command.PullImageResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class EventsCmdExecTest extends AbstractNettyDockerClientTest { - - private static int KNOWN_NUM_EVENTS = 4; - - private static String getEpochTime() { - return String.valueOf(System.currentTimeMillis() / 1000); - } - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - /* - * This specific test may fail with boot2docker as time may not in sync with host system - */ - @Test - public void testEventStreamTimeBound() throws Exception { - // Don't include other tests events - TimeUnit.SECONDS.sleep(1); - - String startTime = getEpochTime(); - int expectedEvents = generateEvents(); - String endTime = getEpochTime(); - - CountDownLatch countDownLatch = new CountDownLatch(expectedEvents); - EventsTestCallback eventCallback = new EventsTestCallback(countDownLatch); - - dockerClient.eventsCmd().withSince(startTime).withUntil(endTime).exec(eventCallback); - - Boolean zeroCount = countDownLatch.await(10, TimeUnit.SECONDS); - - eventCallback.close(); - - assertTrue(zeroCount, "Received only: " + eventCallback.getEvents()); - } - - @Test - public void testEventStreaming1() throws Exception { - // Don't include other tests events - TimeUnit.SECONDS.sleep(1); - - CountDownLatch countDownLatch = new CountDownLatch(KNOWN_NUM_EVENTS); - EventsTestCallback eventCallback = new EventsTestCallback(countDownLatch); - - dockerClient.eventsCmd().withSince(getEpochTime()).exec(eventCallback); - - generateEvents(); - - Boolean zeroCount = countDownLatch.await(10, TimeUnit.SECONDS); - - eventCallback.close(); - assertTrue(zeroCount, "Received only: " + eventCallback.getEvents()); - } - - @Test - public void testEventStreaming2() throws Exception { - // Don't include other tests events - TimeUnit.SECONDS.sleep(1); - - CountDownLatch countDownLatch = new CountDownLatch(KNOWN_NUM_EVENTS); - EventsTestCallback eventCallback = new EventsTestCallback(countDownLatch); - - dockerClient.eventsCmd().withSince(getEpochTime()).exec(eventCallback); - - generateEvents(); - - Boolean zeroCount = countDownLatch.await(10, TimeUnit.SECONDS); - - eventCallback.close(); - assertTrue(zeroCount, "Received only: " + eventCallback.getEvents()); - } - - public void testEventStreamingWithFilter() throws Exception { - // Don't include other tests events - TimeUnit.SECONDS.sleep(1); - - CountDownLatch countDownLatch = new CountDownLatch(1); - EventsTestCallback eventCallback = dockerClient.eventsCmd().withEventFilter("start") - .exec(new EventsTestCallback(countDownLatch)); - - generateEvents(); - - Boolean zeroCount = countDownLatch.await(10, TimeUnit.SECONDS); - - eventCallback.close(); - assertTrue(zeroCount, "Received only: " + eventCallback.getEvents()); - } - - /** - * This method generates {#link KNOWN_NUM_EVENTS} events - */ - private int generateEvents() throws Exception { - String testImage = "busybox"; - - dockerClient.pullImageCmd(testImage).exec(new PullImageResultCallback()).awaitSuccess(); - - CreateContainerResponse container = dockerClient.createContainerCmd(testImage).withCmd("sleep", "9999").exec(); - dockerClient.startContainerCmd(container.getId()).exec(); - dockerClient.stopContainerCmd(container.getId()).exec(); - return KNOWN_NUM_EVENTS; - } - - private class EventsTestCallback extends EventsResultCallback { - - private final CountDownLatch countDownLatch; - - private final List events = new ArrayList(); - - public EventsTestCallback(CountDownLatch countDownLatch) { - this.countDownLatch = countDownLatch; - } - - public void onNext(Event event) { - LOG.info("Received event #{}: {}", countDownLatch.getCount(), event); - countDownLatch.countDown(); - events.add(event); - } - - public List getEvents() { - return new ArrayList(events); - } - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/ExecCreateCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/ExecCreateCmdExecTest.java deleted file mode 100644 index d0ec28ac9..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/ExecCreateCmdExecTest.java +++ /dev/null @@ -1,61 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; -import java.security.SecureRandom; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.ExecCreateCmdResponse; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class ExecCreateCmdExecTest extends AbstractNettyDockerClientTest { - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void execCreateTest() { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("top") - .withName(containerName).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - ExecCreateCmdResponse execCreateCmdResponse = dockerClient.execCreateCmd(container.getId()) - .withCmd("touch", "file.log").exec(); - - assertThat(execCreateCmdResponse.getId(), not(isEmptyString())); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/ExecStartCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/ExecStartCmdExecTest.java deleted file mode 100644 index 52db7fe44..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/ExecStartCmdExecTest.java +++ /dev/null @@ -1,218 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_22; -import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_23; -import static com.github.dockerjava.utils.TestUtils.getVersion; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.containsString; - -import java.io.ByteArrayInputStream; -import java.io.ByteArrayOutputStream; -import java.io.InputStream; -import java.lang.reflect.Method; -import java.security.SecureRandom; -import java.util.concurrent.TimeUnit; - -import com.github.dockerjava.core.RemoteApiVersion; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.ExecCreateCmdResponse; -import com.github.dockerjava.core.command.ExecStartResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class ExecStartCmdExecTest extends AbstractNettyDockerClientTest { - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test() - public void execStart() throws Exception { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("top") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - ExecCreateCmdResponse execCreateCmdResponse = dockerClient.execCreateCmd(container.getId()) - .withAttachStdout(true) - .withCmd("touch", "/execStartTest.log") - .exec(); - - dockerClient.execStartCmd(execCreateCmdResponse.getId()) - .withDetach(false) - .exec(new ExecStartResultCallback(System.out, System.err)) - .awaitCompletion(); - - InputStream response = dockerClient.copyArchiveFromContainerCmd(container.getId(), "/execStartTest.log").exec(); - - Boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied from the container."); - - // read the stream fully. Otherwise, the underlying stream will not be closed. - String responseAsString = asString(response); - assertNotNull(responseAsString); - assertTrue(responseAsString.length() > 0); - } - - @Test() - public void execStartAttached() throws Exception { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - ExecCreateCmdResponse execCreateCmdResponse = dockerClient.execCreateCmd(container.getId()) - .withAttachStdout(true) - .withCmd("touch", "/execStartTest.log") - .exec(); - - dockerClient.execStartCmd(execCreateCmdResponse.getId()) - .withDetach(false) - .exec(new ExecStartResultCallback(System.out, System.err)) - .awaitCompletion(); - - InputStream response = dockerClient.copyArchiveFromContainerCmd(container.getId(), "/execStartTest.log").exec(); - Boolean bytesAvailable = response.available() > 0; - assertTrue(bytesAvailable, "The file was not copied from the container."); - - // read the stream fully. Otherwise, the underlying stream will not be closed. - String responseAsString = asString(response); - assertNotNull(responseAsString); - assertTrue(responseAsString.length() > 0); - } - - @Test() - public void execStartAttachStdin() throws Exception { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InputStream stdin = new ByteArrayInputStream("STDIN\n".getBytes("UTF-8")); - - ByteArrayOutputStream stdout = new ByteArrayOutputStream(); - - ExecCreateCmdResponse execCreateCmdResponse = dockerClient.execCreateCmd(container.getId()) - .withAttachStdout(true) - .withAttachStdin(true) - .withCmd("cat") - .exec(); - - boolean completed = dockerClient.execStartCmd(execCreateCmdResponse.getId()) - .withDetach(false) - .withTty(true) - .withStdIn(stdin) - .exec(new ExecStartResultCallback(stdout, System.err)) - .awaitCompletion(5, TimeUnit.SECONDS); - - assertTrue(completed, "The process was not finished."); - assertEquals(stdout.toString("UTF-8"), "STDIN\n"); - } - - @Test() - public void execStartAttachStdinToShell() throws Exception { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient - .createContainerCmd("busybox") - .withCmd("sleep", "9999") - .withName(containerName) - .exec(); - - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InputStream stdin = new ByteArrayInputStream("ls\n".getBytes()); - - ByteArrayOutputStream stdout = new ByteArrayOutputStream(); - - ExecCreateCmdResponse execCreateCmdResponse = dockerClient.execCreateCmd(container.getId()) - .withAttachStdout(true) - .withAttachStdin(true) - .withTty(false) - .withCmd("/bin/sh").exec(); - - dockerClient.execStartCmd(execCreateCmdResponse.getId()) - .withDetach(false) - .withStdIn(stdin) - .exec(new ExecStartResultCallback(stdout, System.err)) - .awaitCompletion(); - - assertThat(stdout.toString(), containsString("etc\n")); - } - - @Test() - public void execStartNotAttachedStdin() throws Exception { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InputStream stdin = new ByteArrayInputStream("echo STDIN\n".getBytes()); - - ByteArrayOutputStream stdout = new ByteArrayOutputStream(); - - ExecCreateCmdResponse execCreateCmdResponse = dockerClient.execCreateCmd(container.getId()) - .withAttachStdout(true) - .withAttachStdin(false) - .withCmd("/bin/sh") - .exec(); - - boolean completed = dockerClient.execStartCmd(execCreateCmdResponse.getId()) - .withDetach(false) - .withStdIn(stdin) - .exec(new ExecStartResultCallback(stdout, System.err)) - .awaitCompletion(5, TimeUnit.SECONDS); - - assertEquals(stdout.toString(), ""); - - if (getVersion(dockerClient).isGreaterOrEqual(VERSION_1_23)) { - assertFalse(completed, "The process was not finished."); - } else { - assertTrue(completed, "with v1.22 of the remote api the server closed the connection when no stdin " + - "was attached while exec create, so completed was true"); - } - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/InfoCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/InfoCmdExecTest.java deleted file mode 100644 index 850f98a74..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/InfoCmdExecTest.java +++ /dev/null @@ -1,71 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyOrNullString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Info; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class InfoCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void info() throws DockerException { - // Make sure that there is at least one container for the assertion - // TODO extract this into a shared method - if (dockerClient.listContainersCmd().withShowAll(true).exec().size() == 0) { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withName("docker-java-itest-info").withCmd("touch", "/test").exec(); - - LOG.info("Created container: {}", container); - assertThat(container.getId(), not(isEmptyOrNullString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - } - - Info dockerInfo = dockerClient.infoCmd().exec(); - LOG.info(dockerInfo.toString()); - - assertTrue(dockerInfo.toString().contains("containers")); - assertTrue(dockerInfo.toString().contains("images")); - assertTrue(dockerInfo.toString().contains("debug")); - - assertTrue(dockerInfo.getContainers() > 0); - assertTrue(dockerInfo.getImages() > 0); - assertTrue(dockerInfo.getNFd() > 0); - assertTrue(dockerInfo.getNGoroutines() > 0); - assertTrue(dockerInfo.getNCPU() > 0); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/InspectContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/InspectContainerCmdExecTest.java deleted file mode 100644 index af32cebce..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/InspectContainerCmdExecTest.java +++ /dev/null @@ -1,120 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; -import java.security.SecureRandom; - -import com.github.dockerjava.api.command.InspectContainerCmd; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class InspectContainerCmdExecTest extends AbstractNettyDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(InspectContainerCmdExecTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test() - public void inspectContainer() throws DockerException { - - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("top") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse containerInfo = dockerClient.inspectContainerCmd(container.getId()).exec(); - assertEquals(containerInfo.getId(), container.getId()); - - } - - @Test() - public void inspectContainerWithSize() throws DockerException { - - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("top") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerCmd command = dockerClient.inspectContainerCmd(container.getId()) - .withSize(true); - assertTrue(command.getSize()); - InspectContainerResponse containerInfo = command.exec(); - assertEquals(containerInfo.getId(), container.getId()); - assertNotNull(containerInfo.getSizeRootFs()); - assertTrue(containerInfo.getSizeRootFs().intValue() > 0 ); - } - - @Test(expectedExceptions = NotFoundException.class) - public void inspectNonExistingContainer() throws DockerException { - dockerClient.inspectContainerCmd("non-existing").exec(); - } - - @Test - public void inspectContainerRestartCount() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("env").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getRestartCount(), equalTo(0)); - } - - @Test - public void inspectContainerNetworkSettings() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("env").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertFalse(inspectContainerResponse.getNetworkSettings().getHairpinMode()); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/InspectExecCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/InspectExecCmdExecTest.java deleted file mode 100644 index d0bb5ca1f..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/InspectExecCmdExecTest.java +++ /dev/null @@ -1,145 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_22; -import static com.github.dockerjava.utils.TestUtils.getVersion; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.notNullValue; -import static org.hamcrest.Matchers.nullValue; - -import java.io.IOException; -import java.lang.reflect.Method; -import java.security.SecureRandom; - -import com.github.dockerjava.core.RemoteApiVersion; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.ExecCreateCmdResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.command.InspectExecResponse; -import com.github.dockerjava.core.command.ExecStartResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; -import com.github.dockerjava.test.serdes.JSONTestHelper; - -@Test(groups = "integration") -public class InspectExecCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void inspectExec() throws Exception { - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - // Check that file does not exist - ExecCreateCmdResponse checkFileExec1 = dockerClient.execCreateCmd(container.getId()).withAttachStdout(true) - .withAttachStderr(true).withCmd("test", "-e", "/marker").exec(); - LOG.info("Created exec {}", checkFileExec1.toString()); - assertThat(checkFileExec1.getId(), not(isEmptyString())); - dockerClient.execStartCmd(checkFileExec1.getId()).withDetach(false) - .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); - InspectExecResponse first = dockerClient.inspectExecCmd(checkFileExec1.getId()).exec(); - assertThat(first.isRunning(), is(false)); - assertThat(first.getExitCode(), is(1)); - - // Create the file - ExecCreateCmdResponse touchFileExec = dockerClient.execCreateCmd(container.getId()).withAttachStdout(true) - .withAttachStderr(true).withCmd("touch", "/marker").exec(); - LOG.info("Created exec {}", touchFileExec.toString()); - assertThat(touchFileExec.getId(), not(isEmptyString())); - dockerClient.execStartCmd(touchFileExec.getId()).withDetach(false) - .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); - InspectExecResponse second = dockerClient.inspectExecCmd(touchFileExec.getId()).exec(); - assertThat(second.isRunning(), is(false)); - assertThat(second.getExitCode(), is(0)); - - // Check that file does exist now - ExecCreateCmdResponse checkFileExec2 = dockerClient.execCreateCmd(container.getId()).withAttachStdout(true) - .withAttachStderr(true).withCmd("test", "-e", "/marker").exec(); - LOG.info("Created exec {}", checkFileExec2.toString()); - assertThat(checkFileExec2.getId(), not(isEmptyString())); - dockerClient.execStartCmd(checkFileExec2.getId()).withDetach(false) - .exec(new ExecStartResultCallback(System.out, System.err)).awaitCompletion(); - InspectExecResponse third = dockerClient.inspectExecCmd(checkFileExec2.getId()).exec(); - assertThat(third.isRunning(), is(false)); - assertThat(third.getExitCode(), is(0)); - - // Get container info and check its roundtrip to ensure the consistency - InspectContainerResponse containerInfo = dockerClient.inspectContainerCmd(container.getId()).exec(); - assertEquals(containerInfo.getId(), container.getId()); - JSONTestHelper.testRoundTrip(containerInfo); - } - - @Test(groups = "ignoreInCircleCi") - public void inspectExecNetworkSettings() throws IOException { - final RemoteApiVersion apiVersion = getVersion(dockerClient); - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - ExecCreateCmdResponse exec = dockerClient.execCreateCmd(container.getId()).withAttachStdout(true) - .withAttachStderr(true).withCmd("/bin/bash").exec(); - LOG.info("Created exec {}", exec.toString()); - assertThat(exec.getId(), not(isEmptyString())); - - InspectExecResponse inspectExecResponse = dockerClient.inspectExecCmd(exec.getId()).exec(); - - if (apiVersion.isGreaterOrEqual(RemoteApiVersion.VERSION_1_22)) { - assertThat(inspectExecResponse.getExitCode(), is(nullValue())); - assertThat(inspectExecResponse.getCanRemove(), is(false)); - assertThat(inspectExecResponse.getContainerID(), is(container.getId())); - } else { - assertThat(inspectExecResponse.getExitCode(), is(0)); - assertNotNull(inspectExecResponse.getContainer().getNetworkSettings().getNetworks().get("bridge")); - } - - assertThat(inspectExecResponse.isOpenStdin(), is(false)); - assertThat(inspectExecResponse.isOpenStdout(), is(true)); - assertThat(inspectExecResponse.isRunning(), is(false)); - - final InspectExecResponse.Container inspectContainer = inspectExecResponse.getContainer(); - if (apiVersion.isGreaterOrEqual(VERSION_1_22)) { - assertThat(inspectContainer, nullValue()); - } else { - assertThat(inspectContainer, notNullValue()); - assertNotNull(inspectContainer.getNetworkSettings().getNetworks().get("bridge")); - } - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/InspectNetworkCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/InspectNetworkCmdExecTest.java deleted file mode 100644 index 7cbab660b..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/InspectNetworkCmdExecTest.java +++ /dev/null @@ -1,57 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; -import java.util.List; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; - -@Test(groups = "integration") -public class InspectNetworkCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void inspectNetwork() throws DockerException { - - List networks = dockerClient.listNetworksCmd().exec(); - - Network expected = findNetwork(networks, "bridge"); - - Network network = dockerClient.inspectNetworkCmd().withNetworkId(expected.getId()).exec(); - - assertThat(network.getName(), equalTo(expected.getName())); - assertThat(network.getScope(), equalTo(expected.getScope())); - assertThat(network.getDriver(), equalTo(expected.getDriver())); - assertThat(network.getIpam().getConfig().get(0).getSubnet(), equalTo(expected.getIpam().getConfig().get(0).getSubnet())); - assertThat(network.getIpam().getDriver(), equalTo(expected.getIpam().getDriver())); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/InspectVolumeCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/InspectVolumeCmdExecTest.java deleted file mode 100644 index 191767f87..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/InspectVolumeCmdExecTest.java +++ /dev/null @@ -1,65 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.InspectVolumeResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class InspectVolumeCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void inspectVolume() throws DockerException { - - String volumeName = "volume1"; - - dockerClient.createVolumeCmd().withName(volumeName).withDriver("local").exec(); - - InspectVolumeResponse inspectVolumeResponse = dockerClient.inspectVolumeCmd(volumeName).exec(); - - assertThat(inspectVolumeResponse.getName(), equalTo("volume1")); - assertThat(inspectVolumeResponse.getDriver(), equalTo("local")); - assertThat(inspectVolumeResponse.getMountpoint(), containsString("/volume1/")); - } - - @Test(expectedExceptions = NotFoundException.class) - public void inspectNonExistentVolume() throws DockerException { - - String volumeName = "non-existing"; - - dockerClient.inspectVolumeCmd(volumeName).exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/KillContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/KillContainerCmdExecTest.java deleted file mode 100644 index d354c8df3..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/KillContainerCmdExecTest.java +++ /dev/null @@ -1,76 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class KillContainerCmdExecTest extends AbstractNettyDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(KillContainerCmdExecTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void killContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - LOG.info("Killing container: {}", container.getId()); - dockerClient.killContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); - assertThat(inspectContainerResponse.getState().getExitCode(), not(equalTo(0))); - - } - - @Test(expectedExceptions = NotFoundException.class) - public void killNonExistingContainer() throws DockerException { - - dockerClient.killContainerCmd("non-existing").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/ListContainersCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/ListContainersCmdExecTest.java deleted file mode 100644 index 40464c6f7..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/ListContainersCmdExecTest.java +++ /dev/null @@ -1,172 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static ch.lambdaj.Lambda.filter; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.notNullValue; -import static org.hamcrest.Matchers.startsWith; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; -import java.util.List; -import java.util.Map; - -import org.hamcrest.Matcher; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.model.Container; -import com.github.dockerjava.core.command.PullImageResultCallback; -import com.github.dockerjava.core.util.FiltersBuilder; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; -import com.google.common.collect.ImmutableMap; - -@Test(groups = "integration") -public class ListContainersCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void testListContainers() throws Exception { - - String testImage = "busybox"; - - // // need to block until image is pulled completely - // dockerClient.pullImageCmd(testImage).exec(new PullImageResultCallback()).awaitSuccess(); - - List containers = dockerClient.listContainersCmd().withShowAll(true).exec(); - assertThat(containers, notNullValue()); - LOG.info("Container List: {}", containers); - - int size = containers.size(); - - CreateContainerResponse container1 = dockerClient.createContainerCmd(testImage).withCmd("echo").exec(); - - assertThat(container1.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container1.getId()).exec(); - - assertThat(inspectContainerResponse.getConfig().getImage(), is(equalTo(testImage))); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - LOG.info("container id: " + container1.getId()); - - List containers2 = dockerClient.listContainersCmd().withShowAll(true).exec(); - - for (Container container : containers2) { - LOG.info("listContainer: id=" + container.getId() + " image=" + container.getImage()); - } - - assertThat(size + 1, is(equalTo(containers2.size()))); - Matcher matcher = hasItem(hasField("id", startsWith(container1.getId()))); - assertThat(containers2, matcher); - - List filteredContainers = filter(hasField("id", startsWith(container1.getId())), containers2); - assertThat(filteredContainers.size(), is(equalTo(1))); - - for (Container container : filteredContainers) { - LOG.info("filteredContainer: " + container); - } - - Container container2 = filteredContainers.get(0); - assertThat(container2.getCommand(), not(isEmptyString())); - assertThat(container2.getImage(), startsWith(testImage)); - } - - @Test - public void testListContainersWithLabelsFilter() throws Exception { - - String testImage = "busybox"; - - // need to block until image is pulled completely - dockerClient.pullImageCmd(testImage).exec(new PullImageResultCallback()).awaitCompletion(); - - List containers = dockerClient.listContainersCmd().withShowAll(true).exec(); - assertThat(containers, notNullValue()); - LOG.info("Container List: {}", containers); - - int size = containers.size(); - - CreateContainerResponse container1 = dockerClient.createContainerCmd(testImage).withCmd("echo").exec(); - - assertThat(container1.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container1.getId()).exec(); - - assertThat(inspectContainerResponse.getConfig().getImage(), is(equalTo(testImage))); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - LOG.info("container id: " + container1.getId()); - - List containers2 = dockerClient.listContainersCmd().withShowAll(true).exec(); - - for (Container container : containers2) { - LOG.info("listContainer: id=" + container.getId() + " image=" + container.getImage()); - } - - assertThat(size + 1, is(equalTo(containers2.size()))); - Matcher matcher = hasItem(hasField("id", startsWith(container1.getId()))); - assertThat(containers2, matcher); - - List filteredContainers = filter(hasField("id", startsWith(container1.getId())), containers2); - assertThat(filteredContainers.size(), is(equalTo(1))); - - for (Container container : filteredContainers) { - LOG.info("filteredContainer: " + container); - } - - Container container2 = filteredContainers.get(0); - assertThat(container2.getCommand(), not(isEmptyString())); - assertThat(container2.getImage(), startsWith(testImage)); - - Map labels = ImmutableMap.of("test", "docker-java"); - - // list with filter by label - dockerClient.createContainerCmd(testImage).withCmd("echo").withLabels(labels).exec(); - filteredContainers = dockerClient.listContainersCmd().withShowAll(true) - .withLabelFilter(labels).exec(); - assertThat(filteredContainers.size(), is(equalTo(1))); - Container container3 = filteredContainers.get(0); - assertThat(container3.getCommand(), not(isEmptyString())); - assertThat(container3.getImage(), startsWith(testImage)); - - filteredContainers = dockerClient.listContainersCmd().withShowAll(true) - .withLabelFilter("test").exec(); - assertThat(filteredContainers.size(), is(equalTo(1))); - container3 = filteredContainers.get(0); - assertThat(container3.getCommand(), not(isEmptyString())); - assertThat(container3.getImage(), startsWith(testImage)); - assertEquals(container3.getLabels(), labels); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/ListImagesCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/ListImagesCmdExecTest.java deleted file mode 100644 index 170a8479f..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/ListImagesCmdExecTest.java +++ /dev/null @@ -1,112 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.emptyArray; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.greaterThan; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.notNullValue; - -import java.lang.reflect.Method; -import java.util.List; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Image; -import com.github.dockerjava.api.model.Info; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class ListImagesCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void listImages() throws DockerException { - List images = dockerClient.listImagesCmd().withShowAll(true).exec(); - assertThat(images, notNullValue()); - LOG.info("Images List: {}", images); - Info info = dockerClient.infoCmd().exec(); - - assertThat(images.size(), equalTo(info.getImages())); - - Image img = images.get(0); - assertThat(img.getCreated(), is(greaterThan(0L))); - assertThat(img.getVirtualSize(), is(greaterThan(0L))); - assertThat(img.getId(), not(isEmptyString())); - assertThat(img.getRepoTags(), not(emptyArray())); - } - - @Test(groups = "ignoreInCircleCi") - public void listImagesWithDanglingFilter() throws DockerException { - String imageId = createDanglingImage(); - List images = dockerClient.listImagesCmd().withDanglingFilter(true).withShowAll(true).exec(); - assertThat(images, notNullValue()); - LOG.info("Images List: {}", images); - assertThat(images.size(), is(greaterThan(0))); - Boolean imageInFilteredList = isImageInFilteredList(images, imageId); - assertTrue(imageInFilteredList); - } - - @Test - public void listImagesWithNameFilter() throws DockerException { - String imageId = createDanglingImage(); - dockerClient.tagImageCmd(imageId, "test_repository", "latest").exec(); - List images = dockerClient.listImagesCmd().withImageNameFilter("test_repository:latest").exec(); - assertThat(images, notNullValue()); - LOG.info("Images List: {}", images); - assertThat(images.size(), is(equalTo(1))); - Boolean imageInFilteredList = isImageInFilteredList(images, imageId); - assertTrue(imageInFilteredList); - } - - private boolean isImageInFilteredList(List images, String expectedImageId) { - for (Image image : images) { - if (expectedImageId.equals(image.getId())) { - return true; - } - } - return false; - } - - private String createDanglingImage() { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - LOG.info("Committing container {}", container.toString()); - String imageId = dockerClient.commitCmd(container.getId()).exec(); - - dockerClient.stopContainerCmd(container.getId()).exec(); - dockerClient.removeContainerCmd(container.getId()).exec(); - return imageId; - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/ListNetworksCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/ListNetworksCmdExecTest.java deleted file mode 100644 index 8a5d975f0..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/ListNetworksCmdExecTest.java +++ /dev/null @@ -1,55 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; - -import java.lang.reflect.Method; -import java.util.List; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class ListNetworksCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void listNetworks() throws DockerException { - - List networks = dockerClient.listNetworksCmd().exec(); - - Network network = findNetwork(networks, "bridge"); - - assertThat(network.getName(), equalTo("bridge")); - assertThat(network.getScope(), equalTo("local")); - assertThat(network.getDriver(), equalTo("bridge")); - assertThat(network.getIpam().getDriver(), equalTo("default")); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/ListVolumesCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/ListVolumesCmdExecTest.java deleted file mode 100644 index bca5ebece..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/ListVolumesCmdExecTest.java +++ /dev/null @@ -1,59 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.greaterThanOrEqualTo; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateVolumeResponse; -import com.github.dockerjava.api.command.ListVolumesResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class ListVolumesCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void listVolumes() throws DockerException { - - CreateVolumeResponse createVolumeResponse = dockerClient.createVolumeCmd().withName("volume1") - .withDriver("local").exec(); - - assertThat(createVolumeResponse.getName(), equalTo("volume1")); - assertThat(createVolumeResponse.getDriver(), equalTo("local")); - assertThat(createVolumeResponse.getMountpoint(), containsString("/volume1/")); - - ListVolumesResponse listVolumesResponse = dockerClient.listVolumesCmd().exec(); - - assertThat(listVolumesResponse.getVolumes().size(), greaterThanOrEqualTo(1)); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/LoadImageCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/LoadImageCmdExecTest.java deleted file mode 100644 index d978f5662..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/LoadImageCmdExecTest.java +++ /dev/null @@ -1,76 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.model.Image; - -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; -import com.github.dockerjava.utils.TestResources; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.io.InputStream; -import java.lang.reflect.Method; -import java.nio.file.Files; -import java.util.List; - -import static java.util.Arrays.asList; -import static java.util.Collections.singletonList; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.core.IsNull.notNullValue; -import static org.hamcrest.core.IsEqual.equalTo; - -@Test(groups = "integration") -public class LoadImageCmdExecTest extends AbstractNettyDockerClientTest { - - private String expectedImageId; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - expectedImageId = "sha256:56031f66eb0cef2e2e5cb2d1dabafaa0ebcd0a18a507d313b5bdb8c0472c5eba"; - if (findImageWithId(expectedImageId, dockerClient.listImagesCmd().exec()) != null) { - dockerClient.removeImageCmd(expectedImageId).exec(); - } - } - - @AfterMethod - public void afterMethod(ITestResult result) { - dockerClient.removeImageCmd(expectedImageId).exec(); - super.afterMethod(result); - } - - @Test - public void loadImageFromTar() throws Exception { - try (InputStream uploadStream = Files.newInputStream(TestResources.getApiImagesLoadTestTarball())) { - dockerClient.loadImageCmd(uploadStream).exec(); - } - - final Image image = findImageWithId(expectedImageId, dockerClient.listImagesCmd().exec()); - - assertThat("Can't find expected image after loading from a tar archive!", image, notNullValue()); - assertThat("Image after loading from a tar archive has wrong tags!", - asList(image.getRepoTags()), equalTo(singletonList("docker-java/load:1.0"))); - } - - private Image findImageWithId(final String id, final List images) { - for (Image image : images) { - if (id.equals(image.getId())) { - return image; - } - } - return null; - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/LogContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/LogContainerCmdExecTest.java deleted file mode 100644 index fe408ac18..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/LogContainerCmdExecTest.java +++ /dev/null @@ -1,223 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.containsString; - -import java.io.IOException; -import java.lang.reflect.Method; -import java.util.concurrent.TimeUnit; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.StreamType; -import com.github.dockerjava.core.command.WaitContainerResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class LogContainerCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void asyncLogContainerWithTtyEnabled() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("/bin/sh", "-c", "while true; do echo hello; sleep 1; done") - .withTty(true) - .exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()) - .exec(); - - LogContainerTestCallback loggingCallback = new LogContainerTestCallback(true); - - // this essentially test the since=0 case - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .withFollowStream(true) - .withTailAll() - .exec(loggingCallback); - - loggingCallback.awaitCompletion(3, TimeUnit.SECONDS); - - assertTrue(loggingCallback.toString().contains("hello")); - - assertEquals(loggingCallback.getCollectedFrames().get(0).getStreamType(), StreamType.RAW); - } - - @Test - public void asyncLogContainerWithTtyDisabled() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("/bin/sh", "-c", "while true; do echo hello; sleep 1; done") - .withTty(false) - .exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()) - .exec(); - - LogContainerTestCallback loggingCallback = new LogContainerTestCallback(true); - - // this essentially test the since=0 case - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .withFollowStream(true) - .withTailAll() - .exec(loggingCallback); - - loggingCallback.awaitCompletion(3, TimeUnit.SECONDS); - - assertTrue(loggingCallback.toString().contains("hello")); - - assertEquals(loggingCallback.getCollectedFrames().get(0).getStreamType(), StreamType.STDOUT); - } - - @Test - public void asyncLogNonExistingContainer() throws Exception { - - LogContainerTestCallback loggingCallback = new LogContainerTestCallback() { - @Override - public void onError(Throwable throwable) { - - assertEquals(throwable.getClass().getName(), NotFoundException.class.getName()); - - try { - // close the callback to prevent the call to onComplete - close(); - } catch (IOException e) { - throw new RuntimeException(); - } - - super.onError(throwable); - } - - public void onComplete() { - super.onComplete(); - fail("expected NotFoundException"); - }; - }; - - dockerClient.logContainerCmd("non-existing").withStdErr(true).withStdOut(true).exec(loggingCallback) - .awaitCompletion(); - } - - @Test - public void asyncMultipleLogContainer() throws Exception { - - String snippet = "hello world"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("/bin/echo", snippet) - .exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - int exitCode = dockerClient.waitContainerCmd(container.getId()) - .exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - - assertThat(exitCode, equalTo(0)); - - LogContainerTestCallback loggingCallback = new LogContainerTestCallback(); - - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .exec(loggingCallback); - - loggingCallback.close(); - - loggingCallback = new LogContainerTestCallback(); - - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .exec(loggingCallback); - - loggingCallback.close(); - - loggingCallback = new LogContainerTestCallback(); - - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .exec(loggingCallback); - - loggingCallback.awaitCompletion(); - - assertTrue(loggingCallback.toString().contains(snippet)); - } - - @Test - public void asyncLogContainerWithSince() throws Exception { - String snippet = "hello world"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withCmd("/bin/echo", snippet) - .exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - int timestamp = (int) (System.currentTimeMillis() / 1000); - - dockerClient.startContainerCmd(container.getId()).exec(); - - int exitCode = dockerClient.waitContainerCmd(container.getId()) - .exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - - assertThat(exitCode, equalTo(0)); - - LogContainerTestCallback loggingCallback = new LogContainerTestCallback(); - - dockerClient.logContainerCmd(container.getId()) - .withStdErr(true) - .withStdOut(true) - .withSince(timestamp) - .exec(loggingCallback); - - loggingCallback.awaitCompletion(); - - assertThat(loggingCallback.toString(), containsString(snippet)); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/PauseCmdImplTest.java b/src/test/java/com/github/dockerjava/netty/exec/PauseCmdImplTest.java deleted file mode 100644 index 612c3efd2..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/PauseCmdImplTest.java +++ /dev/null @@ -1,103 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.InternalServerErrorException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; -import com.github.dockerjava.utils.ContainerUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -@Test(groups = {"integration", "ignoreInCircleCi"}) -public class PauseCmdImplTest extends AbstractNettyDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(PauseCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void pauseRunningContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.pauseContainer(dockerClient, container); - } - - @Test(expectedExceptions = NotFoundException.class) - public void pauseNonExistingContainer() { - - dockerClient.pauseContainerCmd("non-existing").exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void pauseStoppedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.stopContainer(dockerClient, container); - - dockerClient.pauseContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void pausePausedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.pauseContainer(dockerClient, container); - - dockerClient.pauseContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void pauseCreatedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.pauseContainerCmd(container.getId()).exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/PingCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/PingCmdExecTest.java deleted file mode 100644 index 128c1969a..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/PingCmdExecTest.java +++ /dev/null @@ -1,43 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class PingCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void ping() throws DockerException { - dockerClient.pingCmd().exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/PullImageCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/PullImageCmdExecTest.java deleted file mode 100644 index 53fa8d660..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/PullImageCmdExecTest.java +++ /dev/null @@ -1,119 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.lessThanOrEqualTo; -import static org.hamcrest.Matchers.notNullValue; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.async.ResultCallback; -import com.github.dockerjava.api.command.InspectImageResponse; -import com.github.dockerjava.api.command.PullImageCmd; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Info; -import com.github.dockerjava.api.model.PullResponseItem; -import com.github.dockerjava.core.command.PullImageCmdImpl; -import com.github.dockerjava.core.command.PullImageResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class PullImageCmdExecTest extends AbstractNettyDockerClientTest { - - private static final PullImageCmd.Exec NOP_EXEC = new PullImageCmd.Exec() { - public Void exec(PullImageCmd command, ResultCallback resultCallback) { - return null; - }; - }; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void nullAuthConfig() throws Exception { - PullImageCmdImpl pullImageCmd = new PullImageCmdImpl(NOP_EXEC, null, ""); - try { - pullImageCmd.withAuthConfig(null); - fail(); - } catch (Exception e) { - assertEquals(e.getMessage(), "authConfig was not specified"); - } finally { - pullImageCmd.close(); - } - } - - @Test - public void testPullImage() throws Exception { - Info info = dockerClient.infoCmd().exec(); - LOG.info("Client info: {}", info.toString()); - - int imgCount = info.getImages(); - LOG.info("imgCount1: {}", imgCount); - - // This should be an image that is not used by other repositories - // already - // pulled down, preferably small in size. If tag is not used pull will - // download all images in that repository but tmpImgs will only - // deleted 'latest' image but not images with other tags - String testImage = "hackmann/empty"; - - LOG.info("Removing image: {}", testImage); - - try { - dockerClient.removeImageCmd(testImage).withForce(true).exec(); - } catch (NotFoundException e) { - // just ignore if not exist - } - - info = dockerClient.infoCmd().exec(); - LOG.info("Client info: {}", info.toString()); - - imgCount = info.getImages(); - LOG.info("imgCount2: {}", imgCount); - - LOG.info("Pulling image: {}", testImage); - - dockerClient.pullImageCmd(testImage).exec(new PullImageResultCallback()).awaitSuccess(); - - info = dockerClient.infoCmd().exec(); - LOG.info("Client info after pull, {}", info.toString()); - - assertThat(imgCount, lessThanOrEqualTo(info.getImages())); - - InspectImageResponse inspectImageResponse = dockerClient.inspectImageCmd(testImage).exec(); - LOG.info("Image Inspect: {}", inspectImageResponse.toString()); - assertThat(inspectImageResponse, notNullValue()); - } - - @Test - public void testPullNonExistingImage() throws Exception { - - // does not throw an exception - // stream needs to be fully read in order to close the underlying connection - dockerClient.pullImageCmd("xvxcv/foo").exec(new PullImageResultCallback()).awaitCompletion(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/PushImageCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/PushImageCmdExecTest.java deleted file mode 100644 index 73e99af6d..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/PushImageCmdExecTest.java +++ /dev/null @@ -1,79 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.core.command.PullImageResultCallback; -import com.github.dockerjava.core.command.PushImageResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = {"integration", "integration-auth"}) -public class PushImageCmdExecTest extends AbstractNettyDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(PushImageCmdExecTest.class); - - String username; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - username = dockerClient.authConfig().getUsername(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void pushLatest() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - LOG.info("Committing container: {}", container.toString()); - String imageId = dockerClient.commitCmd(container.getId()).withRepository(username + "/busybox").exec(); - - // we have to block until image is pushed - dockerClient.pushImageCmd(username + "/busybox").exec(new PushImageResultCallback()).awaitSuccess(); - - LOG.info("Removing image: {}", imageId); - dockerClient.removeImageCmd(imageId).exec(); - - dockerClient.pullImageCmd(username + "/busybox").exec(new PullImageResultCallback()).awaitSuccess(); - } - - @Test(expectedExceptions = DockerClientException.class) - public void pushNonExistentImage() throws Exception { - - dockerClient.pushImageCmd(username + "/xxx").exec(new PushImageResultCallback()).awaitSuccess(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/RemoveContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/RemoveContainerCmdExecTest.java deleted file mode 100644 index 1e7abebdb..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/RemoveContainerCmdExecTest.java +++ /dev/null @@ -1,78 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.startsWith; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; -import java.util.List; - -import org.hamcrest.Matcher; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Container; -import com.github.dockerjava.core.command.WaitContainerResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class RemoveContainerCmdExecTest extends AbstractNettyDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(RemoveContainerCmdExecTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void removeContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true").exec(); - - dockerClient.startContainerCmd(container.getId()).exec(); - dockerClient.waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()).awaitStatusCode(); - - LOG.info("Removing container: {}", container.getId()); - dockerClient.removeContainerCmd(container.getId()).exec(); - - List containers2 = dockerClient.listContainersCmd().withShowAll(true).exec(); - - Matcher matcher = not(hasItem(hasField("id", startsWith(container.getId())))); - assertThat(containers2, matcher); - - } - - @Test(expectedExceptions = NotFoundException.class) - public void removeNonExistingContainer() throws DockerException { - - dockerClient.removeContainerCmd("non-existing").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/RemoveImageCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/RemoveImageCmdExecTest.java deleted file mode 100644 index 857bec714..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/RemoveImageCmdExecTest.java +++ /dev/null @@ -1,83 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.startsWith; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; -import java.util.List; - -import org.hamcrest.Matcher; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Container; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class RemoveImageCmdExecTest extends AbstractNettyDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(RemoveImageCmdExecTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void removeImage() throws DockerException, InterruptedException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - LOG.info("Committing container {}", container.toString()); - String imageId = dockerClient.commitCmd(container.getId()).exec(); - - dockerClient.stopContainerCmd(container.getId()).exec(); - dockerClient.removeContainerCmd(container.getId()).exec(); - - LOG.info("Removing image: {}", imageId); - dockerClient.removeImageCmd(imageId).exec(); - - List containers = dockerClient.listContainersCmd().withShowAll(true).exec(); - - Matcher matcher = not(hasItem(hasField("id", startsWith(imageId)))); - assertThat(containers, matcher); - } - - @Test(expectedExceptions = NotFoundException.class) - public void removeNonExistingImage() throws DockerException, InterruptedException { - - dockerClient.removeImageCmd("non-existing").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/RemoveNetworkCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/RemoveNetworkCmdExecTest.java deleted file mode 100644 index f8e7d26ca..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/RemoveNetworkCmdExecTest.java +++ /dev/null @@ -1,73 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.CreateNetworkResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Network; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; -import org.hamcrest.Matcher; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; -import java.util.List; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.startsWith; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -@Test(groups = "integration") -public class RemoveNetworkCmdExecTest extends AbstractNettyDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(RemoveNetworkCmdExecTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void removeNetwork() throws DockerException { - - CreateNetworkResponse network = dockerClient.createNetworkCmd().withName("test-network").exec(); - - LOG.info("Removing network: {}", network.getId()); - dockerClient.removeNetworkCmd(network.getId()).exec(); - - List networks = dockerClient.listNetworksCmd().exec(); - - Matcher matcher = not(hasItem(hasField("id", startsWith(network.getId())))); - assertThat(networks, matcher); - - } - - @Test(expectedExceptions = NotFoundException.class) - public void removeNonExistingContainer() throws DockerException { - - dockerClient.removeNetworkCmd("non-existing").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/RemoveVolumeCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/RemoveVolumeCmdExecTest.java deleted file mode 100644 index d7995861c..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/RemoveVolumeCmdExecTest.java +++ /dev/null @@ -1,60 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.containsString; -import static org.hamcrest.Matchers.equalTo; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateVolumeResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class RemoveVolumeCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(expectedExceptions = NotFoundException.class) - public void removeVolume() throws DockerException { - - String volumeName = "volume1"; - - CreateVolumeResponse createVolumeResponse = dockerClient.createVolumeCmd().withName(volumeName) - .withDriver("local").exec(); - - assertThat(createVolumeResponse.getName(), equalTo(volumeName)); - assertThat(createVolumeResponse.getDriver(), equalTo("local")); - assertThat(createVolumeResponse.getMountpoint(), containsString("/volume1/")); - - dockerClient.removeVolumeCmd(volumeName).exec(); - - dockerClient.inspectVolumeCmd(volumeName).exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/RenameContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/RenameContainerCmdExecTest.java deleted file mode 100644 index 100a7d772..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/RenameContainerCmdExecTest.java +++ /dev/null @@ -1,76 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -@Test(groups = "integration") -public class RenameContainerCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void renameContainer() throws DockerException { - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - String name1 = inspectContainerResponse.getName(); - - dockerClient.renameContainerCmd(container.getId()) - .withName(getClass().getCanonicalName() + "renameContainer") - .exec(); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect After Rename: {}", inspectContainerResponse2.toString()); - - String name2 = inspectContainerResponse2.getName(); - - assertNotEquals(name1, name2); - - dockerClient.killContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = NotFoundException.class) - public void renameNonExistingContainer() throws DockerException, InterruptedException { - dockerClient.renameContainerCmd("non-existing") - .withName(getClass().getCanonicalName() + "renameExistingContainer") - .exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/RestartContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/RestartContainerCmdExecTest.java deleted file mode 100644 index 3b1ada5c2..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/RestartContainerCmdExecTest.java +++ /dev/null @@ -1,80 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class RestartContainerCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void restartContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - String startTime = inspectContainerResponse.getState().getStartedAt(); - - dockerClient.restartContainerCmd(container.getId()).withtTimeout(2).exec(); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect After Restart: {}", inspectContainerResponse2.toString()); - - String startTime2 = inspectContainerResponse2.getState().getStartedAt(); - - assertThat(startTime, not(equalTo(startTime2))); - - assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(true))); - - dockerClient.killContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = NotFoundException.class) - public void restartNonExistingContainer() throws DockerException, InterruptedException { - - dockerClient.restartContainerCmd("non-existing").exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/SaveImageCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/SaveImageCmdExecTest.java deleted file mode 100644 index 0527b793f..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/SaveImageCmdExecTest.java +++ /dev/null @@ -1,55 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.greaterThan; - -import java.io.InputStream; -import java.lang.reflect.Method; - -import org.apache.commons.io.IOUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class SaveImageCmdExecTest extends AbstractNettyDockerClientTest { - public static final Logger LOG = LoggerFactory.getLogger(SaveImageCmdExecTest.class); - - String username; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void saveImage() throws Exception { - - InputStream image = IOUtils.toBufferedInputStream(dockerClient.saveImageCmd("busybox").exec()); - assertThat(image.available(), greaterThan(0)); - - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/SearchImagesCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/SearchImagesCmdExecTest.java deleted file mode 100644 index 70e102335..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/SearchImagesCmdExecTest.java +++ /dev/null @@ -1,59 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static ch.lambdaj.Lambda.filter; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.is; -import static org.testinfected.hamcrest.jpa.HasFieldWithValue.hasField; - -import java.lang.reflect.Method; -import java.util.List; - -import org.hamcrest.Matcher; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.SearchItem; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class SearchImagesCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void searchImages() throws DockerException { - List dockerSearch = dockerClient.searchImagesCmd("busybox").exec(); - LOG.info("Search returned {}", dockerSearch.toString()); - - Matcher matcher = hasItem(hasField("name", equalTo("busybox"))); - assertThat(dockerSearch, matcher); - - assertThat(filter(hasField("name", is("busybox")), dockerSearch).size(), equalTo(1)); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/StartContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/StartContainerCmdExecTest.java deleted file mode 100644 index eed82fcd9..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/StartContainerCmdExecTest.java +++ /dev/null @@ -1,554 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.command.StartContainerCmd; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.InternalServerErrorException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.Bind; -import com.github.dockerjava.api.model.Device; -import com.github.dockerjava.api.model.ExposedPort; -import com.github.dockerjava.api.model.Link; -import com.github.dockerjava.api.model.Ports; -import com.github.dockerjava.api.model.RestartPolicy; -import com.github.dockerjava.api.model.Volume; -import com.github.dockerjava.api.model.VolumesFrom; -import com.github.dockerjava.api.model.Ports.Binding; -import com.github.dockerjava.core.command.WaitContainerResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; -import java.util.Arrays; -import java.util.List; -import java.util.UUID; - -import static com.github.dockerjava.api.model.AccessMode.ro; -import static com.github.dockerjava.api.model.Capability.MKNOD; -import static com.github.dockerjava.api.model.Capability.NET_ADMIN; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.contains; -import static org.hamcrest.Matchers.containsInAnyOrder; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.hasSize; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; -import static org.hamcrest.Matchers.notNullValue; -import static org.hamcrest.Matchers.startsWith; - -@Test(groups = "integration") -public class StartContainerCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void startContainerWithVolumes() throws DockerException { - - // see http://docs.docker.io/use/working_with_volumes/ - Volume volume1 = new Volume("/opt/webapp1"); - - Volume volume2 = new Volume("/opt/webapp2"); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox") - .withVolumes(volume1, volume2) - .withCmd("true") - .withBinds(new Bind("/src/webapp1", volume1, ro), new Bind("/src/webapp2", volume2)) - .exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getConfig().getVolumes().keySet(), contains("/opt/webapp1", "/opt/webapp2")); - - dockerClient.startContainerCmd(container.getId()).exec(); - - dockerClient.waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()).awaitStatusCode(); - - inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse, mountedVolumes(containsInAnyOrder(volume1, volume2))); - - final List mounts = inspectContainerResponse.getMounts(); - - assertThat(mounts, hasSize(2)); - - final InspectContainerResponse.Mount mount1 = new InspectContainerResponse.Mount() - .withRw(false).withMode("ro").withDestination(volume1).withSource("/src/webapp1"); - final InspectContainerResponse.Mount mount2 = new InspectContainerResponse.Mount() - .withRw(true).withMode("rw").withDestination(volume2).withSource("/src/webapp2"); - - assertThat(mounts, containsInAnyOrder(mount1, mount2)); - } - - @Test - public void startContainerWithVolumesFrom() throws DockerException { - - Volume volume1 = new Volume("/opt/webapp1"); - Volume volume2 = new Volume("/opt/webapp2"); - - String container1Name = UUID.randomUUID().toString(); - - CreateContainerResponse container1 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName(container1Name) - .withBinds(new Bind("/src/webapp1", volume1), new Bind("/src/webapp2", volume2)).exec(); - LOG.info("Created container1 {}", container1.toString()); - - dockerClient.startContainerCmd(container1.getId()).exec(); - LOG.info("Started container1 {}", container1.toString()); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - - assertThat(inspectContainerResponse1, mountedVolumes(containsInAnyOrder(volume1, volume2))); - - CreateContainerResponse container2 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withVolumesFrom(new VolumesFrom(container1Name)).exec(); - LOG.info("Created container2 {}", container2.toString()); - - dockerClient.startContainerCmd(container2.getId()).exec(); - LOG.info("Started container2 {}", container2.toString()); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - - assertThat(inspectContainerResponse2, mountedVolumes(containsInAnyOrder(volume1, volume2))); - } - - @Test - public void startContainerWithDns() throws DockerException { - - String aDnsServer = "8.8.8.8"; - String anotherDnsServer = "8.8.4.4"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withDns(aDnsServer, anotherDnsServer).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDns()), - contains(aDnsServer, anotherDnsServer)); - } - - @Test - public void startContainerWithDnsSearch() throws DockerException { - - String dnsSearch = "example.com"; - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withDnsSearch(dnsSearch).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - dockerClient.startContainerCmd(container.getId()).exec(); - - inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDnsSearch()), contains(dnsSearch)); - } - - @Test - public void startContainerWithPortBindings() throws DockerException { - - ExposedPort tcp22 = ExposedPort.tcp(22); - ExposedPort tcp23 = ExposedPort.tcp(23); - - Ports portBindings = new Ports(); - portBindings.bind(tcp22, Binding.bindPort(11022)); - portBindings.bind(tcp23, Binding.bindPort(11023)); - portBindings.bind(tcp23, Binding.bindPort(11024)); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withExposedPorts(tcp22, tcp23).withPortBindings(portBindings).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - dockerClient.startContainerCmd(container.getId()).exec(); - - inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getConfig().getExposedPorts()), contains(tcp22, tcp23)); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp22)[0], - is(equalTo(Binding.bindPort(11022)))); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[0], - is(equalTo(Binding.bindPort(11023)))); - - assertThat(inspectContainerResponse.getHostConfig().getPortBindings().getBindings().get(tcp23)[1], - is(equalTo(Binding.bindPort(11024)))); - - } - - @Test - public void startContainerWithRandomPortBindings() throws DockerException { - - ExposedPort tcp22 = ExposedPort.tcp(22); - ExposedPort tcp23 = ExposedPort.tcp(23); - - Ports portBindings = new Ports(); - portBindings.bind(tcp22, Binding.empty()); - portBindings.bind(tcp23, Binding.empty()); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withExposedPorts(tcp22, tcp23).withPortBindings(portBindings).withPublishAllPorts(true).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(Arrays.asList(inspectContainerResponse.getConfig().getExposedPorts()), contains(tcp22, tcp23)); - - assertThat(inspectContainerResponse.getNetworkSettings().getPorts().getBindings().get(tcp22)[0].getHostPortSpec(), - is(not(equalTo(String.valueOf(tcp22.getPort()))))); - - assertThat(inspectContainerResponse.getNetworkSettings().getPorts().getBindings().get(tcp23)[0].getHostPortSpec(), - is(not(equalTo(String.valueOf(tcp23.getPort()))))); - - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void startContainerWithConflictingPortBindings() throws DockerException { - - ExposedPort tcp22 = ExposedPort.tcp(22); - ExposedPort tcp23 = ExposedPort.tcp(23); - - Ports portBindings = new Ports(); - portBindings.bind(tcp22, Binding.bindPort(11022)); - portBindings.bind(tcp23, Binding.bindPort(11022)); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withExposedPorts(tcp22, tcp23).withPortBindings(portBindings).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - } - - @Test - public void startContainerWithLinkingDeprecated() throws DockerException { - - CreateContainerResponse container1 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container1").exec(); - - LOG.info("Created container1 {}", container1.toString()); - assertThat(container1.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); - - assertThat(inspectContainerResponse1.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse1.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getId(), startsWith(container1.getId())); - assertThat(inspectContainerResponse1.getName(), equalTo("/container1")); - assertThat(inspectContainerResponse1.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getState(), is(notNullValue())); - assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); - - if (!inspectContainerResponse1.getState().getRunning()) { - assertThat(inspectContainerResponse1.getState().getExitCode(), is(equalTo(0))); - } - - CreateContainerResponse container2 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container2").withLinks(new Link("container1", "container1Link")).exec(); - - LOG.info("Created container2 {}", container2.toString()); - assertThat(container2.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container2.getId()).exec(); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - LOG.info("Container2 Inspect: {}", inspectContainerResponse2.toString()); - - assertThat(inspectContainerResponse2.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse2.getHostConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[] {new Link("container1", - "container1Link")})); - assertThat(inspectContainerResponse2.getId(), startsWith(container2.getId())); - assertThat(inspectContainerResponse2.getName(), equalTo("/container2")); - assertThat(inspectContainerResponse2.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse2.getState(), is(notNullValue())); - assertThat(inspectContainerResponse2.getState().getRunning(), is(true)); - - } - - @Test - public void startContainerWithLinking() throws DockerException { - - CreateContainerResponse container1 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container1").exec(); - - LOG.info("Created container1 {}", container1.toString()); - assertThat(container1.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container1.getId()).exec(); - - InspectContainerResponse inspectContainerResponse1 = dockerClient.inspectContainerCmd(container1.getId()) - .exec(); - LOG.info("Container1 Inspect: {}", inspectContainerResponse1.toString()); - - assertThat(inspectContainerResponse1.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse1.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getId(), startsWith(container1.getId())); - assertThat(inspectContainerResponse1.getName(), equalTo("/container1")); - assertThat(inspectContainerResponse1.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse1.getState(), is(notNullValue())); - assertThat(inspectContainerResponse1.getState().getRunning(), is(true)); - - if (!inspectContainerResponse1.getState().getRunning()) { - assertThat(inspectContainerResponse1.getState().getExitCode(), is(equalTo(0))); - } - - CreateContainerResponse container2 = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withName("container2").withLinks(new Link("container1", "container1Link")).exec(); - - LOG.info("Created container2 {}", container2.toString()); - assertThat(container2.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container2.getId()).exec(); - - InspectContainerResponse inspectContainerResponse2 = dockerClient.inspectContainerCmd(container2.getId()) - .exec(); - LOG.info("Container2 Inspect: {}", inspectContainerResponse2.toString()); - - assertThat(inspectContainerResponse2.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getId(), not(isEmptyString())); - assertThat(inspectContainerResponse2.getHostConfig(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), is(notNullValue())); - assertThat(inspectContainerResponse2.getHostConfig().getLinks(), equalTo(new Link[] {new Link("container1", - "container1Link")})); - assertThat(inspectContainerResponse2.getId(), startsWith(container2.getId())); - assertThat(inspectContainerResponse2.getName(), equalTo("/container2")); - assertThat(inspectContainerResponse2.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse2.getState(), is(notNullValue())); - assertThat(inspectContainerResponse2.getState().getRunning(), is(true)); - - } - - @Test - public void startContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd(new String[] {"top"}) - .exec(); - - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - assertThat(inspectContainerResponse.getConfig(), is(notNullValue())); - assertThat(inspectContainerResponse.getId(), not(isEmptyString())); - - assertThat(inspectContainerResponse.getId(), startsWith(container.getId())); - - assertThat(inspectContainerResponse.getImageId(), not(isEmptyString())); - assertThat(inspectContainerResponse.getState(), is(notNullValue())); - - assertThat(inspectContainerResponse.getState().getRunning(), is(true)); - - if (!inspectContainerResponse.getState().getRunning()) { - assertThat(inspectContainerResponse.getState().getExitCode(), is(equalTo(0))); - } - } - - @Test(expectedExceptions = NotFoundException.class) - public void testStartNonExistingContainer() throws DockerException { - - dockerClient.startContainerCmd("non-existing").exec(); - } - - /** - * This tests support for --net option for the docker run command: --net="bridge" Set the Network mode for the container 'bridge': - * creates a new network stack for the container on the docker bridge 'none': no networking for this container 'container:': reuses - * another container network stack 'host': use the host network stack inside the container. Note: the host mode gives the container full - * access to local system services such as D-bus and is therefore considered insecure. - */ - @Test - public void startContainerWithNetworkMode() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true") - .withNetworkMode("host").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - dockerClient.startContainerCmd(container.getId()).exec(); - - inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getHostConfig().getNetworkMode(), is(equalTo("host"))); - } - - @Test - public void startContainerWithCapAddAndCapDrop() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withCapAdd(NET_ADMIN).withCapDrop(MKNOD).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getState().getRunning(), is(true)); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapAdd()), contains(NET_ADMIN)); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getCapDrop()), contains(MKNOD)); - } - - @Test - public void startContainerWithDevices() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withDevices(new Device("rwm", "/dev/nulo", "/dev/zero")).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getState().getRunning(), is(true)); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDevices()), contains(new Device("rwm", - "/dev/nulo", "/dev/zero"))); - } - - @Test - public void startContainerWithExtraHosts() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withExtraHosts("dockerhost:127.0.0.1").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getState().getRunning(), is(true)); - - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getExtraHosts()), - contains("dockerhost:127.0.0.1")); - } - - @Test - public void startContainerWithRestartPolicy() throws DockerException { - - RestartPolicy restartPolicy = RestartPolicy.onFailureRestart(5); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999") - .withRestartPolicy(restartPolicy).exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - assertThat(inspectContainerResponse.getState().getRunning(), is(true)); - - assertThat(inspectContainerResponse.getHostConfig().getRestartPolicy(), is(equalTo(restartPolicy))); - } - - @Test - public void existingHostConfigIsPreservedByBlankStartCmd() throws DockerException { - - String dnsServer = "8.8.8.8"; - - // prepare a container with custom DNS - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withDns(dnsServer) - .withCmd("true").exec(); - - LOG.info("Created container {}", container.toString()); - - assertThat(container.getId(), not(isEmptyString())); - - // start container _without_any_customization_ (important!) - dockerClient.startContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - - // The DNS setting survived. - assertThat(inspectContainerResponse.getHostConfig().getDns(), is(notNullValue())); - assertThat(Arrays.asList(inspectContainerResponse.getHostConfig().getDns()), contains(dnsServer)); - } - - @Test - public void anUnconfiguredCommandSerializesToEmptyJson() throws Exception { - ObjectMapper objectMapper = new ObjectMapper(); - StartContainerCmd command = dockerClient.startContainerCmd(""); - assertThat(objectMapper.writeValueAsString(command), is("{}")); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/StatsCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/StatsCmdExecTest.java deleted file mode 100644 index 2b84ae6b0..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/StatsCmdExecTest.java +++ /dev/null @@ -1,106 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.io.IOException; -import java.lang.reflect.Method; -import java.security.SecureRandom; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.TimeUnit; - -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.model.Statistics; -import com.github.dockerjava.core.async.ResultCallbackTemplate; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class StatsCmdExecTest extends AbstractNettyDockerClientTest { - - private static int NUM_STATS = 5; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void testStatsStreaming() throws InterruptedException, IOException { - TimeUnit.SECONDS.sleep(1); - - CountDownLatch countDownLatch = new CountDownLatch(NUM_STATS); - - String containerName = "generated_" + new SecureRandom().nextInt(); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("top") - .withName(containerName).exec(); - LOG.info("Created container {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - StatsCallbackTest statsCallback = dockerClient.statsCmd(container.getId()) - .exec(new StatsCallbackTest(countDownLatch)); - - countDownLatch.await(3, TimeUnit.SECONDS); - Boolean gotStats = statsCallback.gotStats(); - - LOG.info("Stop stats collection"); - - statsCallback.close(); - - LOG.info("Stopping container"); - dockerClient.stopContainerCmd(container.getId()).exec(); - dockerClient.removeContainerCmd(container.getId()).exec(); - - LOG.info("Completed test"); - assertTrue(gotStats, "Expected true"); - - } - - private class StatsCallbackTest extends ResultCallbackTemplate { - private final CountDownLatch countDownLatch; - - private Boolean gotStats = false; - - public StatsCallbackTest(CountDownLatch countDownLatch) { - this.countDownLatch = countDownLatch; - } - - @Override - public void onNext(Statistics stats) { - LOG.info("Received stats #{}: {}", countDownLatch.getCount(), stats); - if (stats != null) { - gotStats = true; - } - countDownLatch.countDown(); - } - - public Boolean gotStats() { - return gotStats; - } - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/StopContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/StopContainerCmdExecTest.java deleted file mode 100644 index 1dca12890..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/StopContainerCmdExecTest.java +++ /dev/null @@ -1,85 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import static com.github.dockerjava.core.RemoteApiVersion.VERSION_1_22; -import static com.github.dockerjava.utils.TestUtils.getVersion; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -import java.lang.reflect.Method; - -import com.github.dockerjava.core.RemoteApiVersion; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class StopContainerCmdExecTest extends AbstractNettyDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(StopContainerCmdExecTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test(groups = "ignoreInCircleCi") - public void testStopContainer() throws DockerException { - final RemoteApiVersion apiVersion = getVersion(dockerClient); - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - dockerClient.startContainerCmd(container.getId()).exec(); - - LOG.info("Stopping container: {}", container.getId()); - dockerClient.stopContainerCmd(container.getId()).withTimeout(2).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); - - final Integer exitCode = inspectContainerResponse.getState().getExitCode(); - if (apiVersion.equals(VERSION_1_22)) { - assertThat(exitCode, is(0)); - } else { - assertThat(exitCode, not(0)); - } - } - - @Test(expectedExceptions = NotFoundException.class) - public void testStopNonExistingContainer() throws DockerException { - - dockerClient.stopContainerCmd("non-existing").withTimeout(2).exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/TagImageCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/TagImageCmdExecTest.java deleted file mode 100644 index 6c84775ee..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/TagImageCmdExecTest.java +++ /dev/null @@ -1,59 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.lang.reflect.Method; - -import org.apache.commons.lang.math.RandomUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class TagImageCmdExecTest extends AbstractNettyDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(TagImageCmdExecTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void tagImage() throws Exception { - String tag = "" + RandomUtils.nextInt(Integer.MAX_VALUE); - - dockerClient.tagImageCmd("busybox:latest", "docker-java/busybox", tag).exec(); - - dockerClient.removeImageCmd("docker-java/busybox:" + tag).exec(); - } - - @Test(expectedExceptions = NotFoundException.class) - public void tagNonExistingImage() throws Exception { - - String tag = "" + RandomUtils.nextInt(Integer.MAX_VALUE); - dockerClient.tagImageCmd("non-existing", "docker-java/busybox", tag).exec(); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/UnpauseCmdImplTest.java b/src/test/java/com/github/dockerjava/netty/exec/UnpauseCmdImplTest.java deleted file mode 100644 index db91bc9f9..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/UnpauseCmdImplTest.java +++ /dev/null @@ -1,118 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.exception.InternalServerErrorException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; -import com.github.dockerjava.utils.ContainerUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.isEmptyString; -import static org.hamcrest.Matchers.not; - -@Test(groups = {"integration", "ignoreInCircleCi"}) -public class UnpauseCmdImplTest extends AbstractNettyDockerClientTest { - - public static final Logger LOG = LoggerFactory.getLogger(UnpauseCmdImplTest.class); - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void unpausePausedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.pauseContainer(dockerClient, container); - - ContainerUtils.unpaseContainer(dockerClient, container); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void unpauseRunningContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - dockerClient.unpauseContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void unpauseStoppedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.stopContainer(dockerClient, container); - - dockerClient.unpauseContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = NotFoundException.class) - public void unpauseNonExistingContainer() { - - dockerClient.unpauseContainerCmd("non-existing").exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void unpauseCreatedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.unpauseContainerCmd(container.getId()).exec(); - } - - @Test(expectedExceptions = InternalServerErrorException.class) - public void unpauseUnpausedContainer() { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - ContainerUtils.startContainer(dockerClient, container); - - ContainerUtils.pauseContainer(dockerClient, container); - - dockerClient.unpauseContainerCmd(container.getId()).exec(); - dockerClient.unpauseContainerCmd(container.getId()).exec(); - } -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/UpdateContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/UpdateContainerCmdExecTest.java deleted file mode 100644 index 1134afd6d..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/UpdateContainerCmdExecTest.java +++ /dev/null @@ -1,107 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.HostConfig; -import com.github.dockerjava.api.model.UpdateContainerResponse; -import com.github.dockerjava.core.RemoteApiVersion; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.testng.ITestResult; -import org.testng.SkipException; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.io.IOException; -import java.lang.reflect.Method; - -import static com.github.dockerjava.utils.TestUtils.getVersion; -import static org.hamcrest.CoreMatchers.is; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.collection.IsCollectionWithSize.hasSize; - -/** - * @author Kanstantsin Shautsou - */ -@Test(groups = "integration") -public class UpdateContainerCmdExecTest extends AbstractNettyDockerClientTest { - public static final Logger LOG = LoggerFactory.getLogger(UpdateContainerCmdExecTest.class); - private static final String BUSYBOX_IMAGE = "busybox"; - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void updateContainer() throws DockerException, IOException { - final RemoteApiVersion apiVersion = getVersion(dockerClient); - - if (!apiVersion.isGreaterOrEqual(RemoteApiVersion.VERSION_1_22)) { - throw new SkipException("API version should be >= 1.22"); - } - - CreateContainerResponse response = dockerClient.createContainerCmd(BUSYBOX_IMAGE) - .withCmd("sleep", "9999") - .exec(); - String containerId = response.getId(); - dockerClient.startContainerCmd(containerId).exec(); - - InspectContainerResponse inspectBefore = dockerClient.inspectContainerCmd(containerId).exec(); - final HostConfig beforeHostConfig = inspectBefore.getHostConfig(); - - final UpdateContainerResponse updateResponse = dockerClient.updateContainerCmd(containerId) - .withBlkioWeight(300) - .withCpuShares(512) - .withCpuPeriod(100000) - .withCpuQuota(50000) -// .withCpusetCpus("0") // depends on env - .withCpusetMems("0") - .withMemory(314572800L) - .withMemorySwap(514288000L) - .withMemoryReservation(209715200L) -// .withKernelMemory(52428800) Can not update kernel memory to a running container, please stop it first. - .exec(); - - // found only on docker toolbox (1.10.1) -// assertThat(updateResponse.getWarnings(), hasSize(1)); -// assertThat(updateResponse.getWarnings().get(0), -// is("Your kernel does not support Block I/O weight. Weight discarded.")); - - InspectContainerResponse inspectAfter = dockerClient.inspectContainerCmd(containerId).exec(); - final HostConfig afterHostConfig = inspectAfter.getHostConfig(); - - assertThat(afterHostConfig.getMemory(), - is(314572800L)); - -// assertThat(afterHostConfig.getBlkioWeight(), is(300)); - assertThat(afterHostConfig.getCpuShares(), is(512)); - assertThat(afterHostConfig.getCpuPeriod(), is(100000)); - assertThat(afterHostConfig.getCpuQuota(), is(50000)); - assertThat(afterHostConfig.getCpusetMems(), is("0")); - - assertThat(afterHostConfig.getMemoryReservation(), is(209715200L)); - assertThat(afterHostConfig.getMemorySwap(), is(514288000L)); - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/VersionCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/VersionCmdExecTest.java deleted file mode 100644 index 7709fa56d..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/VersionCmdExecTest.java +++ /dev/null @@ -1,52 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import java.lang.reflect.Method; - -import org.apache.commons.lang.StringUtils; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.model.Version; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; - -@Test(groups = "integration") -public class VersionCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void version() throws DockerException { - Version version = dockerClient.versionCmd().exec(); - LOG.info(version.toString()); - - assertTrue(version.getGoVersion().length() > 0); - assertTrue(version.getVersion().length() > 0); - - assertEquals(StringUtils.split(version.getVersion(), ".").length, 3); - - } - -} diff --git a/src/test/java/com/github/dockerjava/netty/exec/WaitContainerCmdExecTest.java b/src/test/java/com/github/dockerjava/netty/exec/WaitContainerCmdExecTest.java deleted file mode 100644 index 42898974c..000000000 --- a/src/test/java/com/github/dockerjava/netty/exec/WaitContainerCmdExecTest.java +++ /dev/null @@ -1,128 +0,0 @@ -package com.github.dockerjava.netty.exec; - -import com.github.dockerjava.api.command.CreateContainerResponse; -import com.github.dockerjava.api.command.InspectContainerResponse; -import com.github.dockerjava.api.exception.DockerClientException; -import com.github.dockerjava.api.exception.DockerException; -import com.github.dockerjava.api.exception.NotFoundException; -import com.github.dockerjava.api.model.WaitResponse; -import com.github.dockerjava.core.command.WaitContainerResultCallback; -import com.github.dockerjava.netty.AbstractNettyDockerClientTest; -import org.testng.ITestResult; -import org.testng.annotations.AfterMethod; -import org.testng.annotations.AfterTest; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; - -import java.lang.reflect.Method; -import java.util.concurrent.TimeUnit; - -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.*; - -@Test(groups = "integration") -public class WaitContainerCmdExecTest extends AbstractNettyDockerClientTest { - - @BeforeTest - public void beforeTest() throws Exception { - super.beforeTest(); - } - - @AfterTest - public void afterTest() { - super.afterTest(); - } - - @BeforeMethod - public void beforeMethod(Method method) { - super.beforeMethod(method); - } - - @AfterMethod - public void afterMethod(ITestResult result) { - super.afterMethod(result); - } - - @Test - public void testWaitContainer() throws DockerException { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("true").exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - int exitCode = dockerClient.waitContainerCmd(container.getId()).exec(new WaitContainerResultCallback()) - .awaitStatusCode(); - LOG.info("Container exit code: {}", exitCode); - - assertThat(exitCode, equalTo(0)); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); - assertThat(inspectContainerResponse.getState().getExitCode(), is(equalTo(exitCode))); - } - - @Test(expectedExceptions = NotFoundException.class) - public void testWaitNonExistingContainer() throws DockerException { - - WaitContainerResultCallback callback = new WaitContainerResultCallback() { - public void onNext(WaitResponse waitResponse) { - fail("expected NotFoundException"); - } - }; - - dockerClient.waitContainerCmd("non-existing").exec(callback).awaitStatusCode(); - } - - @Test - public void testWaitContainerAbort() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "9999").exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - WaitContainerResultCallback callback = dockerClient.waitContainerCmd(container.getId()).exec( - new WaitContainerResultCallback()); - - Thread.sleep(5000); - - System.err.println("closing callback"); - - callback.close(); - - dockerClient.killContainerCmd(container.getId()).exec(); - - InspectContainerResponse inspectContainerResponse = dockerClient.inspectContainerCmd(container.getId()).exec(); - LOG.info("Container Inspect: {}", inspectContainerResponse.toString()); - - assertThat(inspectContainerResponse.getState().getRunning(), is(equalTo(false))); - } - - @Test - public void testWaitContainerTimeout() throws Exception { - - CreateContainerResponse container = dockerClient.createContainerCmd("busybox").withCmd("sleep", "10").exec(); - - LOG.info("Created container: {}", container.toString()); - assertThat(container.getId(), not(isEmptyString())); - - dockerClient.startContainerCmd(container.getId()).exec(); - - WaitContainerResultCallback callback = dockerClient.waitContainerCmd(container.getId()).exec( - new WaitContainerResultCallback()); - try { - callback.awaitStatusCode(100, TimeUnit.MILLISECONDS); - fail("Should throw exception on timeout."); - } catch(DockerClientException e){ - LOG.info(e.getMessage()); - } - } -} diff --git a/src/test/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandlerTest.java b/src/test/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandlerTest.java deleted file mode 100644 index 6652f3eba..000000000 --- a/src/test/java/com/github/dockerjava/netty/handler/HttpResponseStreamHandlerTest.java +++ /dev/null @@ -1,53 +0,0 @@ -package com.github.dockerjava.netty.handler; - -import static org.testng.Assert.assertTrue; - -import java.io.InputStream; - -import io.netty.buffer.ByteBuf; -import io.netty.buffer.ByteBufInputStream; -import io.netty.buffer.Unpooled; -import io.netty.channel.ChannelHandlerContext; -import org.apache.commons.io.IOUtils; -import org.mockito.Mockito; -import org.testng.annotations.Test; - -import com.github.dockerjava.core.async.ResultCallbackTemplate; - -/** - * @author Alexander Koshevoy - */ -public class HttpResponseStreamHandlerTest { - @Test - public void testNoBytesSkipped() throws Exception { - ResultCallbackTest callback = new ResultCallbackTest(); - HttpResponseStreamHandler streamHandler = new HttpResponseStreamHandler(callback); - ChannelHandlerContext ctx = Mockito.mock(ChannelHandlerContext.class); - ByteBuf buffer = generateByteBuf(); - streamHandler.channelRead0(ctx, buffer); - streamHandler.channelReadComplete(ctx); - - assertTrue(IOUtils.contentEquals(callback.getInputStream(), new ByteBufInputStream(buffer))); - } - - private ByteBuf generateByteBuf() { - byte[] array = new byte[256]; - for (int i = 0; i < array.length; i++) { - array[i] = (byte) i; - } - return Unpooled.copiedBuffer(array); - } - - private static class ResultCallbackTest extends ResultCallbackTemplate { - private InputStream stream; - - @Override - public void onNext(InputStream stream) { - this.stream = stream; - } - - public InputStream getInputStream() { - return stream; - } - } -} diff --git a/src/test/java/com/github/dockerjava/test/serdes/JSONSamples.java b/src/test/java/com/github/dockerjava/test/serdes/JSONSamples.java deleted file mode 100644 index 1b5337aa6..000000000 --- a/src/test/java/com/github/dockerjava/test/serdes/JSONSamples.java +++ /dev/null @@ -1,66 +0,0 @@ -package com.github.dockerjava.test.serdes; - -import com.fasterxml.jackson.core.type.TypeReference; -import com.fasterxml.jackson.databind.JavaType; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.github.dockerjava.core.RemoteApiVersion; -import org.apache.commons.io.FileUtils; - -import java.io.File; -import java.io.IOException; - -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertNotSame; - -/** - * Samples helper - * - * @author Kanstantsin Shautsou - */ -public class JSONSamples { - - /** - * Access to samples storage. - * - * @param version docker version of json sample - * @param context path to file for interested dump - * @return Content of JSON sample - * @throws IOException - */ - public static String getSampleContent(RemoteApiVersion version, String context) throws IOException { - File resource = new File("src/test/resources/samples/" + version.getVersion() + "/" + context); - return FileUtils.readFileToString(resource, "UTF-8"); - } - - public static TClass testRoundTrip(RemoteApiVersion version, String context, - JavaType type) throws IOException { - ObjectMapper mapper = new ObjectMapper(); - final TClass tObject = mapper.readValue(getSampleContent(version, context), type); - return testRoundTrip(tObject, type); - } - - /** - * Same as {@link JSONTestHelper#testRoundTrip(java.lang.Object, java.lang.Class)} - * but via {@link TypeReference} - */ - public static TClass testRoundTrip(TClass item, JavaType type) - throws IOException, AssertionError { - ObjectMapper mapper = new ObjectMapper(); - - String serialized1 = mapper.writeValueAsString(item); - JsonNode json1 = mapper.readTree(serialized1); - - TClass deserialized1 = mapper.readValue(serialized1, type); - String serialized2 = mapper.writeValueAsString(deserialized1); - - JsonNode json2 = mapper.readTree(serialized2); - TClass deserialized2 = mapper.readValue(serialized2, type); - - assertEquals(json2, json1, "JSONs must be equal after the second roundtrip"); - assertEquals(deserialized2, deserialized2, "Objects must be equal after the second roundtrip"); - assertNotSame(deserialized2, deserialized1, "Objects must be not the same"); - - return deserialized2; - } -} diff --git a/src/test/java/com/github/dockerjava/utils/TestResources.java b/src/test/java/com/github/dockerjava/utils/TestResources.java deleted file mode 100644 index 35ece680f..000000000 --- a/src/test/java/com/github/dockerjava/utils/TestResources.java +++ /dev/null @@ -1,14 +0,0 @@ -package com.github.dockerjava.utils; - -import java.nio.file.Path; -import java.nio.file.Paths; - -public class TestResources { - - private TestResources() { - } - - public static Path getApiImagesLoadTestTarball() { - return Paths.get("src/test/resources/api/images/load/image.tar"); - } -} diff --git a/src/test/java/com/github/dockerjava/utils/TestUtils.java b/src/test/java/com/github/dockerjava/utils/TestUtils.java deleted file mode 100644 index b40273ecb..000000000 --- a/src/test/java/com/github/dockerjava/utils/TestUtils.java +++ /dev/null @@ -1,17 +0,0 @@ -package com.github.dockerjava.utils; - -import com.github.dockerjava.api.DockerClient; -import com.github.dockerjava.core.RemoteApiVersion; - -/** - * @author Kanstantsin Shautsou - */ -public class TestUtils { - private TestUtils() { - } - - public static RemoteApiVersion getVersion(DockerClient client) { - final String serverVersion = client.versionCmd().exec().getApiVersion(); - return RemoteApiVersion.parseConfig(serverVersion); - } -} diff --git a/src/test/resources/attachContainerTestDockerfile/Dockerfile b/src/test/resources/attachContainerTestDockerfile/Dockerfile deleted file mode 100644 index 90cca5e64..000000000 --- a/src/test/resources/attachContainerTestDockerfile/Dockerfile +++ /dev/null @@ -1,8 +0,0 @@ -FROM ubuntu:latest - -ADD ./echo.sh /tmp/ - -RUN cp /tmp/echo.sh /usr/local/bin/ && chmod +x /usr/local/bin/echo.sh - -CMD ["echo.sh"] - diff --git a/src/test/resources/attachContainerTestDockerfile/echo.sh b/src/test/resources/attachContainerTestDockerfile/echo.sh deleted file mode 100644 index 88b444bf0..000000000 --- a/src/test/resources/attachContainerTestDockerfile/echo.sh +++ /dev/null @@ -1,2 +0,0 @@ -#!/bin/sh -while sleep 2; do echo stdout && echo stderr >&2; done \ No newline at end of file diff --git a/src/test/resources/buildTests/ADD/file/Dockerfile b/src/test/resources/buildTests/ADD/file/Dockerfile deleted file mode 100644 index 8a4a67760..000000000 --- a/src/test/resources/buildTests/ADD/file/Dockerfile +++ /dev/null @@ -1,9 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ADD ./testrun.sh /tmp/ - -RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh - -CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/ADD/fileInSubfolder/Dockerfile b/src/test/resources/buildTests/ADD/fileInSubfolder/Dockerfile deleted file mode 100644 index eac6b0e99..000000000 --- a/src/test/resources/buildTests/ADD/fileInSubfolder/Dockerfile +++ /dev/null @@ -1,9 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ADD ./files/testrun.sh /tmp/ - -RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh - -CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/ADD/files/Dockerfile b/src/test/resources/buildTests/ADD/files/Dockerfile deleted file mode 100644 index 52c5bfc6c..000000000 --- a/src/test/resources/buildTests/ADD/files/Dockerfile +++ /dev/null @@ -1,5 +0,0 @@ -FROM ubuntu:latest - -# Copy multiple source files into the container - -ADD src1 src2 /tmp/ diff --git a/src/test/resources/buildTests/ADD/filesViaWildcard/Dockerfile b/src/test/resources/buildTests/ADD/filesViaWildcard/Dockerfile deleted file mode 100644 index a9a9ad2b9..000000000 --- a/src/test/resources/buildTests/ADD/filesViaWildcard/Dockerfile +++ /dev/null @@ -1,9 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ADD ./folder*/* /tmp/ - -RUN cp /tmp/*.sh /usr/local/bin/ && chmod +x /usr/local/bin/*.sh - -CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/ADD/folder/Dockerfile b/src/test/resources/buildTests/ADD/folder/Dockerfile deleted file mode 100644 index 183e831c3..000000000 --- a/src/test/resources/buildTests/ADD/folder/Dockerfile +++ /dev/null @@ -1,11 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ADD . /src/ - -run ls -la /src - -run cp /src/folderA/testAddFolder.sh /usr/local/bin/ && chmod +x /usr/local/bin/testAddFolder.sh - -CMD ["testAddFolder.sh"] diff --git a/src/test/resources/buildTests/ADD/url/Dockerfile b/src/test/resources/buildTests/ADD/url/Dockerfile deleted file mode 100644 index 2b10c34e5..000000000 --- a/src/test/resources/buildTests/ADD/url/Dockerfile +++ /dev/null @@ -1,10 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ADD http://www.example.com/index.html /tmp/some.html -ADD ./testrun.sh /tmp/ - -RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh - -CMD ["testrun.sh"] \ No newline at end of file diff --git a/src/test/resources/buildTests/AUTHOR/Dockerfile b/src/test/resources/buildTests/AUTHOR/Dockerfile deleted file mode 100644 index 32ffcedc7..000000000 --- a/src/test/resources/buildTests/AUTHOR/Dockerfile +++ /dev/null @@ -1,12 +0,0 @@ -# Nginx -# -# VERSION 0.0.1 - -FROM ubuntu:latest -MAINTAINER Guillaume J. Charmes "guillaume@dotcloud.com" - -# make sure the package repository is up to date -RUN echo "deb http://archive.ubuntu.com/ubuntu precise main universe" > /etc/apt/sources.list -RUN apt-get update - -RUN apt-get install -y nginx \ No newline at end of file diff --git a/src/test/resources/buildTests/ENV/Dockerfile b/src/test/resources/buildTests/ENV/Dockerfile deleted file mode 100644 index 134f7a60f..000000000 --- a/src/test/resources/buildTests/ENV/Dockerfile +++ /dev/null @@ -1,11 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ENV variable abc123 -ADD ./testrun.sh /tmp/ -ADD ./subst-file-$variable.txt /tmp/ -COPY ./subst-file-2-${variable}.txt /tmp/ -RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh - -CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/FROM/privateRegistry/Dockerfile b/src/test/resources/buildTests/FROM/privateRegistry/Dockerfile deleted file mode 100644 index b3d914c6c..000000000 --- a/src/test/resources/buildTests/FROM/privateRegistry/Dockerfile +++ /dev/null @@ -1 +0,0 @@ -FROM localhost:5000/testuser/busybox:latest diff --git a/src/test/resources/buildTests/ONBUILD/parent/Dockerfile b/src/test/resources/buildTests/ONBUILD/parent/Dockerfile deleted file mode 100644 index e482cd833..000000000 --- a/src/test/resources/buildTests/ONBUILD/parent/Dockerfile +++ /dev/null @@ -1,8 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ONBUILD ADD ./testrun.sh /tmp/ -ONBUILD RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh - -CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/buildArgs/Dockerfile b/src/test/resources/buildTests/buildArgs/Dockerfile deleted file mode 100644 index c8dd0ae36..000000000 --- a/src/test/resources/buildTests/buildArgs/Dockerfile +++ /dev/null @@ -1,5 +0,0 @@ -FROM ubuntu:latest - -ARG testArg - -LABEL "test"=$testArg diff --git a/src/test/resources/buildTests/dockerfileNotInBaseDirectory/dockerfileFolder/Dockerfile b/src/test/resources/buildTests/dockerfileNotInBaseDirectory/dockerfileFolder/Dockerfile deleted file mode 100644 index 9bf10088b..000000000 --- a/src/test/resources/buildTests/dockerfileNotInBaseDirectory/dockerfileFolder/Dockerfile +++ /dev/null @@ -1,7 +0,0 @@ -FROM ubuntu:latest - -ADD testrunFolder/testrun.sh /tmp/ - -RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh - -CMD ["testrun.sh"] \ No newline at end of file diff --git a/src/test/resources/buildTests/dockerignore/DockerfileIgnored/Dockerfile b/src/test/resources/buildTests/dockerignore/DockerfileIgnored/Dockerfile deleted file mode 100644 index 8a4a67760..000000000 --- a/src/test/resources/buildTests/dockerignore/DockerfileIgnored/Dockerfile +++ /dev/null @@ -1,9 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ADD ./testrun.sh /tmp/ - -RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh - -CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/dockerignore/DockerfileNotIgnored/Dockerfile b/src/test/resources/buildTests/dockerignore/DockerfileNotIgnored/Dockerfile deleted file mode 100644 index 6a8106b86..000000000 --- a/src/test/resources/buildTests/dockerignore/DockerfileNotIgnored/Dockerfile +++ /dev/null @@ -1,3 +0,0 @@ -FROM ubuntu:latest - -CMD ["echo", "Success"] diff --git a/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/Dockerfile b/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/Dockerfile deleted file mode 100644 index 65fa210c7..000000000 --- a/src/test/resources/buildTests/dockerignore/EffectiveDockerignorePatterns/Dockerfile +++ /dev/null @@ -1,3 +0,0 @@ -FROM ubuntu:latest - -CMD ["echo", "Success"] diff --git a/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/Dockerfile b/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/Dockerfile deleted file mode 100644 index 65fa210c7..000000000 --- a/src/test/resources/buildTests/dockerignore/IneffectiveDockerignorePattern/Dockerfile +++ /dev/null @@ -1,3 +0,0 @@ -FROM ubuntu:latest - -CMD ["echo", "Success"] diff --git a/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/Dockerfile b/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/Dockerfile deleted file mode 100644 index 8a4a67760..000000000 --- a/src/test/resources/buildTests/dockerignore/InvalidDockerignorePattern/Dockerfile +++ /dev/null @@ -1,9 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ADD ./testrun.sh /tmp/ - -RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh - -CMD ["testrun.sh"] diff --git a/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/Dockerfile b/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/Dockerfile deleted file mode 100644 index 65fa210c7..000000000 --- a/src/test/resources/buildTests/dockerignore/NestedDirsDockerignore/Dockerfile +++ /dev/null @@ -1,3 +0,0 @@ -FROM ubuntu:latest - -CMD ["echo", "Success"] diff --git a/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/Dockerfile b/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/Dockerfile deleted file mode 100644 index 97b93b899..000000000 --- a/src/test/resources/buildTests/dockerignore/ValidDockerignorePattern/Dockerfile +++ /dev/null @@ -1,10 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ADD ./testrun.sh /tmp/ -ADD ./a /tmp/a - -RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh - -CMD ["testrun.sh"] diff --git a/src/test/resources/logback.xml b/src/test/resources/logback.xml deleted file mode 100644 index b4309b868..000000000 --- a/src/test/resources/logback.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - - - %d{HH:mm:ss.SSS} %-5level %logger{36} - %msg%n - - - - - - - - - - - - - - \ No newline at end of file diff --git a/src/test/resources/someHomeDir/.docker/config.json b/src/test/resources/someHomeDir/.docker/config.json deleted file mode 100644 index 630394039..000000000 --- a/src/test/resources/someHomeDir/.docker/config.json +++ /dev/null @@ -1,9 +0,0 @@ -{ - "auths":{ - "https://index.docker.io/v1/":{ - "auth":"XXXX=", - "email":"foo.bar@test.com" - } - - } -} \ No newline at end of file diff --git a/src/test/resources/testReadFile/Dockerfile b/src/test/resources/testReadFile/Dockerfile deleted file mode 100644 index 0f73b8ea8..000000000 --- a/src/test/resources/testReadFile/Dockerfile +++ /dev/null @@ -1,9 +0,0 @@ -FROM ubuntu:latest - -# Copy testrun.sh files into the container - -ADD ./testrun.sh /tmp/ -ADD ./oldFile.txt / -RUN cp /tmp/testrun.sh /usr/local/bin/ && chmod +x /usr/local/bin/testrun.sh - -CMD ["testrun.sh"]