2018-09-03 22:54:02 +00:00
|
|
|
#include "http_server.h"
|
|
|
|
#include "../common/socket_helper.h"
|
2018-09-08 21:26:34 +00:00
|
|
|
#include "socket_server.h"
|
2018-09-03 22:54:02 +00:00
|
|
|
#include <errno.h>
|
2018-09-10 03:34:29 +00:00
|
|
|
#include <stdbool.h>
|
2018-09-03 22:54:02 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2018-09-09 23:51:51 +00:00
|
|
|
#include <string.h>
|
2018-09-10 03:34:29 +00:00
|
|
|
#include <sys/signal.h>
|
2018-09-08 21:26:34 +00:00
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <unistd.h>
|
2018-09-03 22:54:02 +00:00
|
|
|
|
2018-09-10 03:34:29 +00:00
|
|
|
bool running = true;
|
|
|
|
|
|
|
|
void handle_signal(int signal) {
|
2018-09-10 04:09:28 +00:00
|
|
|
if (signal == SIGPIPE) {
|
2018-09-10 04:28:30 +00:00
|
|
|
// We must ignore SIGPIPEs such that EPIPEs can be returned from socket read/write functions.
|
2018-09-10 04:09:28 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (signal == SIGINT || signal == SIGTERM) {
|
|
|
|
running = false;
|
|
|
|
}
|
2018-09-10 03:34:29 +00:00
|
|
|
}
|
|
|
|
|
2018-09-03 22:54:02 +00:00
|
|
|
int main(int argc, char* argv[]) {
|
|
|
|
if (argc != 2) {
|
|
|
|
printf("%s", USAGE_STRING);
|
|
|
|
}
|
|
|
|
char *port = argv[1];
|
|
|
|
long port_num = strtol(port, NULL, 10);
|
|
|
|
if (port_num < MIN_PORT || port_num > MAX_PORT || errno == ERANGE || errno == EINVAL) {
|
2018-09-10 04:23:39 +00:00
|
|
|
printf("%s\n", PORT_ERROR);
|
2018-09-10 03:34:29 +00:00
|
|
|
return 1;
|
2018-09-03 22:54:02 +00:00
|
|
|
}
|
2018-09-10 03:34:29 +00:00
|
|
|
struct sigaction sa = {.sa_handler = &handle_signal, .sa_flags = 0};
|
|
|
|
sigemptyset(&sa.sa_mask);
|
|
|
|
sigaction(SIGINT, &sa, NULL);
|
|
|
|
sigaction(SIGTERM, &sa, NULL);
|
2018-09-10 04:09:28 +00:00
|
|
|
sigaction(SIGPIPE, &sa, NULL);
|
2018-09-10 03:34:29 +00:00
|
|
|
|
2018-09-08 21:26:34 +00:00
|
|
|
struct server_info info = setup(port_num);
|
2018-09-09 23:51:51 +00:00
|
|
|
if (info.status == STATUS_ERROR) {
|
|
|
|
printf("%s\n", strerror(errno));
|
2018-09-10 03:34:29 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
while (running) {
|
|
|
|
enum socket_result serve_result = serve_one_request(info.sock_fd);
|
|
|
|
if (serve_result != RESULT_OK && errno != EINTR) {
|
|
|
|
printf("Got error code %d when serving a request.\n", serve_result);
|
2018-09-10 04:23:39 +00:00
|
|
|
if (serve_result == RESULT_MALFORMED) {
|
|
|
|
printf("\tMalformed request. This may be due to the client closing the socket prematurely.\n");
|
|
|
|
} else if (serve_result == RESULT_READ_ERROR) {
|
|
|
|
printf("\tThere was an error reading from the socket.\n");
|
|
|
|
} else if (serve_result == RESULT_PROCESSING_ERROR) {
|
2018-09-11 02:24:28 +00:00
|
|
|
printf("\tThere was an error while processing the data from the socket. This may be due to a malformed request.\n");
|
2018-09-10 04:23:39 +00:00
|
|
|
} else if (serve_result == RESULT_WRITE_ERROR) {
|
2018-09-10 04:32:27 +00:00
|
|
|
printf("\tThere was an error writing to the socket.\n");
|
2018-09-10 04:23:39 +00:00
|
|
|
}
|
2018-09-10 03:34:29 +00:00
|
|
|
}
|
2018-09-09 23:51:51 +00:00
|
|
|
}
|
2018-09-08 21:26:34 +00:00
|
|
|
close(info.sock_fd);
|
2018-09-03 22:54:02 +00:00
|
|
|
}
|