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
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
|
# A file loader server (?)
# The idea here is that we receive requests with a socket, then respond to each
# with a response (ideally a document.)
# For now, the protocol looks like:
# C: Request
# S: res (0 => success, _ => error)
# if success:
# S: status code
# S: headers
# S: response body
#
# The body is passed to the stream as-is, so effectively nothing can follow it.
import nativesockets
import options
import streams
import tables
import net
when defined(posix):
import posix
import bindings/curl
import io/about
import io/file
import io/http
import io/request
import io/urlfilter
import ips/serialize
import ips/serversocket
import ips/socketstream
import types/cookie
import types/mime
import types/referer
import types/url
import utils/twtstr
type
FileLoader* = object
process*: Pid
LoaderCommand = enum
LOAD, QUIT
LoaderConfig* = object
defaultheaders*: HeaderList
filter*: URLFilter
cookiejar*: CookieJar
referrerpolicy*: ReferrerPolicy
proc loadResource(request: Request, ostream: Stream) =
case request.url.scheme
of "file":
loadFile(request.url, ostream)
of "http", "https":
loadHttp(request, ostream)
of "about":
loadAbout(request, ostream)
else:
ostream.swrite(-1) # error
ostream.flush()
var ssock: ServerSocket
proc runFileLoader*(fd: cint, config: LoaderConfig) =
if curl_global_init(CURL_GLOBAL_ALL) != CURLE_OK:
raise newException(Defect, "Failed to initialize libcurl.")
ssock = initServerSocket()
# The server has been initialized, so the main process can resume execution.
var writef: File
if not open(writef, FileHandle(fd), fmWrite):
raise newException(Defect, "Failed to open input handle.")
writef.write(char(0u8))
writef.flushFile()
close(writef)
discard close(fd)
onSignal SIGTERM, SIGINT:
curl_global_cleanup()
ssock.close()
quit(1)
while true:
let stream = ssock.acceptSocketStream()
try:
var cmd: LoaderCommand
stream.sread(cmd)
case cmd
of LOAD:
var request: Request
stream.sread(request)
if not config.filter.match(request.url):
stream.swrite(-1) # error
stream.flush()
else:
for k, v in config.defaultHeaders.table:
if k notin request.headers.table:
request.headers.table[k] = v
if config.cookiejar != nil and config.cookiejar.cookies.len > 0:
if "Cookie" notin request.headers.table:
let cookie = config.cookiejar.serialize(request.url)
if cookie != "":
request.headers["Cookie"] = cookie
if request.referer != nil and "Referer" notin request.headers.table:
let r = getReferer(request.referer, request.url, config.referrerpolicy)
if r != "":
request.headers["Referer"] = r
loadResource(request, stream)
stream.close()
of QUIT:
stream.close()
break
except EOFError:
# End-of-file, quit.
break
stream.close()
curl_global_cleanup()
ssock.close()
quit(0)
#TODO async requests...
proc doRequest*(loader: FileLoader, request: Request, blocking = true): Response =
new(result)
let stream = connectSocketStream(loader.process, false, blocking = true)
stream.swrite(LOAD)
stream.swrite(request)
stream.flush()
stream.sread(result.res)
if result.res == 0:
stream.sread(result.status)
stream.sread(result.headers)
if "Content-Type" in result.headers.table:
result.contenttype = result.headers.table["Content-Type"][0].until(';')
else:
result.contenttype = guessContentType($request.url.path)
if "Location" in result.headers.table:
let location = result.headers.table["Location"][0]
result.redirect = parseUrl(location, some(request.url))
# Only a stream of the response body may arrive after this point.
result.body = stream
if not blocking:
stream.source.getFd().setBlocking(blocking)
proc quit*(loader: FileLoader) =
let stream = connectSocketStream(loader.process)
if stream != nil:
stream.swrite(QUIT)
|