Compare commits
21 Commits
main
...
b8bb637fb2
| Author | SHA1 | Date | |
|---|---|---|---|
| b8bb637fb2 | |||
| 5b1e185867 | |||
| e6a76a28ca | |||
| 87349b1fc8 | |||
| e8b7bda08f | |||
| 5e5c5eeda2 | |||
| 45da665ce0 | |||
| 1ec01bdf99 | |||
| 83c611bb20 | |||
| 3ec54a128a | |||
| e39a4b8cea | |||
| 30e8216aac | |||
| aac8c97a53 | |||
| 5c6b04eda8 | |||
| 71bbecf3a2 | |||
| e60d189fd7 | |||
| ec857c7af8 | |||
| 824e8e340d | |||
| 110724cd60 | |||
| 101b7e5e22 | |||
| f0f9c0800d |
40
.github/workflows/main.yml
vendored
40
.github/workflows/main.yml
vendored
@@ -1,40 +0,0 @@
|
||||
name: Release
|
||||
|
||||
on:
|
||||
release:
|
||||
types:
|
||||
- created
|
||||
|
||||
jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
|
||||
- name: Set up QEMU
|
||||
uses: docker/setup-qemu-action@v1
|
||||
|
||||
- name: Set up Docker Buildx
|
||||
uses: docker/setup-buildx-action@v1
|
||||
|
||||
- name: Login to DockerHub
|
||||
uses: docker/login-action@v1
|
||||
with:
|
||||
username: ${{ secrets.DOCKER_HUB_USERNAME }}
|
||||
password: ${{ secrets.DOCKER_HUB_PASSWORD }}
|
||||
|
||||
- name: Build and push
|
||||
id: docker_build
|
||||
uses: docker/build-push-action@v2
|
||||
with:
|
||||
context: ./
|
||||
file: ./Dockerfile
|
||||
push: true
|
||||
tags: |
|
||||
sulthonzh/docker-remote-deployment-action:latest
|
||||
sulthonzh/docker-remote-deployment-action:${{ github.event.release.tag_name }}
|
||||
|
||||
|
||||
|
||||
14
Dockerfile
14
Dockerfile
@@ -1,12 +1,16 @@
|
||||
FROM docker/compose:1.29.2
|
||||
FROM docker
|
||||
COPY --from=docker/compose-bin:edge /docker-compose /usr/libexec/docker/cli-plugins/docker-compose
|
||||
RUN docker compose version
|
||||
|
||||
LABEL 'name'='Docker Deployment Action'
|
||||
LABEL 'com.github.actions.name'='Docker Deployment'
|
||||
LABEL 'com.github.actions.description'='supports docker-compose and Docker Swarm deployments'
|
||||
LABEL 'name'='Docker Compose Remote Deployment Action'
|
||||
LABEL 'com.github.actions.name'='Docker Compose Remote Deployment'
|
||||
LABEL 'com.github.actions.description'='supports docker-compose remotely through ssh.'
|
||||
LABEL 'com.github.actions.icon'='send'
|
||||
LABEL 'com.github.actions.color'='green'
|
||||
|
||||
RUN apk --no-cache add openssh-client
|
||||
RUN sed -i 's/dl-cdn.alpinelinux.org/mirrors.ustc.edu.cn/g' /etc/apk/repositories
|
||||
|
||||
RUN apk --no-cache add openssh-client socat
|
||||
|
||||
COPY docker-entrypoint.sh /docker-entrypoint.sh
|
||||
|
||||
|
||||
20
README.md
20
README.md
@@ -25,25 +25,23 @@ Below is a brief example on how the action can be used:
|
||||
|
||||
## Input Configurations
|
||||
### `remote_docker_host`
|
||||
Remote Docker host ie (user@host).
|
||||
Remote Docker host.
|
||||
### `remote_docker_user`
|
||||
Remote Docker user.
|
||||
### `remote_docker_port`
|
||||
Remote Docker ssh port ie (22).
|
||||
### `ssh_public_key`
|
||||
Remote Docker SSH public key eg (~/.ssh/rsa_id.pub).
|
||||
### `ssh_private_key`
|
||||
SSH private key used to connect to the docker host eg (~/.ssh/rsa_id).
|
||||
### `ssh_proxy_cmd`
|
||||
SSH ProxyCommand used to connect to the docker host.
|
||||
### `args`
|
||||
Deployment command args.
|
||||
### `deployment_mode`
|
||||
Deployment mode either docker-swarm or docker-compose. Default is docker-compose.
|
||||
### `copy_stack_file`
|
||||
Copy stack file to remote server and deploy from the server. Default is false.
|
||||
### `deploy_path`
|
||||
The path where the stack files will be copied to. Default ~/docker-deployment.
|
||||
The absolute path on remote docker host to deploy, default rely on repo working directory.
|
||||
### `stack_file_name`
|
||||
Docker stack file used. Default is docker-compose.yml.
|
||||
### `keep_files`
|
||||
Number of the files to be kept on the server. Default is 3.
|
||||
Compose configuration files.
|
||||
### `env_file_name`
|
||||
Specify an alternate environment file.
|
||||
### `docker_prune`
|
||||
A boolean input to trigger docker prune command. Default is false.
|
||||
### `pre_deployment_command_args`
|
||||
|
||||
26
action.yml
26
action.yml
@@ -2,35 +2,32 @@ name: Docker Remote (SSH) Deployment
|
||||
description: A GitHub Action that supports docker-compose and Docker Swarm deployments
|
||||
inputs:
|
||||
remote_docker_host:
|
||||
description: Remote Docker host ie (user@host).
|
||||
description: Remote Docker host.
|
||||
required: true
|
||||
remote_docker_user:
|
||||
description: Remote Docker user.
|
||||
required: true
|
||||
remote_docker_port:
|
||||
description: Remote Docker ssh port ie (22).
|
||||
required: false
|
||||
default: '22'
|
||||
ssh_public_key:
|
||||
description: Remote Docker SSH public key eg (~/.ssh/rsa_id.pub).
|
||||
required: true
|
||||
ssh_private_key:
|
||||
description: SSH private key used to connect to the docker host eg (~/.ssh/rsa_id).
|
||||
required: true
|
||||
ssh_proxy_cmd:
|
||||
description: SSH ProxyCommand used to connect to the docker host.
|
||||
required: false
|
||||
args:
|
||||
description: Deployment command args.
|
||||
required: true
|
||||
deployment_mode:
|
||||
description: Deployment mode either docker-swarm or docker-compose. Default is docker-compose.
|
||||
required: false
|
||||
copy_stack_file:
|
||||
description: Copy stack file to remote server and deploy from the server. Default is false.
|
||||
required: false
|
||||
deploy_path:
|
||||
description: The path where the stack files will be copied to. Default ~/docker-deployment.
|
||||
description: The absolute path on remote docker host to deploy, default rely on repo working directory.
|
||||
required: false
|
||||
stack_file_name:
|
||||
description: Docker stack file used. Default is docker-compose.yml.
|
||||
description: Compose configuration files.
|
||||
required: false
|
||||
keep_files:
|
||||
description: Number of the files to be kept on the server. Default is 3.
|
||||
env_file_name:
|
||||
description: Specify an alternate environment file.
|
||||
required: false
|
||||
docker_prune:
|
||||
description: A boolean input to trigger docker prune command. Default is false.
|
||||
@@ -40,7 +37,6 @@ inputs:
|
||||
required: false
|
||||
pull_images_first:
|
||||
description: Pull docker images before deploying. Default is false.
|
||||
required: false
|
||||
docker_registry_username:
|
||||
description: The docker registry username.
|
||||
required: false
|
||||
|
||||
@@ -1,85 +1,43 @@
|
||||
#!/bin/sh
|
||||
set -eu
|
||||
|
||||
execute_ssh(){
|
||||
echo "Execute Over SSH: $@"
|
||||
ssh -q -t -i "$HOME/.ssh/id_rsa" \
|
||||
-o UserKnownHostsFile=/dev/null \
|
||||
-p $INPUT_REMOTE_DOCKER_PORT \
|
||||
-o StrictHostKeyChecking=no "$INPUT_REMOTE_DOCKER_HOST" "$@"
|
||||
}
|
||||
|
||||
if [ -z "${INPUT_REMOTE_DOCKER_PORT+x}" ]; then
|
||||
if [ -z "${INPUT_REMOTE_DOCKER_PORT+x}" ] || [ -z "$INPUT_REMOTE_DOCKER_PORT" ]; then
|
||||
INPUT_REMOTE_DOCKER_PORT=22
|
||||
fi
|
||||
|
||||
if [ -z "${INPUT_REMOTE_DOCKER_HOST+x}" ]; then
|
||||
if [ -z "${INPUT_REMOTE_DOCKER_HOST+x}" ] || [ -z "$INPUT_REMOTE_DOCKER_HOST" ]; then
|
||||
echo "Input remote_docker_host is required!"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [ -z "${INPUT_SSH_PUBLIC_KEY+x}" ]; then
|
||||
echo "Input ssh_public_key is required!"
|
||||
if [ -z "${INPUT_REMOTE_DOCKER_USER+x}" ] || [ -z "$INPUT_REMOTE_DOCKER_USER" ]; then
|
||||
echo "Input remote_docker_user is required!"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [ -z "${INPUT_SSH_PRIVATE_KEY+x}" ]; then
|
||||
if [ -z "${INPUT_SSH_PRIVATE_KEY+x}" ] || [ -z "$INPUT_SSH_PRIVATE_KEY" ]; then
|
||||
echo "Input ssh_private_key is required!"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [ -z "${INPUT_ARGS+x}" ]; then
|
||||
if [ -z "${INPUT_ARGS+x}" ] || [ -z "$INPUT_ARGS" ]; then
|
||||
echo "Input input_args is required!"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [ -z "${INPUT_DEPLOY_PATH+x}" ]; then
|
||||
INPUT_DEPLOY_PATH=~/docker-deployment
|
||||
if ! [ -z "${INPUT_DEPLOY_PATH+x}" ] && ! [ -z "$INPUT_DEPLOY_PATH" ]; then
|
||||
echo "Change working directory"
|
||||
echo "Current: `pwd`"
|
||||
echo "Target: $INPUT_DEPLOY_PATH"
|
||||
mkdir -p $INPUT_DEPLOY_PATH
|
||||
cp -rfp ./. $INPUT_DEPLOY_PATH
|
||||
cd $INPUT_DEPLOY_PATH
|
||||
fi
|
||||
|
||||
if [ -z "${INPUT_STACK_FILE_NAME+x}" ]; then
|
||||
INPUT_STACK_FILE_NAME=docker-compose.yaml
|
||||
fi
|
||||
|
||||
if [ -z "${INPUT_KEEP_FILES+x}" ]; then
|
||||
INPUT_KEEP_FILES=4
|
||||
else
|
||||
INPUT_KEEP_FILES=$((INPUT_KEEP_FILES+1))
|
||||
fi
|
||||
|
||||
if [ -z "${INPUT_DOCKER_REGISTRY_URI+x}" ]; then
|
||||
if [ -z "${INPUT_DOCKER_REGISTRY_URI+x}" ] || [ -z "$INPUT_DOCKER_REGISTRY_URI" ]; then
|
||||
INPUT_DOCKER_REGISTRY_URI=https://registry.hub.docker.com
|
||||
fi
|
||||
|
||||
if [ -z "${INPUT_COPY_STACK_FILE+x}" ]; then
|
||||
INPUT_COPY_STACK_FILE=false
|
||||
fi
|
||||
|
||||
STACK_FILE=${INPUT_STACK_FILE_NAME}
|
||||
DEPLOYMENT_COMMAND_OPTIONS=""
|
||||
|
||||
|
||||
if [ "$INPUT_COPY_STACK_FILE" == "true" ]; then
|
||||
STACK_FILE="$INPUT_DEPLOY_PATH/$STACK_FILE"
|
||||
else
|
||||
DEPLOYMENT_COMMAND_OPTIONS=" --log-level debug --host ssh://$INPUT_REMOTE_DOCKER_HOST:$INPUT_REMOTE_DOCKER_PORT"
|
||||
fi
|
||||
|
||||
case $INPUT_DEPLOYMENT_MODE in
|
||||
|
||||
docker-swarm)
|
||||
DEPLOYMENT_COMMAND="docker $DEPLOYMENT_COMMAND_OPTIONS stack deploy --compose-file $STACK_FILE"
|
||||
;;
|
||||
|
||||
*)
|
||||
INPUT_DEPLOYMENT_MODE="docker-compose"
|
||||
DEPLOYMENT_COMMAND="docker-compose -f $STACK_FILE $DEPLOYMENT_COMMAND_OPTIONS"
|
||||
;;
|
||||
esac
|
||||
|
||||
|
||||
SSH_HOST=${INPUT_REMOTE_DOCKER_HOST#*@}
|
||||
|
||||
echo "Registering SSH keys..."
|
||||
|
||||
# register the private key with the agent.
|
||||
@@ -87,58 +45,61 @@ mkdir -p ~/.ssh
|
||||
ls ~/.ssh
|
||||
printf '%s\n' "$INPUT_SSH_PRIVATE_KEY" > ~/.ssh/id_rsa
|
||||
chmod 600 ~/.ssh/id_rsa
|
||||
printf '%s\n' "$INPUT_SSH_PUBLIC_KEY" > ~/.ssh/id_rsa.pub
|
||||
chmod 600 ~/.ssh/id_rsa.pub
|
||||
#chmod 600 "~/.ssh"
|
||||
eval $(ssh-agent)
|
||||
ssh-add ~/.ssh/id_rsa
|
||||
|
||||
echo "Add known hosts"
|
||||
ssh-keyscan -p $INPUT_REMOTE_DOCKER_PORT "$SSH_HOST" >> ~/.ssh/known_hosts
|
||||
ssh-keyscan -p $INPUT_REMOTE_DOCKER_PORT "$SSH_HOST" >> /etc/ssh/ssh_known_hosts
|
||||
echo "Add REMOTE_DOCKER_HOST alias to ~/.ssh/config"
|
||||
touch ~/.ssh/config
|
||||
echo >> ~/.ssh/config
|
||||
echo "Host REMOTE_DOCKER_HOST" >> ~/.ssh/config
|
||||
echo " HostName $INPUT_REMOTE_DOCKER_HOST" >> ~/.ssh/config
|
||||
echo " User $INPUT_REMOTE_DOCKER_USER" >> ~/.ssh/config
|
||||
echo " Port $INPUT_REMOTE_DOCKER_PORT" >> ~/.ssh/config
|
||||
echo " IdentityFile ~/.ssh/id_rsa" >> ~/.ssh/config
|
||||
echo " StrictHostKeyChecking no" >> ~/.ssh/config
|
||||
echo " UserKnownHostsFile /dev/null" >> ~/.ssh/config
|
||||
echo " ConnectTimeout 300" >> ~/.ssh/config
|
||||
if ! [ -z "${INPUT_SSH_PROXY_CMD+x}" ] && ! [ -z "$INPUT_SSH_PROXY_CMD" ]; then
|
||||
echo "Add ProxyCommand: $INPUT_SSH_PROXY_CMD"
|
||||
echo " ProxyCommand $INPUT_SSH_PROXY_CMD" >> ~/.ssh/config
|
||||
fi
|
||||
echo >> ~/.ssh/config
|
||||
chmod 600 ~/.ssh/config
|
||||
|
||||
docker -v
|
||||
docker compose version
|
||||
|
||||
set context
|
||||
echo "Create docker context"
|
||||
docker context create remote --docker "host=ssh://$INPUT_REMOTE_DOCKER_HOST:$INPUT_REMOTE_DOCKER_PORT"
|
||||
docker context create remote --docker "host=ssh://REMOTE_DOCKER_HOST"
|
||||
docker context use remote
|
||||
|
||||
if ! [ -z "${INPUT_DOCKER_REGISTRY_USERNAME+x}" ] && ! [ -z "${INPUT_DOCKER_REGISTRY_PASSWORD+x}" ]; then
|
||||
echo "Connecting to $INPUT_REMOTE_DOCKER_HOST... Command: docker login"
|
||||
if ! [ -z "${INPUT_DOCKER_REGISTRY_USERNAME+x}" ] && ! [ -z "$INPUT_DOCKER_REGISTRY_USERNAME" ] &&
|
||||
! [ -z "${INPUT_DOCKER_REGISTRY_PASSWORD+x}" ] && ! [ -z "$INPUT_DOCKER_REGISTRY_PASSWORD" ]; then
|
||||
echo "Connecting to $INPUT_DOCKER_REGISTRY_URI... Command: docker login"
|
||||
echo "$INPUT_DOCKER_REGISTRY_PASSWORD" | docker login -u "$INPUT_DOCKER_REGISTRY_USERNAME" --password-stdin "$INPUT_DOCKER_REGISTRY_URI"
|
||||
fi
|
||||
|
||||
if ! [ -z "${INPUT_DOCKER_PRUNE+x}" ] && [ $INPUT_DOCKER_PRUNE = 'true' ] ; then
|
||||
yes | docker --log-level debug --host "ssh://$INPUT_REMOTE_DOCKER_HOST:$INPUT_REMOTE_DOCKER_PORT" system prune -a 2>&1
|
||||
yes | docker --log-level debug system prune -a 2>&1
|
||||
fi
|
||||
|
||||
if ! [ -z "${INPUT_COPY_STACK_FILE+x}" ] && [ $INPUT_COPY_STACK_FILE = 'true' ] ; then
|
||||
execute_ssh "mkdir -p $INPUT_DEPLOY_PATH/stacks || true"
|
||||
FILE_NAME="docker-stack-$(date +%Y%m%d%s).yaml"
|
||||
DEPLOYMENT_COMMAND="docker --log-level=debug compose"
|
||||
|
||||
scp -i "$HOME/.ssh/id_rsa" \
|
||||
-o UserKnownHostsFile=/dev/null \
|
||||
-o StrictHostKeyChecking=no \
|
||||
-P $INPUT_REMOTE_DOCKER_PORT \
|
||||
$INPUT_STACK_FILE_NAME "$INPUT_REMOTE_DOCKER_HOST:$INPUT_DEPLOY_PATH/stacks/$FILE_NAME"
|
||||
|
||||
DIRS=$(dirname "$INPUT_DEPLOY_PATH/$INPUT_STACK_FILE_NAME")
|
||||
mkdir -p "$DIRS" || exit 0
|
||||
|
||||
execute_ssh "ln -nfs $INPUT_DEPLOY_PATH/stacks/$FILE_NAME $INPUT_DEPLOY_PATH/$INPUT_STACK_FILE_NAME"
|
||||
execute_ssh "ls -t $INPUT_DEPLOY_PATH/stacks/docker-stack-* 2>/dev/null | tail -n +$INPUT_KEEP_FILES | xargs rm -- 2>/dev/null || true"
|
||||
|
||||
if ! [ -z "${INPUT_PULL_IMAGES_FIRST+x}" ] && [ $INPUT_PULL_IMAGES_FIRST = 'true' ] && [ $INPUT_DEPLOYMENT_MODE = 'docker-compose' ] ; then
|
||||
execute_ssh ${DEPLOYMENT_COMMAND} "pull"
|
||||
fi
|
||||
|
||||
if ! [ -z "${INPUT_PRE_DEPLOYMENT_COMMAND_ARGS+x}" ] && [ $INPUT_DEPLOYMENT_MODE = 'docker-compose' ] ; then
|
||||
execute_ssh "${DEPLOYMENT_COMMAND} $INPUT_PRE_DEPLOYMENT_COMMAND_ARGS" 2>&1
|
||||
fi
|
||||
|
||||
execute_ssh ${DEPLOYMENT_COMMAND} "$INPUT_ARGS" 2>&1
|
||||
else
|
||||
echo "Connecting to $INPUT_REMOTE_DOCKER_HOST... Command: ${DEPLOYMENT_COMMAND} ${INPUT_ARGS}"
|
||||
${DEPLOYMENT_COMMAND} ${INPUT_ARGS} 2>&1
|
||||
if ! [ -z "${INPUT_STACK_FILE_NAME+x}" ] && ! [ -z "$INPUT_STACK_FILE_NAME" ]; then
|
||||
DEPLOYMENT_COMMAND="$DEPLOYMENT_COMMAND -f ${INPUT_STACK_FILE_NAME//,/ -f }"
|
||||
fi
|
||||
|
||||
if ! [ -z "${INPUT_ENV_FILE_NAME+x}" ] && ! [ -z "$INPUT_ENV_FILE_NAME" ]; then
|
||||
DEPLOYMENT_COMMAND="$DEPLOYMENT_COMMAND --env-file ${INPUT_ENV_FILE_NAME//,/ --env-file }"
|
||||
fi
|
||||
|
||||
if ! [ -z "${INPUT_PULL_IMAGES_FIRST+x}" ] && [ $INPUT_PULL_IMAGES_FIRST = 'true' ] ; then
|
||||
echo "Connecting to $INPUT_REMOTE_DOCKER_HOST... Command: ${DEPLOYMENT_COMMAND} pull"
|
||||
${DEPLOYMENT_COMMAND} "pull"
|
||||
fi
|
||||
|
||||
# DEPLOYMENT_COMMAND_OPTIONS=""
|
||||
|
||||
echo "Connecting to $INPUT_REMOTE_DOCKER_HOST... Command: ${DEPLOYMENT_COMMAND} ${INPUT_ARGS}"
|
||||
${DEPLOYMENT_COMMAND} ${INPUT_ARGS} 2>&1
|
||||
@@ -1,5 +0,0 @@
|
||||
{
|
||||
"extends": [
|
||||
"config:base"
|
||||
]
|
||||
}
|
||||
Reference in New Issue
Block a user