initial commit
This commit is contained in:
commit
37f9100f97
15
README.md
Normal file
15
README.md
Normal 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
52
git-build
Executable 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
55
shellsrv
Executable 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()
|
Loading…
Reference in New Issue
Block a user