189 lines
4.3 KiB
C
189 lines
4.3 KiB
C
/* reply.c
|
|
*
|
|
* This file is part of httpd.
|
|
*
|
|
* 02/17/1996 Michael Temari <Michael@TemWare.Com>
|
|
* 07/07/1996 Initial Release Michael Temari <Michael@TemWare.Com>
|
|
* 12/29/2002 Michael Temari <Michael@TemWare.Com>
|
|
*
|
|
*/
|
|
#include <sys/types.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <signal.h>
|
|
|
|
#include "http.h"
|
|
#include "utility.h"
|
|
#include "net.h"
|
|
#include "config.h"
|
|
|
|
#define SERVER "Server: "VERSION
|
|
|
|
_PROTOTYPE(static void GotAlarm, (int sig));
|
|
_PROTOTYPE(static int sendout, (char *data));
|
|
|
|
static void GotAlarm(sig)
|
|
int sig;
|
|
{
|
|
}
|
|
|
|
static int sendout(data)
|
|
char *data;
|
|
{
|
|
if(strlen(data) > 0)
|
|
write(1, data, strlen(data));
|
|
write(1, "\r\n", 2);
|
|
if(dbglog != (FILE *)NULL) {
|
|
fprintf(dbglog, "REPLY: %s\n", data);
|
|
fflush(dbglog);
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
int sendreply(rp, rq)
|
|
struct http_reply *rp;
|
|
struct http_request *rq;
|
|
{
|
|
int s;
|
|
int s2;
|
|
int e;
|
|
static char buffer[8192];
|
|
|
|
if(rq->type != HTTP_REQUEST_TYPE_PROXY)
|
|
/* We're receiving data from a */
|
|
if(rq->method == HTTP_METHOD_POST ||
|
|
(rq->method == HTTP_METHOD_PUT && rp->status == HTTP_STATUS_OK)) {
|
|
if(rq->type != HTTP_REQUEST_TYPE_FULL)
|
|
return(0);
|
|
if(rq->method == HTTP_METHOD_PUT)
|
|
rp->status = HTTP_STATUS_CREATED;
|
|
else
|
|
rp->status = HTTP_STATUS_OK;
|
|
while(rq->size != 0) {
|
|
s = read(0, buffer, (rq->size > sizeof(buffer)) ? sizeof(buffer) : rq->size);
|
|
if(s <= 0) {
|
|
rp->status = HTTP_STATUS_SERVER_ERROR;
|
|
strcpy(rp->statusmsg, strerror(errno));
|
|
close(rp->fd);
|
|
close(rp->ofd);
|
|
break;
|
|
}
|
|
rq->size -= s;
|
|
s2 = write(rp->ofd, buffer, s);
|
|
if(s2 != s) break;
|
|
}
|
|
}
|
|
|
|
if(rp->status != HTTP_STATUS_OK && rp->status != HTTP_STATUS_CREATED &&
|
|
rp->status != HTTP_STATUS_NOT_MODIFIED)
|
|
rp->keepopen = 0;
|
|
|
|
if(rp->status == HTTP_STATUS_NOT_MODIFIED) {
|
|
sprintf(buffer, "<h2>Error %03d %s</h2>",
|
|
rp->status, rp->statusmsg);
|
|
rp->size = strlen(buffer);
|
|
rp->keepopen = rq->keepopen;
|
|
}
|
|
|
|
if(!rp->headers) {
|
|
|
|
if((rq->type == HTTP_REQUEST_TYPE_PROXY && rp->status != HTTP_STATUS_OK) ||
|
|
rq->type == HTTP_REQUEST_TYPE_FULL) {
|
|
sprintf(buffer, "HTTP/%d.%d %03d %s",
|
|
rq->vmajor, rq->vminor, rp->status, rp->statusmsg);
|
|
sendout(buffer);
|
|
sendout(SERVER);
|
|
if(rp->status == HTTP_STATUS_MOVED_PERM ||
|
|
rp->status == HTTP_STATUS_MOVED_TEMP) {
|
|
#if 1
|
|
sprintf(buffer, "Location: %s", rq->url);
|
|
#else
|
|
sprintf(buffer, "Location: http://%s%s", myhostname, rq->url);
|
|
#endif
|
|
sendout(buffer);
|
|
}
|
|
if(rp->keepopen)
|
|
sendout("Connection: Keep-Alive");
|
|
else
|
|
sendout("Connection: Close");
|
|
if(rp->status == HTTP_STATUS_UNAUTHORIZED && rp->auth != NULL) {
|
|
sprintf(buffer, "WWW-Authenticate: Basic realm=\"%s\"", rp->auth->desc);
|
|
sendout(buffer);
|
|
}
|
|
if(rp->status == HTTP_STATUS_PROXY_AUTH_REQRD && proxyauth != NULL) {
|
|
sprintf(buffer, "Proxy-Authenticate: Basic realm=\"%s\"", proxyauth->desc);
|
|
sendout(buffer);
|
|
}
|
|
if(rp->modtime != (time_t) -1) {
|
|
sprintf(buffer, "Last-Modified: %s", httpdate(&rp->modtime));
|
|
sendout(buffer);
|
|
}
|
|
if(rp->size != 0) {
|
|
sprintf(buffer, "Content-Length: %lu", rp->size);
|
|
sendout(buffer);
|
|
}
|
|
if(rp->status == HTTP_STATUS_OK) {
|
|
sprintf(buffer, "Content-Type: %s", rp->mtype);
|
|
sendout(buffer);
|
|
} else
|
|
sendout("Content-Type: text/html");
|
|
if(!rp->headers)
|
|
sendout("");
|
|
} else
|
|
if(rp->status != HTTP_STATUS_OK)
|
|
return(0);
|
|
}
|
|
|
|
if(rp->status != HTTP_STATUS_OK && rp->status != HTTP_STATUS_CREATED) {
|
|
sprintf(buffer, "<h2>Error %03d %s</h2>",
|
|
rp->status, rp->statusmsg);
|
|
sendout(buffer);
|
|
return(0);
|
|
}
|
|
|
|
if(rq->type == HTTP_REQUEST_TYPE_PROXY) {
|
|
proxy(rq, rp);
|
|
return(0);
|
|
}
|
|
|
|
/* send out entity body */
|
|
if(rq->method == HTTP_METHOD_GET || rq->method == HTTP_METHOD_POST) {
|
|
errno = 0;
|
|
while(1) {
|
|
alarm(0);
|
|
signal(SIGALRM, GotAlarm);
|
|
alarm(10);
|
|
s = read(rp->fd, buffer, sizeof(buffer));
|
|
e = errno;
|
|
alarm(0);
|
|
if(s > 0) {
|
|
s2 = write(1, buffer, s);
|
|
e = errno;
|
|
if(s2 != s) break;
|
|
continue;
|
|
}
|
|
if(s == 0) break;
|
|
if(s < 0 && e != EINTR) break;
|
|
signal(SIGALRM, GotAlarm);
|
|
alarm(2);
|
|
s = read(0, buffer, 1);
|
|
e = errno;
|
|
alarm(0);
|
|
if(s < 0 && e != EINTR) break;
|
|
}
|
|
}
|
|
|
|
close(rp->fd);
|
|
rp->fd = -1;
|
|
if(rp->ofd != -1)
|
|
close(rp->ofd);
|
|
if(rp->pid != 0 && e != 0) {
|
|
kill(-rp->pid, SIGHUP);
|
|
rp->pid = 0;
|
|
}
|
|
|
|
return(0);
|
|
}
|