server.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401
  1. from flask import Flask
  2. from flask_restful import Resource, Api
  3. from flask import jsonify
  4. from flask import Response
  5. from flask import request
  6. from threading import Thread
  7. from OpenSSL import crypto
  8. import docker
  9. import uuid
  10. import signal
  11. import time
  12. import os
  13. import re
  14. import sys
  15. import ssl
  16. import socket
  17. docker_client = docker.DockerClient(base_url='unix://var/run/docker.sock', version='auto')
  18. app = Flask(__name__)
  19. api = Api(app)
  20. class containers_get(Resource):
  21. def get(self):
  22. containers = {}
  23. try:
  24. for container in docker_client.containers.list(all=True):
  25. containers.update({container.attrs['Id']: container.attrs})
  26. return containers
  27. except Exception as e:
  28. return jsonify(type='danger', msg=str(e))
  29. class container_get(Resource):
  30. def get(self, container_id):
  31. if container_id and container_id.isalnum():
  32. try:
  33. for container in docker_client.containers.list(all=True, filters={"id": container_id}):
  34. return container.attrs
  35. except Exception as e:
  36. return jsonify(type='danger', msg=str(e))
  37. else:
  38. return jsonify(type='danger', msg='no or invalid id defined')
  39. class container_post(Resource):
  40. def post(self, container_id, post_action):
  41. if container_id and container_id.isalnum() and post_action:
  42. if post_action == 'stop':
  43. try:
  44. for container in docker_client.containers.list(all=True, filters={"id": container_id}):
  45. container.stop()
  46. return jsonify(type='success', msg='command completed successfully')
  47. except Exception as e:
  48. return jsonify(type='danger', msg=str(e))
  49. elif post_action == 'start':
  50. try:
  51. for container in docker_client.containers.list(all=True, filters={"id": container_id}):
  52. container.start()
  53. return jsonify(type='success', msg='command completed successfully')
  54. except Exception as e:
  55. return jsonify(type='danger', msg=str(e))
  56. elif post_action == 'restart':
  57. try:
  58. for container in docker_client.containers.list(all=True, filters={"id": container_id}):
  59. container.restart()
  60. return jsonify(type='success', msg='command completed successfully')
  61. except Exception as e:
  62. return jsonify(type='danger', msg=str(e))
  63. elif post_action == 'top':
  64. try:
  65. for container in docker_client.containers.list(all=True, filters={"id": container_id}):
  66. return jsonify(type='success', msg=container.top())
  67. except Exception as e:
  68. return jsonify(type='danger', msg=str(e))
  69. elif post_action == 'stats':
  70. try:
  71. for container in docker_client.containers.list(all=True, filters={"id": container_id}):
  72. return jsonify(type='success', msg=container.stats(decode=True, stream=False))
  73. except Exception as e:
  74. return jsonify(type='danger', msg=str(e))
  75. elif post_action == 'exec':
  76. if not request.json or not 'cmd' in request.json:
  77. return jsonify(type='danger', msg='cmd is missing')
  78. if request.json['cmd'] == 'mailq':
  79. if 'items' in request.json:
  80. r = re.compile("^[0-9a-fA-F]+$")
  81. filtered_qids = filter(r.match, request.json['items'])
  82. if filtered_qids:
  83. if request.json['task'] == 'delete':
  84. flagged_qids = ['-d %s' % i for i in filtered_qids]
  85. sanitized_string = str(' '.join(flagged_qids));
  86. try:
  87. for container in docker_client.containers.list(filters={"id": container_id}):
  88. postsuper_r = container.exec_run(["/bin/bash", "-c", "/usr/sbin/postsuper " + sanitized_string])
  89. return exec_run_handler('generic', postsuper_r)
  90. except Exception as e:
  91. return jsonify(type='danger', msg=str(e))
  92. if request.json['task'] == 'hold':
  93. flagged_qids = ['-h %s' % i for i in filtered_qids]
  94. sanitized_string = str(' '.join(flagged_qids));
  95. try:
  96. for container in docker_client.containers.list(filters={"id": container_id}):
  97. postsuper_r = container.exec_run(["/bin/bash", "-c", "/usr/sbin/postsuper " + sanitized_string])
  98. return exec_run_handler('generic', postsuper_r)
  99. except Exception as e:
  100. return jsonify(type='danger', msg=str(e))
  101. if request.json['task'] == 'unhold':
  102. flagged_qids = ['-H %s' % i for i in filtered_qids]
  103. sanitized_string = str(' '.join(flagged_qids));
  104. try:
  105. for container in docker_client.containers.list(filters={"id": container_id}):
  106. postsuper_r = container.exec_run(["/bin/bash", "-c", "/usr/sbin/postsuper " + sanitized_string])
  107. return exec_run_handler('generic', postsuper_r)
  108. except Exception as e:
  109. return jsonify(type='danger', msg=str(e))
  110. if request.json['task'] == 'deliver':
  111. flagged_qids = ['-i %s' % i for i in filtered_qids]
  112. try:
  113. for container in docker_client.containers.list(filters={"id": container_id}):
  114. for i in flagged_qids:
  115. postqueue_r = container.exec_run(["/bin/bash", "-c", "/usr/sbin/postqueue " + i], user='postfix')
  116. # todo: check each exit code
  117. return jsonify(type='success', msg=str("Scheduled immediate delivery"))
  118. except Exception as e:
  119. return jsonify(type='danger', msg=str(e))
  120. elif request.json['task'] == 'list':
  121. try:
  122. for container in docker_client.containers.list(filters={"id": container_id}):
  123. mailq_return = container.exec_run(["/usr/sbin/postqueue", "-j"], user='postfix')
  124. return exec_run_handler('utf8_text_only', mailq_return)
  125. except Exception as e:
  126. return jsonify(type='danger', msg=str(e))
  127. elif request.json['task'] == 'flush':
  128. try:
  129. for container in docker_client.containers.list(filters={"id": container_id}):
  130. postqueue_r = container.exec_run(["/usr/sbin/postqueue", "-f"], user='postfix')
  131. return exec_run_handler('generic', postqueue_r)
  132. except Exception as e:
  133. return jsonify(type='danger', msg=str(e))
  134. elif request.json['task'] == 'super_delete':
  135. try:
  136. for container in docker_client.containers.list(filters={"id": container_id}):
  137. postsuper_r = container.exec_run(["/usr/sbin/postsuper", "-d", "ALL"])
  138. return exec_run_handler('generic', postsuper_r)
  139. except Exception as e:
  140. return jsonify(type='danger', msg=str(e))
  141. elif request.json['cmd'] == 'system':
  142. if request.json['task'] == 'fts_rescan':
  143. if 'username' in request.json:
  144. try:
  145. for container in docker_client.containers.list(filters={"id": container_id}):
  146. rescan_return = container.exec_run(["/bin/bash", "-c", "/usr/local/bin/doveadm fts rescan -u '" + request.json['username'].replace("'", "'\\''") + "'"], user='vmail')
  147. if rescan_return.exit_code == 0:
  148. return jsonify(type='success', msg='fts_rescan: rescan triggered')
  149. else:
  150. return jsonify(type='warning', msg='fts_rescan error')
  151. except Exception as e:
  152. return jsonify(type='danger', msg=str(e))
  153. if 'all' in request.json:
  154. try:
  155. for container in docker_client.containers.list(filters={"id": container_id}):
  156. rescan_return = container.exec_run(["/bin/bash", "-c", "/usr/local/bin/doveadm fts rescan -A"], user='vmail')
  157. if rescan_return.exit_code == 0:
  158. return jsonify(type='success', msg='fts_rescan: rescan triggered')
  159. else:
  160. return jsonify(type='warning', msg='fts_rescan error')
  161. except Exception as e:
  162. return jsonify(type='danger', msg=str(e))
  163. elif request.json['task'] == 'df':
  164. if 'dir' in request.json:
  165. try:
  166. for container in docker_client.containers.list(filters={"id": container_id}):
  167. df_return = container.exec_run(["/bin/bash", "-c", "/bin/df -H '" + request.json['dir'].replace("'", "'\\''") + "' | /usr/bin/tail -n1 | /usr/bin/tr -s [:blank:] | /usr/bin/tr ' ' ','"], user='nobody')
  168. if df_return.exit_code == 0:
  169. return df_return.output.rstrip()
  170. else:
  171. return "0,0,0,0,0,0"
  172. except Exception as e:
  173. return jsonify(type='danger', msg=str(e))
  174. elif request.json['task'] == 'mysql_upgrade':
  175. try:
  176. for container in docker_client.containers.list(filters={"id": container_id}):
  177. sql_shell = container.exec_run(["/bin/bash"], stdin=True, socket=True, user='mysql')
  178. upgrade_cmd = "/usr/bin/mysql_upgrade -uroot -p'" + os.environ['DBROOT'].replace("'", "'\\''") + "'\n"
  179. sql_socket = sql_shell.output;
  180. try :
  181. sql_socket.sendall(upgrade_cmd.encode('utf-8'))
  182. sql_socket.shutdown(socket.SHUT_WR)
  183. except socket.error:
  184. return jsonify(type='danger', msg=str('socket error'))
  185. worker_response = recv_socket_data(sql_socket)
  186. matched = False
  187. for line in worker_response.split("\n"):
  188. if 'is already upgraded to' in line:
  189. matched = True
  190. if matched:
  191. return jsonify(type='success', msg='mysql_upgrade: already upgraded')
  192. else:
  193. container.restart()
  194. return jsonify(type='warning', msg='mysql_upgrade: upgrade was applied')
  195. except Exception as e:
  196. return jsonify(type='danger', msg=str(e))
  197. elif request.json['cmd'] == 'reload':
  198. if request.json['task'] == 'dovecot':
  199. try:
  200. for container in docker_client.containers.list(filters={"id": container_id}):
  201. reload_return = container.exec_run(["/bin/bash", "-c", "/usr/local/sbin/dovecot reload"])
  202. return exec_run_handler('generic', reload_return)
  203. except Exception as e:
  204. return jsonify(type='danger', msg=str(e))
  205. if request.json['task'] == 'postfix':
  206. try:
  207. for container in docker_client.containers.list(filters={"id": container_id}):
  208. reload_return = container.exec_run(["/bin/bash", "-c", "/usr/sbin/postfix reload"])
  209. return exec_run_handler('generic', reload_return)
  210. except Exception as e:
  211. return jsonify(type='danger', msg=str(e))
  212. if request.json['task'] == 'nginx':
  213. try:
  214. for container in docker_client.containers.list(filters={"id": container_id}):
  215. reload_return = container.exec_run(["/bin/sh", "-c", "/usr/sbin/nginx -s reload"])
  216. return exec_run_handler('generic', reload_return)
  217. except Exception as e:
  218. return jsonify(type='danger', msg=str(e))
  219. elif request.json['cmd'] == 'sieve':
  220. if request.json['task'] == 'list':
  221. if 'username' in request.json:
  222. try:
  223. for container in docker_client.containers.list(filters={"id": container_id}):
  224. sieve_return = container.exec_run(["/bin/bash", "-c", "/usr/local/bin/doveadm sieve list -u '" + request.json['username'].replace("'", "'\\''") + "'"])
  225. return exec_run_handler('utf8_text_only', sieve_return)
  226. except Exception as e:
  227. return jsonify(type='danger', msg=str(e))
  228. elif request.json['task'] == 'print':
  229. if 'username' in request.json and 'script_name' in request.json:
  230. try:
  231. for container in docker_client.containers.list(filters={"id": container_id}):
  232. sieve_return = container.exec_run(["/bin/bash", "-c", "/usr/local/bin/doveadm sieve get -u '" + request.json['username'].replace("'", "'\\''") + "' '" + request.json['script_name'].replace("'", "'\\''") + "'"])
  233. return exec_run_handler('utf8_text_only', sieve_return)
  234. except Exception as e:
  235. return jsonify(type='danger', msg=str(e))
  236. elif request.json['cmd'] == 'maildir':
  237. if request.json['task'] == 'cleanup':
  238. if 'maildir' in request.json:
  239. try:
  240. for container in docker_client.containers.list(filters={"id": container_id}):
  241. sane_name = re.sub(r'\W+', '', request.json['maildir'])
  242. maildir_cleanup = container.exec_run(["/bin/bash", "-c", "if [[ -d '/var/vmail/" + request.json['maildir'].replace("'", "'\\''") + "' ]]; then /bin/mv '/var/vmail/" + request.json['maildir'].replace("'", "'\\''") + "' '/var/vmail/_garbage/" + str(int(time.time())) + "_" + sane_name + "'; fi"], user='vmail')
  243. return exec_run_handler('generic', maildir_cleanup)
  244. except Exception as e:
  245. return jsonify(type='danger', msg=str(e))
  246. elif request.json['cmd'] == 'rspamd':
  247. if request.json['task'] == 'worker_password':
  248. if 'raw' in request.json:
  249. try:
  250. for container in docker_client.containers.list(filters={"id": container_id}):
  251. worker_shell = container.exec_run(["/bin/bash"], stdin=True, socket=True, user='_rspamd')
  252. worker_cmd = "/usr/bin/rspamadm pw -e -p '" + request.json['raw'].replace("'", "'\\''") + "' 2> /dev/null\n"
  253. worker_socket = worker_shell.output;
  254. try :
  255. worker_socket.sendall(worker_cmd.encode('utf-8'))
  256. worker_socket.shutdown(socket.SHUT_WR)
  257. except socket.error:
  258. return jsonify(type='danger', msg=str('socket error'))
  259. worker_response = recv_socket_data(worker_socket)
  260. matched = False
  261. for line in worker_response.split("\n"):
  262. if '$2$' in line:
  263. matched = True
  264. hash = line.strip()
  265. hash_out = re.search('\$2\$.+$', hash).group(0)
  266. f = open("/access.inc", "w")
  267. f.write('enable_password = "' + re.sub('[^0-9a-zA-Z\$]+', '', hash_out.rstrip()) + '";\n')
  268. f.close()
  269. container.restart()
  270. if matched:
  271. return jsonify(type='success', msg='command completed successfully')
  272. else:
  273. return jsonify(type='danger', msg='command did not complete')
  274. except Exception as e:
  275. return jsonify(type='danger', msg=str(e))
  276. else:
  277. return jsonify(type='danger', msg='Unknown command')
  278. else:
  279. return jsonify(type='danger', msg='invalid action')
  280. else:
  281. return jsonify(type='danger', msg='invalid container id or missing action')
  282. class GracefulKiller:
  283. kill_now = False
  284. def __init__(self):
  285. signal.signal(signal.SIGINT, self.exit_gracefully)
  286. signal.signal(signal.SIGTERM, self.exit_gracefully)
  287. def exit_gracefully(self, signum, frame):
  288. self.kill_now = True
  289. def startFlaskAPI():
  290. create_self_signed_cert()
  291. try:
  292. ctx = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH)
  293. ctx.check_hostname = False
  294. ctx.load_cert_chain(certfile='/cert.pem', keyfile='/key.pem')
  295. except:
  296. print "Cannot initialize TLS, retrying in 5s..."
  297. time.sleep(5)
  298. app.run(debug=False, host='0.0.0.0', port=443, threaded=True, ssl_context=ctx)
  299. def recv_socket_data(c_socket, timeout=10):
  300. c_socket.setblocking(0)
  301. total_data=[];
  302. data='';
  303. begin=time.time()
  304. while True:
  305. if total_data and time.time()-begin > timeout:
  306. break
  307. elif time.time()-begin > timeout*2:
  308. break
  309. try:
  310. data = c_socket.recv(8192)
  311. if data:
  312. total_data.append(data)
  313. #change the beginning time for measurement
  314. begin=time.time()
  315. else:
  316. #sleep for sometime to indicate a gap
  317. time.sleep(0.1)
  318. break
  319. except:
  320. pass
  321. return ''.join(total_data)
  322. def exec_run_handler(type, output):
  323. if type == 'generic':
  324. if output.exit_code == 0:
  325. return jsonify(type='success', msg='command completed successfully')
  326. else:
  327. return jsonify(type='danger', msg='command failed: ' + output.output)
  328. if type == 'utf8_text_only':
  329. r = Response(response=output.output, status=200, mimetype="text/plain")
  330. r.headers["Content-Type"] = "text/plain; charset=utf-8"
  331. return r
  332. def create_self_signed_cert():
  333. success = False
  334. while not success:
  335. try:
  336. pkey = crypto.PKey()
  337. pkey.generate_key(crypto.TYPE_RSA, 2048)
  338. cert = crypto.X509()
  339. cert.get_subject().O = "mailcow"
  340. cert.get_subject().CN = "dockerapi"
  341. cert.set_serial_number(int(uuid.uuid4()))
  342. cert.gmtime_adj_notBefore(0)
  343. cert.gmtime_adj_notAfter(10*365*24*60*60)
  344. cert.set_issuer(cert.get_subject())
  345. cert.set_pubkey(pkey)
  346. cert.sign(pkey, 'sha512')
  347. cert = crypto.dump_certificate(crypto.FILETYPE_PEM, cert)
  348. pkey = crypto.dump_privatekey(crypto.FILETYPE_PEM, pkey)
  349. with os.fdopen(os.open('/cert.pem', os.O_WRONLY | os.O_CREAT, 0o644), 'w') as handle:
  350. handle.write(cert)
  351. with os.fdopen(os.open('/key.pem', os.O_WRONLY | os.O_CREAT, 0o600), 'w') as handle:
  352. handle.write(pkey)
  353. success = True
  354. except:
  355. time.sleep(1)
  356. try:
  357. os.remove('/cert.pem')
  358. os.remove('/key.pem')
  359. except OSError:
  360. pass
  361. api.add_resource(containers_get, '/containers/json')
  362. api.add_resource(container_get, '/containers/<string:container_id>/json')
  363. api.add_resource(container_post, '/containers/<string:container_id>/<string:post_action>')
  364. if __name__ == '__main__':
  365. api_thread = Thread(target=startFlaskAPI)
  366. api_thread.daemon = True
  367. api_thread.start()
  368. killer = GracefulKiller()
  369. while True:
  370. time.sleep(1)
  371. if killer.kill_now:
  372. break
  373. print "Stopping dockerapi-mailcow"