import std/options import std/os import std/posix import std/streams import std/tables import config/config import display/winattrs import io/posixstream import io/serialize import io/serversocket import io/urlfilter import loader/headers import loader/loader import server/buffer import types/cookie import types/urimethodmap import types/url import utils/strwidth type ForkCommand* = enum FORK_BUFFER, FORK_LOADER, REMOVE_CHILD, LOAD_CONFIG, FORK_BUFFER_WITH_LOADER ForkServer* = ref object istream: Stream ostream: Stream estream*: PosixStream ForkServerContext = object istream: Stream ostream: Stream children: seq[(Pid, Pid)] proc newFileLoader*(forkserver: ForkServer, defaultHeaders: Headers, proxy: URL, urimethodmap: URIMethodMap, cgiDir: seq[string], acceptProxy, w3mCGICompat: bool): FileLoader = forkserver.ostream.swrite(FORK_LOADER) let config = LoaderConfig( defaultHeaders: defaultHeaders, filter: newURLFilter(default = true), proxy: proxy, acceptProxy: acceptProxy, urimethodmap: urimethodmap, w3mCGICompat: w3mCGICompat, cgiDir: cgiDir ) forkserver.ostream.swrite(config) forkserver.ostream.flush() var process: Pid forkserver.istream.sread(process) return FileLoader(process: process, clientPid: getCurrentProcessId()) proc loadForkServerConfig*(forkserver: ForkServer, config: Config) = forkserver.ostream.swrite(LOAD_CONFIG) forkserver.ostream.swrite(config.getForkServerConfig()) forkserver.ostream.flush() proc removeChild*(forkserver: ForkServer, pid: Pid) = forkserver.ostream.swrite(REMOVE_CHILD) forkserver.ostream.swrite(pid) forkserver.ostream.flush() proc forkBuffer*(forkserver: ForkServer, source: BufferSource, config: BufferConfig, attrs: WindowAttributes): tuple[process, loaderPid: Pid] = forkserver.ostream.swrite(FORK_BUFFER) forkserver.ostream.swrite(source) forkserver.ostream.swrite(config) forkserver.ostream.swrite(attrs) forkserver.ostream.flush() var process: Pid var loaderPid: Pid forkserver.istream.sread(process) forkserver.istream.sread(loaderPid) return (process, loaderPid) proc forkBufferWithLoader*(forkserver: ForkServer, source: BufferSource, config: BufferConfig, attrs: WindowAttributes, loaderPid: Pid): Pid = forkserver.ostream.swrite(FORK_BUFFER_WITH_LOADER) forkserver.ostream.swrite(source) forkserver.ostream.swrite(config) forkserver.ostream.swrite(attrs) forkserver.ostream.swrite(loaderPid) forkserver.ostream.flush() var bufferPid: Pid forkserver.istream.sread(bufferPid) return bufferPid proc trapSIGINT() = # trap SIGINT, so e.g. an external editor receiving an interrupt in the # same process group can't just kill the process # Note that the main process normally quits on interrupt (thus terminating # all child processes as well). setControlCHook(proc() {.noconv.} = discard) proc forkLoader(ctx: var ForkServerContext, config: LoaderConfig): Pid = var pipefd: array[2, cint] if pipe(pipefd) == -1: raise newException(Defect, "Failed to open pipe.") let pid = fork() if pid == 0: # child process trapSIGINT() for i in 0 ..< ctx.children.len: ctx.children[i] = (Pid(0), Pid(0)) ctx.children.setLen(0) zeroMem(addr ctx, sizeof(ctx)) discard close(pipefd[0]) # close read try: runFileLoader(pipefd[1], config) except CatchableError: let e = getCurrentException() # taken from system/excpt.nim let msg = e.getStackTrace() & "Error: unhandled exception: " & e.msg & " [" & $e.name & "]\n" stderr.write(msg) quit(1) doAssert false let readfd = pipefd[0] # get read discard close(pipefd[1]) # close write var readf: File if not open(readf, FileHandle(readfd), fmRead): raise newException(Defect, "Failed to open output handle.") let c = readf.readChar() assert c == char(0u8) close(readf) discard close(pipefd[0]) return pid var gssock: ServerSocket proc forkBuffer0(ctx: var ForkServerContext, source: BufferSource, config: BufferConfig, attrs: WindowAttributes, loaderPid: Pid): Pid = var pipefd: array[2, cint] if pipe(pipefd) == -1: raise newException(Defect, "Failed to open pipe.") let pid = fork() if pid == -1: raise newException(Defect, "Failed to fork process.") if pid == 0: # child process trapSIGINT() for i in 0 ..< ctx.children.len: ctx.children[i] = (Pid(0), Pid(0)) ctx.children.setLen(0) zeroMem(addr ctx, sizeof(ctx)) discard close(pipefd[0]) # close read let ssock = initServerSocket(buffered = false) gssock = ssock onSignal SIGTERM: # This will be overridden after buffer has been set up; it is only # necessary to avoid a race condition when buffer is killed before that. discard sig gssock.close() let ps = newPosixStream(pipefd[1]) ps.write(char(0)) ps.close() discard close(stdin.getFileHandle()) discard close(stdout.getFileHandle()) let loader = FileLoader( process: loaderPid, clientPid: getCurrentProcessId() ) try: launchBuffer(config, source, attrs, loader, ssock) except CatchableError: let e = getCurrentException() # taken from system/excpt.nim let msg = e.getStackTrace() & "Error: unhandled exception: " & e.msg & " [" & $e.name & "]\n" stderr.write(msg) quit(1) doAssert false discard close(pipefd[1]) # close write let ps = newPosixStream(pipefd[0]) let c = ps.readChar() assert c == char(0) ps.close() ctx.children.add((pid, loaderPid)) return pid proc forkBuffer(ctx: var ForkServerContext): tuple[process, loaderPid: Pid] = var source: BufferSource var config: BufferConfig var attrs: WindowAttributes ctx.istream.sread(source) ctx.istream.sread(config) ctx.istream.sread(attrs) let loaderPid = ctx.forkLoader(config.loaderConfig) let process = ctx.forkBuffer0(source, config, attrs, loaderPid) return (process, loaderPid) proc forkBufferWithLoader(ctx: var ForkServerContext): Pid = var source: BufferSource var config: BufferConfig var attrs: WindowAttributes var loaderPid: Pid ctx.istream.sread(source) ctx.istream.sread(config) ctx.istream.sread(attrs) ctx.istream.sread(loaderPid) FileLoader(process: loaderPid).addref() return ctx.forkBuffer0(source, config, attrs, loaderPid) proc runForkServer() = var ctx = ForkServerContext( istream: newPosixStream(stdin.getFileHandle()), ostream: newPosixStream(stdout.getFileHandle()) ) while true: try: var cmd: ForkCommand ctx.istream.sread(cmd) case cmd of REMOVE_CHILD: var pid: Pid ctx.istream.sread(pid) for i in 0 .. ctx.children.high: if ctx.children[i][0] == pid: ctx.children.del(i) break of FORK_BUFFER: ctx.ostream.swrite(ctx.forkBuffer()) of FORK_BUFFER_WITH_LOADER: ctx.ostream.swrite(ctx.forkBufferWithLoader()) of FORK_LOADER: var config: LoaderConfig ctx.istream.sread(config) let pid = ctx.forkLoader(config) ctx.ostream.swrite(pid) ctx.children.add((pid, Pid(-1))) of LOAD_CONFIG: var config: ForkServerConfig ctx.istream.sread(config) set_cjk_ambiguous(config.ambiguous_double) SocketDirectory = config.tmpdir ctx.ostream.flush() except EOFError: # EOF break ctx.istream.close() ctx.ostream.close() # Clean up when the main process crashed. for childpair in ctx.children: let a = childpair[0] let b = childpair[1] discard kill(cint(a), cint(SIGTERM)) if b != -1: discard kill(cint(b), cint(SIGTERM)) quit(0) proc newForkServer*(): ForkServer = var pipefd_in: array[2, cint] # stdin in forkserver var pipefd_out: array[2, cint] # stdout in forkserver var pipefd_err: array[2, cint] # stderr in forkserver if pipe(pipefd_in) == -1: raise newException(Defect, "Failed to open input pipe.") if pipe(pipefd_out) == -1: raise newException(Defect, "Failed to open output pipe.") if pipe(pipefd_err) == -1: raise newException(Defect, "Failed to open error pipe.") let pid = fork() if pid == -1: raise newException(Defect, "Failed to fork the fork process.") elif pid == 0: # child process trapSIGINT() discard close(pipefd_in[1]) # close write discard close(pipefd_out[0]) # close read discard close(pipefd_err[0]) # close read let readfd = pipefd_in[0] let writefd = pipefd_out[1] let errfd = pipefd_err[1] discard dup2(readfd, stdin.getFileHandle()) discard dup2(writefd, stdout.getFileHandle()) discard dup2(errfd, stderr.getFileHandle()) stderr.flushFile() discard close(pipefd_in[0]) discard close(pipefd_out[1]) discard close(pipefd_err[1]) runForkServer() doAssert false else: discard close(pipefd_in[0]) # close read discard close(pipefd_out[1]) # close write discard close(pipefd_err[1]) # close write var writef, readf: File if not open(writef, pipefd_in[1], fmWrite): raise newException(Defect, "Failed to open output handle") if not open(readf, pipefd_out[0], fmRead): raise newException(Defect, "Failed to open input handle") let estream = newPosixStream(pipefd_err[0]) estream.setBlocking(false) return ForkServer( ostream: newFileStream(writef), istream: newFileStream(readf), estream: estream )