forked from autodesk-cloud/ochothon
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcli.py
executable file
·106 lines (91 loc) · 3.72 KB
/
cli.py
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
#! /usr/bin/env python
#
# Copyright (c) 2015 Autodesk Inc.
# All rights reserved
#
# 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 cmd
import json
import os
import sys
from os.path import basename, expanduser, isfile
from subprocess import Popen, PIPE
from sys import exit
def cli():
"""
Minimalistic self-contained wrapper performing the curl calls to the ochopod proxy. The input
is turned into a POST -H X-Shell:<> to the proxy at port TCP 9000. Any token from that input that
matches a local file (wherever the script is running from) will force an upload for the said file.
This mechanism is used for instance to upload the container definition YAML files when deploying a
new cluster.
The proxy IP is either passed as the first command-line argument or vi a $OCHOPOD_PROXY.
"""
class Shell(cmd.Cmd):
def __init__(self, ip):
cmd.Cmd.__init__(self)
self.prompt = '%s > ' % ip
self.ruler = '-'
def precmd(self, line):
return 'shell %s' % line if line not in ['exit'] else line
def emptyline(self):
pass
def do_exit(self, _):
raise KeyboardInterrupt
def do_shell(self, line):
if line:
tokens = line.split(' ')
#
# - update from steven -> reformat the input line to handle indirect paths transparently
# - for instance ../foo.bar will become foo.bar with the actual file included in the multi-part post
#
files = ['-F %s=@%s' % (basename(token), expanduser(token)) for token in tokens if isfile(expanduser(token))]
line = ' '.join([basename(token) if isfile(expanduser(token)) else token for token in tokens])
snippet = 'curl -X POST -H "X-Shell:%s" %s %s:9000/shell' % (line, ' '.join(files), ip)
code, out = self._exec(snippet)
js = json.loads(out.decode('utf-8'))
print(js['out'] if code is 0 else 'i/o failure (is the proxy down ?)')
def _exec(self, snippet):
pid = Popen(snippet, shell=True, stdout=PIPE, stderr=PIPE)
pid.wait()
return pid.returncode, pid.stdout.read()
try:
#
# - partition ip and args by looking for OCHOPOD_PROXY first.
# - if OCHOPOD_PROXY is not used, treat the first argument as the ip.
#
if 'OCHOPOD_PROXY' in os.environ:
ip = os.environ['OCHOPOD_PROXY']
args = sys.argv[1:]
elif len(sys.argv) > 1:
ip = sys.argv[1]
args = sys.argv[2:]
else:
print('either set $OCHOPOD_PROXY or pass the proxy IP as an argument')
exit(1)
#
# - determine whether to run in interactive or non-interactive mode.
#
if len(args):
command = " ".join(args)
Shell(ip).do_shell(command)
else:
print('welcome to the ocho CLI ! (CTRL-C or exit to get out)')
Shell(ip).cmdloop()
except KeyboardInterrupt:
exit(0)
except Exception as failure:
print('internal failure <- %s' % str(failure))
exit(1)
if __name__ == "__main__":
cli()