forked from codefresh-io/remote-docker
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrdocker.sh
executable file
·197 lines (170 loc) · 6.59 KB
/
rdocker.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
#!/bin/bash
set -e
# use ssh port; allow to override
SSH_PORT=${SSH_PORT:-22}
# use Python as Docker image by default
PYTHON_DOCKER=${PYTHON_DOCKER:-true}
re='^[0-9]+$'
if [[ $# -eq 0 || $1 == "-h" || $1 == "-help" ]]; then
echo "Usage: rdocker [-h|-help] [user@]hostname [port] [cmd]"
echo ""
echo " -h -help print this message"
echo " user@hostname ssh remote login address"
echo " port local port used to forward the remote docker daemon, if not present a free random port will be used"
echo " cmd when provided, it is the only command run on the remote host (no bash session is created)"
echo " SSH_PORT set ssh port through environment variable (default: 22)"
exit
fi
# use SSH_KEY environment variable to create key file, if not exists
ssh_key_file="$HOME/.ssh/id_rdocker"
if [[ ! -f "$ssh_key_file" ]]; then
if [[ ! -z "${SSH_KEY}" ]]; then
echo "SSH key passed through SSH_KEY environment variable: lenght check ${#SSH_KEY}"
mkdir -p ~/.ssh
if [[ ! -z "${SPLIT_CHAR}" ]]; then
echo "${SSH_KEY}" | tr \'"${SPLIT_CHAR}"\' '\n' > "$ssh_key_file"
else
echo "${SSH_KEY}" > "$ssh_key_file"
fi
chmod 600 "$ssh_key_file"
fi
else
echo "Found $ssh_key_file file"
fi
#Extracting parameters
remote_host=${1}
if [[ $2 =~ $re ]]; then
local_port=${2}
command=${*:3} #third parameter and beyond
else
command=${*:2} #second parameter and beyond
fi
control_path="$HOME/.rdocker-master-$(date +%s%N)"
ssh "${remote_host}" -p ${SSH_PORT} -i "$ssh_key_file" -nNf -o "StrictHostKeyChecking no" -o ControlMaster=yes -o ControlPath="${control_path}" -o ControlPersist=yes
if [ ! -S "${control_path}" ]; then
exit
fi
find_port_code="import socket;s=socket.socket(socket.AF_INET, socket.SOCK_STREAM);s.bind(('', 0));print(s.getsockname()[1]);s.close()"
remote_port=$(ssh "${remote_host}" -p ${SSH_PORT} -i "$ssh_key_file" -o ControlPath="${control_path}" python -c \"$find_port_code\")
if [ -z "$remote_port" ]; then
echo "ERROR: Failed to find a free port. This usually happens when python is not installed on the remote host."
#clear the ssh control connection
ssh -O exit -o ControlPath="$control_path" "$remote_host" -p ${SSH_PORT} "" 2> /dev/null
rm -f "$control_path"
exit 1
fi
success_msg="Connection established"
forwarder="
import threading,socket,select,signal,sys,os
running = True
remote_port = $remote_port
def signal_handler(signal, frame):
global running
running = False
sys.exit(0)
signal.signal(signal.SIGINT, signal_handler)
signal.signal(signal.SIGTERM, signal_handler)
def main():
global running
addr = (\"localhost\", remote_port)
server = socket.socket()
server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
try:
server.bind(addr)
server.listen(50)
except socket.error as msg:
server.close()
print(\"Port \" + remote_port + \" is already in use.\")
sys.exit(0)
print(\"${success_msg}.\")
rlist = [server]
while running:
readable, writable, exceptional = select.select(rlist, [], [], 0.5)
if os.getppid() == 1:
running=False
break
if server in readable:
client, connection = server.accept()
threading.Thread(target=serve_client, args=(client,)).start()
server.shutdown(socket.SHUT_RDWR)
server.close()
def serve_client(tcp_socket):
global running
uds_socket = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
try:
uds_socket.connect(\"/var/run/docker.sock\")
except socket.error as msg:
uds_socket.close()
tcp_socket.close()
return
rlist = [tcp_socket, uds_socket]
while running:
readable, writable, exceptional = select.select(rlist, [], [], 0.5)
if tcp_socket in readable:
data = tcp_socket.recv(4096)
if not data or not running: break
else:
uds_socket.sendall(data)
if uds_socket in readable:
data = uds_socket.recv(4096)
if not data or not running: break
else:
tcp_socket.sendall(data)
tcp_socket.shutdown(socket.SHUT_RDWR)
tcp_socket.close()
uds_socket.shutdown(socket.SHUT_RDWR)
uds_socket.close()
if __name__ == \"__main__\":
main()
"
# find an unused local port
# create a temporary named pipe and attach it to file descriptor 3
PIPE=$(mktemp -u); mkfifo "$PIPE"
exec 3<>"$PIPE"; rm "$PIPE"
# find a free port or use the provided one
local_port=${local_port:-$(python -c "$find_port_code")}
remote_script_path="/tmp/rdocker-forwarder.py"
remote_python="python"
if [[ "$PYTHON_DOCKER" == true ]]; then
remote_python="docker run -i --rm --name remote_python --network=host -v /tmp:/tmp -v /var/run/docker.sock:/var/run/docker.sock frolvlad/alpine-python2 python"
fi
printf "%s" "$forwarder" | ssh -i "$ssh_key_file" "$remote_host" -p ${SSH_PORT} -o ControlPath="$control_path" -L "$local_port:localhost:$remote_port" "cat > ${remote_script_path}""; exec ${remote_python} -u ${remote_script_path}" 1>&3 &
CONNECTION_PID=$!
# wait for it's output
read -r -u 3 -d . line
exec 3>&-
if [[ "$line" == "$success_msg" ]]; then
export DOCKER_HOST="tcp://localhost:${local_port}"
if [[ -n "$command" ]]; then
bash -c "$command"
exit_status=$?
kill -15 $CONNECTION_PID
#clear the ssh control connection
if [[ "$PYTHON_DOCKER" == true ]]; then
ssh -i "$ssh_key_file" "$remote_host" -p ${SSH_PORT} "docker rm -f remote_python" 1>& 2> /dev/null
fi
ssh -O exit -o ControlPath="$control_path" "$remote_host" -p ${SSH_PORT} "" 2> /dev/null
rm -f "$control_path"
#exit with the same status as the command
exit $exit_status
else
COLOR='\033[0;33m'
NC='\033[0m'
RED='\033[1;31m'
echo -e "Local docker client connected to ${COLOR}${remote_host}${NC} docker daemon."
echo "Route: localhost:${local_port} -> ${remote_host}:${remote_port}"
echo -e "Press ${RED}Ctrl+D${NC} to stop forwarding and exit the bash session."
export PROMPT_COMMAND="echo -en \"${COLOR}docker:${remote_host}${NC} \""
bash
fi
kill -15 $CONNECTION_PID
echo -e "Disconnected from ${COLOR}$remote_host${NC} docker daemon."
else
echo "$line"
fi
#clear the ssh control connection
if [[ "$PYTHON_DOCKER" == true ]]; then
ssh -i "$ssh_key_file" "$remote_host" -p ${SSH_PORT} "docker rm -f remote_python" 1>& 2> /dev/null
fi
ssh -O exit -o ControlPath="$control_path" "$remote_host" -p ${SSH_PORT} "" 2> /dev/null
rm -f "$control_path"