initial commit

This commit is contained in:
Anton Lydike 2021-09-10 14:31:02 +02:00
commit 37f9100f97
3 changed files with 122 additions and 0 deletions

15
README.md Normal file
View File

@ -0,0 +1,15 @@
# Jank Pipeline
This is a minimal CI/CD pipeline built just as a fun side project. You might be able to use it on a private project for a quick and dirty hack, but should probably not use it for anything more serious.
It has two components: `shellsrv` and `git-build`.
# `shellsrv`
This tool is a quick way to glue networking capabilities to shell scripts.
You call it by running `./shellsrv -p <port> COMMAND [ARGS...]`
It sets up an https server, and whenever a GET/PUSH/POST etc html request arrives, it calls `COMMAND ARGS <HTTP VERB> <URL>` and passes the request body as stdin to the created process...
# `git-build`
This is designed to interface with the shellsrv script. It reads a ref from stdin json and then clones a git repo.

52
git-build Executable file
View File

@ -0,0 +1,52 @@
#!/usr/bin/env fish
# called using shellsrv git-build <repo url>
set -l repo $argv[1]
set -l path $argv[3]
set -l local_location (pwd)
if test "$path" != '/build-pls'
echo "invalid path"
exit 1
end
set -l tmp_dir /tmp/build_(random)
git clone $repo $tmp_dir
cd $tmp_dir
# read json from stding - which is the request body
git checkout (jq -r '.after')
# read config
set mount /pipeline
set envfile ""
for l in (cat .ezpipe)
if test -z "$l"
continue
end
set -l parts (string split -m 1 "=" "$l")
set $parts[1] $parts[2]
end
set docker_args
if test -z "$image"
echo "image required"
exit 1
end
if test -z "$entrypoint"
echo "entrypoint required"
exit 1
end
if test ! -z "$envfile"
set -a docker_args '--env-file'
set -a docker_args $local_location/$envfile
end
docker run --rm $docker_args -v (pwd):$mount -w $mount $image $mount/$entrypoint
cd ..
rm -rf $tmp_dir

55
shellsrv Executable file
View File

@ -0,0 +1,55 @@
#!/usr/bin/env python3
from http.server import BaseHTTPRequestHandler, HTTPServer
from subprocess import Popen, PIPE
from shlex import quote
import argparse
import sys
def server_class_factory(command):
class MyServer(BaseHTTPRequestHandler):
def do_GET(self):
self.handle_all()
def do_POST(self):
self.handle_all()
def do_PUT(self):
self.handle_all()
def handle_all(self):
self.send_response(200)
self.send_header("Content-type", "text/html")
self.end_headers()
content = self.rfile.read(int(self.headers.get('Content-Length', 0)))
p = Popen(command + [quote(self.command), quote(self.path)], stdin=PIPE, shell=False)
p.communicate(input=content)
def log_message(self, format, *args):
return
return MyServer
if __name__ == "__main__":
parser = argparse.ArgumentParser(description='shellsrv - Translate HTTP requests to command invocations on the host system.\n\n\nEach GET request results in the execution of COMMAND ARGS GET <path>. Other HTTP verbs might be added later.', prog='shellsrv')
parser.add_argument('--host', '-H', type=str, help='HTTP server host, default 0.0.0.0', default="0.0.0.0")
parser.add_argument('--port', '-p', type=int, help='HTTP server port, default 8080', default=8080)
parser.add_argument('--limit', '-n', type=int, help='Close the server after limit requests', default=-1)
parser.add_argument('command', metavar='COMMAND', type=str, nargs=1, help='command to call')
parser.add_argument('args', metavar='ARG', type=str, nargs='*', help='args of the command to call')
args = parser.parse_args()
command = args.command + args.args
request_limit = args.limit
server = HTTPServer((args.host, args.port), server_class_factory(command))
print("Server started http://%s:%s" % (args.host, args.port), file=sys.stderr)
if request_limit < 0:
try:
server.serve_forever()
except KeyboardInterrupt:
pass
else:
for _ in range(request_limit):
server.handle_request()
server.server_close()