diff options
-rw-r--r-- | adapter/img/jebp.nim | 67 | ||||
-rw-r--r-- | adapter/img/stbi.nim | 107 | ||||
-rw-r--r-- | adapter/protocol/http.nim | 18 |
3 files changed, 131 insertions, 61 deletions
diff --git a/adapter/img/jebp.nim b/adapter/img/jebp.nim index ec3c9073..03d72d59 100644 --- a/adapter/img/jebp.nim +++ b/adapter/img/jebp.nim @@ -1,5 +1,6 @@ import std/options import std/os +import std/posix import std/strutils import utils/sandbox @@ -17,6 +18,9 @@ else: {.passc: "-I" & currentSourcePath().parentDir().} +const STDIN_FILENO = 0 +const STDOUT_FILENO = 1 + {.push header: "jebp.h".} type jebp_io_callbacks {.importc.} = object @@ -48,13 +52,36 @@ proc jebp_free_image(image: ptr jebp_image_t) {.importc.} {.pop.} proc myRead(data: pointer; size: csize_t; user: pointer): csize_t {.cdecl.} = - return csize_t(stdin.readBuffer(data, size)) + var n = csize_t(0) + while n < size: + let i = read(STDIN_FILENO, addr cast[ptr UncheckedArray[char]](data)[n], + int(size - n)) + if i == 0: + break + n += csize_t(i) + return n proc stbir_resize_uint8(input_pixels: ptr uint8; input_w, input_h, input_stride_in_bytes: cint; output_pixels: ptr uint8; output_w, output_h, output_stride_in_bytes, num_channels: cint): cint {.importc.} +proc writeAll(data: pointer; size: int) = + var n = 0 + while n < size: + let i = write(STDOUT_FILENO, addr cast[ptr UncheckedArray[uint8]](data)[n], + int(size) - n) + assert i >= 0 + n += i + +proc puts(s: string) = + if s.len > 0: + writeAll(unsafeAddr s[0], s.len) + +proc die(s: string) {.noreturn.} = + puts(s) + quit(1) + proc main() = enterNetworkSandbox() let scheme = getEnv("MAPPED_URI_SCHEME") @@ -62,8 +89,7 @@ proc main() = case getEnv("MAPPED_URI_PATH") of "decode": if f != "webp": - stdout.write("Cha-Control: ConnectionError 1 unknown format " & f) - return + die("Cha-Control: ConnectionError 1 unknown format " & f) let headers = getEnv("REQUEST_HEADERS") var targetWidth = cint(-1) var targetHeight = cint(-1) @@ -76,13 +102,11 @@ proc main() = of "Cha-Image-Target-Dimensions": let s = v.split('x') if s.len != 2: - stdout.write("Cha-Control: ConnectionError 1 wrong dimensions") - return + die("Cha-Control: ConnectionError 1 wrong dimensions\n") let w = parseUInt32(s[0], allowSign = false) let h = parseUInt32(s[1], allowSign = false) if w.isNone or w.isNone: - stdout.write("Cha-Control: ConnectionError 1 wrong dimensions") - return + die("Cha-Control: ConnectionError 1 wrong dimensions\n") targetWidth = cint(w.get) targetHeight = cint(h.get) var image = jebp_image_t() @@ -90,32 +114,33 @@ proc main() = if infoOnly: let res = jebp_read_size_from_callbacks(addr image, addr cb, nil) if res == 0: - stdout.write("Cha-Image-Dimensions: " & $image.width & "x" & + puts("Cha-Image-Dimensions: " & $image.width & "x" & $image.height & "\n\n") + quit(0) else: - stdout.write("Cha-Control: ConnectionError 1 jepb error " & + die("Cha-Control: ConnectionError 1 jepb error " & $jebp_error_string(res)) - return let res = jebp_read_from_callbacks(addr image, addr cb, nil) if res != 0: - stdout.write("Cha-Control: ConnectionError 1 jebp error " & + die("Cha-Control: ConnectionError 1 jebp error " & $jebp_error_string(res)) elif targetWidth != -1 and targetHeight != -1: - let p2 = cast[ptr uint8](alloc(targetWidth * targetHeight * 4)) + let hdr = "Cha-Image-Dimensions: " & $targetWidth & "x" & $targetHeight & + "\n\n" + let p2 = cast[ptr UncheckedArray[uint8]](alloc(hdr.len + + targetWidth * targetHeight * 4)) + copyMem(addr p2[0], unsafeAddr hdr[0], hdr.len) doAssert stbir_resize_uint8(cast[ptr uint8](image.pixels), image.width, - image.height, 0, p2, targetWidth, targetHeight, 0, 4) == 1 - stdout.write("Cha-Image-Dimensions: " & $targetWidth & "x" & - $targetHeight & "\n\n") - discard stdout.writeBuffer(p2, targetWidth * targetHeight * 4) + image.height, 0, addr p2[hdr.len], targetWidth, targetHeight, 0, 4) == 1 + writeAll(p2, hdr.len + targetWidth * targetHeight * 4) dealloc(p2) jebp_free_image(addr image) else: - stdout.write("Cha-Image-Dimensions: " & $image.width & "x" & - $image.height & "\n\n") - discard stdout.writeBuffer(cast[ptr uint8](image.pixels), image.width * - image.height * 4) + puts("Cha-Image-Dimensions: " & $image.width & "x" & $image.height & + "\n\n") + writeAll(image.pixels, image.width * image.height * 4) jebp_free_image(addr image) of "encode": - stdout.write("Cha-Control: ConnectionError 1 not supported") + die("Cha-Control: ConnectionError 1 not supported") main() diff --git a/adapter/img/stbi.nim b/adapter/img/stbi.nim index f4a5faa2..8a1442f2 100644 --- a/adapter/img/stbi.nim +++ b/adapter/img/stbi.nim @@ -1,5 +1,6 @@ import std/options import std/os +import std/posix import std/strutils import utils/sandbox @@ -30,18 +31,35 @@ proc stbi_image_free(retval_from_stbi_load: pointer) {.importc.} {.pop.} -proc myRead(user: pointer; data: ptr char; size: cint): cint {.cdecl.} = - return cint(stdin.readBuffer(data, size)) +type StbiUser = object + atEof: bool + +const STDIN_FILENO = 0 +const STDOUT_FILENO = 1 -proc mySkip(user: pointer; n: cint) {.cdecl.} = +proc myRead(user: pointer; data: ptr char; size: cint): cint {.cdecl.} = + var n = cint(0) + while n < size: + let i = read(STDIN_FILENO, addr cast[ptr UncheckedArray[char]](data)[n], + int(size - n)) + if i == 0: + cast[ptr StbiUser](user)[].atEof = true + break + n += cint(i) + return n + +proc mySkip(user: pointer; size: cint) {.cdecl.} = var data: array[4096, uint8] - let n = int(n) - var i = 0 - while i < n: - i += stdin.readBuffer(addr data[0], min(n - i, data.len)) + var n = cint(0) + while n < size: + let i = read(STDIN_FILENO, addr data[0], min(int(size - n), data.len)) + if i == 0: + cast[ptr StbiUser](user)[].atEof = true + break + n += cint(i) proc myEof(user: pointer): cint {.cdecl.} = - return cint(stdin.endOfFile()) + return cint(cast[ptr StbiUser](user)[].atEof) type stbi_write_func = proc(context, data: pointer; size: cint) {.cdecl.} @@ -54,8 +72,16 @@ proc stbi_write_jpg_to_func(fun: stbi_write_func; context: pointer; w, h, comp: cint; data: pointer; quality: cint) {.importc.} {.pop.} +proc writeAll(data: pointer; size: int) = + var n = 0 + while n < size: + let i = write(STDOUT_FILENO, addr cast[ptr UncheckedArray[uint8]](data)[n], + int(size) - n) + assert i >= 0 + n += i + proc myWriteFunc(context, data: pointer; size: cint) {.cdecl.} = - discard stdout.writeBuffer(data, size) + writeAll(data, int(size)) {.push header: "stb_image_resize.h".} proc stbir_resize_uint8(input_pixels: ptr uint8; @@ -64,6 +90,14 @@ proc stbir_resize_uint8(input_pixels: ptr uint8; {.importc.} {.pop.} +proc puts(s: string) = + if s.len > 0: + writeAll(unsafeAddr s[0], s.len) + +proc die(s: string) {.noreturn.} = + puts(s) + quit(1) + proc main() = enterNetworkSandbox() let scheme = getEnv("MAPPED_URI_SCHEME") @@ -71,8 +105,8 @@ proc main() = case getEnv("MAPPED_URI_PATH") of "decode": if f notin ["jpeg", "gif", "bmp", "png", "x-unknown"]: - stdout.write("Cha-Control: ConnectionError 1 unknown format " & f) - return + die("Cha-Control: ConnectionError 1 unknown format " & f) + var user = StbiUser() var x: cint var y: cint var channels_in_file: cint @@ -93,38 +127,40 @@ proc main() = of "Cha-Image-Target-Dimensions": let s = v.split('x') if s.len != 2: - stdout.write("Cha-Control: ConnectionError 1 wrong dimensions") - return + die("Cha-Control: ConnectionError 1 wrong dimensions\n") let w = parseUInt32(s[0], allowSign = false) let h = parseUInt32(s[1], allowSign = false) if w.isNone or w.isNone: - stdout.write("Cha-Control: ConnectionError 1 wrong dimensions") - return + die("Cha-Control: ConnectionError 1 wrong dimensions\n") targetWidth = cint(w.get) targetHeight = cint(h.get) if infoOnly: - if stbi_info_from_callbacks(addr clbk, nil, x, y, channels_in_file) == 1: - stdout.write("Cha-Image-Dimensions: " & $x & "x" & $y & "\n\n") + if stbi_info_from_callbacks(addr clbk, addr user, x, y, + channels_in_file) == 1: + puts("Cha-Image-Dimensions: " & $x & "x" & $y & "\n\n") + quit(0) else: - stdout.write("Cha-Control: ConnectionError 1 stbi error " & + die("Cha-Control: ConnectionError 1 stbi error " & $stbi_failure_reason()) - return - let p = stbi_load_from_callbacks(addr clbk, nil, x, y, channels_in_file, 4) + let p = stbi_load_from_callbacks(addr clbk, addr user, x, y, + channels_in_file, 4) if p == nil: - stdout.write("Cha-Control: ConnectionError 1 stbi error " & + die("Cha-Control: ConnectionError 1 stbi error " & $stbi_failure_reason()) elif targetWidth != -1 and targetHeight != -1: - let p2 = cast[ptr uint8](alloc(targetWidth * targetHeight * 4)) - doAssert stbir_resize_uint8(p, x, y, 0, p2, targetWidth, targetHeight, 0, - 4) == 1 - stdout.write("Cha-Image-Dimensions: " & $targetWidth & "x" & - $targetHeight & "\n\n") - discard stdout.writeBuffer(p2, targetWidth * targetHeight * 4) + let hdr = "Cha-Image-Dimensions: " & $targetWidth & "x" & + $targetHeight & "\n\n" + let p2 = cast[ptr UncheckedArray[uint8]](alloc(hdr.len + + targetWidth * targetHeight * 4)) + copyMem(addr p2[0], unsafeAddr hdr[0], hdr.len) + doAssert stbir_resize_uint8(p, x, y, 0, addr p2[hdr.len], targetWidth, + targetHeight, 0, 4) == 1 + writeAll(p2, hdr.len + targetWidth * targetHeight * 4) dealloc(p2) stbi_image_free(p) else: - stdout.write("Cha-Image-Dimensions: " & $x & "x" & $y & "\n\n") - discard stdout.writeBuffer(p, x * y * 4) + puts("Cha-Image-Dimensions: " & $x & "x" & $y & "\n\n") + writeAll(p, x * y * 4) stbi_image_free(p) of "encode": let headers = getEnv("REQUEST_HEADERS") @@ -138,22 +174,19 @@ proc main() = let w = parseUInt32(s[0], allowSign = false) let h = parseUInt32(s[1], allowSign = false) if w.isNone or w.isNone: - stdout.write("Cha-Control: ConnectionError 1 wrong dimensions") - return + die("Cha-Control: ConnectionError 1 wrong dimensions") width = cint(w.get) height = cint(h.get) of "Cha-Image-Quality": let s = hdr.after(':').strip() let q = parseUInt32(s, allowSign = false).get(101) if q < 1 or 100 < q: - stdout.write("Cha-Control: ConnectionError 1 wrong quality") - return + die("Cha-Control: ConnectionError 1 wrong quality") quality = cint(q) let s = stdin.readAll() if s.len != width * height * 4: - stdout.write("Cha-Control: ConnectionError 1 wrong size") - return - stdout.write("Cha-Image-Dimensions: " & $width & 'x' & $height & "\n\n") + die("Cha-Control: ConnectionError 1 wrong size") + puts("Cha-Image-Dimensions: " & $width & 'x' & $height & "\n\n") let p = unsafeAddr s[0] case f of "png": @@ -165,6 +198,6 @@ proc main() = stbi_write_jpg_to_func(myWriteFunc, nil, cint(width), cint(height), 4, p, quality) else: - stdout.write("Cha-Control: ConnectionError 1 unknown format " & f) + die("Cha-Control: ConnectionError 1 unknown format " & f) main() diff --git a/adapter/protocol/http.nim b/adapter/protocol/http.nim index d00479e4..69542288 100644 --- a/adapter/protocol/http.nim +++ b/adapter/protocol/http.nim @@ -23,8 +23,20 @@ type earlyhint: EarlyHintState slist: curl_slist +const STDIN_FILENO = 0 +const STDOUT_FILENO = 1 + +proc writeAll(data: pointer; size: int) = + var n = 0 + while n < size: + let i = write(STDOUT_FILENO, addr cast[ptr UncheckedArray[uint8]](data)[n], + int(size) - n) + assert i >= 0 + n += i + proc puts(s: string) = - discard write(1, unsafeAddr s[0], s.len) + if s.len > 0: + writeAll(unsafeAddr s[0], s.len) proc curlWriteHeader(p: cstring; size, nitems: csize_t; userdata: pointer): csize_t {.cdecl.} = @@ -63,12 +75,12 @@ proc curlWriteHeader(p: cstring; size, nitems: csize_t; userdata: pointer): # From the documentation: size is always 1. proc curlWriteBody(p: cstring; size, nmemb: csize_t; userdata: pointer): csize_t {.cdecl.} = - return csize_t(write(stdout.getFileHandle(), p, int(nmemb))) + return csize_t(write(STDOUT_FILENO, p, int(nmemb))) # From the documentation: size is always 1. proc readFromStdin(p: pointer; size, nitems: csize_t; userdata: pointer): csize_t {.cdecl.} = - return csize_t(read(0, p, int(nitems))) + return csize_t(read(STDIN_FILENO, p, int(nitems))) proc curlPreRequest(clientp: pointer; conn_primary_ip, conn_local_ip: cstring; conn_primary_port, conn_local_port: cint): cint {.cdecl.} = |