1 | /* reply.c
|
---|
2 | *
|
---|
3 | * This file is part of httpd.
|
---|
4 | *
|
---|
5 | * 02/17/1996 Michael Temari <Michael@TemWare.Com>
|
---|
6 | * 07/07/1996 Initial Release Michael Temari <Michael@TemWare.Com>
|
---|
7 | * 12/29/2002 Michael Temari <Michael@TemWare.Com>
|
---|
8 | *
|
---|
9 | */
|
---|
10 | #include <sys/types.h>
|
---|
11 | #include <stdio.h>
|
---|
12 | #include <string.h>
|
---|
13 | #include <unistd.h>
|
---|
14 | #include <errno.h>
|
---|
15 | #include <signal.h>
|
---|
16 |
|
---|
17 | #include "http.h"
|
---|
18 | #include "utility.h"
|
---|
19 | #include "net.h"
|
---|
20 | #include "config.h"
|
---|
21 |
|
---|
22 | #define SERVER "Server: "VERSION
|
---|
23 |
|
---|
24 | _PROTOTYPE(static void GotAlarm, (int sig));
|
---|
25 | _PROTOTYPE(static int sendout, (char *data));
|
---|
26 |
|
---|
27 | static void GotAlarm(sig)
|
---|
28 | int sig;
|
---|
29 | {
|
---|
30 | }
|
---|
31 |
|
---|
32 | static int sendout(data)
|
---|
33 | char *data;
|
---|
34 | {
|
---|
35 | if(strlen(data) > 0)
|
---|
36 | write(1, data, strlen(data));
|
---|
37 | write(1, "\r\n", 2);
|
---|
38 | if(dbglog != (FILE *)NULL) {
|
---|
39 | fprintf(dbglog, "REPLY: %s\n", data);
|
---|
40 | fflush(dbglog);
|
---|
41 | }
|
---|
42 |
|
---|
43 | return(0);
|
---|
44 | }
|
---|
45 |
|
---|
46 | int sendreply(rp, rq)
|
---|
47 | struct http_reply *rp;
|
---|
48 | struct http_request *rq;
|
---|
49 | {
|
---|
50 | int s;
|
---|
51 | int s2;
|
---|
52 | int e;
|
---|
53 | static char buffer[8192];
|
---|
54 |
|
---|
55 | if(rq->type != HTTP_REQUEST_TYPE_PROXY)
|
---|
56 | /* We're receiving data from a */
|
---|
57 | if(rq->method == HTTP_METHOD_POST ||
|
---|
58 | (rq->method == HTTP_METHOD_PUT && rp->status == HTTP_STATUS_OK)) {
|
---|
59 | if(rq->type != HTTP_REQUEST_TYPE_FULL)
|
---|
60 | return(0);
|
---|
61 | if(rq->method == HTTP_METHOD_PUT)
|
---|
62 | rp->status = HTTP_STATUS_CREATED;
|
---|
63 | else
|
---|
64 | rp->status = HTTP_STATUS_OK;
|
---|
65 | while(rq->size != 0) {
|
---|
66 | s = read(0, buffer, (rq->size > sizeof(buffer)) ? sizeof(buffer) : rq->size);
|
---|
67 | if(s <= 0) {
|
---|
68 | rp->status = HTTP_STATUS_SERVER_ERROR;
|
---|
69 | strcpy(rp->statusmsg, strerror(errno));
|
---|
70 | close(rp->fd);
|
---|
71 | close(rp->ofd);
|
---|
72 | break;
|
---|
73 | }
|
---|
74 | rq->size -= s;
|
---|
75 | s2 = write(rp->ofd, buffer, s);
|
---|
76 | if(s2 != s) break;
|
---|
77 | }
|
---|
78 | }
|
---|
79 |
|
---|
80 | if(rp->status != HTTP_STATUS_OK && rp->status != HTTP_STATUS_CREATED &&
|
---|
81 | rp->status != HTTP_STATUS_NOT_MODIFIED)
|
---|
82 | rp->keepopen = 0;
|
---|
83 |
|
---|
84 | if(rp->status == HTTP_STATUS_NOT_MODIFIED) {
|
---|
85 | sprintf(buffer, "<h2>Error %03d %s</h2>",
|
---|
86 | rp->status, rp->statusmsg);
|
---|
87 | rp->size = strlen(buffer);
|
---|
88 | rp->keepopen = rq->keepopen;
|
---|
89 | }
|
---|
90 |
|
---|
91 | if(!rp->headers) {
|
---|
92 |
|
---|
93 | if((rq->type == HTTP_REQUEST_TYPE_PROXY && rp->status != HTTP_STATUS_OK) ||
|
---|
94 | rq->type == HTTP_REQUEST_TYPE_FULL) {
|
---|
95 | sprintf(buffer, "HTTP/%d.%d %03d %s",
|
---|
96 | rq->vmajor, rq->vminor, rp->status, rp->statusmsg);
|
---|
97 | sendout(buffer);
|
---|
98 | sendout(SERVER);
|
---|
99 | if(rp->status == HTTP_STATUS_MOVED_PERM ||
|
---|
100 | rp->status == HTTP_STATUS_MOVED_TEMP) {
|
---|
101 | #if 1
|
---|
102 | sprintf(buffer, "Location: %s", rq->url);
|
---|
103 | #else
|
---|
104 | sprintf(buffer, "Location: http://%s%s", myhostname, rq->url);
|
---|
105 | #endif
|
---|
106 | sendout(buffer);
|
---|
107 | }
|
---|
108 | if(rp->keepopen)
|
---|
109 | sendout("Connection: Keep-Alive");
|
---|
110 | else
|
---|
111 | sendout("Connection: Close");
|
---|
112 | if(rp->status == HTTP_STATUS_UNAUTHORIZED && rp->auth != NULL) {
|
---|
113 | sprintf(buffer, "WWW-Authenticate: Basic realm=\"%s\"", rp->auth->desc);
|
---|
114 | sendout(buffer);
|
---|
115 | }
|
---|
116 | if(rp->status == HTTP_STATUS_PROXY_AUTH_REQRD && proxyauth != NULL) {
|
---|
117 | sprintf(buffer, "Proxy-Authenticate: Basic realm=\"%s\"", proxyauth->desc);
|
---|
118 | sendout(buffer);
|
---|
119 | }
|
---|
120 | if(rp->modtime != (time_t) -1) {
|
---|
121 | sprintf(buffer, "Last-Modified: %s", httpdate(&rp->modtime));
|
---|
122 | sendout(buffer);
|
---|
123 | }
|
---|
124 | if(rp->size != 0) {
|
---|
125 | sprintf(buffer, "Content-Length: %lu", rp->size);
|
---|
126 | sendout(buffer);
|
---|
127 | }
|
---|
128 | if(rp->status == HTTP_STATUS_OK) {
|
---|
129 | sprintf(buffer, "Content-Type: %s", rp->mtype);
|
---|
130 | sendout(buffer);
|
---|
131 | } else
|
---|
132 | sendout("Content-Type: text/html");
|
---|
133 | if(!rp->headers)
|
---|
134 | sendout("");
|
---|
135 | } else
|
---|
136 | if(rp->status != HTTP_STATUS_OK)
|
---|
137 | return(0);
|
---|
138 | }
|
---|
139 |
|
---|
140 | if(rp->status != HTTP_STATUS_OK && rp->status != HTTP_STATUS_CREATED) {
|
---|
141 | sprintf(buffer, "<h2>Error %03d %s</h2>",
|
---|
142 | rp->status, rp->statusmsg);
|
---|
143 | sendout(buffer);
|
---|
144 | return(0);
|
---|
145 | }
|
---|
146 |
|
---|
147 | if(rq->type == HTTP_REQUEST_TYPE_PROXY) {
|
---|
148 | proxy(rq, rp);
|
---|
149 | return(0);
|
---|
150 | }
|
---|
151 |
|
---|
152 | /* send out entity body */
|
---|
153 | if(rq->method == HTTP_METHOD_GET || rq->method == HTTP_METHOD_POST) {
|
---|
154 | errno = 0;
|
---|
155 | while(1) {
|
---|
156 | alarm(0);
|
---|
157 | signal(SIGALRM, GotAlarm);
|
---|
158 | alarm(10);
|
---|
159 | s = read(rp->fd, buffer, sizeof(buffer));
|
---|
160 | e = errno;
|
---|
161 | alarm(0);
|
---|
162 | if(s > 0) {
|
---|
163 | s2 = write(1, buffer, s);
|
---|
164 | e = errno;
|
---|
165 | if(s2 != s) break;
|
---|
166 | continue;
|
---|
167 | }
|
---|
168 | if(s == 0) break;
|
---|
169 | if(s < 0 && e != EINTR) break;
|
---|
170 | signal(SIGALRM, GotAlarm);
|
---|
171 | alarm(2);
|
---|
172 | s = read(0, buffer, 1);
|
---|
173 | e = errno;
|
---|
174 | alarm(0);
|
---|
175 | if(s < 0 && e != EINTR) break;
|
---|
176 | }
|
---|
177 | }
|
---|
178 |
|
---|
179 | close(rp->fd);
|
---|
180 | rp->fd = -1;
|
---|
181 | if(rp->ofd != -1)
|
---|
182 | close(rp->ofd);
|
---|
183 | if(rp->pid != 0 && e != 0) {
|
---|
184 | kill(-rp->pid, SIGHUP);
|
---|
185 | rp->pid = 0;
|
---|
186 | }
|
---|
187 |
|
---|
188 | return(0);
|
---|
189 | }
|
---|