0dfe9a15cb0177143e7ca1b65e2adb9d583bf8c3
[ashd.git] / python3 / scgi-wsgi3
1 #!/usr/bin/python3
2
3 import sys, os, getopt, logging, collections
4 import socket
5 import ashd.scgi, ashd.serve
6
7 def usage(out):
8     out.write("usage: scgi-wsgi3 [-hAL] [-p MODPATH] [-T [HOST:]PORT] HANDLER-MODULE [ARGS...]\n")
9
10 sk = None
11 modwsgi_compat = False
12 setlog = True
13 opts, args = getopt.getopt(sys.argv[1:], "+hALp:T:")
14 for o, a in opts:
15     if o == "-h":
16         usage(sys.stdout)
17         sys.exit(0)
18     elif o == "-p":
19         sys.path.insert(0, a)
20     elif o == "-L":
21         setlog = False
22     elif o == "-T":
23         sk = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
24         sk.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
25         p = a.rfind(":")
26         if p < 0:
27             bindhost = "localhost"
28             bindport = int(a)
29         else:
30             bindhost = a[:p]
31             bindport = int(a[p + 1:])
32         sk.bind((bindhost, bindport))
33         sk.listen(32)
34     elif o == "-A":
35         modwsgi_compat = True
36 if len(args) < 1:
37     usage(sys.stderr)
38     sys.exit(1)
39 if setlog:
40     logging.basicConfig(format="scgi-wsgi3(%(name)s): %(levelname)s: %(message)s")
41
42 if sk is None:
43     # This is suboptimal, since the socket on stdin is not necessarily
44     # AF_UNIX, but Python does not seem to offer any way around it,
45     # that I can find.
46     sk = socket.fromfd(0, socket.AF_UNIX, socket.SOCK_STREAM)
47
48 try:
49     handlermod = __import__(args[0], fromlist = ["dummy"])
50 except ImportError as exc:
51     sys.stderr.write("scgi-wsgi3: handler %s not found: %s\n" % (args[0], exc.args[0]))
52     sys.exit(1)
53 if not modwsgi_compat:
54     if not hasattr(handlermod, "wmain"):
55         sys.stderr.write("scgi-wsgi3: handler %s has no `wmain' function\n" % args[0])
56         sys.exit(1)
57     handler = handlermod.wmain(*args[1:])
58 else:
59     if not hasattr(handlermod, "application"):
60         sys.stderr.write("scgi-wsgi3: handler %s has no `application' object\n" % args[0])
61         sys.exit(1)
62     handler = handlermod.application
63
64 def mkenv(head, sk):
65     try:
66         env = ashd.scgi.decodehead(head, "utf-8")
67         env["wsgi.uri_encoding"] = "utf-8"
68     except UnicodeError:
69         env = ashd.scgi.decodehead(head, "latin-1")
70         env["wsgi.uri_encoding"] = "latin-1"
71     env["wsgi.version"] = 1, 0
72     if "HTTP_X_ASH_PROTOCOL" in env:
73         env["wsgi.url_scheme"] = env["HTTP_X_ASH_PROTOCOL"]
74     elif "HTTPS" in env:
75         env["wsgi.url_scheme"] = "https"
76     else:
77         env["wsgi.url_scheme"] = "http"
78     env["wsgi.input"] = sk
79     env["wsgi.errors"] = sys.stderr
80     env["wsgi.multithread"] = True
81     env["wsgi.multiprocess"] = False
82     env["wsgi.run_once"] = False
83     return env
84
85 def recode(thing):
86     if isinstance(thing, collections.ByteString):
87         return thing
88     else:
89         return str(thing).encode("latin-1")
90
91 class reqthread(ashd.serve.wsgithread):
92     def __init__(self, sk):
93         super().__init__()
94         self.bsk = sk.dup()
95         self.sk = self.bsk.makefile("rwb")
96
97     def handlewsgi(self):
98         return handler(self.env, self.startreq)
99
100     def writehead(self, status, headers):
101         buf = bytearray()
102         buf += b"Status: " + recode(status) + b"\n"
103         for nm, val in headers:
104             buf += recode(nm) + b": " + recode(val) + b"\n"
105         buf += b"\n"
106         try:
107             self.sk.write(buf)
108         except IOError:
109             raise ashd.serve.closed()
110
111     def writedata(self, data):
112         try:
113             self.sk.write(data)
114             self.sk.flush()
115         except IOError:
116             raise ashd.serve.closed()
117
118     def handle(self):
119         head = ashd.scgi.readhead(self.sk)
120         self.env = mkenv(head, self.sk)
121         super().handle()
122
123     def run(self):
124         try:
125             super().run()
126         finally:
127             self.sk.close()
128             self.bsk.close()
129
130 while True:
131     nsk, addr = sk.accept()
132     try:
133         reqthread(nsk).start()
134     finally:
135         nsk.close()