Программирование для Linux. Профессиональный подход
Шрифт:
/* Закрытие библиотеки. */
dlclose(module->handle);
/* Удаление строки с именем модуля. */
free((char*)module->name);
/* Удаление объекта module. */
free(module);
}
Каждый модуль содержится в файле совместно используемой библиотеки (см. раздел 2.3.2, "Совместно используемые библиотеки") и должен экспортировать функцию
module_generate
. Эта функция генерирует HTML-код Web-страницы и записывает его в сокет, дескриптор
В файле
module.c
определены две функции. ■ Функция
module_open
пытается загрузить серверный модуль с указанным именем. Файл модуля имеет расширение .so
, так как это совместно используемая библиотека. Функция открывает библиотеку с помощью функции dlopen
и ищет в библиотеке константу module_generate
посредством функции dlsym
(описаны в разделе 2.3.6, "Динамическая загрузка и выгрузка"). Если библиотеку не удалось открыть или в ней не обнаружена экспортируемая константа module_generate
, возвращается значение NULL
. В противном случае выделяется и возвращается объект module
. ■ Функция
module_close
закрывает совместно используемую библиотеку, соответствующую указанному модулю, и удаляет объект module
. В файле module.c определена также глобальная переменная module_dir. В ней записано имя каталога, в котором функция module_open будет искать совместно используемые библиотеки.
11.2.3. Сервер
Файл
server.c
(листинг 11.4) представляет собой реализацию простейшего HTTP-сервера. Листинг 11.4. (server.c) Реализация HTTP-сервера
#include <arpa/inet.h>
#include <assert.h>
#include <errno.h>
#include <netinet/in.h>
#include <signal.h>
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/wait.h>
#include <unistd.h>
#include "server.h"
/* HTTP-ответ и заголовок, возвращаемые в случае
успешной обработки запроса. */
static char* ok_response =
"HTTP/1.0 100 OK\n"
"Content-type: text/html\n"
"\n";
/* HTTP-ответ, заголовок и тело страницы на случай
непонятного запроса. */
static char* bad_request_response =
"HTTP/1.0 400 Bad Reguest\n"
"Content-type: text/html\n"
"\n"
"<html>\n"
" <body>\n"
" <h1>Bad Request</h1>\n"
" This server did not understand your request.\n"
" </body>\n"
"</html>\n";
/* HTTP-ответ,
заголовок и шаблон страницы на случай,
когда запрашиваемый документ не найден. */
static char* not_found_response_template =
"HTTP/1.0 404 Not Found\n"
"Content-type: text/html\n"
"\n"
"<html>\n"
" <body>\n"
" <h1>Not Found</h1>\n"
" The requested URL %s was not found on this server.\n"
" </body>\n"
"</html>\n";
/* HTTP-ответ, заголовок к шаблон страницы на случай,
когда запрашивается непонятный метод */
static char* bad_method_response_template =
"HTTP/1.0 501 Method Not Implemented\n"
"Content-type: text/html\n"
"\n"
"<html>\n"
" <body>\n"
" <h1>Method Not Implemented</h1>\n"
" The method %s is not implemented by this server.\n"
" </body>\n"
"</html>\n";
/* Обработчик сигнала SIGCHLD, удаляющий завершившиеся
дочерние процессы. */
static void clean_up_child_process(int signal_number) {
int status;
wait(&status);
}
/* Обработка HTTP-запроса "GET" к странице PAGE и
запись результата в файл с дескриптором CONNECTION_FD. */
static void handle_get(int connection_fd, const char* page) {
struct server_module* module = NULL;
/* Убеждаемся, что имя страницы начинается с косой черты и
не содержит других символов косой черты, так как
подкаталоги не поддерживаются. */
if (*page == '/' && strchr(page + 1, '/') == NULL) {
char module_file_name[64];
/* Имя страницы правильно. Формируем имя модуля, добавляя
расширение ".so" к имени страницы. */
snprintf(module_file_name, sizeof(module_file_name),
"%s.so", page + 1);
/* Попытка открытия модуля. */
module = module_open(module_file_name);
Поделиться с друзьями: