2017-10-06 05:38:33 +08:00
from flask import Flask
from flask_restful import Resource , Api
2017-10-06 16:20:40 +08:00
from flask import jsonify
2017-11-04 03:26:09 +08:00
from flask import request
2017-10-27 17:22:39 +08:00
from threading import Thread
2017-10-06 16:20:40 +08:00
import docker
2017-10-27 17:22:39 +08:00
import signal
import time
2017-12-09 20:15:24 +08:00
import os
import re
import sys
2017-10-06 05:38:33 +08:00
2017-12-02 06:41:37 +08:00
docker_client = docker . DockerClient ( base_url = ' unix://var/run/docker.sock ' , version = ' auto ' )
2017-10-06 05:38:33 +08:00
app = Flask ( __name__ )
api = Api ( app )
2017-10-06 16:20:40 +08:00
class containers_get ( Resource ) :
2017-10-27 17:22:39 +08:00
def get ( self ) :
containers = { }
2017-11-04 03:26:09 +08:00
try :
2017-12-11 16:43:01 +08:00
for container in docker_client . containers . list ( all = True ) :
2017-11-04 03:26:09 +08:00
containers . update ( { container . attrs [ ' Id ' ] : container . attrs } )
return containers
except Exception as e :
2018-02-01 20:38:42 +08:00
return jsonify ( type = ' danger ' , msg = str ( e ) )
2017-10-06 05:38:33 +08:00
2017-10-06 16:20:40 +08:00
class container_get ( Resource ) :
2017-10-27 17:22:39 +08:00
def get ( self , container_id ) :
if container_id and container_id . isalnum ( ) :
2017-11-04 03:26:09 +08:00
try :
2017-12-11 16:43:01 +08:00
for container in docker_client . containers . list ( all = True , filters = { " id " : container_id } ) :
2017-11-04 03:26:09 +08:00
return container . attrs
except Exception as e :
2018-02-01 20:38:42 +08:00
return jsonify ( type = ' danger ' , msg = str ( e ) )
2017-10-27 17:22:39 +08:00
else :
2017-11-04 03:26:09 +08:00
return jsonify ( type = ' danger ' , msg = ' no or invalid id defined ' )
2017-10-06 05:38:33 +08:00
2017-10-06 16:20:40 +08:00
class container_post ( Resource ) :
2017-10-27 17:22:39 +08:00
def post ( self , container_id , post_action ) :
if container_id and container_id . isalnum ( ) and post_action :
if post_action == ' stop ' :
try :
2017-12-11 16:43:01 +08:00
for container in docker_client . containers . list ( all = True , filters = { " id " : container_id } ) :
2017-10-27 17:22:39 +08:00
container . stop ( )
2017-11-04 03:26:09 +08:00
return jsonify ( type = ' success ' , msg = ' command completed successfully ' )
except Exception as e :
2018-02-01 20:38:42 +08:00
return jsonify ( type = ' danger ' , msg = str ( e ) )
2017-11-04 03:26:09 +08:00
2017-10-27 17:22:39 +08:00
elif post_action == ' start ' :
try :
2017-12-11 16:43:01 +08:00
for container in docker_client . containers . list ( all = True , filters = { " id " : container_id } ) :
2017-10-27 17:22:39 +08:00
container . start ( )
2017-11-04 03:26:09 +08:00
return jsonify ( type = ' success ' , msg = ' command completed successfully ' )
except Exception as e :
2018-02-01 20:38:42 +08:00
return jsonify ( type = ' danger ' , msg = str ( e ) )
2017-11-04 03:26:09 +08:00
2017-10-27 17:22:39 +08:00
elif post_action == ' restart ' :
try :
2017-12-11 16:43:01 +08:00
for container in docker_client . containers . list ( all = True , filters = { " id " : container_id } ) :
2017-10-27 17:22:39 +08:00
container . restart ( )
2017-11-04 03:26:09 +08:00
return jsonify ( type = ' success ' , msg = ' command completed successfully ' )
except Exception as e :
2018-02-01 20:38:42 +08:00
return jsonify ( type = ' danger ' , msg = str ( e ) )
2017-11-04 03:26:09 +08:00
elif post_action == ' exec ' :
if not request . json or not ' cmd ' in request . json :
return jsonify ( type = ' danger ' , msg = ' cmd is missing ' )
2018-02-22 16:21:37 +08:00
if request . json [ ' cmd ' ] == ' df ' and request . json [ ' dir ' ] :
2017-11-04 03:26:09 +08:00
try :
2017-12-11 16:43:01 +08:00
for container in docker_client . containers . list ( filters = { " id " : container_id } ) :
2018-02-22 16:21:37 +08:00
# Should be changed to be able to validate a path
directory = re . sub ( ' [^0-9a-zA-Z/]+ ' , ' ' , request . json [ ' dir ' ] )
df_return = container . exec_run ( [ " /bin/bash " , " -c " , " /bin/df -H " + directory + " | /usr/bin/tail -n1 | /usr/bin/tr -s [:blank:] | /usr/bin/tr ' ' ' , ' " ] , user = ' nobody ' )
if df_return . exit_code == 0 :
return df_return . output . rstrip ( )
else :
return " 0,0,0,0,0,0 "
except Exception as e :
return jsonify ( type = ' danger ' , msg = str ( e ) )
elif request . json [ ' cmd ' ] == ' sieve_list ' and request . json [ ' username ' ] :
try :
for container in docker_client . containers . list ( filters = { " id " : container_id } ) :
sieve_return = container . exec_run ( [ " /bin/bash " , " -c " , " /usr/local/bin/doveadm sieve list -u ' " + request . json [ ' username ' ] . replace ( " ' " , " ' \\ ' ' " ) + " ' " ] , user = ' vmail ' )
return sieve_return . output
2017-11-04 03:26:09 +08:00
except Exception as e :
2018-02-01 20:38:42 +08:00
return jsonify ( type = ' danger ' , msg = str ( e ) )
2017-11-04 03:26:09 +08:00
elif request . json [ ' cmd ' ] == ' sieve_print ' and request . json [ ' script_name ' ] and request . json [ ' username ' ] :
try :
2017-12-11 16:43:01 +08:00
for container in docker_client . containers . list ( filters = { " id " : container_id } ) :
2017-11-04 03:26:09 +08:00
return container . exec_run ( [ " /bin/bash " , " -c " , " /usr/local/bin/doveadm sieve get -u ' " + request . json [ ' username ' ] . replace ( " ' " , " ' \\ ' ' " ) + " ' ' " + request . json [ ' script_name ' ] . replace ( " ' " , " ' \\ ' ' " ) + " ' " ] , user = ' vmail ' )
except Exception as e :
2018-02-01 20:38:42 +08:00
return jsonify ( type = ' danger ' , msg = str ( e ) )
2017-12-09 20:15:24 +08:00
elif request . json [ ' cmd ' ] == ' worker_password ' and request . json [ ' raw ' ] :
try :
2017-12-11 16:43:01 +08:00
for container in docker_client . containers . list ( filters = { " id " : container_id } ) :
2018-01-15 01:44:06 +08:00
hash = container . exec_run ( [ " /bin/bash " , " -c " , " /usr/bin/rspamadm pw -e -p ' " + request . json [ ' raw ' ] . replace ( " ' " , " ' \\ ' ' " ) + " ' 2> /dev/null " ] , user = ' _rspamd ' )
2018-02-09 05:29:06 +08:00
if hash . exit_code == 0 :
hash = str ( hash . output )
f = open ( " /access.inc " , " w " )
f . write ( ' enable_password = " ' + re . sub ( ' [^0-9a-zA-Z \ $]+ ' , ' ' , hash . rstrip ( ) ) + ' " ; \n ' )
f . close ( )
container . restart ( )
return jsonify ( type = ' success ' , msg = ' command completed successfully ' )
else :
return jsonify ( type = ' danger ' , msg = ' command did not complete, exit code was ' + int ( hash . exit_code ) )
2017-12-09 20:15:24 +08:00
except Exception as e :
2018-02-01 20:38:42 +08:00
return jsonify ( type = ' danger ' , msg = str ( e ) )
2017-12-09 20:15:24 +08:00
2017-10-06 05:38:33 +08:00
else :
2017-11-04 03:26:09 +08:00
return jsonify ( type = ' danger ' , msg = ' Unknown command ' )
2017-10-27 17:22:39 +08:00
else :
2017-11-04 03:26:09 +08:00
return jsonify ( type = ' danger ' , msg = ' invalid action ' )
2017-10-27 17:22:39 +08:00
else :
2017-11-04 03:26:09 +08:00
return jsonify ( type = ' danger ' , msg = ' invalid container id or missing action ' )
2017-10-27 17:22:39 +08:00
class GracefulKiller :
kill_now = False
def __init__ ( self ) :
signal . signal ( signal . SIGINT , self . exit_gracefully )
signal . signal ( signal . SIGTERM , self . exit_gracefully )
def exit_gracefully ( self , signum , frame ) :
self . kill_now = True
def startFlaskAPI ( ) :
2017-12-18 23:41:04 +08:00
app . run ( debug = False , host = ' 0.0.0.0 ' , port = 8080 , threaded = True )
2017-10-06 05:38:33 +08:00
2017-10-06 16:20:40 +08:00
api . add_resource ( containers_get , ' /containers/json ' )
api . add_resource ( container_get , ' /containers/<string:container_id>/json ' )
api . add_resource ( container_post , ' /containers/<string:container_id>/<string:post_action> ' )
2017-10-06 05:38:33 +08:00
if __name__ == ' __main__ ' :
2017-10-27 17:22:39 +08:00
api_thread = Thread ( target = startFlaskAPI )
api_thread . daemon = True
api_thread . start ( )
killer = GracefulKiller ( )
while True :
time . sleep ( 1 )
if killer . kill_now :
break
print " Stopping dockerapi-mailcow "
2017-12-11 16:43:01 +08:00