about summary refs log tree commit diff stats
path: root/adapter/protocol/gopher.nim
blob: 14e34b7aa56cebd4762291ec93c8acbaced73514 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
when NimMajor >= 2:
  import std/envvars
else:
  import std/os

import curl
import curlerrors
import curlwrap

import ../gophertypes

import loader/connecterror
import utils/twtstr

type GopherHandle = ref object
  curl: CURL
  t: GopherType
  statusline: bool

proc onStatusLine(op: GopherHandle) =
  let s = case op.t
  of gtDirectory, gtSearch: "Content-Type: text/gopher\n"
  of gtHTML: "Content-Type: text/html\n"
  of gtGif: "Content-Type: image/gif\n"
  of gtPng: "Content-Type: image/png\n"
  of gtTextFile, gtError: "Content-Type: text/plain\n"
  else: ""
  stdout.write(s & "\n")

proc loadSearch(op: GopherHandle; surl: string) =
  stdout.write("""
Content-Type: text/html

<!DOCTYPE HTML>
<HTML>
<HEAD>
<BASE HREF="""" & surl & """">
</HEAD>
<BODY>
<H1>Search """ & htmlEscape(surl) & """</H1>
<FORM>
<INPUT TYPE=SEARCH NAME="NAME">
</FORM>
</BODY>
</HTML>
""")

# From the documentation: size is always 1.
proc curlWriteBody(p: cstring; size, nmemb: csize_t; userdata: pointer):
    csize_t {.cdecl.} =
  let op = cast[GopherHandle](userdata)
  if not op.statusline:
    op.statusline = true
    op.onStatusLine()
  return csize_t(stdout.writeBuffer(p, int(nmemb)))

proc main() =
  let curl = curl_easy_init()
  doAssert curl != nil
  if getEnv("REQUEST_METHOD") != "GET":
    stdout.write("Cha-Control: ConnectionError " & $int(ERROR_INVALID_METHOD))
    return
  var path = getEnv("MAPPED_URI_PATH")
  if path.len < 1:
    path &= '/'
  if path.len < 2:
    path &= '1'
  let url = curl_url()
  const flags = cuint(CURLU_PATH_AS_IS)
  url.set(CURLUPART_SCHEME, getEnv("MAPPED_URI_SCHEME"), flags)
  url.set(CURLUPART_HOST, getEnv("MAPPED_URI_HOST"), flags)
  let port = getEnv("MAPPED_URI_PORT")
  if port != "":
    url.set(CURLUPART_PORT, port, flags)
  url.set(CURLUPART_PATH, path, flags)
  let query = getEnv("MAPPED_URI_QUERY")
  if query != "":
    url.set(CURLUPART_QUERY, query.after('='), flags)
  let op = GopherHandle(
    curl: curl,
    t: gopherType(path[1])
  )
  if op.t == gtSearch and query == "":
    const flags = cuint(CURLU_PUNY2IDN)
    let surl = url.get(CURLUPART_URL, flags)
    if surl == nil:
      stdout.write("Cha-Control: ConnectionError " & $int(ERROR_INVALID_URL))
    else:
      op.loadSearch($surl)
  else:
    curl.setopt(CURLOPT_CURLU, url)
    curl.setopt(CURLOPT_WRITEDATA, op)
    curl.setopt(CURLOPT_WRITEFUNCTION, curlWriteBody)
    let proxy = getEnv("ALL_PROXY")
    if proxy != "":
      curl.setopt(CURLOPT_PROXY, proxy)
    let res = curl_easy_perform(curl)
    if res != CURLE_OK and not op.statusline:
      stdout.write(getCurlConnectionError(res))
  curl_easy_cleanup(curl)

main()