1
0
mirror of https://github.com/IEEE-SB-Passau/pelican-deployment-system.git synced 2017-09-06 16:35:38 +02:00

nice status page

This commit is contained in:
2016-06-15 00:15:11 +02:00
parent 72ba260c8f
commit 7ff1a9d4e5
4 changed files with 112 additions and 3 deletions

7
app.py
View File

@@ -7,6 +7,7 @@ from operator import methodcaller
from bottle import run, default_app from bottle import run, default_app
from wsgiref.simple_server import make_server from wsgiref.simple_server import make_server
import pelican_deploy.webhookbottle import pelican_deploy.webhookbottle
import pelican_deploy.statusbottle
import logging import logging
import atexit import atexit
import sys import sys
@@ -48,6 +49,12 @@ def init_app(configpath):
pelican_deploy.webhookbottle.set_github_secret(config.GITHUB_SECRET) pelican_deploy.webhookbottle.set_github_secret(config.GITHUB_SECRET)
default_app().mount("/hooks/", pelican_deploy.webhookbottle.app) default_app().mount("/hooks/", pelican_deploy.webhookbottle.app)
pelican_deploy.statusbottle.set_auth_basic_fn(getattr(config,
"STATUS_AUTH_BASIC_FN",
lambda us, pw: False))
pelican_deploy.statusbottle.set_runners(**runners)
default_app().mount("/status/", pelican_deploy.statusbottle.app)
return default_app() return default_app()
if __name__ == "__main__": if __name__ == "__main__":

View File

@@ -1,6 +1,7 @@
import os import os
import logging import logging
from apscheduler.triggers.cron import CronTrigger from apscheduler.triggers.cron import CronTrigger
from apscheduler.triggers.date import DateTrigger
if __name__ == "__main__": if __name__ == "__main__":
raise SystemExit("Not meant to be run directly!") raise SystemExit("Not meant to be run directly!")
@@ -59,9 +60,12 @@ RUNNERS = {
# define crojobs as sequence of (runner, trigger) pairs, for cron triggers see # define crojobs as sequence of (runner, trigger) pairs, for cron triggers see
# http://apscheduler.readthedocs.io/en/latest/modules/triggers/cron.html # http://apscheduler.readthedocs.io/en/latest/modules/triggers/cron.html
SCHEDULED_BUILD_JOBS = [ SCHEDULED_BUILD_JOBS = [
("website_master", CronTrigger(minute="*/30")) ("website_master", CronTrigger(minute="*/30")),
("website_master", DateTrigger()) # once at start
] ]
# username, password for /status/... subpages, accepts nothing if not set
def STATUS_AUTH_BASIC_FN(user, passw):
return user == "powerpoint" and passw == "karaoke"

View File

@@ -169,7 +169,8 @@ class DeploymentRunner:
def final_install(self): def final_install(self):
args = shlex.split(self.final_install_command) args = shlex.split(self.final_install_command)
self.update_status(True, "Starting final_install") self.update_status(True, "Starting final_install",
payload={"cmd": args})
log.info("%s: Starting final_install `%s`", self.name, args) log.info("%s: Starting final_install `%s`", self.name, args)
proc = Popen(args, stdout=PIPE, stderr=PIPE, universal_newlines=True, proc = Popen(args, stdout=PIPE, stderr=PIPE, universal_newlines=True,
start_new_session=True) start_new_session=True)

View File

@@ -0,0 +1,97 @@
from bottle import route, template, request, post, Bottle, HTTPError,auth_basic
from pprint import pformat
from itertools import islice
import logging
import sys
log = logging.getLogger(__name__)
app = Bottle()
def _auth_basic_fn(us, pw):
return app.config.get("auth_basic_fn", lambda us, pw: False)(us, pw)
def set_runners(**name_runner_mapping):
app.config["deploy.runners"] = name_runner_mapping
def set_auth_basic_fn(fn):
app.config["auth_basic_fn"] = fn
def _get_runner(name):
try:
runners = app.config["deploy.runners"]
except KeyError as e:
sys.exit("you have to call set_runners first")
return runners[name]
@app.route('/')
def status():
tpl = """
<html>
<h1>Runners</h1>
<ul>
% for r in runners:
<%
bs = r.build_status[-1] if r.build_status else None
%>
<li>
% if bs:
<%
date = bs.date.strftime("%Y-%m-%d %H:%M:%S %Z%z")
%>
<a href="{{r.name}}">{{r.name}}</a>:
{{date}} - {{bs.msg}} - running: {{bs.running}} - ok: {{bs.ok}}
% else:
No job was ever running.
% end
</li>
% end
</ul>
</html>
"""
return template(tpl, runners=app.config["deploy.runners"].values())
@app.route('/<name>')
@auth_basic(_auth_basic_fn)
def runnerstatus(name):
runner = _get_runner(name)
rerun = "rerun" in request.query
start = int(request.query.get("start", 0))
end = int(request.query.get("end", 50))
start = start if start >= 0 else 0
end = end if end >= 0 else 0
tpl = """
<html>
<h1>{{runner.name}} status events ({{start}} - {{end}}) --
<a href={{runner.name}}/rerun>(re)start build</a></h1>
<ul>
% for bs in islice(reversed(bss),start,end):
<%
date = bs.date.strftime("%Y-%m-%d %H:%M:%S %Z%z")
%>
<li>{{date}} -- {{bs.msg}}
<ul>
<li>build still running: {{bs.running}}</li>
<li>nothing went wrong: {{bs.ok}}</li>
<li>message payload:<br>
<pre>{{pformat(bs.payload, width=120)}}</pre>
</li>
</ul>
</li>
% end
</ul>
</html>
"""
return template(tpl, runner=runner, bss=runner.build_status, islice=islice,
pformat=pformat, start=start, end=end)
@auth_basic(_auth_basic_fn)
@app.route('/<name>/rerun')
def rerun(name):
runner = _get_runner(name)
runner.build(abort_running=True, ignore_pull_error=True)
return "Restarted the build"