import options import strutils import bindings/curl import io/headers import io/loaderhandle import io/request import types/blob import types/formdata import types/url import utils/opt import utils/twtstr type CurlHandle* = ref CurlHandleObj CurlHandleObj = object curl*: CURL statusline: bool headers: Headers request: Request handle*: LoaderHandle mime: curl_mime slist: curl_slist func newCurlHandle(curl: CURL, request: Request, handle: LoaderHandle): CurlHandle = return CurlHandle( headers: newHeaders(), curl: curl, handle: handle, request: request ) proc cleanup*(handleData: CurlHandle) = handleData.handle.close() if handleData.mime != nil: curl_mime_free(handleData.mime) if handleData.slist != nil: curl_slist_free_all(handleData.slist) curl_easy_cleanup(handleData.curl) template setopt(curl: CURL, opt: CURLoption, arg: typed) = discard curl_easy_setopt(curl, opt, arg) template setopt(curl: CURL, opt: CURLoption, arg: string) = discard curl_easy_setopt(curl, opt, cstring(arg)) template getinfo(curl: CURL, info: CURLINFO, arg: typed) = discard curl_easy_getinfo(curl, info, arg) proc curlWriteHeader(p: cstring, size: csize_t, nitems: csize_t, userdata: pointer): csize_t {.cdecl.} = var line = newString(nitems) for i in 0.. error) if not op.handle.sendHeaders(op.headers): return 0 return nitems let v = line.substr(k.len + 1).strip() op.headers.add(k, v) return nitems # From the documentation: size is always 1. proc curlWriteBody(p: cstring, size: csize_t, nmemb: csize_t, userdata: pointer): csize_t {.cdecl.} = let handleData = cast[CurlHandle](userdata) if nmemb > 0: if not handleData.handle.sendData(p, int(nmemb)): return 0 return nmemb proc applyPostBody(curl: CURL, request: Request, handleData: CurlHandle) = if request.multipart.isOk: handleData.mime = curl_mime_init(curl) doAssert handleData.mime != nil for entry in request.multipart.get: let part = curl_mime_addpart(handleData.mime) doAssert part != nil curl_mime_name(part, cstring(entry.name)) if entry.isstr: curl_mime_data(part, cstring(entry.svalue), csize_t(entry.svalue.len)) else: let blob = entry.value if blob.isfile: #TODO ? curl_mime_filedata(part, cstring(WebFile(blob).path)) else: curl_mime_data(part, blob.buffer, csize_t(blob.size)) # may be overridden by curl_mime_filedata, so set it here curl_mime_filename(part, cstring(entry.filename)) curl.setopt(CURLOPT_MIMEPOST, handleData.mime) elif request.body.issome: curl.setopt(CURLOPT_POSTFIELDS, cstring(request.body.get)) curl.setopt(CURLOPT_POSTFIELDSIZE, request.body.get.len) proc loadHttp*(handle: LoaderHandle, curlm: CURLM, request: Request): CurlHandle = let curl = curl_easy_init() doAssert curl != nil let surl = request.url.serialize() curl.setopt(CURLOPT_URL, surl) let handleData = curl.newCurlHandle(request, handle) curl.setopt(CURLOPT_WRITEDATA, handleData) curl.setopt(CURLOPT_WRITEFUNCTION, curlWriteBody) curl.setopt(CURLOPT_HEADERDATA, handleData) curl.setopt(CURLOPT_HEADERFUNCTION, curlWriteHeader) if request.proxy != nil: let purl = request.proxy.serialize() curl.setopt(CURLOPT_PROXY, purl) case request.httpmethod of HTTP_GET: curl.setopt(CURLOPT_HTTPGET, 1) of HTTP_POST: curl.setopt(CURLOPT_POST, 1) curl.applyPostBody(request, handleData) else: discard #TODO for k, v in request.headers: let header = k & ": " & v handleData.slist = curl_slist_append(handleData.slist, cstring(header)) if handleData.slist != nil: curl.setopt(CURLOPT_HTTPHEADER, handleData.slist) let res = curl_multi_add_handle(curlm, curl) if res != CURLM_OK: discard handle.sendResult(int(res)) return nil return handleData