Annotation of libwww/Library/src/HTTPReq.c, revision 2.66
2.1 frystyk 1: /* HTTPReq.c
2.23 frystyk 2: ** HTTP REQUEST GENERATION
2.1 frystyk 3: **
2.32 frystyk 4: ** (c) COPYRIGHT MIT 1995.
5: ** Please first read the full copyright statement in the file COPYRIGH.
2.66 ! duerst 6: ** @(#) $Id: HTTPReq.c,v 2.65 2005/01/04 17:47:18 duerst Exp $
2.32 frystyk 7: **
2.1 frystyk 8: ** This module implements the output stream for HTTP used for sending
9: ** requests with or without a entity body.
10: **
11: ** History:
12: ** Jan 95 HFN Written from scratch
13: */
14:
15: /* Library Includes */
2.52 frystyk 16: #include "wwwsys.h"
2.28 frystyk 17: #include "WWWUtil.h"
2.32 frystyk 18: #include "WWWCore.h"
19:
2.23 frystyk 20: #include "HTTPGen.h"
2.21 frystyk 21: #include "HTTPUtil.h"
2.63 duerst 22: #include "HTDNS.h"
2.1 frystyk 23: #include "HTTPReq.h" /* Implements */
24:
2.24 frystyk 25: #define PUTC(c) (*me->target->isa->put_character)(me->target, c)
26: #define PUTS(s) (*me->target->isa->put_string)(me->target, s)
2.1 frystyk 27: #define PUTBLOCK(b, l) (*me->target->isa->put_block)(me->target, b, l)
28:
29: struct _HTStream {
2.29 frystyk 30: const HTStreamClass * isa;
2.1 frystyk 31: HTStream * target;
32: HTRequest * request;
2.18 frystyk 33: SOCKET sockfd;
2.11 frystyk 34: int version;
2.38 frystyk 35: int state;
36: char * url;
2.1 frystyk 37: BOOL transparent;
38: };
39:
40: /* ------------------------------------------------------------------------- */
41: /* HTTP Output Request Stream */
42: /* ------------------------------------------------------------------------- */
43:
2.11 frystyk 44: /* HTTP09Request
45: ** -------------
46: ** Makes a HTTP/0.9 request
2.1 frystyk 47: */
2.38 frystyk 48: PRIVATE int HTTP09Request (HTStream * me, HTRequest * request)
2.11 frystyk 49: {
2.34 frystyk 50: HTParentAnchor * anchor = HTRequest_anchor(request);
51: char * addr = HTAnchor_physical(anchor);
2.38 frystyk 52: if (!me->url) me->url = HTParse(addr, "", PARSE_PATH|PARSE_PUNCTUATION);
53: if (me->state == 0) {
2.47 frystyk 54: int status = PUTS("GET ");
55: if (status != HT_OK) return status;
2.38 frystyk 56: me->state++;
57: }
58: if (me->state == 1) {
59: int status = PUTS(me->url);
60: if (status != HT_OK) return status;
61: me->state++;
62: }
2.24 frystyk 63: PUTC(CR);
64: PUTC(LF);
2.61 frystyk 65: HTTRACE(PROT_TRACE, "HTTP........ Generating HTTP/0.9 Request\n");
2.38 frystyk 66: return HT_OK;
2.11 frystyk 67: }
68:
69: /* HTTPMakeRequest
70: ** ---------------
71: ** Makes a HTTP/1.0-1.1 request header.
72: */
2.38 frystyk 73: PRIVATE int HTTPMakeRequest (HTStream * me, HTRequest * request)
2.1 frystyk 74: {
2.39 frystyk 75: HTMethod method = HTRequest_method(request);
76: HTRqHd request_mask = HTRequest_rqHd(request);
77: HTParentAnchor * anchor = HTRequest_anchor(request);
2.43 frystyk 78: char * etag = HTAnchor_etag(anchor);
2.24 frystyk 79: char crlf[3];
80: char qstr[10];
81: *crlf = CR; *(crlf+1) = LF; *(crlf+2) = '\0';
2.1 frystyk 82:
2.41 frystyk 83: /* Generate the HTTP/1.x RequestLine */
2.38 frystyk 84: if (me->state == 0) {
2.34 frystyk 85: if (method != METHOD_INVALID) {
86: PUTS(HTMethod_name(method));
87: PUTC(' ');
88: } else
89: PUTS("GET ");
2.38 frystyk 90: me->state++;
2.34 frystyk 91: }
2.1 frystyk 92:
2.39 frystyk 93: /*
2.45 frystyk 94: ** Generate the Request URI. If we are using full request URI then it's
95: ** easy. Otherwise we must filter out the path part of the URI.
96: ** In case it's a OPTIONS request then if there is no pathinfo then use
97: ** a * instead. If we use a method different from GET or HEAD then use
98: ** the content-location if available.
2.39 frystyk 99: */
2.45 frystyk 100: if (me->state == 1) {
101: char * abs_location = NULL;
2.38 frystyk 102: char * addr = HTAnchor_physical(anchor);
2.62 kahan 103: char * location;
104:
105: /* JK: If the application specified a content-location (which is
106: stored in the request in default put-name!), we use it instead
107: of the URL that's being saved to. This is like having a user
108: defined Content-Location */
109: location = HTRequest_defaultPutName (request);
110: if (location)
111: {
112: if (HTURL_isAbsolute (location))
113: {
114: char * relative;
115: relative = HTRelative (location, location);
116: abs_location = HTParse (relative + 2, addr, PARSE_ALL);
117: HT_FREE (relative);
118: }
119: else
120: abs_location = HTParse (location, addr, PARSE_ALL);
121: addr = abs_location;
122: }
2.45 frystyk 123:
2.60 frystyk 124: #if 0
125: /*
126: ** We don't use the content-location any more as it is superseeded
127: ** by etags and the combination of the two might do more harm than
128: ** good (The etag is not guaranteed to be unique over multiple URIs)
129: */
130:
2.45 frystyk 131: /*
132: ** If we are using a method different from HEAD and GET then use
133: ** the Content-Location if available, else the Request-URI.
134: */
135: if (!HTMethod_isSafe(method)) {
136: char * location = HTAnchor_location(anchor);
137: if (location) {
138: if (HTURL_isAbsolute(location))
139: addr = location;
140: else {
141: /*
142: ** We have a content-location but it is relative and
143: ** must expand it either to the content-base or to
144: ** the Request-URI itself.
145: */
146: char * base = HTAnchor_base(anchor);
147: abs_location = HTParse(location, base, PARSE_ALL);
148: addr = abs_location;
149: }
150: }
151: }
2.60 frystyk 152: #endif
2.45 frystyk 153:
154: /*
155: ** If we are using a proxy or newer versions of HTTP then we can
156: ** send the full URL. Otherwise we only send the path.
157: */
158: if (HTRequest_fullURI(request))
159: StrAllocCopy(me->url, addr);
160: else {
161: me->url = HTParse(addr, "", PARSE_PATH | PARSE_PUNCTUATION);
2.39 frystyk 162: if (method == METHOD_OPTIONS) {
163: /*
164: ** We don't preserve the final slash or lack of same through
165: ** out the code. This is mainly for optimization reasons
2.44 frystyk 166: ** but it gives a problem OPTIONS. We can either send a "*"
167: ** or a "/" but not both. For now we send a "*".
2.39 frystyk 168: */
2.45 frystyk 169: if (!strcmp(me->url, "/")) *me->url = '*';
2.39 frystyk 170: }
2.1 frystyk 171: }
2.45 frystyk 172: HT_FREE(abs_location);
173: me->state++;
174: }
175:
176: /*
177: ** Now send the URL that we have put together
178: */
179: if (me->state == 2) {
180: int status = HT_OK;
181: if ((status = PUTS(me->url)) != HT_OK) return status;
2.38 frystyk 182: me->state++;
2.48 frystyk 183:
184: #if 0
185: fprintf(stderr, "Requesting '%s'\n", me->url);
186: #endif
187:
2.1 frystyk 188: }
2.25 frystyk 189: PUTC(' ');
2.41 frystyk 190:
191: /*
192: ** Send out the version number. If we know it is a HTTP/1.0 server we
193: ** are talking to then use HTTP/1.0, else use HTTP/1.1 as default version
194: ** number
195: */
196: if (me->version == HTTP_10)
197: PUTS(HTTP_VERSION_10);
198: else
199: PUTS(HTTP_VERSION);
2.24 frystyk 200: PUTBLOCK(crlf, 2);
2.1 frystyk 201:
2.4 frystyk 202: /* Request Headers */
2.34 frystyk 203: if (request_mask & HT_C_ACCEPT_TYPE) {
2.58 frystyk 204: HTFormat format = HTRequest_outputFormat(request);
205:
2.37 frystyk 206: /*
207: ** If caller has specified a specific output format then use this.
208: ** Otherwise use all the registered converters to generate the
209: ** accept header
210: */
2.58 frystyk 211: if (format == WWW_PRESENT) {
2.37 frystyk 212: int list;
213: HTList *cur;
214: BOOL first=YES;
215: for (list=0; list<2; list++) {
216: if ((!list && ((cur = HTFormat_conversion()) != NULL)) ||
217: (list && ((cur = HTRequest_conversion(request))!=NULL))) {
218: HTPresentation * pres;
219: while ((pres=(HTPresentation *) HTList_nextObject(cur))) {
220: if (pres->rep_out==WWW_PRESENT && pres->quality<=1.0) {
221: if (first) {
222: PUTS("Accept: ");
223: first=NO;
224: } else
225: PUTC(',');
226: PUTS(HTAtom_name(pres->rep));
2.46 frystyk 227: if (pres->quality < 1.0 && pres->quality >= 0.0) {
2.37 frystyk 228: sprintf(qstr, ";q=%1.1f", pres->quality);
229: PUTS(qstr);
230: }
2.1 frystyk 231: }
232: }
233: }
234: }
2.37 frystyk 235: if (!first) PUTBLOCK(crlf, 2);
236: } else {
2.58 frystyk 237:
238: /*
239: ** If we have an explicit output format then only send
240: ** this one if not this is an internal libwww format
241: ** of type www/<star>
242: */
243: if (!HTMIMEMatch(WWW_INTERNAL, format)) {
244: PUTS("Accept: ");
245: PUTS(HTAtom_name(format));
246: PUTBLOCK(crlf, 2);
247: }
2.37 frystyk 248: }
2.1 frystyk 249: }
2.34 frystyk 250: if (request_mask & HT_C_ACCEPT_CHAR) {
2.4 frystyk 251: int list;
252: HTList *cur;
2.24 frystyk 253: BOOL first=YES;
2.4 frystyk 254: for (list=0; list<2; list++) {
2.14 frystyk 255: if ((!list && ((cur = HTFormat_charset()) != NULL)) ||
256: (list && ((cur = HTRequest_charset(request)) != NULL))) {
2.4 frystyk 257: HTAcceptNode *pres;
258: while ((pres = (HTAcceptNode *) HTList_nextObject(cur))) {
2.5 frystyk 259: if (first) {
2.24 frystyk 260: PUTS("Accept-Charset: ");
2.5 frystyk 261: first=NO;
2.24 frystyk 262: } else
263: PUTC(',');
264: PUTS(HTAtom_name(pres->atom));
2.46 frystyk 265: if (pres->quality < 1.0 && pres->quality >= 0.0) {
266: sprintf(qstr, ";q=%1.1f", pres->quality);
267: PUTS(qstr);
268: }
2.4 frystyk 269: }
270: }
271: }
2.31 frystyk 272: if (!first) PUTBLOCK(crlf, 2);
2.4 frystyk 273: }
2.34 frystyk 274: if (request_mask & HT_C_ACCEPT_ENC) {
2.4 frystyk 275: int list;
276: HTList *cur;
2.24 frystyk 277: BOOL first=YES;
2.4 frystyk 278: for (list=0; list<2; list++) {
2.33 frystyk 279: if ((!list && ((cur = HTFormat_contentCoding()) != NULL)) ||
2.14 frystyk 280: (list && ((cur = HTRequest_encoding(request)) != NULL))) {
2.33 frystyk 281: HTCoding * pres;
282: while ((pres = (HTCoding *) HTList_nextObject(cur))) {
2.46 frystyk 283: double quality = HTCoding_quality(pres);
2.5 frystyk 284: if (first) {
2.24 frystyk 285: PUTS("Accept-Encoding: ");
2.32 frystyk 286: first = NO;
2.24 frystyk 287: } else
288: PUTC(',');
2.33 frystyk 289: PUTS(HTCoding_name(pres));
2.46 frystyk 290: if (quality < 1.0 && quality >= 0.0) {
291: sprintf(qstr, ";q=%1.1f", quality);
292: PUTS(qstr);
293: }
294: }
295: }
296: }
297: if (!first) PUTBLOCK(crlf, 2);
298: }
299: if (request_mask & HT_C_ACCEPT_TE) {
300: int list;
301: HTList *cur;
302: BOOL first=YES;
303: for (list=0; list<2; list++) {
304: if ((!list && ((cur = HTFormat_transferCoding()) != NULL)) ||
305: (list && ((cur = HTRequest_transfer(request)) != NULL))) {
306: HTCoding * pres;
307: while ((pres = (HTCoding *) HTList_nextObject(cur))) {
308: double quality = HTCoding_quality(pres);
2.55 frystyk 309: const char * coding = HTCoding_name(pres);
2.46 frystyk 310: if (first) {
311: PUTS("TE: ");
312: first = NO;
313: } else
314: PUTC(',');
2.55 frystyk 315:
316: /* Special check for "chunked" which is translated to "trailers" */
317: if (!strcasecomp(coding, "chunked"))
318: PUTS("trailers");
319: else
320: PUTS(coding);
2.46 frystyk 321: if (quality < 1.0 && quality >= 0.0) {
322: sprintf(qstr, ";q=%1.1f", quality);
323: PUTS(qstr);
324: }
2.4 frystyk 325: }
326: }
327: }
2.31 frystyk 328: if (!first) PUTBLOCK(crlf, 2);
2.4 frystyk 329: }
2.34 frystyk 330: if (request_mask & HT_C_ACCEPT_LAN) {
2.4 frystyk 331: int list;
332: HTList *cur;
2.24 frystyk 333: BOOL first=YES;
2.4 frystyk 334: for (list=0; list<2; list++) {
2.14 frystyk 335: if ((!list && ((cur = HTFormat_language()) != NULL)) ||
336: (list && ((cur = HTRequest_language(request)) != NULL))) {
2.4 frystyk 337: HTAcceptNode *pres;
338: while ((pres = (HTAcceptNode *) HTList_nextObject(cur))) {
2.5 frystyk 339: if (first) {
2.24 frystyk 340: PUTS("Accept-Language: ");
2.5 frystyk 341: first=NO;
2.24 frystyk 342: } else
343: PUTC(',');
344: PUTS(HTAtom_name(pres->atom));
2.46 frystyk 345: if (pres->quality < 1.0 && pres->quality >= 0.0) {
2.24 frystyk 346: sprintf(qstr, ";q=%1.1f", pres->quality);
347: PUTS(qstr);
2.5 frystyk 348: }
2.4 frystyk 349: }
350: }
351: }
2.31 frystyk 352: if (!first) PUTBLOCK(crlf, 2);
2.4 frystyk 353: }
2.36 frystyk 354: if (request_mask & HT_C_AUTH) {
2.34 frystyk 355: HTAssocList * cur = HTRequest_credentials(request);
356: if (cur) { /* Access authentication */
357: HTAssoc * pres;
358: while ((pres = (HTAssoc *) HTAssocList_nextObject(cur))) {
359: PUTS(HTAssoc_name(pres));
360: PUTS(": ");
361: PUTS(HTAssoc_value(pres));
362: PUTBLOCK(crlf, 2);
363: }
2.28 frystyk 364: }
2.1 frystyk 365: }
2.46 frystyk 366: if (request_mask & HT_C_EXPECT) {
367: HTAssocList * cur = HTRequest_expect(request);
368: if (cur) {
369: BOOL first=YES;
370: HTAssoc * pres;
371: while ((pres = (HTAssoc *) HTAssocList_nextObject(cur))) {
372: char * value = HTAssoc_value(pres);
373: if (first) {
374: PUTS("Expect: ");
375: first = NO;
376: } else
377: PUTC(',');
378:
379: /* Output the name */
380: PUTS(HTAssoc_name(pres));
381:
382: /* Only output the value if not empty string */
383: if (*value) {
384: PUTS("=");
385: PUTS(value);
386: }
387: }
388: PUTBLOCK(crlf, 2);
389: }
390: }
2.34 frystyk 391: if (request_mask & HT_C_FROM) {
392: HTUserProfile * up = HTRequest_userProfile(request);
393: const char * mailaddress = HTUserProfile_email(up);
2.1 frystyk 394: if (mailaddress) {
2.24 frystyk 395: PUTS("From: ");
396: PUTS(mailaddress);
397: PUTBLOCK(crlf, 2);
2.1 frystyk 398: }
399: }
2.34 frystyk 400: if (request_mask & HT_C_HOST) {
2.13 frystyk 401: char *orig = HTAnchor_address((HTAnchor *) anchor);
2.12 frystyk 402: char *host = HTParse(orig, "", PARSE_HOST);
2.64 duerst 403: char hostace[256];
2.34 frystyk 404: #if 0
405: /* Keep the port number for HTTP/1.1 compliance */
2.12 frystyk 406: char *ptr = strchr(host, ':'); /* Chop off port number */
407: if (ptr) *ptr = '\0';
2.34 frystyk 408: #endif
2.63 duerst 409: PUTS("Host: ");
2.64 duerst 410: /****** still have to check UTF8toACE with port number */
411: if (!HTACEfromUTF8 (host, hostace, 255)) {
2.63 duerst 412: PUTS(hostace);
413: }
414: else {
415: PUTS(host); /* this may be dangerous, but helps server side debugging */
2.66 ! duerst 416: HTTRACE(PROT_TRACE, "HTTP........ Error: Cannot convert to ACE: `%s\'\n" _ host);
2.63 duerst 417: }
2.24 frystyk 418: PUTBLOCK(crlf, 2);
2.26 frystyk 419: HT_FREE(orig);
420: HT_FREE(host);
2.39 frystyk 421: }
2.42 frystyk 422:
423: /*
424: ** In the "If-*" series of headers, the ones related to etags have higher
425: ** priority than the date relates ones. That is, if we have a etag then
2.45 frystyk 426: ** use that, otherwise use the date. First we check for range, match, and
427: ** unmodified-since.
2.42 frystyk 428: */
2.45 frystyk 429: if (request_mask & HT_C_IF_RANGE && etag) {
430: PUTS("If-Range: \"");
431: PUTS(etag);
432: PUTC('"');
433: PUTBLOCK(crlf, 2);
2.61 frystyk 434: HTTRACE(PROT_TRACE, "HTTP........ If-Range using etag `%s\'\n" _ etag);
2.57 frystyk 435: } else if (request_mask & HT_C_IF_MATCH_ANY) {
2.56 frystyk 436: PUTS("If-Match: *");
437: PUTBLOCK(crlf, 2);
2.61 frystyk 438: HTTRACE(PROT_TRACE, "HTTP........ If-Match using `*\'\n");
2.45 frystyk 439: } else if (request_mask & HT_C_IF_MATCH && etag) {
2.43 frystyk 440: PUTS("If-Match: \"");
441: PUTS(etag);
442: PUTC('"');
443: PUTBLOCK(crlf, 2);
2.61 frystyk 444: HTTRACE(PROT_TRACE, "HTTP........ If-Match using etag `%s\'\n" _ etag);
2.43 frystyk 445: } else if (request_mask & HT_C_IF_UNMOD_SINCE) {
2.42 frystyk 446: time_t lm = HTAnchor_lastModified(anchor);
447: if (lm > 0) {
2.43 frystyk 448: PUTS("If-Unmodified-Since: ");
2.42 frystyk 449: PUTS(HTDateTimeStr(&lm, NO));
450: PUTBLOCK(crlf, 2);
2.61 frystyk 451: HTTRACE(PROT_TRACE, "HTTP........ If-Unmodified-Since `%s\'\n" _ HTDateTimeStr(&lm, NO));
2.42 frystyk 452: }
453: }
2.45 frystyk 454:
455: /*
456: ** If-None-Match and If-Modified-Since are equivalent except that the
457: ** first uses etags and the second uses dates. Etags have precedence over
458: ** dates.
459: */
2.57 frystyk 460: if (request_mask & HT_C_IF_NONE_MATCH_ANY) {
2.56 frystyk 461: PUTS("If-None-Match: *");
462: PUTBLOCK(crlf, 2);
2.61 frystyk 463: HTTRACE(PROT_TRACE, "HTTP........ If-None-Match using `*\'\n");
2.56 frystyk 464: } else if (request_mask & HT_C_IF_NONE_MATCH && etag) {
2.43 frystyk 465: PUTS("If-None-Match: \"");
466: PUTS(etag);
467: PUTC('"');
468: PUTBLOCK(crlf, 2);
2.61 frystyk 469: HTTRACE(PROT_TRACE, "HTTP........ If-None-Match `%s\'\n" _ etag);
2.46 frystyk 470: }
471: if (request_mask & HT_C_IMS) {
2.42 frystyk 472: time_t lm = HTAnchor_lastModified(anchor);
473: if (lm > 0) {
2.43 frystyk 474: PUTS("If-Modified-Since: ");
2.42 frystyk 475: PUTS(HTDateTimeStr(&lm, NO));
476: PUTBLOCK(crlf, 2);
2.61 frystyk 477: HTTRACE(PROT_TRACE, "HTTP........ If-Modified-Since `%s\'\n" _ HTDateTimeStr(&lm, NO));
2.42 frystyk 478: }
479: }
480:
2.45 frystyk 481: /*
482: ** Max forwards is mainly for TRACE where we want to be able to stop the
483: ** TRACE at a specific location un the message path.
484: */
2.39 frystyk 485: if (request_mask & HT_C_MAX_FORWARDS) {
486: int hops = HTRequest_maxForwards(request);
487: if (hops >= 0) {
488: sprintf(qstr, "%d", hops);
489: PUTS("Max-Forwards: ");
490: PUTS(qstr);
491: PUTBLOCK(crlf, 2);
492: }
2.42 frystyk 493: }
2.45 frystyk 494:
495: /*
496: ** Range requests. For now, we only take the first entry registered for
497: ** this request. This means that you can only send a single "unit" and
498: ** then a set of range within this unit. This is in accordance with
499: ** HTTP/1.1. Multiple units will go on multiple lines.
500: */
2.42 frystyk 501: if (request_mask & HT_C_RANGE) {
2.45 frystyk 502: HTAssocList * cur = HTRequest_range(request);
503: if (cur) { /* Range requests */
504: HTAssoc * pres;
505: while ((pres = (HTAssoc *) HTAssocList_nextObject(cur))) {
506: PUTS("Range: ");
507: PUTS(HTAssoc_name(pres)); /* Unit */
508: PUTS("=");
509: PUTS(HTAssoc_value(pres)); /* Ranges within this unit */
510: PUTBLOCK(crlf, 2);
511: }
512: }
2.1 frystyk 513: }
2.34 frystyk 514: if (request_mask & HT_C_REFERER) {
515: HTParentAnchor * parent_anchor = HTRequest_parent(request);
516: if (parent_anchor) {
517: char * act = HTAnchor_address((HTAnchor *) anchor);
518: char * parent = HTAnchor_address((HTAnchor *) parent_anchor);
2.50 frystyk 519: #if 1
520: char * relative = HTRelative(parent, act);
521: #else
2.34 frystyk 522: char * relative = HTParse(parent, act,
523: PARSE_ACCESS|PARSE_HOST|PARSE_PATH|PARSE_PUNCTUATION);
2.50 frystyk 524: #endif
2.34 frystyk 525: if (relative && *relative) {
526: PUTS("Referer: ");
2.50 frystyk 527: PUTS(relative);
2.34 frystyk 528: PUTBLOCK(crlf, 2);
529: }
530: HT_FREE(act);
531: HT_FREE(parent);
532: HT_FREE(relative);
2.1 frystyk 533: }
534: }
2.34 frystyk 535: if (request_mask & HT_C_USER_AGENT) {
2.24 frystyk 536: PUTS("User-Agent: ");
537: PUTS(HTLib_appName());
538: PUTC('/');
539: PUTS(HTLib_appVersion());
540: PUTC(' ');
541: PUTS(HTLib_name());
542: PUTC('/');
543: PUTS(HTLib_version());
544: PUTBLOCK(crlf, 2);
2.1 frystyk 545: }
2.61 frystyk 546: HTTRACE(PROT_TRACE, "HTTP........ Generating HTTP/1.x Request Headers\n");
2.38 frystyk 547: return HT_OK;
2.1 frystyk 548: }
549:
2.29 frystyk 550: PRIVATE int HTTPRequest_put_block (HTStream * me, const char * b, int l)
2.1 frystyk 551: {
2.7 frystyk 552: if (!me->target) {
2.1 frystyk 553: return HT_WOULD_BLOCK;
2.7 frystyk 554: } else if (me->transparent)
2.11 frystyk 555: return b ? PUTBLOCK(b, l) : HT_OK;
2.1 frystyk 556: else {
2.38 frystyk 557: int status = HT_OK;
558: if (me->version == HTTP_09) {
559: status = HTTP09Request(me, me->request);
560: if (status != HT_OK) return status;
561: } else {
562: status = HTTPMakeRequest(me, me->request);
563: if (status != HT_OK) return status;
2.1 frystyk 564: me->transparent = YES;
2.11 frystyk 565: return b ? PUTBLOCK(b, l) : HT_OK;
2.1 frystyk 566: }
567: return status;
568: }
569: }
570:
2.16 frystyk 571: PRIVATE int HTTPRequest_put_character (HTStream * me, char c)
2.1 frystyk 572: {
2.11 frystyk 573: return HTTPRequest_put_block(me, &c, 1);
2.1 frystyk 574: }
575:
2.29 frystyk 576: PRIVATE int HTTPRequest_put_string (HTStream * me, const char * s)
2.1 frystyk 577: {
2.11 frystyk 578: return HTTPRequest_put_block(me, s, strlen(s));
2.1 frystyk 579: }
580:
581: /*
582: ** Flushes data but doesn't free stream object
583: */
2.16 frystyk 584: PRIVATE int HTTPRequest_flush (HTStream * me)
2.1 frystyk 585: {
2.13 frystyk 586: int status = HTTPRequest_put_block(me, NULL, 0);
587: return status==HT_OK ? (*me->target->isa->flush)(me->target) : status;
2.1 frystyk 588: }
589:
590: /*
591: ** Flushes data and frees stream object
592: */
2.16 frystyk 593: PRIVATE int HTTPRequest_free (HTStream * me)
2.1 frystyk 594: {
2.11 frystyk 595: int status = HTTPRequest_flush(me);
596: if (status != HT_WOULD_BLOCK) {
597: if ((status = (*me->target->isa->_free)(me->target)) == HT_WOULD_BLOCK)
598: return HT_WOULD_BLOCK;
2.38 frystyk 599: HT_FREE(me->url);
2.26 frystyk 600: HT_FREE(me);
2.1 frystyk 601: }
2.7 frystyk 602: return status;
2.1 frystyk 603: }
604:
2.16 frystyk 605: PRIVATE int HTTPRequest_abort (HTStream * me, HTList * e)
2.1 frystyk 606: {
2.61 frystyk 607: HTTRACE(PROT_TRACE, "HTTPRequest. ABORTING...\n");
2.51 kahan 608: /* JK: Added protection against NULL pointers */
2.49 frystyk 609: if (me) {
2.51 kahan 610: if (me->target && me->target->isa)
611: (*me->target->isa->abort)(me->target, e);
612: if (me->url)
613: HT_FREE(me->url);
2.49 frystyk 614: HT_FREE(me);
615: }
2.1 frystyk 616: return HT_ERROR;
617: }
618:
619: /* HTTPRequest Stream
620: ** -----------------
621: */
2.29 frystyk 622: PRIVATE const HTStreamClass HTTPRequestClass =
2.1 frystyk 623: {
624: "HTTPRequest",
625: HTTPRequest_flush,
626: HTTPRequest_free,
627: HTTPRequest_abort,
628: HTTPRequest_put_character,
629: HTTPRequest_put_string,
630: HTTPRequest_put_block
631: };
632:
2.23 frystyk 633: PUBLIC HTStream * HTTPRequest_new (HTRequest * request, HTStream * target,
2.45 frystyk 634: BOOL endHeader, int version)
2.1 frystyk 635: {
2.26 frystyk 636: HTStream * me;
637: if ((me = (HTStream *) HT_CALLOC(1, sizeof(HTStream))) == NULL)
638: HT_OUTOFMEM("HTTPRequest_new");
2.1 frystyk 639: me->isa = &HTTPRequestClass;
640: me->target = target;
641: me->request = request;
2.45 frystyk 642: me->version = version;
2.1 frystyk 643: me->transparent = NO;
2.46 frystyk 644:
645: /*
646: ** If sending a body in the request then we want a 100 code!
647: */
648: if (HTMethod_hasEntity(HTRequest_method(request)))
649: HTRequest_addExpect(request, "100-continue", "");
650:
2.23 frystyk 651:
652: /* Return general HTTP header stream */
2.45 frystyk 653: return HTTPGen_new(request, me, endHeader, version);
2.1 frystyk 654: }
Webmaster