diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/loader/connecterror.nim | 90 | ||||
-rw-r--r-- | src/loader/loader.nim | 53 |
2 files changed, 87 insertions, 56 deletions
diff --git a/src/loader/connecterror.nim b/src/loader/connecterror.nim index d8b06881..878bc8d0 100644 --- a/src/loader/connecterror.nim +++ b/src/loader/connecterror.nim @@ -1,36 +1,64 @@ -type ConnectErrorCode* = enum - ERROR_CGI_CACHED_BODY_NOT_FOUND = (-18, "cached request body not found") - ERROR_FAILED_TO_REDIRECT = (-17, "failed to redirect request body") - ERROR_URL_NOT_IN_CACHE = (-16, "URL was not found in the cache") - ERROR_FILE_NOT_IN_CACHE = (-15, "file was not found in the cache") - ERROR_FAILED_TO_EXECUTE_CGI_SCRIPT = (-14, "failed to execute CGI script") - ERROR_CGI_NO_DATA = (-13, "CGI script returned no data") - ERROR_CGI_MALFORMED_HEADER = (-12, "CGI script returned a malformed header") - ERROR_CGI_INVALID_CHA_CONTROL = (-11, "CGI got invalid Cha-Control header") - ERROR_TOO_MANY_REWRITES = (-10, "too many URI method map rewrites") - ERROR_INVALID_URI_METHOD_ENTRY = (-9, "invalid URI method entry") - ERROR_CGI_FILE_NOT_FOUND = (-8, "CGI file not found") - ERROR_INVALID_CGI_PATH = (-7, "invalid CGI path") - ERROR_FAIL_SETUP_CGI = (-6, "failed to set up CGI script") - ERROR_NO_CGI_DIR = (-5, "no local-CGI directory configured") - ERROR_SOURCE_NOT_FOUND = (-4, "clone source could not be found") - ERROR_LOADER_KILLED = (-3, "loader killed during transfer") - ERROR_DISALLOWED_URL = (-2, "url not allowed by filter") - ERROR_UNKNOWN_SCHEME = (-1, "unknown scheme") - CONNECTION_SUCCESS = (0, "connection successful") - ERROR_INTERNAL = (1, "internal error") - ERROR_INVALID_METHOD = (2, "invalid method") - ERROR_INVALID_URL = (3, "invalid URL") - ERROR_FILE_NOT_FOUND = (4, "file not found") - ERROR_CONNECTION_REFUSED = (5, "connection refused") - ERROR_PROXY_REFUSED_TO_CONNECT = (6, "proxy refused to connect") - ERROR_FAILED_TO_RESOLVE_HOST = (7, "failed to resolve host") - ERROR_FAILED_TO_RESOLVE_PROXY = (8, "failed to resolve proxy") +type ConnectionError* = enum + ceCGIOutputHandleNotFound = -17 + ceCGIFailedToOpenCacheOutput = -16 + ceCGICachedBodyNotFound = -15 + ceFailedToRedirect = -14 + ceURLNotInCache = -13 + ceFileNotInCache = -12 + ceFailedToExecuteCGIScript = -11 + ceCGIMalformedHeader = -10 + ceCGIInvalidChaControl = -9 + ceTooManyRewrites = -8 + ceInvalidURIMethodEntry = -7 + ceCGIFileNotFound = -6 + ceInvalidCGIPath = -5 + ceFailedToSetUpCGI = -4 + ceNoCGIDir = -3 + ceDisallowedURL = -2 + ceUnknownScheme = -1 + ceNone = 0 + ceInternalError = (1, "InternalError") + ceInvalidMethod = (2, "InvalidMethod") + ceInvalidURL = (3, "InvalidURL") + ceFileNotFound = (4, "FileNotFound") + ceConnectionRefused = (5, "ConnectionRefused") + ceProxyRefusedToConnect = (6, "ProxyRefusedToConnect") + ceFailedToResolveHost = (7, "FailedToResolveHost") + ceFailedToResolveProxy = (8, "FailedToResolveProxy") -converter toInt*(code: ConnectErrorCode): int = +const ErrorMessages* = [ + ceCGIOutputHandleNotFound: "request body output handle not found", + ceCGIFailedToOpenCacheOutput: "failed to open cache output", + ceCGICachedBodyNotFound: "cached request body not found", + ceFailedToRedirect: "failed to redirect request body", + ceURLNotInCache: "URL was not found in the cache", + ceFileNotInCache: "file was not found in the cache", + ceFailedToExecuteCGIScript: "failed to execute CGI script", + ceCGIMalformedHeader: "CGI script returned a malformed header", + ceCGIInvalidChaControl: "CGI got invalid Cha-Control header", + ceTooManyRewrites: "too many URI method map rewrites", + ceInvalidURIMethodEntry: "invalid URI method entry", + ceCGIFileNotFound: "CGI file not found", + ceInvalidCGIPath: "invalid CGI path", + ceFailedToSetUpCGI: "failed to set up CGI script", + ceNoCGIDir: "no local-CGI directory configured", + ceDisallowedURL: "url not allowed by filter", + ceUnknownScheme: "unknown scheme", + ceNone: "connection successful", + ceInternalError: "internal error", + ceInvalidMethod: "invalid method", + ceInvalidURL: "invalid URL", + ceFileNotFound: "file not found", + ceConnectionRefused: "connection refused", + ceProxyRefusedToConnect: "proxy refused to connect", + ceFailedToResolveHost: "failed to resolve host", + ceFailedToResolveProxy: "failed to resolve proxy", +] + +converter toInt*(code: ConnectionError): int = return int(code) func getLoaderErrorMessage*(code: int): string = - if code in int(ConnectErrorCode.low)..int(ConnectErrorCode.high): - return $ConnectErrorCode(code) + if code in int(ConnectionError.low)..int(ConnectionError.high): + return ErrorMessages[ConnectionError(code)] return "unexpected error code " & $code diff --git a/src/loader/loader.nim b/src/loader/loader.nim index 732336f9..9ebc6917 100644 --- a/src/loader/loader.nim +++ b/src/loader/loader.nim @@ -100,7 +100,7 @@ func canRewriteForCGICompat(ctx: LoaderContext; path: string): bool = return true return false -proc rejectHandle(handle: InputHandle; code: ConnectErrorCode; msg = "") = +proc rejectHandle(handle: InputHandle; code: ConnectionError; msg = "") = handle.sendResult(code, msg) handle.close() @@ -307,7 +307,7 @@ proc handleFirstLine(handle: InputHandle; line: string; headers: Headers; let k = line.until(':') if k.len == line.len: # invalid - handle.sendResult(ERROR_CGI_MALFORMED_HEADER) + handle.sendResult(ceCGIMalformedHeader) return crError let v = line.substr(k.len + 1).strip() if k.equalsIgnoreCase("Status"): @@ -321,10 +321,13 @@ proc handleFirstLine(handle: InputHandle; line: string; headers: Headers; elif v.startsWithIgnoreCase("ConnectionError"): let errs = v.split(' ') if errs.len <= 1: - handle.sendResult(ERROR_CGI_INVALID_CHA_CONTROL) + handle.sendResult(ceCGIInvalidChaControl) else: - let fb = int32(ERROR_CGI_INVALID_CHA_CONTROL) - let code = int(parseInt32(errs[1]).get(fb)) + var code = int32(ceCGIInvalidChaControl) + if (let x = parseInt32(errs[1]); x.isSome): + code = x.get + elif (let x = strictParseEnum[ConnectionError](errs[1]); x.isSome): + code = int32(x.get) var message = "" if errs.len > 2: message &= errs[2] @@ -335,7 +338,7 @@ proc handleFirstLine(handle: InputHandle; line: string; headers: Headers; return crError elif v.startsWithIgnoreCase("ControlDone"): return crDone - handle.sendResult(ERROR_CGI_INVALID_CHA_CONTROL) + handle.sendResult(ceCGIInvalidChaControl) return crError handle.sendResult(0) # success headers.add(k, v) @@ -584,20 +587,20 @@ proc parseCGIPath(ctx: LoaderContext; request: Request): CGIPath = proc loadCGI(ctx: LoaderContext; client: ClientData; handle: InputHandle; request: Request; prevURL: URL; config: LoaderClientConfig) = if ctx.config.cgiDir.len == 0: - handle.sendResult(ERROR_NO_CGI_DIR) + handle.sendResult(ceNoCGIDir) return let cpath = ctx.parseCGIPath(request) if cpath.cmd == "" or cpath.basename in ["", ".", ".."] or cpath.basename[0] == '~': - handle.sendResult(ERROR_INVALID_CGI_PATH) + handle.sendResult(ceInvalidCGIPath) return if not fileExists(cpath.cmd): - handle.sendResult(ERROR_CGI_FILE_NOT_FOUND) + handle.sendResult(ceCGIFileNotFound) return # Pipe the response body as stdout. var pipefd: array[2, cint] # child -> parent if pipe(pipefd) == -1: - handle.sendResult(ERROR_FAIL_SETUP_CGI) + handle.sendResult(ceFailedToSetUpCGI) return let istreamOut = newPosixStream(pipefd[0]) # read by loader var ostreamOut = newPosixStream(pipefd[1]) # written by child @@ -610,7 +613,7 @@ proc loadCGI(ctx: LoaderContext; client: ClientData; handle: InputHandle; # RDWR, otherwise mmap won't work ostreamOut = newPosixStream(tmpf, O_CREAT or O_RDWR, 0o600) if ostreamOut == nil: - handle.sendResult(ERROR_FAIL_SETUP_CGI) + handle.sendResult(ceCGIFailedToOpenCacheOutput) return let cacheId = handle.output.outputId # welp client.cacheMap.add(CachedItem( @@ -629,7 +632,7 @@ proc loadCGI(ctx: LoaderContext; client: ClientData; handle: InputHandle; var n: int (istream, n) = client.openCachedItem(request.body.cacheId) if istream == nil: - handle.sendResult(ERROR_CGI_CACHED_BODY_NOT_FOUND) + handle.sendResult(ceCGICachedBodyNotFound) return cachedHandle = ctx.findCachedHandle(request.body.cacheId) if cachedHandle != nil: # cached item still open, switch to streaming mode @@ -637,13 +640,13 @@ proc loadCGI(ctx: LoaderContext; client: ClientData; handle: InputHandle; elif request.body.t == rbtOutput: outputIn = ctx.findOutput(request.body.outputId, client) if outputIn == nil: - handle.sendResult(ERROR_FAIL_SETUP_CGI) + handle.sendResult(ceCGIOutputHandleNotFound) return if request.body.t in {rbtString, rbtMultipart, rbtOutput} or request.body.t == rbtCache and istream2 != nil: var pipefdRead: array[2, cint] # parent -> child if pipe(pipefdRead) == -1: - handle.sendResult(ERROR_FAIL_SETUP_CGI) + handle.sendResult(ceFailedToSetUpCGI) return istream = newPosixStream(pipefdRead[0]) ostream = newPosixStream(pipefdRead[1]) @@ -651,7 +654,7 @@ proc loadCGI(ctx: LoaderContext; client: ClientData; handle: InputHandle; stderr.flushFile() let pid = fork() if pid == -1: - handle.sendResult(ERROR_FAIL_SETUP_CGI) + handle.sendResult(ceFailedToSetUpCGI) elif pid == 0: istreamOut.sclose() # close read discard dup2(ostreamOut.fd, 1) # dup stdout @@ -679,7 +682,7 @@ proc loadCGI(ctx: LoaderContext; client: ClientData; handle: InputHandle; if ctx.handleMap[i] != nil: discard close(cint(i)) discard execl(cstring(cpath.cmd), cstring(cpath.basename), nil) - let code = int(ERROR_FAILED_TO_EXECUTE_CGI_SCRIPT) + let code = int(ceFailedToExecuteCGIScript) stdout.write("Cha-Control: ConnectionError " & $code & " " & ($strerror(errno)).deleteChars({'\n', '\r'})) quit(1) @@ -738,7 +741,7 @@ proc loadStream(ctx: LoaderContext; client: ClientData; handle: InputHandle; # not loading from cache, so cachedHandle is nil ctx.loadStreamRegular(handle, nil) do: - handle.sendResult(ERROR_FILE_NOT_FOUND, "stream not found") + handle.sendResult(ceFileNotFound, "stream not found") proc loadFromCache(ctx: LoaderContext; client: ClientData; handle: InputHandle; request: Request) = @@ -750,7 +753,7 @@ proc loadFromCache(ctx: LoaderContext; client: ClientData; handle: InputHandle; ps.seek(startFrom) handle.stream = ps if ps == nil: - handle.rejectHandle(ERROR_FILE_NOT_IN_CACHE) + handle.rejectHandle(ceFileNotInCache) client.cacheMap.del(n) return handle.sendResult(0) @@ -760,7 +763,7 @@ proc loadFromCache(ctx: LoaderContext; client: ClientData; handle: InputHandle; let cachedHandle = ctx.findCachedHandle(id) ctx.loadStreamRegular(handle, cachedHandle) else: - handle.sendResult(ERROR_URL_NOT_IN_CACHE) + handle.sendResult(ceURLNotInCache) # Data URL handler. # Moved back into loader from CGI, because data URLs can get extremely long @@ -796,7 +799,7 @@ proc loadData(ctx: LoaderContext; handle: InputHandle; request: Request) = let url = request.url var ct = url.path.s.until(',') if AllChars - Ascii + Controls - {'\t', ' '} in ct: - handle.sendResult(ERROR_INVALID_URL, "invalid data URL") + handle.sendResult(ceInvalidURL, "invalid data URL") handle.close() return let sd = ct.len + 1 # data start @@ -804,7 +807,7 @@ proc loadData(ctx: LoaderContext; handle: InputHandle; request: Request) = if ct.endsWith(";base64"): let d = atob0(body) # decode from ct end + 1 if d.isNone: - handle.sendResult(ERROR_INVALID_URL, "invalid data URL") + handle.sendResult(ceInvalidURL, "invalid data URL") handle.close() return ct.setLen(ct.len - ";base64".len) # remove base64 indicator @@ -853,11 +856,11 @@ proc loadResource(ctx: LoaderContext; client: ClientData; inc tries redo = true of ummrWrongURL: - handle.rejectHandle(ERROR_INVALID_URI_METHOD_ENTRY) + handle.rejectHandle(ceInvalidURIMethodEntry) of ummrNotFound: - handle.rejectHandle(ERROR_UNKNOWN_SCHEME) + handle.rejectHandle(ceUnknownScheme) if tries >= MaxRewrites: - handle.rejectHandle(ERROR_TOO_MANY_REWRITES) + handle.rejectHandle(ceTooManyRewrites) proc setupRequestDefaults(request: Request; config: LoaderClientConfig) = for k, v in config.defaultHeaders.table: @@ -880,7 +883,7 @@ proc load(ctx: LoaderContext; stream: SocketStream; request: Request; handle.url = request.url handle.output.url = request.url if not config.filter.match(request.url): - handle.rejectHandle(ERROR_DISALLOWED_URL) + handle.rejectHandle(ceDisallowedURL) else: request.setupRequestDefaults(config) ctx.loadResource(client, config, request, handle) |