-rw-r--r-- | src/Makefile.am | 4 | ||||
-rw-r--r-- | src/cgi_gateway.cc | 4 | ||||
-rw-r--r-- | src/cookies.cc | 242 | ||||
-rw-r--r-- | src/headers.cc | 40 | ||||
-rw-r--r-- | src/util.cc | 53 |
5 files changed, 342 insertions, 1 deletions
diff --git a/src/Makefile.am b/src/Makefile.am index 12bb1f8..2d462c8 100644 --- a/src/Makefile.am +++ b/src/Makefile.am | |||
@@ -1,25 +1,27 @@ | |||
1 | lib_LTLIBRARIES = libkingate.la libkingate-plaincgi.la | 1 | lib_LTLIBRARIES = libkingate.la libkingate-plaincgi.la |
2 | 2 | ||
3 | if HAVE_FCGI | 3 | if HAVE_FCGI |
4 | lib_LTLIBRARIES += libkingate-fcgi.la | 4 | lib_LTLIBRARIES += libkingate-fcgi.la |
5 | endif | 5 | endif |
6 | 6 | ||
7 | INCLUDES = -I${top_srcdir}/include -I${top_srcdir} | 7 | INCLUDES = -I${top_srcdir}/include -I${top_srcdir} |
8 | AM_CXXFLAGS = ${KONFORKA_CFLAGS} | 8 | AM_CXXFLAGS = ${KONFORKA_CFLAGS} |
9 | LDADD = ${KONFORKA_LIBS} | 9 | LDADD = ${KONFORKA_LIBS} |
10 | 10 | ||
11 | libkingate_la_SOURCES = \ | 11 | libkingate_la_SOURCES = \ |
12 | cgi_gateway.cc \ | 12 | cgi_gateway.cc \ |
13 | cgi_interface.cc \ | 13 | cgi_interface.cc \ |
14 | util.cc | 14 | util.cc \ |
15 | cookies.cc \ | ||
16 | headers.cc | ||
15 | libkingate_la_LDFLAGS = -version-info 2:0:0 | 17 | libkingate_la_LDFLAGS = -version-info 2:0:0 |
16 | 18 | ||
17 | libkingate_fcgi_la_SOURCES = \ | 19 | libkingate_fcgi_la_SOURCES = \ |
18 | fastcgi.cc | 20 | fastcgi.cc |
19 | libkingate_fcgi_la_LDFLAGS = -version-info 1:0:0 | 21 | libkingate_fcgi_la_LDFLAGS = -version-info 1:0:0 |
20 | 22 | ||
21 | libkingate_plaincgi_la_SOURCES = \ | 23 | libkingate_plaincgi_la_SOURCES = \ |
22 | plaincgi.cc | 24 | plaincgi.cc |
23 | libkingate_plaincgi_la_LDFLAGS = -version-info 1:0:0 | 25 | libkingate_plaincgi_la_LDFLAGS = -version-info 1:0:0 |
24 | 26 | ||
25 | EXTRA_DIST = ${libkingate_fcgi_la_SOURCES} | 27 | EXTRA_DIST = ${libkingate_fcgi_la_SOURCES} |
diff --git a/src/cgi_gateway.cc b/src/cgi_gateway.cc index 30410f2..ab48f78 100644 --- a/src/cgi_gateway.cc +++ b/src/cgi_gateway.cc | |||
@@ -1,241 +1,245 @@ | |||
1 | #include <errno.h> | 1 | #include <errno.h> |
2 | #include <ctype.h> | 2 | #include <ctype.h> |
3 | #include "kingate/cgi_gateway.h" | 3 | #include "kingate/cgi_gateway.h" |
4 | #include "kingate/util.h" | 4 | #include "kingate/util.h" |
5 | #include "kingate/exception.h" | 5 | #include "kingate/exception.h" |
6 | 6 | ||
7 | namespace kingate { | 7 | namespace kingate { |
8 | 8 | ||
9 | static string empty_string; | 9 | static string empty_string; |
10 | 10 | ||
11 | cgi_gateway::cgi_gateway(cgi_interface& ci) | 11 | cgi_gateway::cgi_gateway(cgi_interface& ci) |
12 | : iface(ci), b_parsed_content(false) { | 12 | : iface(ci), b_parsed_content(false) { |
13 | // Fetch GET content | 13 | // Fetch GET content |
14 | try { | 14 | try { |
15 | string qs = get_meta("QUERY_STRING"); | 15 | string qs = get_meta("QUERY_STRING"); |
16 | parse_query(qs,get); | 16 | parse_query(qs,get); |
17 | }catch(exception_notfound& enf) { } | 17 | }catch(exception_notfound& enf) { } |
18 | // Fetch POST content | 18 | // Fetch POST content |
19 | if(!strcasecmp(content_type().c_str(),"application/x-www-form-urlencoded")) { | 19 | if(!strcasecmp(content_type().c_str(),"application/x-www-form-urlencoded")) { |
20 | unsigned long cl = content_length(); | 20 | unsigned long cl = content_length(); |
21 | if(cl) { | 21 | if(cl) { |
22 | char * tmp = new char[cl]; | 22 | char * tmp = new char[cl]; |
23 | iface.in().read(tmp,cl); | 23 | iface.in().read(tmp,cl); |
24 | string qs(tmp,cl); | 24 | string qs(tmp,cl); |
25 | delete tmp; | 25 | delete tmp; |
26 | parse_query(qs,post); | 26 | parse_query(qs,post); |
27 | } | 27 | } |
28 | b_parsed_content = true; | 28 | b_parsed_content = true; |
29 | } | 29 | } |
30 | // Parse cookies | ||
31 | try { | ||
32 | cookies.parse_cookies(get_meta("HTTP_COOKIE")); | ||
33 | }catch(exception_notfound& enf) { } | ||
30 | } | 34 | } |
31 | 35 | ||
32 | bool cgi_gateway::has_GET(const string& n) const { | 36 | bool cgi_gateway::has_GET(const string& n) const { |
33 | return get.find(n) != get.end(); | 37 | return get.find(n) != get.end(); |
34 | } | 38 | } |
35 | const string& cgi_gateway::get_GET(const string& n) const { | 39 | const string& cgi_gateway::get_GET(const string& n) const { |
36 | params_t::const_iterator i = get.find(n); | 40 | params_t::const_iterator i = get.find(n); |
37 | if(i==get.end()) | 41 | if(i==get.end()) |
38 | throw exception_notfound(CODEPOINT,"no such parameter"); | 42 | throw exception_notfound(CODEPOINT,"no such parameter"); |
39 | return i->second; | 43 | return i->second; |
40 | } | 44 | } |
41 | bool cgi_gateway::has_POST(const string& n) const { | 45 | bool cgi_gateway::has_POST(const string& n) const { |
42 | return post.find(n) != post.end(); | 46 | return post.find(n) != post.end(); |
43 | } | 47 | } |
44 | const string& cgi_gateway::get_POST(const string& n) const { | 48 | const string& cgi_gateway::get_POST(const string& n) const { |
45 | params_t::const_iterator i = post.find(n); | 49 | params_t::const_iterator i = post.find(n); |
46 | if(i==post.end()) | 50 | if(i==post.end()) |
47 | throw exception_notfound(CODEPOINT,"no such parameter"); | 51 | throw exception_notfound(CODEPOINT,"no such parameter"); |
48 | return i->second; | 52 | return i->second; |
49 | } | 53 | } |
50 | bool cgi_gateway::has_param(const string& n) const { | 54 | bool cgi_gateway::has_param(const string& n) const { |
51 | return has_GET(n) || has_POST(n); | 55 | return has_GET(n) || has_POST(n); |
52 | } | 56 | } |
53 | const string& cgi_gateway::get_param(const string& n) const { | 57 | const string& cgi_gateway::get_param(const string& n) const { |
54 | params_t::const_iterator i = get.find(n); | 58 | params_t::const_iterator i = get.find(n); |
55 | if(i!=get.end()) | 59 | if(i!=get.end()) |
56 | return i->second; | 60 | return i->second; |
57 | i = post.find(n); | 61 | i = post.find(n); |
58 | if(i!=post.end()) | 62 | if(i!=post.end()) |
59 | return i->second; | 63 | return i->second; |
60 | throw exception_notfound(CODEPOINT,"no such parameter"); | 64 | throw exception_notfound(CODEPOINT,"no such parameter"); |
61 | } | 65 | } |
62 | 66 | ||
63 | /* | 67 | /* |
64 | * deprecated stuff. | 68 | * deprecated stuff. |
65 | */ | 69 | */ |
66 | const string& cgi_gateway::get_content_type() const { | 70 | const string& cgi_gateway::get_content_type() const { |
67 | if(!has_meta("CONTENT_TYPE")) | 71 | if(!has_meta("CONTENT_TYPE")) |
68 | return empty_string; | 72 | return empty_string; |
69 | return get_meta("CONTENT_TYPE"); | 73 | return get_meta("CONTENT_TYPE"); |
70 | } | 74 | } |
71 | unsigned long cgi_gateway::get_content_length() const { | 75 | unsigned long cgi_gateway::get_content_length() const { |
72 | if(!has_meta("CONTENT_LENGTH")) | 76 | if(!has_meta("CONTENT_LENGTH")) |
73 | return 0; | 77 | return 0; |
74 | string cl = get_meta("CONTENT_LENGTH"); | 78 | string cl = get_meta("CONTENT_LENGTH"); |
75 | return strtol(cl.c_str(),NULL,10); | 79 | return strtol(cl.c_str(),NULL,10); |
76 | } | 80 | } |
77 | /* | 81 | /* |
78 | * | 82 | * |
79 | */ | 83 | */ |
80 | 84 | ||
81 | const string& cgi_gateway::http_request_header(const string& hn) const { | 85 | const string& cgi_gateway::http_request_header(const string& hn) const { |
82 | string mvn = "HTTP_"; | 86 | string mvn = "HTTP_"; |
83 | for(const char* p=hn.c_str();*p;p++) { | 87 | for(const char* p=hn.c_str();*p;p++) { |
84 | if(*p=='-') | 88 | if(*p=='-') |
85 | mvn += '_'; | 89 | mvn += '_'; |
86 | else | 90 | else |
87 | mvn += toupper(*p); | 91 | mvn += toupper(*p); |
88 | } | 92 | } |
89 | return get_meta(mvn); | 93 | return get_meta(mvn); |
90 | } | 94 | } |
91 | 95 | ||
92 | const string& cgi_gateway::auth_type() const { | 96 | const string& cgi_gateway::auth_type() const { |
93 | try { | 97 | try { |
94 | return get_meta("AUTH_TYPE"); | 98 | return get_meta("AUTH_TYPE"); |
95 | }catch(exception_notfound& enf) { | 99 | }catch(exception_notfound& enf) { |
96 | return empty_string; | 100 | return empty_string; |
97 | } | 101 | } |
98 | } | 102 | } |
99 | unsigned long cgi_gateway::content_length() const { | 103 | unsigned long cgi_gateway::content_length() const { |
100 | try { | 104 | try { |
101 | const string& cl = get_meta("CONTENT_LENGTH"); | 105 | const string& cl = get_meta("CONTENT_LENGTH"); |
102 | errno = 0; | 106 | errno = 0; |
103 | const char *clp = cl.c_str(); | 107 | const char *clp = cl.c_str(); |
104 | unsigned long rv = strtol(clp,(char**)&clp,10); | 108 | unsigned long rv = strtol(clp,(char**)&clp,10); |
105 | if(errno || *clp) | 109 | if(errno || *clp) |
106 | throw server_error(CODEPOINT,"Invalid CONTENT_LENGTH value passed from server"); | 110 | throw server_error(CODEPOINT,"Invalid CONTENT_LENGTH value passed from server"); |
107 | return rv; | 111 | return rv; |
108 | }catch(exception_notfound& enf) { | 112 | }catch(exception_notfound& enf) { |
109 | return 0; | 113 | return 0; |
110 | } | 114 | } |
111 | } | 115 | } |
112 | const string& cgi_gateway::content_type() const { | 116 | const string& cgi_gateway::content_type() const { |
113 | try { | 117 | try { |
114 | return get_meta("CONTENT_TYPE"); | 118 | return get_meta("CONTENT_TYPE"); |
115 | }catch(exception_notfound& enf) { | 119 | }catch(exception_notfound& enf) { |
116 | return empty_string; | 120 | return empty_string; |
117 | } | 121 | } |
118 | } | 122 | } |
119 | const string& cgi_gateway::gateway_interface() const { | 123 | const string& cgi_gateway::gateway_interface() const { |
120 | try { | 124 | try { |
121 | return get_meta("GATEWAY_INTERFACE"); | 125 | return get_meta("GATEWAY_INTERFACE"); |
122 | }catch(exception_notfound& enf) { | 126 | }catch(exception_notfound& enf) { |
123 | return empty_string; | 127 | return empty_string; |
124 | } | 128 | } |
125 | } | 129 | } |
126 | const string& cgi_gateway::path_info() const { | 130 | const string& cgi_gateway::path_info() const { |
127 | try { | 131 | try { |
128 | return get_meta("PATH_INFO"); | 132 | return get_meta("PATH_INFO"); |
129 | }catch(exception_notfound& enf) { | 133 | }catch(exception_notfound& enf) { |
130 | return empty_string; | 134 | return empty_string; |
131 | } | 135 | } |
132 | } | 136 | } |
133 | const string& cgi_gateway::path_translated() const { | 137 | const string& cgi_gateway::path_translated() const { |
134 | try { | 138 | try { |
135 | return get_meta("PATH_TRANSLATED"); | 139 | return get_meta("PATH_TRANSLATED"); |
136 | }catch(exception_notfound& enf) { | 140 | }catch(exception_notfound& enf) { |
137 | return empty_string; | 141 | return empty_string; |
138 | } | 142 | } |
139 | } | 143 | } |
140 | const string& cgi_gateway::query_string() const { | 144 | const string& cgi_gateway::query_string() const { |
141 | try { | 145 | try { |
142 | return get_meta("QUERY_STRING"); | 146 | return get_meta("QUERY_STRING"); |
143 | }catch(exception_notfound& enf) { | 147 | }catch(exception_notfound& enf) { |
144 | return empty_string; | 148 | return empty_string; |
145 | } | 149 | } |
146 | } | 150 | } |
147 | const string& cgi_gateway::remote_addr() const { | 151 | const string& cgi_gateway::remote_addr() const { |
148 | try { | 152 | try { |
149 | return get_meta("REMOTE_ADDR"); | 153 | return get_meta("REMOTE_ADDR"); |
150 | }catch(exception_notfound& enf) { | 154 | }catch(exception_notfound& enf) { |
151 | return empty_string; | 155 | return empty_string; |
152 | } | 156 | } |
153 | } | 157 | } |
154 | const string& cgi_gateway::remote_host() const { | 158 | const string& cgi_gateway::remote_host() const { |
155 | try { | 159 | try { |
156 | return get_meta("REMOTE_HOST"); | 160 | return get_meta("REMOTE_HOST"); |
157 | }catch(exception_notfound& enf) { | 161 | }catch(exception_notfound& enf) { |
158 | return remote_addr(); | 162 | return remote_addr(); |
159 | } | 163 | } |
160 | } | 164 | } |
161 | const string& cgi_gateway::remote_ident() const { | 165 | const string& cgi_gateway::remote_ident() const { |
162 | try { | 166 | try { |
163 | return get_meta("REMOTE_IDENT"); | 167 | return get_meta("REMOTE_IDENT"); |
164 | }catch(exception_notfound& enf) { | 168 | }catch(exception_notfound& enf) { |
165 | return empty_string; | 169 | return empty_string; |
166 | } | 170 | } |
167 | } | 171 | } |
168 | const string& cgi_gateway::remote_user() const { | 172 | const string& cgi_gateway::remote_user() const { |
169 | try { | 173 | try { |
170 | return get_meta("REMOTE_USER"); | 174 | return get_meta("REMOTE_USER"); |
171 | }catch(exception_notfound& enf) { | 175 | }catch(exception_notfound& enf) { |
172 | return empty_string; | 176 | return empty_string; |
173 | } | 177 | } |
174 | } | 178 | } |
175 | const string& cgi_gateway::request_method() const { | 179 | const string& cgi_gateway::request_method() const { |
176 | try { | 180 | try { |
177 | return get_meta("REQUEST_METHOD"); | 181 | return get_meta("REQUEST_METHOD"); |
178 | }catch(exception_notfound& enf) { | 182 | }catch(exception_notfound& enf) { |
179 | throw server_error(CODEPOINT,"No REQUEST_METHOD passed from server"); | 183 | throw server_error(CODEPOINT,"No REQUEST_METHOD passed from server"); |
180 | } | 184 | } |
181 | } | 185 | } |
182 | const string& cgi_gateway::script_name() const { | 186 | const string& cgi_gateway::script_name() const { |
183 | try { | 187 | try { |
184 | return get_meta("SCRIPT_NAME"); | 188 | return get_meta("SCRIPT_NAME"); |
185 | }catch(exception_notfound& enf) { | 189 | }catch(exception_notfound& enf) { |
186 | throw server_error(CODEPOINT,"No SCRIPT_NAME passed from server"); | 190 | throw server_error(CODEPOINT,"No SCRIPT_NAME passed from server"); |
187 | } | 191 | } |
188 | } | 192 | } |
189 | const string& cgi_gateway::server_name() const { | 193 | const string& cgi_gateway::server_name() const { |
190 | try { | 194 | try { |
191 | return get_meta("SERVER_NAME"); | 195 | return get_meta("SERVER_NAME"); |
192 | }catch(exception_notfound& enf) { | 196 | }catch(exception_notfound& enf) { |
193 | throw server_error(CODEPOINT,"No SERVER_NAME passed from server"); | 197 | throw server_error(CODEPOINT,"No SERVER_NAME passed from server"); |
194 | } | 198 | } |
195 | } | 199 | } |
196 | unsigned int cgi_gateway::server_port() const { | 200 | unsigned int cgi_gateway::server_port() const { |
197 | try { | 201 | try { |
198 | const string& sp = get_meta("SERVER_PORT"); | 202 | const string& sp = get_meta("SERVER_PORT"); |
199 | errno = 0; | 203 | errno = 0; |
200 | const char *spp = sp.c_str(); | 204 | const char *spp = sp.c_str(); |
201 | unsigned int rv = strtol(spp,(char**)&spp,10); | 205 | unsigned int rv = strtol(spp,(char**)&spp,10); |
202 | if(errno || *spp) | 206 | if(errno || *spp) |
203 | throw server_error(CODEPOINT,"Invalid SERVER_PORT value passed from server"); | 207 | throw server_error(CODEPOINT,"Invalid SERVER_PORT value passed from server"); |
204 | return rv; | 208 | return rv; |
205 | }catch(exception_notfound& enf) { | 209 | }catch(exception_notfound& enf) { |
206 | throw server_error(CODEPOINT,"No SERVER_PORT passed from server"); | 210 | throw server_error(CODEPOINT,"No SERVER_PORT passed from server"); |
207 | } | 211 | } |
208 | } | 212 | } |
209 | const string& cgi_gateway::server_protocol() const { | 213 | const string& cgi_gateway::server_protocol() const { |
210 | try { | 214 | try { |
211 | return get_meta("SERVER_PROTOCOL"); | 215 | return get_meta("SERVER_PROTOCOL"); |
212 | }catch(exception_notfound& enf) { | 216 | }catch(exception_notfound& enf) { |
213 | throw server_error(CODEPOINT,"No SERVER_PROTOCOL passed from server"); | 217 | throw server_error(CODEPOINT,"No SERVER_PROTOCOL passed from server"); |
214 | } | 218 | } |
215 | } | 219 | } |
216 | const string& cgi_gateway::server_software() const { | 220 | const string& cgi_gateway::server_software() const { |
217 | try { | 221 | try { |
218 | return get_meta("SERVER_SOFTWARE"); | 222 | return get_meta("SERVER_SOFTWARE"); |
219 | }catch(exception_notfound& enf) { | 223 | }catch(exception_notfound& enf) { |
220 | throw server_error(CODEPOINT,"No SERVER_SOFTWARE passed from server"); | 224 | throw server_error(CODEPOINT,"No SERVER_SOFTWARE passed from server"); |
221 | } | 225 | } |
222 | } | 226 | } |
223 | 227 | ||
224 | void cgi_gateway::parse_query(string& q,params_t& p) { | 228 | void cgi_gateway::parse_query(string& q,params_t& p) { |
225 | while(!q.empty()) { | 229 | while(!q.empty()) { |
226 | string::size_type amp = q.find('&'); | 230 | string::size_type amp = q.find('&'); |
227 | string pp = (amp==string::npos)?q:q.substr(0,amp); | 231 | string pp = (amp==string::npos)?q:q.substr(0,amp); |
228 | if(amp==string::npos) | 232 | if(amp==string::npos) |
229 | q.clear(); | 233 | q.clear(); |
230 | else | 234 | else |
231 | q.erase(0,amp+1); | 235 | q.erase(0,amp+1); |
232 | string::size_type eq = pp.find('='); | 236 | string::size_type eq = pp.find('='); |
233 | if(eq == string::npos) { | 237 | if(eq == string::npos) { |
234 | p.insert(params_t::value_type("",url_decode(pp))); | 238 | p.insert(params_t::value_type("",url_decode(pp))); |
235 | }else{ | 239 | }else{ |
236 | p.insert(params_t::value_type(url_decode(pp.substr(0,eq)),url_decode(pp.substr(eq+1)))); | 240 | p.insert(params_t::value_type(url_decode(pp.substr(0,eq)),url_decode(pp.substr(eq+1)))); |
237 | } | 241 | } |
238 | } | 242 | } |
239 | } | 243 | } |
240 | 244 | ||
241 | } | 245 | } |
diff --git a/src/cookies.cc b/src/cookies.cc new file mode 100644 index 0000000..40a0c8b --- a/dev/null +++ b/src/cookies.cc | |||
@@ -0,0 +1,242 @@ | |||
1 | #include "kingate/cookies.h" | ||
2 | #include "kingate/util.h" | ||
3 | #include "kingate/exception.h" | ||
4 | |||
5 | namespace kingate { | ||
6 | |||
7 | /* | ||
8 | * RFC 2109: | ||
9 | * av-pairs = av-pair *(";" av-pair) | ||
10 | * av-pair = attr ["=" value] ; optional value | ||
11 | * attr = token | ||
12 | * value = word | ||
13 | * word = token | quoted-string | ||
14 | */ | ||
15 | |||
16 | /* RFC 2109: | ||
17 | * | ||
18 | * The origin server effectively ends a session by sending the client a | ||
19 | * Set-Cookie header with Max-Age=0. | ||
20 | * | ||
21 | * An origin server may include multiple Set-Cookie headers in a response. | ||
22 | * Note that an intervening gateway could fold multiple such headers into a | ||
23 | * single header. | ||
24 | * | ||
25 | * | ||
26 | * set-cookie = "Set-Cookie:" cookies | ||
27 | * cookies = 1#cookie | ||
28 | * cookie = NAME "=" VALUE *(";" cookie-av) | ||
29 | * NAME = attr | ||
30 | * VALUE = value | ||
31 | * cookie-av = "Comment" "=" value | ||
32 | * | "Domain" "=" value | ||
33 | * | "Max-Age" "=" value | ||
34 | * | "Path" "=" value | ||
35 | * | "Secure" | ||
36 | * | "Version" "=" 1*DIGIT | ||
37 | * | ||
38 | * | ||
39 | * The origin server should send the following additional HTTP/1.1 | ||
40 | * response headers, depending on circumstances: | ||
41 | * | ||
42 | * * To suppress caching of the Set-Cookie header: Cache-control: no- | ||
43 | * cache="set-cookie". | ||
44 | * | ||
45 | * and one of the following: | ||
46 | * | ||
47 | * * To suppress caching of a private document in shared caches: Cache- | ||
48 | * control: private. | ||
49 | * | ||
50 | * * To allow caching of a document and require that it be validated | ||
51 | * before returning it to the client: Cache-control: must-revalidate. | ||
52 | * | ||
53 | * * To allow caching of a document, but to require that proxy caches | ||
54 | * (not user agent caches) validate it before returning it to the | ||
55 | * client: Cache-control: proxy-revalidate. | ||
56 | * | ||
57 | * * To allow caching of a document and request that it be validated | ||
58 | * before returning it to the client (by "pre-expiring" it): | ||
59 | * Cache-control: max-age=0. Not all caches will revalidate the | ||
60 | * document in every case. | ||
61 | * | ||
62 | * HTTP/1.1 servers must send Expires: old-date (where old-date is a | ||
63 | * date long in the past) on responses containing Set-Cookie response | ||
64 | * headers unless they know for certain (by out of band means) that | ||
65 | * there are no downsteam HTTP/1.0 proxies. HTTP/1.1 servers may send | ||
66 | * other Cache-Control directives that permit caching by HTTP/1.1 | ||
67 | * proxies in addition to the Expires: old-date directive; the Cache- | ||
68 | * Control directive will override the Expires: old-date for HTTP/1.1 | ||
69 | * proxies. | ||
70 | * | ||
71 | */ | ||
72 | |||
73 | void cookie::_set_string(const string& p,const string& v) { | ||
74 | (*this)[p]=v; | ||
75 | } | ||
76 | |||
77 | void cookie::set_comment(const string& c) { | ||
78 | _set_string("comment",c); | ||
79 | } | ||
80 | void cookie::set_domain(const string& d) { | ||
81 | _set_string("domain",d); | ||
82 | } | ||
83 | void cookie::set_max_age(const string& ma) { | ||
84 | _set_string("max-age",ma); | ||
85 | } | ||
86 | void cookie::set_path(const string& p) { | ||
87 | _set_string("path",p); | ||
88 | } | ||
89 | void cookie::set_secure(bool s) { | ||
90 | if(s) | ||
91 | _set_string("secure",""); | ||
92 | else | ||
93 | erase("secure"); | ||
94 | } | ||
95 | |||
96 | void cookie::set_expires(const string& e) { | ||
97 | (*this)["expires"] = e; | ||
98 | } | ||
99 | |||
100 | const string& cookie::_get_string(const string& s) const { | ||
101 | const_iterator i = find(s); | ||
102 | if(i==end()) | ||
103 | throw exception_notfound(CODEPOINT,"No parameter set"); | ||
104 | return i->second; | ||
105 | } | ||
106 | |||
107 | const string& cookie::get_comment() const { | ||
108 | return _get_string("comment"); | ||
109 | } | ||
110 | const string& cookie::get_domain() const { | ||
111 | return _get_string("domain"); | ||
112 | } | ||
113 | const string& cookie::get_max_age() const { | ||
114 | return _get_string("max-age"); | ||
115 | } | ||
116 | const string& cookie::get_path() const { | ||
117 | return _get_string("path"); | ||
118 | } | ||
119 | bool cookie::get_secure() const { | ||
120 | return find("secure")!=end(); | ||
121 | } | ||
122 | |||
123 | const string& cookie::get_expires() const { | ||
124 | return _get_string("expires"); | ||
125 | } | ||
126 | |||
127 | bool cookie::has_comment() const { | ||
128 | return find("comment")!=end(); | ||
129 | } | ||
130 | bool cookie::has_domain() const { | ||
131 | return find("domain")!=end(); | ||
132 | } | ||
133 | bool cookie::has_max_age() const { | ||
134 | return find("max-age")!=end(); | ||
135 | } | ||
136 | bool cookie::has_path() const { | ||
137 | return find("path")!=end(); | ||
138 | } | ||
139 | |||
140 | bool cookie::has_expires() const { | ||
141 | return find("expires")!=end(); | ||
142 | } | ||
143 | |||
144 | void cookie::unset_comment() { | ||
145 | erase("comment"); | ||
146 | } | ||
147 | void cookie::unset_domain() { | ||
148 | erase("domain"); | ||
149 | } | ||
150 | void cookie::unset_max_age() { | ||
151 | erase("max-age"); | ||
152 | } | ||
153 | void cookie::unset_path() { | ||
154 | erase("path"); | ||
155 | } | ||
156 | |||
157 | void cookie::unset_expires() { | ||
158 | erase("expires"); | ||
159 | } | ||
160 | |||
161 | string cookie::set_cookie_header_rfc2109() const { | ||
162 | string rv = name + "=" + http_quoted_string(value); | ||
163 | for(const_iterator i=begin();i!=end();++i) { | ||
164 | if(i->first=="secure") { | ||
165 | rv += "; secure"; | ||
166 | }else{ | ||
167 | rv += "; "+i->first+"="+http_quote(i->second); | ||
168 | } | ||
169 | } | ||
170 | rv += "; Version=1"; | ||
171 | return rv; | ||
172 | } | ||
173 | |||
174 | string cookie::set_cookie_header() const { | ||
175 | string rv = name + "=" + value; | ||
176 | for(const_iterator i=begin();i!=end();++i) { | ||
177 | if(i->first=="secure") { | ||
178 | rv += "; secure"; | ||
179 | }else{ | ||
180 | rv += "; "+i->first+"="+i->second; | ||
181 | } | ||
182 | } | ||
183 | return rv; | ||
184 | } | ||
185 | |||
186 | bool cookies_t::has_cookie(const key_type& n) const { | ||
187 | return find(n)!=end(); | ||
188 | } | ||
189 | |||
190 | const cookie& cookies_t::get_cookie(const key_type& n) const { | ||
191 | const_iterator i=find(n); | ||
192 | if(i==end()) | ||
193 | throw exception_notfound(CODEPOINT,"No cookie with such name found"); | ||
194 | return i->second; | ||
195 | } | ||
196 | |||
197 | cookie& cookies_t::get_cookie(const key_type& n) { | ||
198 | iterator i=find(n); | ||
199 | if(i==end()) | ||
200 | throw exception_notfound(CODEPOINT,"No cookie with such name found"); | ||
201 | return i->second; | ||
202 | } | ||
203 | |||
204 | void cookies_t::parse_cookies(const string& s) { | ||
205 | string str = s; | ||
206 | while(!str.empty()) { | ||
207 | string::size_type sc = str.find(';'); | ||
208 | string s; | ||
209 | if(sc==string::npos) { | ||
210 | s = str; | ||
211 | str.erase(); | ||
212 | }else{ | ||
213 | s = str.substr(0,sc); | ||
214 | str.erase(0,sc+1); | ||
215 | } | ||
216 | string::size_type nsp=s.find_first_not_of(" \t"); | ||
217 | if((nsp!=string::npos) && nsp) | ||
218 | s.erase(0,nsp); | ||
219 | string::size_type eq=s.find('='); | ||
220 | if(eq==string::npos) | ||
221 | continue; | ||
222 | string n = s.substr(0,eq); | ||
223 | s.erase(0,eq+1); | ||
224 | nsp = n.find_last_not_of(" \t"); | ||
225 | n.erase(nsp+1); | ||
226 | nsp = s.find_first_not_of(" \t"); | ||
227 | string v; | ||
228 | if(nsp!=string::npos) | ||
229 | v = s.substr(nsp); | ||
230 | else | ||
231 | v = s; | ||
232 | nsp = v.find_last_not_of(" \t"); | ||
233 | if(nsp==string::npos) | ||
234 | v.erase(); | ||
235 | else | ||
236 | v.erase(nsp+1); | ||
237 | cookie& c = (*this)[n]; | ||
238 | c.set_name(n); c.set_value(v); | ||
239 | } | ||
240 | } | ||
241 | |||
242 | } | ||
diff --git a/src/headers.cc b/src/headers.cc new file mode 100644 index 0000000..89ac519 --- a/dev/null +++ b/src/headers.cc | |||
@@ -0,0 +1,40 @@ | |||
1 | #include "kingate/headers.h" | ||
2 | #include "kingate/exception.h" | ||
3 | |||
4 | namespace kingate { | ||
5 | |||
6 | void headers::set_header(const key_type& h,const mapped_type& c) { | ||
7 | erase(h); | ||
8 | insert(value_type(h,c)); | ||
9 | } | ||
10 | void headers::add_header(const key_type& h,const mapped_type& c) { | ||
11 | insert(value_type(h,c)); | ||
12 | } | ||
13 | void headers::unset_header(const key_type& h) { | ||
14 | erase(h); | ||
15 | } | ||
16 | const headers::mapped_type& headers::get_header(const key_type& h) const { | ||
17 | const_iterator i=find(h); | ||
18 | if(i==end()) | ||
19 | throw exception_notfound(CODEPOINT,"No such header"); | ||
20 | return i->second; | ||
21 | } | ||
22 | headers::mapped_type& headers::get_header(const key_type& h) { | ||
23 | // XXX: or should it fail if there's no such thing, unlike operator[]? | ||
24 | iterator i=find(h); | ||
25 | if(i==end()) | ||
26 | i = insert(value_type(h,"")); | ||
27 | return i->second; | ||
28 | } | ||
29 | |||
30 | pair<headers::const_iterator,headers::const_iterator> headers::get_headers(const key_type& h) const { | ||
31 | return equal_range(h); | ||
32 | } | ||
33 | pair<headers::iterator,headers::iterator> headers::get_headers(const key_type& h) { | ||
34 | return equal_range(h); | ||
35 | } | ||
36 | |||
37 | bool headers::has_header(const key_type& h) const { | ||
38 | return find(h)!=end(); | ||
39 | } | ||
40 | } | ||
diff --git a/src/util.cc b/src/util.cc index 3166e62..48e486a 100644 --- a/src/util.cc +++ b/src/util.cc | |||
@@ -1,53 +1,106 @@ | |||
1 | #include "kingate/util.h" | 1 | #include "kingate/util.h" |
2 | #include "kingate/exception.h" | 2 | #include "kingate/exception.h" |
3 | 3 | ||
4 | namespace kingate { | 4 | namespace kingate { |
5 | 5 | ||
6 | static const char *safeChars = | 6 | static const char *safeChars = |
7 | "ABCDEFGHIJKLMNOPQRSTUVWXYZ" | 7 | "ABCDEFGHIJKLMNOPQRSTUVWXYZ" |
8 | "abcdefghijklmnopqrstuvwxyz" | 8 | "abcdefghijklmnopqrstuvwxyz" |
9 | "0123456789" | 9 | "0123456789" |
10 | "_-" ; | 10 | "_-" ; |
11 | 11 | ||
12 | string url_encode(const string& str) { | 12 | string url_encode(const string& str) { |
13 | string rv = str; | 13 | string rv = str; |
14 | string::size_type screwed = 0; | 14 | string::size_type screwed = 0; |
15 | for(;;) { | 15 | for(;;) { |
16 | screwed = rv.find_first_not_of(safeChars,screwed); | 16 | screwed = rv.find_first_not_of(safeChars,screwed); |
17 | if(screwed == string::npos) | 17 | if(screwed == string::npos) |
18 | break; | 18 | break; |
19 | while(screwed<rv.length() && !strchr(safeChars,rv.at(screwed))) { | 19 | while(screwed<rv.length() && !strchr(safeChars,rv.at(screwed))) { |
20 | char danger = rv.at(screwed); | 20 | char danger = rv.at(screwed); |
21 | if(danger==' ') { | 21 | if(danger==' ') { |
22 | rv.replace(screwed++,1,1,'+'); | 22 | rv.replace(screwed++,1,1,'+'); |
23 | }else{ | 23 | }else{ |
24 | static char tmp[4] = {'%',0,0,0}; | 24 | static char tmp[4] = {'%',0,0,0}; |
25 | snprintf(&tmp[1],3,"%02X",0xFF&(int)danger); | 25 | snprintf(&tmp[1],3,"%02X",0xFF&(int)danger); |
26 | rv.replace(screwed,1,tmp,3); | 26 | rv.replace(screwed,1,tmp,3); |
27 | screwed+=3; | 27 | screwed+=3; |
28 | } | 28 | } |
29 | } | 29 | } |
30 | } | 30 | } |
31 | return rv; | 31 | return rv; |
32 | } | 32 | } |
33 | string url_decode(const string& str) { | 33 | string url_decode(const string& str) { |
34 | string rv = str; | 34 | string rv = str; |
35 | string::size_type unscrewed = 0; | 35 | string::size_type unscrewed = 0; |
36 | for(;;) { | 36 | for(;;) { |
37 | unscrewed = rv.find_first_of("%+",unscrewed); | 37 | unscrewed = rv.find_first_of("%+",unscrewed); |
38 | if(unscrewed == string::npos) | 38 | if(unscrewed == string::npos) |
39 | break; | 39 | break; |
40 | if(rv.at(unscrewed)=='+') { | 40 | if(rv.at(unscrewed)=='+') { |
41 | rv.replace(unscrewed++,1,1,' '); | 41 | rv.replace(unscrewed++,1,1,' '); |
42 | }else{ | 42 | }else{ |
43 | if((rv.length()-unscrewed)<3) | 43 | if((rv.length()-unscrewed)<3) |
44 | throw exception(CODEPOINT,"incorrectly escaped string"); | 44 | throw exception(CODEPOINT,"incorrectly escaped string"); |
45 | // XXX: ensure it's hex? | 45 | // XXX: ensure it's hex? |
46 | int danger = strtol(rv.substr(unscrewed+1,2).c_str(),NULL,16); | 46 | int danger = strtol(rv.substr(unscrewed+1,2).c_str(),NULL,16); |
47 | rv.replace(unscrewed,3,1,danger); | 47 | rv.replace(unscrewed,3,1,danger); |
48 | unscrewed++; | 48 | unscrewed++; |
49 | } | 49 | } |
50 | } | 50 | } |
51 | return rv; | 51 | return rv; |
52 | } | 52 | } |
53 | |||
54 | /* | ||
55 | * RFC 2616: | ||
56 | * | ||
57 | * separators = "(" | ")" | "<" | ">" | "@" | ||
58 | * | "," | ";" | ":" | "\" | <"> | ||
59 | * | "/" | "[" | "]" | "?" | "=" | ||
60 | * | "{" | "}" | SP | HT | ||
61 | */ | ||
62 | |||
63 | /* | ||
64 | * RFC 2616: | ||
65 | * | ||
66 | * token = 1*<any CHAR except CTLs or separators> | ||
67 | */ | ||
68 | |||
69 | static const char *quotible_chars = | ||
70 | "\001\002\003\004\005\006\007\010" | ||
71 | "\011\012\013\014\015\016\017\020" | ||
72 | "\021\022\023\024\025\026\027\030" | ||
73 | "\031\032\033\034\035\036\037\040" | ||
74 | "()<>@,;:\\\"/[]?={}" /* separator chars (except for SP and HT mentioned elsewhere */ | ||
75 | "\177" | ||
76 | ; | ||
77 | |||
78 | /* | ||
79 | * RFC 2616: | ||
80 | * | ||
81 | * quoted-string = ( <"> *(qdtext | quoted-pair ) <"> ) | ||
82 | * qdtext = <any TEXT except <">> | ||
83 | * | ||
84 | * The backslash character ("\") MAY be used as a single-character | ||
85 | * quoting mechanism only within quoted-string and comment constructs. | ||
86 | * | ||
87 | * quoted-pair = "\" CHAR | ||
88 | */ | ||
89 | |||
90 | string http_quoted_string(const string& str) { | ||
91 | string rv = str; | ||
92 | string::size_type sp=0; | ||
93 | for(string::size_type q=rv.find('"');(q=rv.find('"',q))!=string::npos;q+=2) | ||
94 | rv.insert(q,1,'\\'); | ||
95 | rv.insert(0,1,'"'); | ||
96 | rv += '"'; | ||
97 | return rv; | ||
98 | } | ||
99 | |||
100 | string http_quote(const string& str) { | ||
101 | if(str.find_first_of(quotible_chars)==string::npos) | ||
102 | return str; | ||
103 | return http_quoted_string(str); | ||
104 | } | ||
105 | |||
53 | } | 106 | } |