123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979 |
- /* snac - A simple, minimalistic ActivityPub instance */
- /* copyright (c) 2022 - 2024 grunfink et al. / MIT license */
- #include "xs.h"
- #include "xs_io.h"
- #include "xs_json.h"
- #include "xs_socket.h"
- #include "xs_unix_socket.h"
- #include "xs_httpd.h"
- #include "xs_mime.h"
- #include "xs_time.h"
- #include "xs_openssl.h"
- #include "xs_fcgi.h"
- #include "xs_html.h"
- #include "snac.h"
- #include <setjmp.h>
- #include <pthread.h>
- #include <semaphore.h>
- #include <fcntl.h>
- #include <stdint.h>
- #include <sys/resource.h> // for getrlimit()
- #include <sys/mman.h>
- #ifdef USE_POLL_FOR_SLEEP
- #include <poll.h>
- #endif
- /** server state **/
- srv_state *p_state = NULL;
- /** job control **/
- /* mutex to access the lists of jobs */
- static pthread_mutex_t job_mutex;
- /* semaphore to trigger job processing */
- static sem_t *job_sem;
- typedef struct job_fifo_item {
- struct job_fifo_item *next;
- xs_val *job;
- } job_fifo_item;
- static job_fifo_item *job_fifo_first = NULL;
- static job_fifo_item *job_fifo_last = NULL;
- /** other global data **/
- static jmp_buf on_break;
- /** code **/
- /* nodeinfo 2.0 template */
- const char *nodeinfo_2_0_template = ""
- "{\"version\":\"2.0\","
- "\"software\":{\"name\":\"snac\",\"version\":\"" VERSION "\"},"
- "\"protocols\":[\"activitypub\"],"
- "\"services\":{\"outbound\":[],\"inbound\":[]},"
- "\"usage\":{\"users\":{\"total\":%d,\"activeMonth\":%d,\"activeHalfyear\":%d},"
- "\"localPosts\":%d},"
- "\"openRegistrations\":false,\"metadata\":{}}";
- xs_str *nodeinfo_2_0(void)
- /* builds a nodeinfo json object */
- {
- int n_utotal = 0;
- int n_umonth = 0;
- int n_uhyear = 0;
- int n_posts = 0;
- xs *users = user_list();
- xs_list *p = users;
- const char *v;
- double now = (double)time(NULL);
- while (xs_list_iter(&p, &v)) {
- /* build the full path name to the last usage log */
- xs *llfn = xs_fmt("%s/user/%s/lastlog.txt", srv_basedir, v);
- double llsecs = now - mtime(llfn);
- if (llsecs < 60 * 60 * 24 * 30 * 6) {
- n_uhyear++;
- if (llsecs < 60 * 60 * 24 * 30)
- n_umonth++;
- }
- n_utotal++;
- /* build the file to each user public.idx */
- xs *pidxfn = xs_fmt("%s/user/%s/public.idx", srv_basedir, v);
- n_posts += index_len(pidxfn);
- }
- return xs_fmt(nodeinfo_2_0_template, n_utotal, n_umonth, n_uhyear, n_posts);
- }
- static xs_str *greeting_html(void)
- /* processes and returns greeting.html */
- {
- /* try to open greeting.html */
- xs *fn = xs_fmt("%s/greeting.html", srv_basedir);
- FILE *f;
- xs_str *s = NULL;
- if ((f = fopen(fn, "r")) != NULL) {
- s = xs_readall(f);
- fclose(f);
- /* replace %host% */
- s = xs_replace_i(s, "%host%", xs_dict_get(srv_config, "host"));
- const char *adm_email = xs_dict_get(srv_config, "admin_email");
- if (xs_is_null(adm_email) || *adm_email == '\0')
- adm_email = "the administrator of this instance";
- /* replace %admin_email */
- s = xs_replace_i(s, "%admin_email%", adm_email);
- /* does it have a %userlist% mark? */
- if (xs_str_in(s, "%userlist%") != -1) {
- const char *host = xs_dict_get(srv_config, "host");
- xs *list = user_list();
- xs_list *p = list;
- const xs_str *uid;
- xs_html *ul = xs_html_tag("ul",
- xs_html_attr("class", "snac-user-list"));
- p = list;
- while (xs_list_iter(&p, &uid)) {
- snac user;
- if (user_open(&user, uid)) {
- xs_html_add(ul,
- xs_html_tag("li",
- xs_html_tag("a",
- xs_html_attr("href", user.actor),
- xs_html_text("@"),
- xs_html_text(uid),
- xs_html_text("@"),
- xs_html_text(host),
- xs_html_text(" ("),
- xs_html_text(xs_dict_get(user.config, "name")),
- xs_html_text(")"))));
- user_free(&user);
- }
- }
- xs *s1 = xs_html_render(ul);
- s = xs_replace_i(s, "%userlist%", s1);
- }
- }
- return s;
- }
- const char *share_page = ""
- "<!DOCTYPE html>\n"
- "<html>\n"
- "<head>\n"
- "<title>%s - snac</title>\n"
- "<meta content=\"width=device-width, initial-scale=1, minimum-scale=1, user-scalable=no\" name=\"viewport\">"
- "<style>:root {color-scheme: light dark}</style>\n"
- "</head>\n"
- "<body><h1>%s link share</h1>\n"
- "<form method=\"get\" action=\"%s/share-bridge\">\n"
- "<textarea name=\"content\" rows=\"6\" wrap=\"virtual\" required=\"required\" style=\"width: 50em\">%s</textarea>\n"
- "<p>Login: <input type=\"text\" name=\"login\" autocapitalize=\"off\" required=\"required\"></p>\n"
- "<input type=\"submit\" value=\"OK\">\n"
- "</form><p>%s</p></body></html>\n"
- "";
- int server_get_handler(xs_dict *req, const char *q_path,
- char **body, int *b_size, char **ctype)
- /* basic server services */
- {
- int status = 0;
- /* is it the server root? */
- if (*q_path == '\0') {
- const xs_dict *q_vars = xs_dict_get(req, "q_vars");
- const char *t = NULL;
- if (xs_type(q_vars) == XSTYPE_DICT && (t = xs_dict_get(q_vars, "t"))) {
- /** search by tag **/
- int skip = 0;
- int show = xs_number_get(xs_dict_get(srv_config, "max_timeline_entries"));
- const char *v;
- if ((v = xs_dict_get(q_vars, "skip")) != NULL)
- skip = atoi(v);
- if ((v = xs_dict_get(q_vars, "show")) != NULL)
- show = atoi(v);
- xs *tl = tag_search(t, skip, show + 1);
- int more = 0;
- if (xs_list_len(tl) >= show + 1) {
- /* drop the last one */
- tl = xs_list_del(tl, -1);
- more = 1;
- }
- const char *accept = xs_dict_get(req, "accept");
- if (!xs_is_null(accept) && strcmp(accept, "application/rss+xml") == 0) {
- xs *link = xs_fmt("%s/?t=%s", srv_baseurl, t);
- *body = timeline_to_rss(NULL, tl, link, link, link);
- *ctype = "application/rss+xml; charset=utf-8";
- }
- else {
- xs *page = xs_fmt("?t=%s", t);
- xs *title = xs_fmt(L("Search results for tag #%s"), t);
- *body = html_timeline(NULL, tl, 0, skip, show, more, title, page, 0, NULL);
- }
- }
- else
- if (xs_type(xs_dict_get(srv_config, "show_instance_timeline")) == XSTYPE_TRUE) {
- /** instance timeline **/
- xs *tl = timeline_instance_list(0, 30);
- *body = html_timeline(NULL, tl, 0, 0, 0, 0,
- L("Recent posts by users in this instance"), NULL, 0, NULL);
- }
- else
- *body = greeting_html();
- if (*body)
- status = HTTP_STATUS_OK;
- }
- else
- if (strcmp(q_path, "/susie.png") == 0 || strcmp(q_path, "/favicon.ico") == 0 ) {
- status = HTTP_STATUS_OK;
- *body = xs_base64_dec(default_avatar_base64(), b_size);
- *ctype = "image/png";
- }
- else
- if (strcmp(q_path, "/.well-known/nodeinfo") == 0) {
- status = HTTP_STATUS_OK;
- *ctype = "application/json; charset=utf-8";
- *body = xs_fmt("{\"links\":["
- "{\"rel\":\"http:/" "/nodeinfo.diaspora.software/ns/schema/2.0\","
- "\"href\":\"%s/nodeinfo_2_0\"}]}",
- srv_baseurl);
- }
- else
- if (strcmp(q_path, "/.well-known/host-meta") == 0) {
- status = HTTP_STATUS_OK;
- *ctype = "application/xrd+xml";
- *body = xs_fmt("<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
- "<XRD>"
- "<Link rel=\"lrdd\" type=\"application/xrd+xml\" template=\"https://%s/.well-known/webfinger?resource={uri}\"/>"
- "</XRD>", xs_dict_get(srv_config, "host"));
- }
- else
- if (strcmp(q_path, "/nodeinfo_2_0") == 0) {
- status = HTTP_STATUS_OK;
- *ctype = "application/json; charset=utf-8";
- *body = nodeinfo_2_0();
- }
- else
- if (strcmp(q_path, "/robots.txt") == 0) {
- status = HTTP_STATUS_OK;
- *ctype = "text/plain";
- *body = xs_str_new("User-agent: *\n"
- "Disallow: /\n");
- }
- else
- if (strcmp(q_path, "/style.css") == 0) {
- FILE *f;
- xs *css_fn = xs_fmt("%s/style.css", srv_basedir);
- if ((f = fopen(css_fn, "r")) != NULL) {
- *body = xs_readall(f);
- fclose(f);
- status = HTTP_STATUS_OK;
- *ctype = "text/css";
- }
- }
- else
- if (strcmp(q_path, "/share") == 0) {
- const xs_dict *q_vars = xs_dict_get(req, "q_vars");
- const char *url = xs_dict_get(q_vars, "url");
- const char *text = xs_dict_get(q_vars, "text");
- xs *s = NULL;
- if (xs_type(text) == XSTYPE_STRING) {
- if (xs_type(url) == XSTYPE_STRING)
- s = xs_fmt("%s:\n\n%s\n", text, url);
- else
- s = xs_fmt("%s\n", text);
- }
- else
- s = xs_fmt("%s\n", url);
- status = HTTP_STATUS_OK;
- *ctype = "text/html";
- *body = xs_fmt(share_page,
- xs_dict_get(srv_config, "host"),
- xs_dict_get(srv_config, "host"),
- srv_baseurl,
- s,
- USER_AGENT
- );
- }
- if (status != 0)
- srv_debug(1, xs_fmt("server_get_handler serving '%s' %d", q_path, status));
- return status;
- }
- void httpd_connection(FILE *f)
- /* the connection processor */
- {
- xs *req;
- const char *method;
- int status = 0;
- xs_str *body = NULL;
- int b_size = 0;
- char *ctype = NULL;
- xs *headers = xs_dict_new();
- xs *q_path = NULL;
- xs *payload = NULL;
- xs *etag = NULL;
- xs *last_modified = NULL;
- xs *link = NULL;
- int p_size = 0;
- const char *p;
- int fcgi_id;
- if (p_state->use_fcgi)
- req = xs_fcgi_request(f, &payload, &p_size, &fcgi_id);
- else
- req = xs_httpd_request(f, &payload, &p_size);
- if (req == NULL) {
- /* probably because a timeout */
- fclose(f);
- return;
- }
- if (!(method = xs_dict_get(req, "method")) || !(p = xs_dict_get(req, "path"))) {
- /* missing needed headers; discard */
- fclose(f);
- return;
- }
- q_path = xs_dup(p);
- /* crop the q_path from leading / and the prefix */
- if (xs_endswith(q_path, "/"))
- q_path = xs_crop_i(q_path, 0, -1);
- p = xs_dict_get(srv_config, "prefix");
- if (xs_startswith(q_path, p))
- q_path = xs_crop_i(q_path, strlen(p), 0);
- if (strcmp(method, "GET") == 0 || strcmp(method, "HEAD") == 0) {
- /* cascade through */
- if (status == 0)
- status = server_get_handler(req, q_path, &body, &b_size, &ctype);
- if (status == 0)
- status = webfinger_get_handler(req, q_path, &body, &b_size, &ctype);
- if (status == 0)
- status = activitypub_get_handler(req, q_path, &body, &b_size, &ctype);
- #ifndef NO_MASTODON_API
- if (status == 0)
- status = oauth_get_handler(req, q_path, &body, &b_size, &ctype);
- if (status == 0)
- status = mastoapi_get_handler(req, q_path, &body, &b_size, &ctype, &link);
- #endif /* NO_MASTODON_API */
- if (status == 0)
- status = html_get_handler(req, q_path, &body, &b_size, &ctype, &etag, &last_modified);
- }
- else
- if (strcmp(method, "POST") == 0) {
- #ifndef NO_MASTODON_API
- if (status == 0)
- status = oauth_post_handler(req, q_path,
- payload, p_size, &body, &b_size, &ctype);
- if (status == 0)
- status = mastoapi_post_handler(req, q_path,
- payload, p_size, &body, &b_size, &ctype);
- #endif
- if (status == 0)
- status = activitypub_post_handler(req, q_path,
- payload, p_size, &body, &b_size, &ctype);
- if (status == 0)
- status = html_post_handler(req, q_path,
- payload, p_size, &body, &b_size, &ctype);
- }
- else
- if (strcmp(method, "PUT") == 0) {
- #ifndef NO_MASTODON_API
- if (status == 0)
- status = mastoapi_put_handler(req, q_path,
- payload, p_size, &body, &b_size, &ctype);
- #endif
- }
- else
- if (strcmp(method, "PATCH") == 0) {
- #ifndef NO_MASTODON_API
- if (status == 0)
- status = mastoapi_patch_handler(req, q_path,
- payload, p_size, &body, &b_size, &ctype);
- #endif
- }
- else
- if (strcmp(method, "OPTIONS") == 0) {
- const char *methods = "OPTIONS, GET, HEAD, POST, PUT, DELETE";
- headers = xs_dict_append(headers, "allow", methods);
- headers = xs_dict_append(headers, "access-control-allow-methods", methods);
- status = HTTP_STATUS_OK;
- }
- else
- if (strcmp(method, "DELETE") == 0) {
- #ifndef NO_MASTODON_API
- if (status == 0)
- status = mastoapi_delete_handler(req, q_path,
- payload, p_size, &body, &b_size, &ctype);
- #endif
- }
- /* unattended? it's an error */
- if (status == 0) {
- srv_archive_error("unattended_method", "unattended method", req, payload);
- srv_debug(1, xs_fmt("httpd_connection unattended %s %s", method, q_path));
- status = HTTP_STATUS_NOT_FOUND;
- }
- if (status == HTTP_STATUS_FORBIDDEN)
- body = xs_str_new("<h1>403 Forbidden</h1>");
- if (status == HTTP_STATUS_NOT_FOUND)
- body = xs_str_new("<h1>404 Not Found</h1>");
- if (status == HTTP_STATUS_BAD_REQUEST && body != NULL)
- body = xs_str_new("<h1>400 Bad Request</h1>");
- if (status == HTTP_STATUS_SEE_OTHER)
- headers = xs_dict_append(headers, "location", body);
- if (status == HTTP_STATUS_UNAUTHORIZED && body) {
- xs *www_auth = xs_fmt("Basic realm=\"@%s@%s snac login\"",
- body, xs_dict_get(srv_config, "host"));
- headers = xs_dict_append(headers, "WWW-Authenticate", www_auth);
- headers = xs_dict_append(headers, "Cache-Control", "no-cache, must-revalidate, max-age=0");
- }
- if (ctype == NULL)
- ctype = "text/html; charset=utf-8";
- headers = xs_dict_append(headers, "content-type", ctype);
- headers = xs_dict_append(headers, "x-creator", USER_AGENT);
- if (!xs_is_null(etag))
- headers = xs_dict_append(headers, "etag", etag);
- if (!xs_is_null(last_modified))
- headers = xs_dict_append(headers, "last-modified", last_modified);
- if (!xs_is_null(link))
- headers = xs_dict_append(headers, "Link", link);
- /* if there are any additional headers, add them */
- const xs_dict *more_headers = xs_dict_get(srv_config, "http_headers");
- if (xs_type(more_headers) == XSTYPE_DICT) {
- const char *k, *v;
- int c = 0;
- while (xs_dict_next(more_headers, &k, &v, &c))
- headers = xs_dict_set(headers, k, v);
- }
- if (b_size == 0 && body != NULL)
- b_size = strlen(body);
- /* if it was a HEAD, no body will be sent */
- if (strcmp(method, "HEAD") == 0)
- body = xs_free(body);
- headers = xs_dict_append(headers, "access-control-allow-origin", "*");
- headers = xs_dict_append(headers, "access-control-allow-headers", "*");
- if (p_state->use_fcgi)
- xs_fcgi_response(f, status, headers, body, b_size, fcgi_id);
- else
- xs_httpd_response(f, status, http_status_text(status), headers, body, b_size);
- fclose(f);
- srv_archive("RECV", NULL, req, payload, p_size, status, headers, body, b_size);
- /* JSON validation check */
- if (!xs_is_null(body) && strcmp(ctype, "application/json") == 0) {
- xs *j = xs_json_loads(body);
- if (j == NULL) {
- srv_log(xs_fmt("bad JSON"));
- srv_archive_error("bad_json", "bad JSON", req, body);
- }
- }
- xs_free(body);
- }
- void job_post(const xs_val *job, int urgent)
- /* posts a job for the threads to process it */
- {
- if (job != NULL) {
- /* lock the mutex */
- pthread_mutex_lock(&job_mutex);
- job_fifo_item *i = xs_realloc(NULL, sizeof(job_fifo_item));
- *i = (job_fifo_item){ NULL, xs_dup(job) };
- if (job_fifo_first == NULL)
- job_fifo_first = job_fifo_last = i;
- else
- if (urgent) {
- /* prepend */
- i->next = job_fifo_first;
- job_fifo_first = i;
- }
- else {
- /* append */
- job_fifo_last->next = i;
- job_fifo_last = i;
- }
- p_state->job_fifo_size++;
- if (p_state->job_fifo_size > p_state->peak_job_fifo_size)
- p_state->peak_job_fifo_size = p_state->job_fifo_size;
- /* unlock the mutex */
- pthread_mutex_unlock(&job_mutex);
- /* ask for someone to attend it */
- sem_post(job_sem);
- }
- }
- void job_wait(xs_val **job)
- /* waits for an available job */
- {
- *job = NULL;
- if (sem_wait(job_sem) == 0) {
- /* lock the mutex */
- pthread_mutex_lock(&job_mutex);
- /* dequeue */
- job_fifo_item *i = job_fifo_first;
- if (i != NULL) {
- job_fifo_first = i->next;
- if (job_fifo_first == NULL)
- job_fifo_last = NULL;
- *job = i->job;
- xs_free(i);
- p_state->job_fifo_size--;
- }
- /* unlock the mutex */
- pthread_mutex_unlock(&job_mutex);
- }
- }
- static void *job_thread(void *arg)
- /* job thread */
- {
- int pid = (int)(uintptr_t)arg;
- srv_debug(1, xs_fmt("job thread %d started", pid));
- for (;;) {
- xs *job = NULL;
- p_state->th_state[pid] = THST_WAIT;
- job_wait(&job);
- if (job == NULL) /* corrupted message? */
- continue;
- if (xs_type(job) == XSTYPE_FALSE) /* special message: exit */
- break;
- else
- if (xs_type(job) == XSTYPE_DATA) {
- /* it's a socket */
- FILE *f = NULL;
- p_state->th_state[pid] = THST_IN;
- xs_data_get(&f, job);
- if (f != NULL)
- httpd_connection(f);
- }
- else {
- /* it's a q_item */
- p_state->th_state[pid] = THST_QUEUE;
- process_queue_item(job);
- }
- }
- p_state->th_state[pid] = THST_STOP;
- srv_debug(1, xs_fmt("job thread %d stopped", pid));
- return NULL;
- }
- /* background thread sleep control */
- static pthread_mutex_t sleep_mutex;
- static pthread_cond_t sleep_cond;
- static void *background_thread(void *arg)
- /* background thread (queue management and other things) */
- {
- time_t purge_time;
- (void)arg;
- /* first purge time */
- purge_time = time(NULL) + 10 * 60;
- srv_log(xs_fmt("background thread started"));
- while (p_state->srv_running) {
- time_t t;
- int cnt = 0;
- p_state->th_state[0] = THST_QUEUE;
- {
- xs *list = user_list();
- char *p;
- const char *uid;
- /* process queues for all users */
- p = list;
- while (xs_list_iter(&p, &uid)) {
- snac snac;
- if (user_open(&snac, uid)) {
- cnt += process_user_queue(&snac);
- user_free(&snac);
- }
- }
- }
- /* global queue */
- cnt += process_queue();
- /* time to purge? */
- if ((t = time(NULL)) > purge_time) {
- /* next purge time is tomorrow */
- purge_time = t + 24 * 60 * 60;
- xs *q_item = xs_dict_new();
- q_item = xs_dict_append(q_item, "type", "purge");
- job_post(q_item, 0);
- }
- if (cnt == 0) {
- /* sleep 3 seconds */
- p_state->th_state[0] = THST_WAIT;
- #ifdef USE_POLL_FOR_SLEEP
- poll(NULL, 0, 3 * 1000);
- #else
- struct timespec ts;
- clock_gettime(CLOCK_REALTIME, &ts);
- ts.tv_sec += 3;
- pthread_mutex_lock(&sleep_mutex);
- while (pthread_cond_timedwait(&sleep_cond, &sleep_mutex, &ts) == 0);
- pthread_mutex_unlock(&sleep_mutex);
- #endif
- }
- }
- p_state->th_state[0] = THST_STOP;
- srv_log(xs_fmt("background thread stopped"));
- return NULL;
- }
- void term_handler(int s)
- {
- (void)s;
- longjmp(on_break, 1);
- }
- srv_state *srv_state_op(xs_str **fname, int op)
- /* opens or deletes the shared memory object */
- {
- int fd;
- srv_state *ss = NULL;
- if (*fname == NULL)
- *fname = xs_fmt("/%s_snac_state", xs_dict_get(srv_config, "host"));
- switch (op) {
- case 0: /* open for writing */
- #ifdef WITHOUT_SHM
- errno = ENOTSUP;
- #else
- if ((fd = shm_open(*fname, O_CREAT | O_RDWR, 0666)) != -1) {
- ftruncate(fd, sizeof(*ss));
- if ((ss = mmap(0, sizeof(*ss), PROT_READ | PROT_WRITE,
- MAP_SHARED, fd, 0)) == MAP_FAILED)
- ss = NULL;
- close(fd);
- }
- #endif
- if (ss == NULL) {
- /* shared memory error: just create a plain structure */
- srv_log(xs_fmt("warning: shm object error (%s)", strerror(errno)));
- ss = malloc(sizeof(*ss));
- }
- /* init structure */
- *ss = (srv_state){0};
- ss->s_size = sizeof(*ss);
- break;
- case 1: /* open for reading */
- #ifdef WITHOUT_SHM
- errno = ENOTSUP;
- #else
- if ((fd = shm_open(*fname, O_RDONLY, 0666)) != -1) {
- if ((ss = mmap(0, sizeof(*ss), PROT_READ, MAP_SHARED, fd, 0)) == MAP_FAILED)
- ss = NULL;
- close(fd);
- }
- #endif
- if (ss == NULL) {
- /* shared memory error */
- srv_log(xs_fmt("error: shm object error (%s) server not running?", strerror(errno)));
- }
- else
- if (ss->s_size != sizeof(*ss)) {
- srv_log(xs_fmt("error: struct size mismatch (%d != %d)",
- ss->s_size, sizeof(*ss)));
- munmap(ss, sizeof(*ss));
- ss = NULL;
- }
- break;
- case 2: /* unlink */
- #ifndef WITHOUT_SHM
- if (*fname)
- shm_unlink(*fname);
- #endif
- break;
- }
- return ss;
- }
- void httpd(void)
- /* starts the server */
- {
- const char *address = NULL;
- const char *port = NULL;
- xs *full_address = NULL;
- int rs;
- pthread_t threads[MAX_THREADS] = {0};
- int n;
- xs *sem_name = NULL;
- xs *shm_name = NULL;
- sem_t anon_job_sem;
- xs *pidfile = xs_fmt("%s/server.pid", srv_basedir);
- int pidfd;
- {
- /* do some pidfile locking acrobatics */
- if ((pidfd = open(pidfile, O_RDWR | O_CREAT, 0660)) == -1) {
- srv_log(xs_fmt("Cannot create pidfile %s -- cannot continue", pidfile));
- return;
- }
- if (lockf(pidfd, F_TLOCK, 1) == -1) {
- srv_log(xs_fmt("Cannot lock pidfile %s -- server already running?", pidfile));
- close(pidfd);
- return;
- }
- ftruncate(pidfd, 0);
- xs *s = xs_fmt("%d\n", (int)getpid());
- write(pidfd, s, strlen(s));
- }
- address = xs_dict_get(srv_config, "address");
- if (*address == '/') {
- rs = xs_unix_socket_server(address, NULL);
- full_address = xs_fmt("unix:%s", address);
- }
- else {
- port = xs_number_str(xs_dict_get(srv_config, "port"));
- full_address = xs_fmt("%s:%s", address, port);
- rs = xs_socket_server(address, port);
- }
- if (rs == -1) {
- srv_log(xs_fmt("cannot bind socket to %s", full_address));
- return;
- }
- /* setup the server stat structure */
- p_state = srv_state_op(&shm_name, 0);
- p_state->srv_start_time = time(NULL);
- p_state->use_fcgi = xs_type(xs_dict_get(srv_config, "fastcgi")) == XSTYPE_TRUE;
- p_state->srv_running = 1;
- signal(SIGPIPE, SIG_IGN);
- signal(SIGTERM, term_handler);
- signal(SIGINT, term_handler);
- srv_log(xs_fmt("httpd%s start %s %s", p_state->use_fcgi ? " (FastCGI)" : "",
- full_address, USER_AGENT));
- /* show the number of usable file descriptors */
- struct rlimit r;
- getrlimit(RLIMIT_NOFILE, &r);
- srv_debug(1, xs_fmt("available (rlimit) fds: %d (cur) / %d (max)",
- (int) r.rlim_cur, (int) r.rlim_max));
- /* initialize the job control engine */
- pthread_mutex_init(&job_mutex, NULL);
- sem_name = xs_fmt("/job_%d", getpid());
- job_sem = sem_open(sem_name, O_CREAT, 0644, 0);
- if (job_sem == NULL) {
- /* error opening a named semaphore; try with an anonymous one */
- if (sem_init(&anon_job_sem, 0, 0) != -1)
- job_sem = &anon_job_sem;
- }
- if (job_sem == NULL) {
- srv_log(xs_fmt("fatal error: cannot create semaphore -- cannot continue"));
- return;
- }
- /* initialize sleep control */
- pthread_mutex_init(&sleep_mutex, NULL);
- pthread_cond_init(&sleep_cond, NULL);
- p_state->n_threads = xs_number_get(xs_dict_get(srv_config, "num_threads"));
- #ifdef _SC_NPROCESSORS_ONLN
- if (p_state->n_threads == 0) {
- /* get number of CPUs on the machine */
- p_state->n_threads = sysconf(_SC_NPROCESSORS_ONLN);
- }
- #endif
- if (p_state->n_threads < 4)
- p_state->n_threads = 4;
- if (p_state->n_threads > MAX_THREADS)
- p_state->n_threads = MAX_THREADS;
- srv_debug(0, xs_fmt("using %d threads", p_state->n_threads));
- /* thread #0 is the background thread */
- pthread_create(&threads[0], NULL, background_thread, NULL);
- /* the rest of threads are for job processing */
- char *ptr = (char *) 0x1;
- for (n = 1; n < p_state->n_threads; n++)
- pthread_create(&threads[n], NULL, job_thread, ptr++);
- if (setjmp(on_break) == 0) {
- for (;;) {
- int cs = xs_socket_accept(rs);
- if (cs != -1) {
- FILE *f = fdopen(cs, "r+");
- xs *job = xs_data_new(&f, sizeof(FILE *));
- job_post(job, 1);
- }
- else
- break;
- }
- }
- p_state->srv_running = 0;
- /* send as many exit jobs as working threads */
- for (n = 1; n < p_state->n_threads; n++)
- job_post(xs_stock(XSTYPE_FALSE), 0);
- /* wait for all the threads to exit */
- for (n = 0; n < p_state->n_threads; n++)
- pthread_join(threads[n], NULL);
- sem_close(job_sem);
- sem_unlink(sem_name);
- srv_state_op(&shm_name, 2);
- xs *uptime = xs_str_time_diff(time(NULL) - p_state->srv_start_time);
- srv_log(xs_fmt("httpd%s stop %s (run time: %s)",
- p_state->use_fcgi ? " (FastCGI)" : "",
- full_address, uptime));
- unlink(pidfile);
- }
|