/* snac - A simple, minimalistic ActivityPub instance */ /* copyright (c) 2022 - 2024 grunfink et al. / MIT license */ #include "xs.h" #include "xs_hex.h" #include "xs_io.h" #include "xs_json.h" #include "xs_openssl.h" #include "xs_glob.h" #include "xs_set.h" #include "xs_time.h" #include "xs_regex.h" #include "xs_match.h" #include "xs_unicode.h" #include "xs_random.h" #include "snac.h" #include #include #include #include #include #include double disk_layout = 2.7; /* storage serializer */ pthread_mutex_t data_mutex = {0}; int snac_upgrade(xs_str **error); int srv_open(const char *basedir, int auto_upgrade) /* opens a server */ { int ret = 0; xs *cfg_file = NULL; FILE *f; xs_str *error = NULL; pthread_mutex_init(&data_mutex, NULL); srv_basedir = xs_str_new(basedir); if (xs_endswith(srv_basedir, "/")) srv_basedir = xs_crop_i(srv_basedir, 0, -1); cfg_file = xs_fmt("%s/server.json", basedir); if ((f = fopen(cfg_file, "r")) == NULL) error = xs_fmt("ERROR: cannot open '%s'", cfg_file); else { /* read full config file */ srv_config = xs_json_load(f); fclose(f); /* parse */ if (srv_config == NULL) error = xs_fmt("ERROR: cannot parse '%s'", cfg_file); else { const char *host; const char *prefix; const char *dbglvl; const char *proto; host = xs_dict_get(srv_config, "host"); prefix = xs_dict_get(srv_config, "prefix"); dbglvl = xs_dict_get(srv_config, "dbglevel"); proto = xs_dict_get_def(srv_config, "protocol", "https"); if (host == NULL || prefix == NULL) error = xs_str_new("ERROR: cannot get server data"); else { srv_baseurl = xs_fmt("%s:/" "/%s%s", proto, host, prefix); dbglevel = (int) xs_number_get(dbglvl); if ((dbglvl = getenv("DEBUG")) != NULL) { dbglevel = atoi(dbglvl); error = xs_fmt("DEBUG level set to %d from environment", dbglevel); } if (auto_upgrade) ret = snac_upgrade(&error); else { if (xs_number_get(xs_dict_get(srv_config, "layout")) < disk_layout) error = xs_fmt("ERROR: disk layout changed - execute 'snac upgrade' first"); else ret = 1; } } } } if (error != NULL) srv_log(error); /* create the queue/ subdir, just in case */ xs *qdir = xs_fmt("%s/queue", srv_basedir); mkdirx(qdir); xs *ibdir = xs_fmt("%s/inbox", srv_basedir); mkdirx(ibdir); xs *tmpdir = xs_fmt("%s/tmp", srv_basedir); mkdirx(tmpdir); #ifdef __APPLE__ /* Apple uses st_atimespec instead of st_atim etc */ #define st_atim st_atimespec #define st_ctim st_ctimespec #define st_mtim st_mtimespec #endif #ifdef __OpenBSD__ if (xs_is_true(xs_dict_get(srv_config, "disable_openbsd_security"))) { srv_debug(1, xs_dup("OpenBSD security disabled by admin")); } else { int smail = !xs_is_true(xs_dict_get(srv_config, "disable_email_notifications")); const char *address = xs_dict_get(srv_config, "address"); srv_debug(1, xs_fmt("Calling unveil()")); unveil(basedir, "rwc"); unveil("/tmp", "rwc"); unveil("/etc/resolv.conf", "r"); unveil("/etc/hosts", "r"); unveil("/etc/ssl/openssl.cnf", "r"); unveil("/etc/ssl/cert.pem", "r"); unveil("/usr/share/zoneinfo", "r"); if (smail) unveil("/usr/sbin/sendmail", "x"); if (*address == '/') unveil(address, "rwc"); unveil(NULL, NULL); srv_debug(1, xs_fmt("Calling pledge()")); xs *p = xs_str_new("stdio rpath wpath cpath flock inet proc dns fattr"); if (smail) p = xs_str_cat(p, " exec"); if (*address == '/') p = xs_str_cat(p, " unix"); pledge(p, NULL); } #endif /* __OpenBSD__ */ /* read (and drop) emojis.json, possibly creating it */ xs_free(emojis()); /* if style.css does not exist, create it */ xs *css_fn = xs_fmt("%s/style.css", srv_basedir); if (mtime(css_fn) == 0) { srv_log(xs_fmt("Writing style.css")); write_default_css(); } /* create the proxy token seed */ { char rnd[16]; xs_rnd_buf(rnd, sizeof(rnd)); srv_proxy_token_seed = xs_hex_enc(rnd, sizeof(rnd)); } return ret; } void srv_free(void) { xs_free(srv_basedir); xs_free(srv_config); xs_free(srv_baseurl); pthread_mutex_destroy(&data_mutex); } void user_free(snac *snac) /* frees a user snac */ { xs_free(snac->uid); xs_free(snac->basedir); xs_free(snac->config); xs_free(snac->config_o); xs_free(snac->key); xs_free(snac->links); xs_free(snac->actor); xs_free(snac->md5); } int user_open(snac *user, const char *uid) /* opens a user */ { int ret = 0; *user = (snac){0}; if (validate_uid(uid)) { xs *cfg_file = NULL; FILE *f; xs *t = xs_fmt("%s/user/%s", srv_basedir, uid); if (mtime(t) == 0.0) { /* user folder does not exist; try with a different case */ xs *lcuid = xs_tolower_i(xs_dup(uid)); xs *ulist = user_list(); xs_list *p = ulist; const xs_str *v; while (xs_list_iter(&p, &v)) { xs *v2 = xs_tolower_i(xs_dup(v)); if (strcmp(lcuid, v2) == 0) { user->uid = xs_dup(v); break; } } } else user->uid = xs_str_new(uid); if (user->uid == NULL) return ret; user->basedir = xs_fmt("%s/user/%s", srv_basedir, user->uid); cfg_file = xs_fmt("%s/user.json", user->basedir); if ((f = fopen(cfg_file, "r")) != NULL) { /* read full config file */ user->config = xs_json_load(f); fclose(f); if (user->config != NULL) { xs *key_file = xs_fmt("%s/key.json", user->basedir); if ((f = fopen(key_file, "r")) != NULL) { user->key = xs_json_load(f); fclose(f); if (user->key != NULL) { user->actor = xs_fmt("%s/%s", srv_baseurl, user->uid); user->md5 = xs_md5_hex(user->actor, strlen(user->actor)); /* everything is ok right now */ ret = 1; /* does it have a configuration override? */ xs *cfg_file_o = xs_fmt("%s/user_o.json", user->basedir); if ((f = fopen(cfg_file_o, "r")) != NULL) { user->config_o = xs_json_load(f); fclose(f); if (user->config_o == NULL) srv_log(xs_fmt("error parsing '%s'", cfg_file_o)); } if (user->config_o == NULL) user->config_o = xs_dict_new(); } else srv_log(xs_fmt("error parsing '%s'", key_file)); } else srv_log(xs_fmt("error opening '%s' %d", key_file, errno)); } else srv_log(xs_fmt("error parsing '%s'", cfg_file)); } else srv_debug(2, xs_fmt("error opening '%s' %d", cfg_file, errno)); /* verified links */ xs *links_file = xs_fmt("%s/links.json", user->basedir); if ((f = fopen(links_file, "r")) != NULL) { user->links = xs_json_load(f); fclose(f); } } else srv_debug(1, xs_fmt("invalid user '%s'", uid)); if (!ret) user_free(user); return ret; } xs_list *user_list(void) /* returns the list of user ids */ { xs *spec = xs_fmt("%s/user/" "*", srv_basedir); return xs_glob(spec, 1, 0); } int user_open_by_md5(snac *snac, const char *md5) /* iterates all users searching by md5 */ { xs *ulist = user_list(); xs_list *p = ulist; const xs_str *v; while (xs_list_iter(&p, &v)) { user_open(snac, v); if (strcmp(snac->md5, md5) == 0) return 1; user_free(snac); } return 0; } int user_persist(snac *snac, int publish) /* store user */ { xs *fn = xs_fmt("%s/user.json", snac->basedir); xs *bfn = xs_fmt("%s.bak", fn); FILE *f; if (publish) { /* check if any of the relevant fields have really changed */ if ((f = fopen(fn, "r")) != NULL) { xs *old = xs_json_load(f); fclose(f); if (old != NULL) { int nw = 0; const char *fields[] = { "header", "avatar", "name", "bio", "metadata", NULL }; for (int n = 0; fields[n]; n++) { const char *of = xs_dict_get(old, fields[n]); const char *nf = xs_dict_get(snac->config, fields[n]); if (of == NULL && nf == NULL) continue; if (xs_type(of) != XSTYPE_STRING || xs_type(nf) != XSTYPE_STRING || strcmp(of, nf)) { nw = 1; break; } } if (!nw) publish = 0; } } } rename(fn, bfn); if ((f = fopen(fn, "w")) != NULL) { xs_json_dump(snac->config, 4, f); fclose(f); } else rename(bfn, fn); history_del(snac, "timeline.html_"); timeline_touch(snac); if (publish) { xs *a_msg = msg_actor(snac); xs *u_msg = msg_update(snac, a_msg); enqueue_message(snac, u_msg); enqueue_verify_links(snac); } return 0; } double mtime_nl(const char *fn, int *n_link) /* returns the mtime and number of links of a file or directory, or 0.0 */ { struct stat st; double r = 0.0; int n = 0; if (fn && stat(fn, &st) != -1) { r = (double) st.st_mtim.tv_sec; n = st.st_nlink; } if (n_link) *n_link = n; return r; } #define MIN(v1, v2) ((v1) < (v2) ? (v1) : (v2)) double f_ctime(const char *fn) /* returns the ctime of a file or directory, or 0.0 */ { struct stat st; double r = 0.0; if (fn && stat(fn, &st) != -1) { /* return the lowest of ctime and mtime; there are operations that change the ctime, like link() */ r = (double) MIN(st.st_ctim.tv_sec, st.st_mtim.tv_sec); } return r; } int is_md5_hex(const char *md5) { return xs_is_hex(md5) && strlen(md5) == MD5_HEX_SIZE - 1; } /** database 2.1+ **/ /** indexes **/ int index_add_md5(const char *fn, const char *md5) /* adds an md5 to an index */ { int status = HTTP_STATUS_CREATED; FILE *f; if (!is_md5_hex(md5)) { srv_log(xs_fmt("index_add_md5: bad md5 %s %s", fn, md5)); return HTTP_STATUS_BAD_REQUEST; } pthread_mutex_lock(&data_mutex); if ((f = fopen(fn, "a")) != NULL) { flock(fileno(f), LOCK_EX); /* ensure the position is at the end after getting the lock */ fseek(f, 0, SEEK_END); fprintf(f, "%s\n", md5); fclose(f); } else status = HTTP_STATUS_INTERNAL_SERVER_ERROR; pthread_mutex_unlock(&data_mutex); return status; } int index_add(const char *fn, const char *id) /* adds an id to an index */ { xs *md5 = xs_md5_hex(id, strlen(id)); return index_add_md5(fn, md5); } int index_del_md5(const char *fn, const char *md5) /* deletes an md5 from an index */ { int status = HTTP_STATUS_NOT_FOUND; FILE *f; pthread_mutex_lock(&data_mutex); if ((f = fopen(fn, "r+")) != NULL) { char line[256]; while (fgets(line, sizeof(line), f) != NULL) { line[MD5_HEX_SIZE - 1] = '\0'; if (strcmp(line, md5) == 0) { /* found! just rewind, overwrite it with garbage and an eventual call to index_gc() will clean it [yes: this breaks index_len()] */ fseek(f, -MD5_HEX_SIZE, SEEK_CUR); fwrite("-", 1, 1, f); status = HTTP_STATUS_OK; break; } } fclose(f); } else status = HTTP_STATUS_GONE; pthread_mutex_unlock(&data_mutex); return status; } int index_del(const char *fn, const char *id) /* deletes an id from an index */ { xs *md5 = xs_md5_hex(id, strlen(id)); return index_del_md5(fn, md5); } int index_gc(const char *fn) /* garbage-collects an index, deleting objects that are not here */ { FILE *i, *o; int gc = -1; pthread_mutex_lock(&data_mutex); if ((i = fopen(fn, "r")) != NULL) { xs *nfn = xs_fmt("%s.new", fn); char line[256]; if ((o = fopen(nfn, "w")) != NULL) { gc = 0; while (fgets(line, sizeof(line), i) != NULL) { line[MD5_HEX_SIZE - 1] = '\0'; if (line[0] != '-' && object_here_by_md5(line)) fprintf(o, "%s\n", line); else gc++; } fclose(o); xs *ofn = xs_fmt("%s.bak", fn); unlink(ofn); link(fn, ofn); rename(nfn, fn); } fclose(i); } pthread_mutex_unlock(&data_mutex); return gc; } int index_in_md5(const char *fn, const char *md5) /* checks if the md5 is already in the index */ { FILE *f; int ret = 0; if ((f = fopen(fn, "r")) != NULL) { flock(fileno(f), LOCK_SH); char line[256]; while (!ret && fgets(line, sizeof(line), f) != NULL) { line[MD5_HEX_SIZE - 1] = '\0'; if (strcmp(line, md5) == 0) ret = 1; } fclose(f); } return ret; } int index_in(const char *fn, const char *id) /* checks if the object id is already in the index */ { xs *md5 = xs_md5_hex(id, strlen(id)); return index_in_md5(fn, md5); } int index_first(const char *fn, char md5[MD5_HEX_SIZE]) /* reads the first entry of an index */ { FILE *f; int ret = 0; if ((f = fopen(fn, "r")) != NULL) { if (fread(md5, MD5_HEX_SIZE, 1, f)) { md5[MD5_HEX_SIZE - 1] = '\0'; ret = 1; } fclose(f); } return ret; } int index_len(const char *fn) /* returns the number of elements in an index */ { struct stat st; int len = 0; if (stat(fn, &st) != -1) len = st.st_size / MD5_HEX_SIZE; return len; } xs_list *index_list(const char *fn, int max) /* returns an index as a list */ { xs_list *list = xs_list_new(); FILE *f; int n = 0; if ((f = fopen(fn, "r")) != NULL) { flock(fileno(f), LOCK_SH); char line[256]; while (n < max && fgets(line, sizeof(line), f) != NULL) { if (line[0] != '-') { line[MD5_HEX_SIZE - 1] = '\0'; list = xs_list_append(list, line); n++; } } fclose(f); } return list; } int index_desc_next(FILE *f, char md5[MD5_HEX_SIZE]) /* reads the next entry of a desc index */ { for (;;) { /* move backwards 2 entries */ if (fseek(f, MD5_HEX_SIZE * -2, SEEK_CUR) == -1) return 0; /* read and md5 */ if (!fread(md5, MD5_HEX_SIZE, 1, f)) return 0; if (md5[0] != '-') break; } md5[MD5_HEX_SIZE - 1] = '\0'; return 1; } int index_desc_first(FILE *f, char md5[MD5_HEX_SIZE], int skip) /* reads the first entry of a desc index */ { /* try to position at the end and then back to the first element */ if (fseek(f, 0, SEEK_END) || fseek(f, (skip + 1) * -MD5_HEX_SIZE, SEEK_CUR)) return 0; /* try to read an md5 */ if (!fread(md5, MD5_HEX_SIZE, 1, f)) return 0; /* null-terminate */ md5[MD5_HEX_SIZE - 1] = '\0'; /* deleted? retry next */ if (md5[0] == '-') return index_desc_next(f, md5); return 1; } xs_list *index_list_desc(const char *fn, int skip, int show) /* returns an index as a list, in reverse order */ { xs_list *list = xs_list_new(); FILE *f; if ((f = fopen(fn, "r")) != NULL) { char md5[MD5_HEX_SIZE]; if (index_desc_first(f, md5, skip)) { int n = 1; do { list = xs_list_append(list, md5); } while (n++ < show && index_desc_next(f, md5)); } fclose(f); } return list; } /** objects **/ static xs_str *_object_fn_by_md5(const char *md5, const char *func) { xs *bfn = xs_fmt("%s/object/%c%c", srv_basedir, md5[0], md5[1]); xs_str *ret; int ok = 1; /* an object deleted from an index; fail but don't bark */ if (md5[0] == '-') ok = 0; else if (!is_md5_hex(md5)) { srv_log(xs_fmt("_object_fn_by_md5() [from %s()]: bad md5 '%s'", func, md5)); ok = 0; } if (ok) { mkdirx(bfn); ret = xs_fmt("%s/%s.json", bfn, md5); } else ret = xs_fmt("%s/object/invalid/invalid.json", srv_basedir); return ret; } static xs_str *_object_fn(const char *id) { xs *md5 = xs_md5_hex(id, strlen(id)); return _object_fn_by_md5(md5, "_object_fn"); } int object_here_by_md5(const char *id) /* checks if an object is already downloaded */ { xs *fn = _object_fn_by_md5(id, "object_here_by_md5"); return mtime(fn) > 0.0; } int object_here(const char *id) /* checks if an object is already downloaded */ { xs *fn = _object_fn(id); return mtime(fn) > 0.0; } int object_get_by_md5(const char *md5, xs_dict **obj) /* returns a stored object, optionally of the requested type */ { int status = HTTP_STATUS_NOT_FOUND; xs *fn = _object_fn_by_md5(md5, "object_get_by_md5"); FILE *f; if ((f = fopen(fn, "r")) != NULL) { *obj = xs_json_load(f); fclose(f); if (*obj) status = HTTP_STATUS_OK; } else *obj = NULL; return status; } int object_get(const char *id, xs_dict **obj) /* returns a stored object, optionally of the requested type */ { xs *md5 = xs_md5_hex(id, strlen(id)); return object_get_by_md5(md5, obj); } int _object_add(const char *id, const xs_dict *obj, int ow) /* stores an object */ { int status = HTTP_STATUS_CREATED; /* Created */ xs *fn = _object_fn(id); FILE *f; if (mtime(fn) > 0.0) { if (!ow) { /* object already here */ srv_debug(1, xs_fmt("object_add object already here %s", id)); return HTTP_STATUS_NO_CONTENT; } else status = HTTP_STATUS_OK; } if ((f = fopen(fn, "w")) != NULL) { flock(fileno(f), LOCK_EX); xs_json_dump(obj, 4, f); fclose(f); /* does this object has a parent? */ const char *in_reply_to = get_in_reply_to(obj); if (!xs_is_null(in_reply_to) && *in_reply_to) { /* update the children index of the parent */ xs *c_idx = _object_fn(in_reply_to); c_idx = xs_replace_i(c_idx, ".json", "_c.idx"); if (!index_in(c_idx, id)) { index_add(c_idx, id); srv_debug(1, xs_fmt("object_add added child %s to %s", id, c_idx)); } else srv_debug(1, xs_fmt("object_add %s child already in %s", id, c_idx)); /* create a one-element index with the parent */ xs *p_idx = xs_replace(fn, ".json", "_p.idx"); if (mtime(p_idx) == 0.0) { index_add(p_idx, in_reply_to); srv_debug(1, xs_fmt("object_add added parent %s to %s", in_reply_to, p_idx)); } } } else { srv_log(xs_fmt("object_add error writing %s (errno: %d)", fn, errno)); status = HTTP_STATUS_INTERNAL_SERVER_ERROR; } srv_debug(1, xs_fmt("object_add %s %s %d", id, fn, status)); return status; } int object_add(const char *id, const xs_dict *obj) /* stores an object */ { return _object_add(id, obj, 0); } int object_add_ow(const char *id, const xs_dict *obj) /* stores an object (overwriting allowed) */ { return _object_add(id, obj, 1); } int object_del_by_md5(const char *md5) /* deletes an object by its md5 */ { int status = HTTP_STATUS_NOT_FOUND; xs *fn = _object_fn_by_md5(md5, "object_del_by_md5"); if (unlink(fn) != -1) { status = HTTP_STATUS_OK; /* also delete associated indexes */ xs *spec = xs_dup(fn); spec = xs_replace_i(spec, ".json", "*.idx"); xs *files = xs_glob(spec, 0, 0); char *p; const char *v; p = files; while (xs_list_iter(&p, &v)) { srv_debug(1, xs_fmt("object_del index %s", v)); unlink(v); } } srv_debug(1, xs_fmt("object_del %s %d", fn, status)); return status; } int object_del(const char *id) /* deletes an object */ { xs *md5 = xs_md5_hex(id, strlen(id)); return object_del_by_md5(md5); } int object_del_if_unref(const char *id) /* deletes an object if its n_links < 2 */ { xs *fn = _object_fn(id); int n_links; int ret = 0; if (mtime_nl(fn, &n_links) > 0.0 && n_links < 2) ret = object_del(id); return ret; } double object_ctime_by_md5(const char *md5) { xs *fn = _object_fn_by_md5(md5, "object_ctime_by_md5"); return f_ctime(fn); } double object_ctime(const char *id) { xs *md5 = xs_md5_hex(id, strlen(id)); return object_ctime_by_md5(md5); } double object_mtime_by_md5(const char *md5) { xs *fn = _object_fn_by_md5(md5, "object_mtime_by_md5"); return mtime(fn); } double object_mtime(const char *id) { xs *md5 = xs_md5_hex(id, strlen(id)); return object_mtime_by_md5(md5); } void object_touch(const char *id) { xs *md5 = xs_md5_hex(id, strlen(id)); xs *fn = _object_fn_by_md5(md5, "object_touch"); if (mtime(fn)) utimes(fn, NULL); } xs_str *_object_index_fn(const char *id, const char *idxsfx) /* returns the filename of an object's index */ { xs_str *fn = _object_fn(id); return xs_replace_i(fn, ".json", idxsfx); } int object_likes_len(const char *id) /* returns the number of likes (without reading the index) */ { xs *fn = _object_index_fn(id, "_l.idx"); return index_len(fn); } int object_announces_len(const char *id) /* returns the number of announces (without reading the index) */ { xs *fn = _object_index_fn(id, "_a.idx"); return index_len(fn); } xs_list *object_children(const char *id) /* returns the list of an object's children */ { xs *fn = _object_index_fn(id, "_c.idx"); return index_list(fn, XS_ALL); } xs_list *object_likes(const char *id) { xs *fn = _object_index_fn(id, "_l.idx"); return index_list(fn, XS_ALL); } xs_list *object_announces(const char *id) { xs *fn = _object_index_fn(id, "_a.idx"); return index_list(fn, XS_ALL); } int object_parent(const char *md5, char parent[MD5_HEX_SIZE]) /* returns the object parent, if any */ { xs *fn = _object_fn_by_md5(md5, "object_parent"); fn = xs_replace_i(fn, ".json", "_p.idx"); return index_first(fn, parent); } int object_admire(const char *id, const char *actor, int like) /* actor likes or announces this object */ { int status = HTTP_STATUS_OK; xs *fn = _object_fn(id); fn = xs_replace_i(fn, ".json", like ? "_l.idx" : "_a.idx"); if (!index_in(fn, actor)) { status = index_add(fn, actor); srv_debug(1, xs_fmt("object_admire (%s) %s %s", like ? "Like" : "Announce", actor, fn)); } return status; } int object_unadmire(const char *id, const char *actor, int like) /* actor no longer likes or announces this object */ { int status; xs *fn = _object_fn(id); fn = xs_replace_i(fn, ".json", like ? "_l.idx" : "_a.idx"); status = index_del(fn, actor); if (valid_status(status)) index_gc(fn); srv_debug(0, xs_fmt("object_unadmire (%s) %s %s %d", like ? "Like" : "Announce", actor, fn, status)); return status; } xs_str *object_user_cache_fn_by_md5(snac *user, const char *md5, const char *cachedir) { return xs_fmt("%s/%s/%s.json", user->basedir, cachedir, md5); } xs_str *object_user_cache_fn(snac *user, const char *id, const char *cachedir) { xs *md5 = xs_md5_hex(id, strlen(id)); return object_user_cache_fn_by_md5(user, md5, cachedir); } xs_str *object_user_cache_index_fn(snac *user, const char *cachedir) { return xs_fmt("%s/%s.idx", user->basedir, cachedir); } int _object_user_cache(snac *user, const char *id, const char *cachedir, int del) /* adds or deletes from a user cache */ { xs *ofn = _object_fn(id); xs *cfn = object_user_cache_fn(user, id, cachedir); xs *idx = object_user_cache_index_fn(user, cachedir); int ret; if (del) { ret = unlink(cfn); index_del(idx, id); } else { /* create the subfolder, if it does not exist */ xs *dir = xs_fmt("%s/%s/", user->basedir, cachedir); mkdirx(dir); if ((ret = link(ofn, cfn)) != -1) index_add(idx, id); } return ret; } int object_user_cache_add(snac *user, const char *id, const char *cachedir) /* caches an object into a user cache */ { return _object_user_cache(user, id, cachedir, 0); } int object_user_cache_del(snac *user, const char *id, const char *cachedir) /* deletes an object from a user cache */ { return _object_user_cache(user, id, cachedir, 1); } int object_user_cache_in(snac *user, const char *id, const char *cachedir) /* checks if an object is stored in a cache */ { xs *cfn = object_user_cache_fn(user, id, cachedir); return !!(mtime(cfn) != 0.0); } int object_user_cache_in_by_md5(snac *user, const char *md5, const char *cachedir) /* checks if an object is stored in a cache */ { xs *cfn = object_user_cache_fn_by_md5(user, md5, cachedir); return !!(mtime(cfn) != 0.0); } xs_list *object_user_cache_list(snac *user, const char *cachedir, int max, int inv) /* returns the objects in a cache as a list */ { xs *idx = xs_fmt("%s/%s.idx", user->basedir, cachedir); return inv ? index_list_desc(idx, 0, max) : index_list(idx, max); } /** specialized functions **/ /** followers **/ int follower_add(snac *snac, const char *actor) /* adds a follower */ { int ret = object_user_cache_add(snac, actor, "followers"); snac_debug(snac, 2, xs_fmt("follower_add %s", actor)); return ret == -1 ? HTTP_STATUS_INTERNAL_SERVER_ERROR : HTTP_STATUS_OK; } int follower_del(snac *snac, const char *actor) /* deletes a follower */ { int ret = object_user_cache_del(snac, actor, "followers"); snac_debug(snac, 2, xs_fmt("follower_del %s", actor)); return ret == -1 ? HTTP_STATUS_NOT_FOUND : HTTP_STATUS_OK; } int follower_check(snac *snac, const char *actor) /* checks if someone is a follower */ { return object_user_cache_in(snac, actor, "followers"); } xs_list *follower_list(snac *snac) /* returns the list of followers */ { xs *list = object_user_cache_list(snac, "followers", XS_ALL, 0); xs_list *fwers = xs_list_new(); char *p; const char *v; /* resolve the list of md5 to be a list of actors */ p = list; while (xs_list_iter(&p, &v)) { xs *a_obj = NULL; if (valid_status(object_get_by_md5(v, &a_obj))) { const char *actor = xs_dict_get(a_obj, "id"); if (!xs_is_null(actor)) { /* check if the actor is still cached */ xs *fn = xs_fmt("%s/followers/%s.json", snac->basedir, v); if (mtime(fn) > 0.0) fwers = xs_list_append(fwers, actor); } } } return fwers; } /** pending followers **/ int pending_add(snac *user, const char *actor, const xs_dict *msg) /* stores the follow message for later confirmation */ { xs *dir = xs_fmt("%s/pending", user->basedir); xs *md5 = xs_md5_hex(actor, strlen(actor)); xs *fn = xs_fmt("%s/%s.json", dir, md5); FILE *f; mkdirx(dir); if ((f = fopen(fn, "w")) == NULL) return -1; xs_json_dump(msg, 4, f); fclose(f); return 0; } int pending_check(snac *user, const char *actor) /* checks if there is a pending follow confirmation for the actor */ { xs *md5 = xs_md5_hex(actor, strlen(actor)); xs *fn = xs_fmt("%s/pending/%s.json", user->basedir, md5); return mtime(fn) != 0; } xs_dict *pending_get(snac *user, const char *actor) /* returns the pending follow confirmation for the actor */ { xs *md5 = xs_md5_hex(actor, strlen(actor)); xs *fn = xs_fmt("%s/pending/%s.json", user->basedir, md5); xs_dict *msg = NULL; FILE *f; if ((f = fopen(fn, "r")) != NULL) { msg = xs_json_load(f); fclose(f); } return msg; } void pending_del(snac *user, const char *actor) /* deletes a pending follow confirmation for the actor */ { xs *md5 = xs_md5_hex(actor, strlen(actor)); xs *fn = xs_fmt("%s/pending/%s.json", user->basedir, md5); unlink(fn); } xs_list *pending_list(snac *user) /* returns a list of pending follow confirmations */ { xs *spec = xs_fmt("%s/pending/""*.json", user->basedir); xs *l = xs_glob(spec, 0, 0); xs_list *r = xs_list_new(); const char *v; xs_list_foreach(l, v) { FILE *f; xs *msg = NULL; if ((f = fopen(v, "r")) == NULL) continue; msg = xs_json_load(f); fclose(f); if (msg == NULL) continue; const char *actor = xs_dict_get(msg, "actor"); if (xs_type(actor) == XSTYPE_STRING) r = xs_list_append(r, actor); } return r; } /** timeline **/ double timeline_mtime(snac *snac) { xs *fn = xs_fmt("%s/private.idx", snac->basedir); return mtime(fn); } int timeline_touch(snac *snac) /* changes the date of the timeline index */ { xs *fn = xs_fmt("%s/private.idx", snac->basedir); return utimes(fn, NULL); } xs_str *timeline_fn_by_md5(snac *snac, const char *md5) /* get the filename of an entry by md5 from any timeline */ { xs_str *fn = NULL; if (is_md5_hex(md5)) { fn = xs_fmt("%s/private/%s.json", snac->basedir, md5); if (mtime(fn) == 0.0) { fn = xs_free(fn); fn = xs_fmt("%s/public/%s.json", snac->basedir, md5); if (mtime(fn) == 0.0) fn = xs_free(fn); } } return fn; } int timeline_here(snac *snac, const char *md5) /* checks if an object is in the user cache */ { xs *fn = timeline_fn_by_md5(snac, md5); return !(fn == NULL); } int timeline_get_by_md5(snac *snac, const char *md5, xs_dict **msg) /* gets a message from the timeline */ { int status = HTTP_STATUS_NOT_FOUND; FILE *f = NULL; xs *fn = timeline_fn_by_md5(snac, md5); if (fn != NULL && (f = fopen(fn, "r")) != NULL) { *msg = xs_json_load(f); fclose(f); if (*msg != NULL) status = HTTP_STATUS_OK; } return status; } int timeline_del(snac *snac, const char *id) /* deletes a message from the timeline */ { /* delete from the user's caches */ object_user_cache_del(snac, id, "public"); object_user_cache_del(snac, id, "private"); unpin(snac, id); unbookmark(snac, id); /* try to delete the object if it's not used elsewhere */ return object_del_if_unref(id); } void timeline_update_indexes(snac *snac, const char *id) /* updates the indexes */ { object_user_cache_add(snac, id, "private"); if (xs_startswith(id, snac->actor)) { xs *msg = NULL; if (valid_status(object_get(id, &msg))) { /* if its ours and is public, also store in public */ if (is_msg_public(msg)) { object_user_cache_add(snac, id, "public"); /* also add it to the instance public timeline */ xs *ipt = xs_fmt("%s/public.idx", srv_basedir); index_add(ipt, id); } } } } int timeline_add(snac *snac, const char *id, const xs_dict *o_msg) /* adds a message to the timeline */ { int ret = object_add(id, o_msg); timeline_update_indexes(snac, id); tag_index(id, o_msg); list_distribute(snac, NULL, o_msg); snac_debug(snac, 1, xs_fmt("timeline_add %s", id)); return ret; } int timeline_admire(snac *snac, const char *id, const char *admirer, int like) /* updates a timeline entry with a new admiration */ { /* if we are admiring this, add to both timelines */ if (!like && strcmp(admirer, snac->actor) == 0) { object_user_cache_add(snac, id, "public"); object_user_cache_add(snac, id, "private"); } int ret = object_admire(id, admirer, like); snac_debug(snac, 1, xs_fmt("timeline_admire (%s) %s %s", like ? "Like" : "Announce", id, admirer)); return ret; } xs_list *timeline_top_level(snac *snac, const xs_list *list) /* returns the top level md5 entries from this index */ { xs_set seen; const xs_str *v; xs_set_init(&seen); int c = 0; while (xs_list_next(list, &v, &c)) { char line[MD5_HEX_SIZE] = ""; strncpy(line, v, sizeof(line)); for (;;) { char line2[MD5_HEX_SIZE]; /* if it doesn't have a parent, use this */ if (!object_parent(line, line2)) break; /* well, there is a parent... but is it here? */ if (!timeline_here(snac, line2)) break; /* it's here! try again with its own parent */ strncpy(line, line2, sizeof(line)); } xs_set_add(&seen, line); } return xs_set_result(&seen); } xs_str *user_index_fn(snac *user, const char *idx_name) /* returns the filename of a user index */ { return xs_fmt("%s/%s.idx", user->basedir, idx_name); } xs_list *timeline_simple_list(snac *user, const char *idx_name, int skip, int show) /* returns a timeline (with all entries) */ { xs *idx = user_index_fn(user, idx_name); return index_list_desc(idx, skip, show); } xs_list *timeline_list(snac *snac, const char *idx_name, int skip, int show) /* returns a timeline (only top level entries) */ { int c_max; /* maximum number of items in the timeline */ c_max = xs_number_get(xs_dict_get(srv_config, "max_timeline_entries")); /* never more timeline entries than the configured maximum */ if (show > c_max) show = c_max; xs *list = timeline_simple_list(snac, idx_name, skip, show); return timeline_top_level(snac, list); } xs_str *instance_index_fn(void) { return xs_fmt("%s/public.idx", srv_basedir); } xs_list *timeline_instance_list(int skip, int show) /* returns the timeline for the full instance */ { xs *idx = instance_index_fn(); return index_list_desc(idx, skip, show); } /** following **/ /* this needs special treatment and cannot use the object db as is, with a link to a cached author, because we need the Follow object in case we need to unfollow (Undo + original Follow) */ xs_str *_following_fn(snac *snac, const char *actor) { xs *md5 = xs_md5_hex(actor, strlen(actor)); return xs_fmt("%s/following/%s.json", snac->basedir, md5); } int following_add(snac *snac, const char *actor, const xs_dict *msg) /* adds to the following list */ { int ret = HTTP_STATUS_CREATED; xs *fn = _following_fn(snac, actor); FILE *f; xs *p_object = NULL; if (valid_status(following_get(snac, actor, &p_object))) { /* object already exists; if it's of type Accept, the actor is already being followed and confirmed, so do nothing */ const char *type = xs_dict_get(p_object, "type"); if (!xs_is_null(type) && strcmp(type, "Accept") == 0) { snac_debug(snac, 1, xs_fmt("following_add actor already confirmed %s", actor)); return HTTP_STATUS_OK; } } if ((f = fopen(fn, "w")) != NULL) { xs_json_dump(msg, 4, f); fclose(f); /* get the filename of the actor object */ xs *actor_fn = _object_fn(actor); /* increase its reference count */ fn = xs_replace_i(fn, ".json", "_a.json"); link(actor_fn, fn); } else ret = HTTP_STATUS_INTERNAL_SERVER_ERROR; snac_debug(snac, 2, xs_fmt("following_add %s %s", actor, fn)); return ret; } int following_del(snac *snac, const char *actor) /* we're not following this actor any longer */ { xs *fn = _following_fn(snac, actor); snac_debug(snac, 2, xs_fmt("following_del %s %s", actor, fn)); unlink(fn); /* also delete the reference to the author */ fn = xs_replace_i(fn, ".json", "_a.json"); unlink(fn); return HTTP_STATUS_OK; } int following_check(snac *snac, const char *actor) /* checks if we are following this actor */ { xs *fn = _following_fn(snac, actor); return !!(mtime(fn) != 0.0); } int following_get(snac *snac, const char *actor, xs_dict **data) /* returns the 'Follow' object */ { xs *fn = _following_fn(snac, actor); FILE *f; int status = HTTP_STATUS_OK; if ((f = fopen(fn, "r")) != NULL) { *data = xs_json_load(f); fclose(f); } else status = HTTP_STATUS_NOT_FOUND; return status; } xs_list *following_list(snac *snac) /* returns the list of people being followed */ { xs *spec = xs_fmt("%s/following/" "*.json", snac->basedir); xs *glist = xs_glob(spec, 0, 0); xs_list *p; const xs_str *v; xs_list *list = xs_list_new(); /* iterate the list of files */ p = glist; while (xs_list_iter(&p, &v)) { FILE *f; /* load the follower data */ if ((f = fopen(v, "r")) != NULL) { xs *o = xs_json_load(f); fclose(f); if (o != NULL) { const char *type = xs_dict_get(o, "type"); if (!xs_is_null(type) && strcmp(type, "Accept") == 0) { const char *actor = xs_dict_get(o, "actor"); if (!xs_is_null(actor)) { list = xs_list_append(list, actor); /* check if there is a link to the actor object */ xs *v2 = xs_replace(v, ".json", "_a.json"); if (mtime(v2) == 0.0) { /* no; add a link to it */ xs *actor_fn = _object_fn(actor); link(actor_fn, v2); } } } } } } return list; } xs_str *_muted_fn(snac *snac, const char *actor) { xs *md5 = xs_md5_hex(actor, strlen(actor)); return xs_fmt("%s/muted/%s", snac->basedir, md5); } void mute(snac *snac, const char *actor) /* mutes a moron */ { xs *fn = _muted_fn(snac, actor); FILE *f; if ((f = fopen(fn, "w")) != NULL) { fprintf(f, "%s\n", actor); fclose(f); snac_debug(snac, 2, xs_fmt("muted %s %s", actor, fn)); } } void unmute(snac *snac, const char *actor) /* actor is no longer a moron */ { xs *fn = _muted_fn(snac, actor); unlink(fn); snac_debug(snac, 2, xs_fmt("unmuted %s %s", actor, fn)); } int is_muted(snac *snac, const char *actor) /* check if someone is muted */ { xs *fn = _muted_fn(snac, actor); return !!(mtime(fn) != 0.0); } xs_list *muted_list(snac *user) /* returns the list (actor URLs) of the muted morons */ { xs_list *l = xs_list_new(); xs *spec = xs_fmt("%s/muted/" "*", user->basedir); xs *files = xs_glob(spec, 0, 0); const char *fn; xs_list_foreach(files, fn) { FILE *f; if ((f = fopen(fn, "r")) != NULL) { xs *actor = xs_strip_i(xs_readline(f)); fclose(f); l = xs_list_append(l, actor); } } return l; } /** bookmarking **/ int is_bookmarked(snac *user, const char *id) /* returns true if this note is bookmarked */ { return object_user_cache_in(user, id, "bookmark"); } int bookmark(snac *user, const char *id) /* bookmarks a post */ { if (is_bookmarked(user, id)) return -3; return object_user_cache_add(user, id, "bookmark"); } int unbookmark(snac *user, const char *id) /* unbookmarks a post */ { return object_user_cache_del(user, id, "bookmark"); } xs_list *bookmark_list(snac *user) /* return the lists of bookmarked posts */ { return object_user_cache_list(user, "bookmark", XS_ALL, 1); } xs_str *bookmark_index_fn(snac *user) { return object_user_cache_index_fn(user, "bookmark"); } /** pinning **/ int is_pinned(snac *user, const char *id) /* returns true if this note is pinned */ { return object_user_cache_in(user, id, "pinned"); } int is_pinned_by_md5(snac *user, const char *md5) { return object_user_cache_in_by_md5(user, md5, "pinned"); } int pin(snac *user, const char *id) /* pins a message */ { int ret = -2; if (xs_startswith(id, user->actor)) { if (is_pinned(user, id)) ret = -3; else ret = object_user_cache_add(user, id, "pinned"); } return ret; } int unpin(snac *user, const char *id) /* unpin a message */ { return object_user_cache_del(user, id, "pinned"); } xs_list *pinned_list(snac *user) /* return the lists of pinned posts */ { return object_user_cache_list(user, "pinned", XS_ALL, 1); } /** drafts **/ int is_draft(snac *user, const char *id) /* returns true if this note is a draft */ { return object_user_cache_in(user, id, "draft"); } void draft_del(snac *user, const char *id) /* delete a message from the draft cache */ { object_user_cache_del(user, id, "draft"); } void draft_add(snac *user, const char *id, const xs_dict *msg) /* store the message as a draft */ { /* delete from the index, in case it was already there */ draft_del(user, id); /* overwrite object */ object_add_ow(id, msg); /* [re]add to the index */ object_user_cache_add(user, id, "draft"); } xs_list *draft_list(snac *user) /* return the lists of drafts */ { return object_user_cache_list(user, "draft", XS_ALL, 1); } /** hiding **/ xs_str *_hidden_fn(snac *snac, const char *id) { xs *md5 = xs_md5_hex(id, strlen(id)); return xs_fmt("%s/hidden/%s", snac->basedir, md5); } void hide(snac *snac, const char *id) /* hides a message tree */ { xs *fn = _hidden_fn(snac, id); FILE *f; if ((f = fopen(fn, "w")) != NULL) { fprintf(f, "%s\n", id); fclose(f); snac_debug(snac, 2, xs_fmt("hidden %s %s", id, fn)); /* hide all the children */ xs *chld = object_children(id); char *p; const char *v; p = chld; while (xs_list_iter(&p, &v)) { xs *co = NULL; /* resolve to get the id */ if (valid_status(object_get_by_md5(v, &co))) { const char *id = xs_dict_get(co, "id"); if (id != NULL) hide(snac, id); } } } } int is_hidden(snac *snac, const char *id) /* check is id is hidden */ { xs *fn = _hidden_fn(snac, id); return !!(mtime(fn) != 0.0); } int actor_add(const char *actor, const xs_dict *msg) /* adds an actor */ { return object_add_ow(actor, msg); } int actor_get(const char *actor, xs_dict **data) /* returns an already downloaded actor */ { int status = HTTP_STATUS_OK; xs_dict *d = NULL; if (xs_startswith(actor, srv_baseurl)) { /* it's a (possible) local user */ xs *l = xs_split(actor, "/"); const char *uid = xs_list_get(l, -1); snac user; if (!xs_is_null(uid) && user_open(&user, uid)) { if (data) *data = msg_actor(&user); user_free(&user); return HTTP_STATUS_OK; } else return HTTP_STATUS_NOT_FOUND; } /* read the object */ if (!valid_status(status = object_get(actor, &d))) { d = xs_free(d); return status; } /* if the object is corrupted, discard it */ if (xs_is_null(xs_dict_get(d, "id")) || xs_is_null(xs_dict_get(d, "type"))) { srv_debug(1, xs_fmt("corrupted actor object %s", actor)); d = xs_free(d); return HTTP_STATUS_NOT_FOUND; } if (data) *data = d; else d = xs_free(d); xs *fn = _object_fn(actor); double max_time; /* maximum time for the actor data to be considered stale */ max_time = 3600.0 * 36.0; if (mtime(fn) + max_time < (double) time(NULL)) { /* actor data exists but also stinks */ status = HTTP_STATUS_RESET_CONTENT; /* "110: Response Is Stale" */ } return status; } int actor_get_refresh(snac *user, const char *actor, xs_dict **data) /* gets an actor and requests a refresh if it's stale */ { int status = actor_get(actor, data); if (status == HTTP_STATUS_RESET_CONTENT && user && !xs_startswith(actor, srv_baseurl)) enqueue_actor_refresh(user, actor, 0); return status; } /** user limiting (announce blocks) **/ int limited(snac *user, const char *id, int cmd) /* announce messages from a followed (0: check, 1: limit; 2: unlimit) */ { int ret = 0; xs *dir = xs_fmt("%s/limited", user->basedir); xs *md5 = xs_md5_hex(id, strlen(id)); xs *fn = xs_fmt("%s/%s", dir, md5); switch (cmd) { case 0: /** check **/ ret = !!(mtime(fn) > 0.0); break; case 1: /** limit **/ mkdirx(dir); if (mtime(fn) > 0.0) ret = -1; else { FILE *f; if ((f = fopen(fn, "w")) != NULL) { fprintf(f, "%s\n", id); fclose(f); } else ret = -2; } break; case 2: /** unlimit **/ if (mtime(fn) > 0.0) ret = unlink(fn); else ret = -1; break; } return ret; } /** tag indexing **/ void tag_index(const char *id, const xs_dict *obj) /* update the tag indexes for this object */ { const xs_list *tags = xs_dict_get(obj, "tag"); xs *md5_id = xs_md5_hex(id, strlen(id)); if (is_msg_public(obj) && xs_type(tags) == XSTYPE_LIST && xs_list_len(tags) > 0) { xs *g_tag_dir = xs_fmt("%s/tag", srv_basedir); mkdirx(g_tag_dir); const xs_dict *v; int ct = 0; while (xs_list_next(tags, &v, &ct)) { const char *type = xs_dict_get(v, "type"); const char *name = xs_dict_get(v, "name"); if (!xs_is_null(type) && !xs_is_null(name) && strcmp(type, "Hashtag") == 0) { while (*name == '#' || *name == '@') name++; if (*name == '\0') continue; name = xs_tolower_i((xs_str *)name); xs *md5_tag = xs_md5_hex(name, strlen(name)); xs *tag_dir = xs_fmt("%s/%c%c", g_tag_dir, md5_tag[0], md5_tag[1]); mkdirx(tag_dir); xs *g_tag_idx = xs_fmt("%s/%s.idx", tag_dir, md5_tag); if (!index_in_md5(g_tag_idx, md5_id)) index_add_md5(g_tag_idx, md5_id); FILE *f; xs *g_tag_name = xs_replace(g_tag_idx, ".idx", ".tag"); if ((f = fopen(g_tag_name, "w")) != NULL) { fprintf(f, "%s\n", name); fclose(f); } srv_debug(0, xs_fmt("tagged %s #%s (#%s)", id, name, md5_tag)); } } } } xs_str *tag_fn(const char *tag) { if (*tag == '#') tag++; xs *lw_tag = xs_tolower_i(xs_dup(tag)); xs *md5 = xs_md5_hex(lw_tag, strlen(lw_tag)); return xs_fmt("%s/tag/%c%c/%s.idx", srv_basedir, md5[0], md5[1], md5); } xs_list *tag_search(const char *tag, int skip, int show) /* returns the list of posts tagged with tag */ { xs *idx = tag_fn(tag); return index_list_desc(idx, skip, show); } /** lists **/ xs_val *list_maint(snac *user, const char *list, int op) /* list maintenance */ { xs_val *l = NULL; switch (op) { case 0: /** list of lists **/ { FILE *f; xs *spec = xs_fmt("%s/list/" "*.id", user->basedir); xs *ls = xs_glob(spec, 0, 0); int c = 0; const char *v; l = xs_list_new(); while (xs_list_next(ls, &v, &c)) { if ((f = fopen(v, "r")) != NULL) { xs *title = xs_readline(f); fclose(f); title = xs_strip_i(title); xs *v2 = xs_replace(v, ".id", ""); xs *l2 = xs_split(v2, "/"); /* return [ list_id, list_title ] */ l = xs_list_append(l, xs_list_append(xs_list_new(), xs_list_get(l2, -1), title)); } } } break; case 1: /** create new list (list is the name) **/ { xs *lol = list_maint(user, NULL, 0); int c = 0; const xs_list *v; int add = 1; /* check if this list name already exists */ while (xs_list_next(lol, &v, &c)) { if (strcmp(xs_list_get(v, 1), list) == 0) { add = 0; l = xs_dup(xs_list_get(v, 0)); break; } } if (add) { FILE *f; xs *dir = xs_fmt("%s/list/", user->basedir); l = xs_fmt("%010x", time(NULL)); mkdirx(dir); xs *fn = xs_fmt("%s%s.id", dir, l); if ((f = fopen(fn, "w")) != NULL) { fprintf(f, "%s\n", list); fclose(f); } } } break; case 2: /** delete list (list is the id) **/ { if (xs_is_hex(list)) { xs *fn = xs_fmt("%s/list/%s.id", user->basedir, list); unlink(fn); fn = xs_replace_i(fn, ".id", ".lst"); unlink(fn); fn = xs_replace_i(fn, ".lst", ".idx"); unlink(fn); fn = xs_str_cat(fn, ".bak"); unlink(fn); } } break; case 3: /** get list name **/ if (xs_is_hex(list)) { FILE *f; xs *fn = xs_fmt("%s/list/%s.id", user->basedir, list); if ((f = fopen(fn, "r")) != NULL) { l = xs_strip_i(xs_readline(f)); fclose(f); } } break; } return l; } xs_str *list_timeline_fn(snac *user, const char *list) { return xs_fmt("%s/list/%s.idx", user->basedir, list); } xs_list *list_timeline(snac *user, const char *list, int skip, int show) /* returns the timeline of a list */ { xs_list *l = NULL; if (!xs_is_hex(list)) return NULL; xs *fn = list_timeline_fn(user, list); if (mtime(fn) > 0.0) l = index_list_desc(fn, skip, show); else l = xs_list_new(); return l; } xs_val *list_content(snac *user, const char *list, const char *actor_md5, int op) /* list content management */ { xs_val *l = NULL; if (!xs_is_hex(list)) return NULL; if (actor_md5 != NULL && !xs_is_hex(actor_md5)) return NULL; xs *fn = xs_fmt("%s/list/%s.lst", user->basedir, list); switch (op) { case 0: /** list content **/ l = index_list(fn, XS_ALL); break; case 1: /** append actor to list **/ if (actor_md5 != NULL) { if (!index_in_md5(fn, actor_md5)) index_add_md5(fn, actor_md5); } break; case 2: /** delete actor from list **/ if (actor_md5 != NULL) index_del_md5(fn, actor_md5); break; default: srv_log(xs_fmt("ERROR: list_content: bad op %d", op)); break; } return l; } void list_distribute(snac *user, const char *who, const xs_dict *post) /* distributes the post to all appropriate lists */ { const char *id = xs_dict_get(post, "id"); /* if who is not set, use the attributedTo in the message */ if (xs_is_null(who)) who = get_atto(post); if (xs_type(who) == XSTYPE_STRING && xs_type(id) == XSTYPE_STRING) { xs *a_md5 = xs_md5_hex(who, strlen(who)); xs *i_md5 = xs_md5_hex(id, strlen(id)); xs *spec = xs_fmt("%s/list/" "*.lst", user->basedir); xs *ls = xs_glob(spec, 0, 0); int c = 0; const char *v; while (xs_list_next(ls, &v, &c)) { /* is the actor in this list? */ if (index_in_md5(v, a_md5)) { /* it is; add post md5 to its timeline */ xs *idx = xs_replace(v, ".lst", ".idx"); index_add_md5(idx, i_md5); snac_debug(user, 1, xs_fmt("listed post %s in %s", id, idx)); } } } } /** static data **/ static int _load_raw_file(const char *fn, xs_val **data, int *size, const char *inm, xs_str **etag) /* loads a cached file */ { int status = HTTP_STATUS_NOT_FOUND; if (fn) { double tm = mtime(fn); if (tm > 0.0) { /* file exists; build the etag */ xs *e = xs_fmt("W/\"snac-%.0lf\"", tm); /* if if-none-match is set, check if it's the same */ if (!xs_is_null(inm) && strcmp(e, inm) == 0) { /* client has the newest version */ status = HTTP_STATUS_NOT_MODIFIED; } else { /* newer or never downloaded; read the full file */ FILE *f; if ((f = fopen(fn, "rb")) != NULL) { *size = XS_ALL; *data = xs_read(f, size); fclose(f); status = HTTP_STATUS_OK; } } /* if caller wants the etag, return it */ if (etag != NULL) *etag = xs_dup(e); srv_debug(1, xs_fmt("_load_raw_file(): %s %d", fn, status)); } } return status; } xs_str *_static_fn(snac *snac, const char *id) /* gets the filename for a static file */ { if (strchr(id, '/')) return NULL; else return xs_fmt("%s/static/%s", snac->basedir, id); } int static_get(snac *snac, const char *id, xs_val **data, int *size, const char *inm, xs_str **etag) /* returns static content */ { xs *fn = _static_fn(snac, id); return _load_raw_file(fn, data, size, inm, etag); } void static_put(snac *snac, const char *id, const char *data, int size) /* writes status content */ { xs *fn = _static_fn(snac, id); FILE *f; if (fn && (f = fopen(fn, "wb")) != NULL) { fwrite(data, size, 1, f); fclose(f); } } void static_put_meta(snac *snac, const char *id, const char *str) /* puts metadata (i.e. a media description string) to id */ { xs *fn = _static_fn(snac, id); if (fn) { fn = xs_str_cat(fn, ".txt"); FILE *f; if ((f = fopen(fn, "w")) != NULL) { fprintf(f, "%s\n", str); fclose(f); } } } xs_str *static_get_meta(snac *snac, const char *id) /* gets metadata from a media */ { xs *fn = _static_fn(snac, id); xs_str *r = NULL; if (fn) { fn = xs_str_cat(fn, ".txt"); FILE *f; if ((f = fopen(fn, "r")) != NULL) { r = xs_strip_i(xs_readline(f)); fclose(f); } } else r = xs_str_new(""); return r; } /** history **/ xs_str *_history_fn(snac *snac, const char *id) /* gets the filename for the history */ { if (strchr(id, '/')) return NULL; else return xs_fmt("%s/history/%s", snac->basedir, id); } double history_mtime(snac *snac, const char *id) { double t = 0.0; xs *fn = _history_fn(snac, id); if (fn != NULL) t = mtime(fn); return t; } void history_add(snac *snac, const char *id, const char *content, int size, xs_str **etag) /* adds something to the history */ { xs *fn = _history_fn(snac, id); FILE *f; if (fn && (f = fopen(fn, "w")) != NULL) { fwrite(content, size, 1, f); fclose(f); if (etag) { double tm = mtime(fn); *etag = xs_fmt("W/\"snac-%.0lf\"", tm); } } } int history_get(snac *snac, const char *id, xs_str **content, int *size, const char *inm, xs_str **etag) { xs *fn = _history_fn(snac, id); return _load_raw_file(fn, content, size, inm, etag); } int history_del(snac *snac, const char *id) { xs *fn = _history_fn(snac, id); if (fn) return unlink(fn); else return -1; } xs_list *history_list(snac *snac) { xs *spec = xs_fmt("%s/history/" "*.html", snac->basedir); return xs_glob(spec, 1, 1); } void lastlog_write(snac *snac, const char *source) /* writes the last time the user logged in */ { xs *fn = xs_fmt("%s/lastlog.txt", snac->basedir); FILE *f; if ((f = fopen(fn, "w")) != NULL) { fprintf(f, "%lf %s\n", ftime(), source); fclose(f); } } /** inbox collection **/ void inbox_add(const char *inbox) /* collects a shared inbox */ { /* don't collect ourselves */ if (xs_startswith(inbox, srv_baseurl)) return; xs *md5 = xs_md5_hex(inbox, strlen(inbox)); xs *fn = xs_fmt("%s/inbox/%s", srv_basedir, md5); FILE *f; if ((f = fopen(fn, "w")) != NULL) { fprintf(f, "%s\n", inbox); fclose(f); } } void inbox_add_by_actor(const xs_dict *actor) /* collects an actor's shared inbox, if it has one */ { const char *v; if (!xs_is_null(v = xs_dict_get(actor, "endpoints")) && !xs_is_null(v = xs_dict_get(v, "sharedInbox"))) { /* only collect this inbox if its instance is not blocked */ if (!is_instance_blocked(v)) inbox_add(v); } } xs_list *inbox_list(void) /* returns the collected inboxes as a list */ { xs_list *ibl = xs_list_new(); xs *spec = xs_fmt("%s/inbox/" "*", srv_basedir); xs *files = xs_glob(spec, 0, 0); xs_list *p = files; const xs_val *v; while (xs_list_iter(&p, &v)) { FILE *f; if ((f = fopen(v, "r")) != NULL) { xs *line = xs_readline(f); if (line) { line = xs_strip_i(line); ibl = xs_list_append(ibl, line); } fclose(f); } } return ibl; } /** instance-wide operations **/ xs_str *_instance_block_fn(const char *instance) { xs *s = xs_replace(instance, "http:/" "/", ""); xs *s1 = xs_replace(s, "https:/" "/", ""); xs *l = xs_split(s1, "/"); const char *p = xs_list_get(l, 0); xs *md5 = xs_md5_hex(p, strlen(p)); return xs_fmt("%s/block/%s", srv_basedir, md5); } int is_instance_blocked(const char *instance) { xs *fn = _instance_block_fn(instance); return !!(mtime(fn) != 0.0); } int instance_block(const char *instance) /* blocks a full instance */ { int ret; /* create the subdir */ xs *dir = xs_fmt("%s/block/", srv_basedir); mkdirx(dir); if (!is_instance_blocked(instance)) { xs *fn = _instance_block_fn(instance); FILE *f; if ((f = fopen(fn, "w")) != NULL) { fprintf(f, "%s\n", instance); fclose(f); ret = 0; } else ret = -1; } else ret = -2; return ret; } int instance_unblock(const char *instance) /* unblocks a full instance */ { int ret; if (is_instance_blocked(instance)) { xs *fn = _instance_block_fn(instance); ret = unlink(fn); } else ret = -2; return ret; } /** operations by content **/ int content_match(const char *file, const xs_dict *msg) /* checks if a message's content matches any of the regexes in file */ /* file format: one regex per line */ { xs *fn = xs_fmt("%s/%s", srv_basedir, file); FILE *f; int r = 0; const char *v = xs_dict_get(msg, "content"); if (xs_type(v) == XSTYPE_STRING && *v) { if ((f = fopen(fn, "r")) != NULL) { srv_debug(1, xs_fmt("content_match: loading regexes from %s", fn)); /* massage content (strip HTML tags, etc.) */ xs *c = xs_regex_replace(v, "<[^>]+>", " "); c = xs_regex_replace_i(c, " {2,}", " "); c = xs_tolower_i(c); while (!r && !feof(f)) { xs *rx = xs_strip_i(xs_readline(f)); if (*rx && xs_regex_match(c, rx)) { srv_debug(1, xs_fmt("content_match: match for '%s'", rx)); r = 1; } } fclose(f); } } return r; } xs_list *content_search(snac *user, const char *regex, int priv, int skip, int show, int max_secs, int *timeout) /* returns a list of posts which content matches the regex */ { if (regex == NULL || *regex == '\0') return xs_list_new(); xs *i_regex = xs_utf8_to_lower(regex); xs_set seen; xs_set_init(&seen); if (max_secs == 0) max_secs = 3; time_t t = time(NULL) + max_secs; *timeout = 0; /* iterate all timelines simultaneously */ xs_list *tls[3] = {0}; const char *md5s[3] = {0}; int c[3] = {0}; tls[0] = timeline_simple_list(user, "public", 0, XS_ALL); /* public */ tls[1] = timeline_instance_list(0, XS_ALL); /* instance */ tls[2] = priv ? timeline_simple_list(user, "private", 0, XS_ALL) : xs_list_new(); /* private or none */ /* first positioning */ for (int n = 0; n < 3; n++) xs_list_next(tls[n], &md5s[n], &c[n]); show += skip; while (show > 0) { /* timeout? */ if (time(NULL) > t) { *timeout = 1; break; } /* find the newest post */ int newest = -1; double mtime = 0.0; for (int n = 0; n < 3; n++) { if (md5s[n] != NULL) { xs *fn = _object_fn_by_md5(md5s[n], "content_search"); double mt = mtime(fn); if (mt > mtime) { newest = n; mtime = mt; } } } if (newest == -1) break; const char *md5 = md5s[newest]; /* advance the chosen timeline */ if (!xs_list_next(tls[newest], &md5s[newest], &c[newest])) md5s[newest] = NULL; xs *post = NULL; if (!valid_status(object_get_by_md5(md5, &post))) continue; if (!xs_match(xs_dict_get_def(post, "type", "-"), POSTLIKE_OBJECT_TYPE)) continue; const char *id = xs_dict_get(post, "id"); if (id == NULL || is_hidden(user, id)) continue; /* recalculate the md5 id to be sure it's not repeated (it may have been searched by the "url" field instead of "id") */ xs *new_md5 = xs_md5_hex(id, strlen(id)); md5 = new_md5; /* test for the post URL */ if (strcmp(id, regex) == 0) { if (xs_set_add(&seen, md5) == 1) show--; continue; } /* test for the alternate post id */ const char *url = xs_dict_get(post, "url"); if (xs_type(url) == XSTYPE_STRING && strcmp(url, regex) == 0) { if (xs_set_add(&seen, md5) == 1) show--; continue; } xs *c = xs_str_new(NULL); const char *content = xs_dict_get(post, "content"); const char *name = xs_dict_get(post, "name"); if (!xs_is_null(content)) c = xs_str_cat(c, content); if (!xs_is_null(name)) c = xs_str_cat(c, " ", name); /* add alt-texts from attachments */ const xs_list *atts = xs_dict_get(post, "attachment"); int tc = 0; const xs_dict *att; while (xs_list_next(atts, &att, &tc)) { const char *name = xs_dict_get(att, "name"); if (name != NULL) c = xs_str_cat(c, " ", name); } /* strip HTML */ c = xs_regex_replace_i(c, "<[^>]+>", " "); c = xs_regex_replace_i(c, " {2,}", " "); /* convert to lowercase */ xs *lc = xs_utf8_to_lower(c); /* apply regex */ if (xs_regex_match(lc, i_regex)) { if (xs_set_add(&seen, md5) == 1) show--; } } xs_list *r = xs_set_result(&seen); if (skip) { /* BAD */ while (skip--) { r = xs_list_del(r, 0); } } xs_free(tls[0]); xs_free(tls[1]); xs_free(tls[2]); return r; } /** notifications **/ xs_str *notify_check_time(snac *snac, int reset) /* gets or resets the latest notification check time */ { xs_str *t = NULL; xs *fn = xs_fmt("%s/notifydate.txt", snac->basedir); FILE *f; if (reset) { if ((f = fopen(fn, "w")) != NULL) { t = tid(0); fprintf(f, "%s\n", t); fclose(f); } } else { if ((f = fopen(fn, "r")) != NULL) { t = xs_readline(f); fclose(f); } else /* never set before */ t = xs_fmt("%16.6f", 0.0); } return t; } xs_dict *markers_get(snac *snac, const xs_list *markers) { xs *data = NULL; xs_dict *returns = xs_dict_new(); xs *fn = xs_fmt("%s/markers.json", snac->basedir); const xs_str *v = NULL; FILE *f; if ((f = fopen(fn, "r")) != NULL) { data = xs_json_load(f); fclose(f); } if (xs_is_null(data)) data = xs_dict_new(); xs_list_foreach(markers, v) { const xs_dict *mark = xs_dict_get(data, v); if (!xs_is_null(mark)) { returns = xs_dict_append(returns, v, mark); } } return returns; } xs_dict *markers_set(snac *snac, const char *home_marker, const char *notify_marker) /* gets or sets notification marker */ { xs *data = NULL; xs_dict *written = xs_dict_new(); xs *fn = xs_fmt("%s/markers.json", snac->basedir); FILE *f; if ((f = fopen(fn, "r")) != NULL) { data = xs_json_load(f); fclose(f); } if (xs_is_null(data)) data = xs_dict_new(); if (!xs_is_null(home_marker)) { xs *home = xs_dict_new(); xs *s_tid = tid(0); home = xs_dict_append(home, "last_read_id", home_marker); home = xs_dict_append(home, "version", xs_stock(0)); home = xs_dict_append(home, "updated_at", s_tid); data = xs_dict_set(data, "home", home); written = xs_dict_append(written, "home", home); } if (!xs_is_null(notify_marker)) { xs *notify = xs_dict_new(); xs *s_tid = tid(0); notify = xs_dict_append(notify, "last_read_id", notify_marker); notify = xs_dict_append(notify, "version", xs_stock(0)); notify = xs_dict_append(notify, "updated_at", s_tid); data = xs_dict_set(data, "notifications", notify); written = xs_dict_append(written, "notifications", notify); } if ((f = fopen(fn, "w")) != NULL) { xs_json_dump(data, 4, f); fclose(f); } return written; } void notify_add(snac *snac, const char *type, const char *utype, const char *actor, const char *objid, const xs_dict *msg) /* adds a new notification */ { xs *ntid = tid(0); xs *fn = xs_fmt("%s/notify/", snac->basedir); xs *date = xs_str_utctime(0, ISO_DATE_SPEC); FILE *f; /* create the directory */ mkdirx(fn); fn = xs_str_cat(fn, ntid); fn = xs_str_cat(fn, ".json"); xs *noti = xs_dict_new(); noti = xs_dict_append(noti, "id", ntid); noti = xs_dict_append(noti, "type", type); noti = xs_dict_append(noti, "utype", utype); noti = xs_dict_append(noti, "actor", actor); noti = xs_dict_append(noti, "date", date); noti = xs_dict_append(noti, "msg", msg); if (!xs_is_null(objid)) noti = xs_dict_append(noti, "objid", objid); if ((f = fopen(fn, "w")) != NULL) { xs_json_dump(noti, 4, f); fclose(f); } /* add it to the index if it already exists */ xs *idx = xs_fmt("%s/notify.idx", snac->basedir); if (mtime(idx) != 0.0) { pthread_mutex_lock(&data_mutex); if ((f = fopen(idx, "a")) != NULL) { fprintf(f, "%-32s\n", ntid); fclose(f); } pthread_mutex_unlock(&data_mutex); } } xs_dict *notify_get(snac *snac, const char *id) /* gets a notification */ { /* base file */ xs *fn = xs_fmt("%s/notify/%s", snac->basedir, id); /* strip spaces and add extension */ fn = xs_strip_i(fn); fn = xs_str_cat(fn, ".json"); FILE *f; xs_dict *out = NULL; if ((f = fopen(fn, "r")) != NULL) { out = xs_json_load(f); fclose(f); } return out; } xs_list *notify_list(snac *snac, int skip, int show) /* returns a list of notification ids */ { xs *idx = xs_fmt("%s/notify.idx", snac->basedir); if (mtime(idx) == 0.0) { /* create the index from scratch */ FILE *f; pthread_mutex_lock(&data_mutex); if ((f = fopen(idx, "w")) != NULL) { xs *spec = xs_fmt("%s/notify/" "*.json", snac->basedir); xs *lst = xs_glob(spec, 1, 0); xs_list *p = lst; const char *v; while (xs_list_iter(&p, &v)) { char *p = strrchr(v, '.'); if (p) { *p = '\0'; fprintf(f, "%-32s\n", v); } } fclose(f); } pthread_mutex_unlock(&data_mutex); } return index_list_desc(idx, skip, show); } int notify_new_num(snac *snac) /* counts the number of new notifications */ { xs *t = notify_check_time(snac, 0); xs *lst = notify_list(snac, 0, XS_ALL); int cnt = 0; xs_list *p = lst; const xs_str *v; while (xs_list_iter(&p, &v)) { xs *id = xs_strip_i(xs_dup(v)); /* old? count no more */ if (strcmp(id, t) < 0) break; cnt++; } return cnt; } void notify_clear(snac *snac) /* clears all notifications */ { xs *spec = xs_fmt("%s/notify/" "*", snac->basedir); xs *lst = xs_glob(spec, 0, 0); xs_list *p = lst; const xs_str *v; while (xs_list_iter(&p, &v)) unlink(v); xs *idx = xs_fmt("%s/notify.idx", snac->basedir); if (mtime(idx) != 0.0) { pthread_mutex_lock(&data_mutex); truncate(idx, 0); pthread_mutex_unlock(&data_mutex); } } /** the queue **/ static xs_dict *_enqueue_put(const char *fn, xs_dict *msg) /* writes safely to the queue */ { xs *tfn = xs_fmt("%s.tmp", fn); FILE *f; if ((f = fopen(tfn, "w")) != NULL) { xs_json_dump(msg, 4, f); fclose(f); rename(tfn, fn); } return msg; } static xs_dict *_new_qmsg(const char *type, const xs_val *msg, int retries) /* creates a queue message */ { int qrt = xs_number_get(xs_dict_get(srv_config, "queue_retry_minutes")); xs *ntid = tid(retries * 60 * qrt); xs *rn = xs_number_new(retries); xs_dict *qmsg = xs_dict_new(); qmsg = xs_dict_append(qmsg, "type", type); qmsg = xs_dict_append(qmsg, "message", msg); qmsg = xs_dict_append(qmsg, "retries", rn); qmsg = xs_dict_append(qmsg, "ntid", ntid); return qmsg; } void enqueue_input(snac *snac, const xs_dict *msg, const xs_dict *req, int retries) /* enqueues an input message */ { xs *qmsg = _new_qmsg("input", msg, retries); const char *ntid = xs_dict_get(qmsg, "ntid"); xs *fn = xs_fmt("%s/queue/%s.json", snac->basedir, ntid); qmsg = xs_dict_append(qmsg, "req", req); qmsg = _enqueue_put(fn, qmsg); snac_debug(snac, 1, xs_fmt("enqueue_input %s", xs_dict_get(msg, "id"))); } void enqueue_shared_input(const xs_dict *msg, const xs_dict *req, int retries) /* enqueues an input message from the shared input */ { xs *qmsg = _new_qmsg("input", msg, retries); const char *ntid = xs_dict_get(qmsg, "ntid"); xs *fn = xs_fmt("%s/queue/%s.json", srv_basedir, ntid); qmsg = xs_dict_append(qmsg, "req", req); qmsg = _enqueue_put(fn, qmsg); srv_debug(1, xs_fmt("enqueue_shared_input %s", xs_dict_get(msg, "id"))); } void enqueue_output_raw(const char *keyid, const char *seckey, const xs_dict *msg, const xs_str *inbox, int retries, int p_status) /* enqueues an output message to an inbox */ { xs *qmsg = _new_qmsg("output", msg, retries); const char *ntid = xs_dict_get(qmsg, "ntid"); xs *fn = xs_fmt("%s/queue/%s.json", srv_basedir, ntid); xs *ns = xs_number_new(p_status); qmsg = xs_dict_append(qmsg, "p_status", ns); qmsg = xs_dict_append(qmsg, "inbox", inbox); qmsg = xs_dict_append(qmsg, "keyid", keyid); qmsg = xs_dict_append(qmsg, "seckey", seckey); /* if it's to be sent right now, bypass the disk queue and post the job */ if (retries == 0 && p_state != NULL) job_post(qmsg, 0); else { qmsg = _enqueue_put(fn, qmsg); srv_debug(1, xs_fmt("enqueue_output %s %s %d", inbox, fn, retries)); } } void enqueue_output(snac *snac, const xs_dict *msg, const xs_str *inbox, int retries, int p_status) /* enqueues an output message to an inbox */ { if (xs_startswith(inbox, snac->actor)) { snac_debug(snac, 1, xs_str_new("refusing enqueue to myself")); return; } const char *seckey = xs_dict_get(snac->key, "secret"); enqueue_output_raw(snac->actor, seckey, msg, inbox, retries, p_status); } void enqueue_output_by_actor(snac *snac, const xs_dict *msg, const xs_str *actor, int retries) /* enqueues an output message for an actor */ { xs *inbox = get_actor_inbox(actor, 1); if (!xs_is_null(inbox)) enqueue_output(snac, msg, inbox, retries, 0); else snac_log(snac, xs_fmt("enqueue_output_by_actor cannot get inbox %s", actor)); } void enqueue_email(const xs_str *msg, int retries) /* enqueues an email message to be sent */ { xs *qmsg = _new_qmsg("email", msg, retries); const char *ntid = xs_dict_get(qmsg, "ntid"); xs *fn = xs_fmt("%s/queue/%s.json", srv_basedir, ntid); qmsg = _enqueue_put(fn, qmsg); srv_debug(1, xs_fmt("enqueue_email %d", retries)); } void enqueue_telegram(const xs_str *msg, const char *bot, const char *chat_id) /* enqueues a message to be sent via Telegram */ { xs *qmsg = _new_qmsg("telegram", msg, 0); const char *ntid = xs_dict_get(qmsg, "ntid"); xs *fn = xs_fmt("%s/queue/%s.json", srv_basedir, ntid); qmsg = xs_dict_append(qmsg, "bot", bot); qmsg = xs_dict_append(qmsg, "chat_id", chat_id); qmsg = _enqueue_put(fn, qmsg); srv_debug(1, xs_fmt("enqueue_telegram %s %s", bot, chat_id)); } void enqueue_ntfy(const xs_str *msg, const char *ntfy_server, const char *ntfy_token) /* enqueues a message to be sent via ntfy */ { xs *qmsg = _new_qmsg("ntfy", msg, 0); const char *ntid = xs_dict_get(qmsg, "ntid"); xs *fn = xs_fmt("%s/queue/%s.json", srv_basedir, ntid); qmsg = xs_dict_append(qmsg, "ntfy_server", ntfy_server); qmsg = xs_dict_append(qmsg, "ntfy_token", ntfy_token); qmsg = _enqueue_put(fn, qmsg); srv_debug(1, xs_fmt("enqueue_ntfy %s %s", ntfy_server, ntfy_token)); } void enqueue_message(snac *snac, const xs_dict *msg) /* enqueues an output message */ { xs *qmsg = _new_qmsg("message", msg, 0); const char *ntid = xs_dict_get(qmsg, "ntid"); xs *fn = xs_fmt("%s/queue/%s.json", snac->basedir, ntid); qmsg = _enqueue_put(fn, qmsg); snac_debug(snac, 0, xs_fmt("enqueue_message %s", xs_dict_get(msg, "id"))); } void enqueue_close_question(snac *user, const char *id, int end_secs) /* enqueues the closing of a question */ { xs *qmsg = _new_qmsg("close_question", id, 0); xs *ntid = tid(end_secs); xs *fn = xs_fmt("%s/queue/%s.json", user->basedir, ntid); qmsg = xs_dict_set(qmsg, "ntid", ntid); qmsg = _enqueue_put(fn, qmsg); snac_debug(user, 0, xs_fmt("enqueue_close_question %s", id)); } void enqueue_object_request(snac *user, const char *id, int forward_secs) /* enqueues the request of an object in the future */ { xs *qmsg = _new_qmsg("object_request", id, 0); xs *ntid = tid(forward_secs); xs *fn = xs_fmt("%s/queue/%s.json", user->basedir, ntid); qmsg = xs_dict_set(qmsg, "ntid", ntid); qmsg = _enqueue_put(fn, qmsg); snac_debug(user, 0, xs_fmt("enqueue_object_request %s %d", id, forward_secs)); } void enqueue_verify_links(snac *user) /* enqueues a link verification */ { xs *qmsg = _new_qmsg("verify_links", "", 0); const char *ntid = xs_dict_get(qmsg, "ntid"); xs *fn = xs_fmt("%s/queue/%s.json", user->basedir, ntid); qmsg = _enqueue_put(fn, qmsg); snac_debug(user, 1, xs_fmt("enqueue_verify_links %s", user->actor)); } void enqueue_actor_refresh(snac *user, const char *actor, int forward_secs) /* enqueues an actor refresh */ { xs *qmsg = _new_qmsg("actor_refresh", "", 0); xs *ntid = tid(forward_secs); xs *fn = xs_fmt("%s/queue/%s.json", user->basedir, ntid); qmsg = xs_dict_set(qmsg, "ntid", ntid); qmsg = xs_dict_append(qmsg, "actor", actor); qmsg = _enqueue_put(fn, qmsg); snac_debug(user, 1, xs_fmt("enqueue_actor_refresh %s", actor)); } int was_question_voted(snac *user, const char *id) /* returns true if the user voted in this poll */ { xs *children = object_children(id); int voted = 0; xs_list *p; const xs_str *md5; p = children; while (xs_list_iter(&p, &md5)) { xs *obj = NULL; if (valid_status(object_get_by_md5(md5, &obj))) { const char *atto = get_atto(obj); if (atto && strcmp(atto, user->actor) == 0 && !xs_is_null(xs_dict_get(obj, "name"))) { voted = 1; break; } } } return voted; } xs_list *user_queue(snac *snac) /* returns a list with filenames that can be dequeued */ { xs *spec = xs_fmt("%s/queue/" "*.json", snac->basedir); xs_list *list = xs_list_new(); time_t t = time(NULL); xs_list *p; const xs_val *v; xs *fns = xs_glob(spec, 0, 0); p = fns; while (xs_list_iter(&p, &v)) { /* get the retry time from the basename */ char *bn = strrchr(v, '/'); time_t t2 = atol(bn + 1); if (t2 > t) snac_debug(snac, 2, xs_fmt("user_queue not yet time for %s [%ld]", v, t)); else { list = xs_list_append(list, v); snac_debug(snac, 2, xs_fmt("user_queue ready for %s", v)); } } return list; } xs_list *queue(void) /* returns a list with filenames that can be dequeued */ { xs *spec = xs_fmt("%s/queue/" "*.json", srv_basedir); xs_list *list = xs_list_new(); time_t t = time(NULL); xs_list *p; const xs_val *v; xs *fns = xs_glob(spec, 0, 0); p = fns; while (xs_list_iter(&p, &v)) { /* get the retry time from the basename */ char *bn = strrchr(v, '/'); time_t t2 = atol(bn + 1); if (t2 > t) srv_debug(2, xs_fmt("queue not yet time for %s [%ld]", v, t)); else { list = xs_list_append(list, v); srv_debug(2, xs_fmt("queue ready for %s", v)); } } return list; } xs_dict *queue_get(const char *fn) /* gets a file from a queue */ { FILE *f; xs_dict *obj = NULL; if ((f = fopen(fn, "r")) != NULL) { obj = xs_json_load(f); fclose(f); } return obj; } xs_dict *dequeue(const char *fn) /* dequeues a message */ { xs_dict *obj = queue_get(fn); unlink(fn); return obj; } /** the purge **/ static int _purge_file(const char *fn, time_t mt) /* purge fn if it's older than days */ { int ret = 0; if (mtime(fn) < mt) { /* older than the minimum time: delete it */ unlink(fn); srv_debug(2, xs_fmt("purged %s", fn)); ret = 1; } return ret; } static void _purge_dir(const char *dir, int days) /* purges all files in a directory older than days */ { int cnt = 0; if (days) { time_t mt = time(NULL) - days * 24 * 3600; xs *spec = xs_fmt("%s/" "*", dir); xs *list = xs_glob(spec, 0, 0); xs_list *p; const xs_str *v; p = list; while (xs_list_iter(&p, &v)) cnt += _purge_file(v, mt); srv_debug(1, xs_fmt("purge: %s %d", dir, cnt)); } } static void _purge_user_subdir(snac *snac, const char *subdir, int days) /* purges all files in a user subdir older than days */ { xs *u_subdir = xs_fmt("%s/%s", snac->basedir, subdir); _purge_dir(u_subdir, days); } void purge_server(void) /* purge global server data */ { xs *spec = xs_fmt("%s/object/??", srv_basedir); xs *dirs = xs_glob(spec, 0, 0); xs_list *p; const xs_str *v; int cnt = 0; int icnt = 0; time_t mt = time(NULL) - 7 * 24 * 3600; p = dirs; while (xs_list_iter(&p, &v)) { xs_list *p2; const xs_str *v2; { xs *spec2 = xs_fmt("%s/" "*.json", v); xs *files = xs_glob(spec2, 0, 0); p2 = files; while (xs_list_iter(&p2, &v2)) { int n_link; /* old and with no hard links? */ if (mtime_nl(v2, &n_link) < mt && n_link < 2) { xs *s1 = xs_replace(v2, ".json", ""); xs *l = xs_split(s1, "/"); const char *md5 = xs_list_get(l, -1); object_del_by_md5(md5); cnt++; } } } { /* look for stray indexes */ xs *speci = xs_fmt("%s/" "*_?.idx", v); xs *idxfs = xs_glob(speci, 0, 0); p2 = idxfs; while (xs_list_iter(&p2, &v2)) { /* old enough to consider? */ if (mtime(v2) < mt) { /* check if the indexed object is here */ xs *o = xs_dup(v2); char *ext = strchr(o, '_'); if (ext) { *ext = '\0'; o = xs_str_cat(o, ".json"); if (mtime(o) == 0.0) { /* delete */ unlink(v2); srv_debug(1, xs_fmt("purged %s", v2)); icnt++; } } } } /* delete index backups */ xs *specb = xs_fmt("%s/" "*.bak", v); xs *bakfs = xs_glob(specb, 0, 0); p2 = bakfs; while (xs_list_iter(&p2, &v2)) { unlink(v2); srv_debug(1, xs_fmt("purged %s", v2)); } } } /* purge collected inboxes */ xs *ib_dir = xs_fmt("%s/inbox", srv_basedir); _purge_dir(ib_dir, 7); /* purge the instance timeline */ xs *itl_fn = xs_fmt("%s/public.idx", srv_basedir); int itl_gc = index_gc(itl_fn); /* purge tag indexes */ xs *tag_spec = xs_fmt("%s/tag/??", srv_basedir); xs *tag_dirs = xs_glob(tag_spec, 0, 0); p = tag_dirs; int tag_gc = 0; while (xs_list_iter(&p, &v)) { xs *spec2 = xs_fmt("%s/" "*.idx", v); xs *files = xs_glob(spec2, 0, 0); xs_list *p2; const xs_str *v2; p2 = files; while (xs_list_iter(&p2, &v2)) { tag_gc += index_gc(v2); xs *bak = xs_fmt("%s.bak", v2); unlink(bak); if (index_len(v2) == 0) { /* there are no longer any entry with this tag; purge it completely */ unlink(v2); xs *dottag = xs_replace(v2, ".idx", ".tag"); unlink(dottag); } } } srv_debug(1, xs_fmt("purge: global " "(obj: %d, idx: %d, itl: %d, tag: %d)", cnt, icnt, itl_gc, tag_gc)); } void purge_user(snac *snac) /* do the purge for this user */ { int priv_days, pub_days, user_days = 0; const char *v; int n; priv_days = xs_number_get(xs_dict_get(srv_config, "timeline_purge_days")); pub_days = xs_number_get(xs_dict_get(srv_config, "local_purge_days")); if ((v = xs_dict_get(snac->config_o, "purge_days")) != NULL || (v = xs_dict_get(snac->config, "purge_days")) != NULL) user_days = xs_number_get(v); if (user_days) { /* override admin settings only if they are lesser */ if (priv_days == 0 || user_days < priv_days) priv_days = user_days; if (pub_days == 0 || user_days < pub_days) pub_days = user_days; } _purge_user_subdir(snac, "hidden", priv_days); _purge_user_subdir(snac, "private", priv_days); _purge_user_subdir(snac, "public", pub_days); const char *idxs[] = { "followers.idx", "private.idx", "public.idx", "pinned.idx", "bookmark.idx", "draft.idx", NULL }; for (n = 0; idxs[n]; n++) { xs *idx = xs_fmt("%s/%s", snac->basedir, idxs[n]); int gc = index_gc(idx); srv_debug(1, xs_fmt("purge: %s %d", idx, gc)); } /* purge lists */ { xs *spec = xs_fmt("%s/list/" "*.idx", snac->basedir); xs *lol = xs_glob(spec, 0, 0); int c = 0; const char *v; while (xs_list_next(lol, &v, &c)) { int gc = index_gc(v); srv_debug(1, xs_fmt("purge: %s %d", v, gc)); } } /* unrelated to purging, but it's a janitorial process, so what the hell */ verify_links(snac); } void purge_all(void) /* purge all users */ { snac snac; xs *list = user_list(); char *p; const char *uid; p = list; while (xs_list_iter(&p, &uid)) { if (user_open(&snac, uid)) { purge_user(&snac); user_free(&snac); } } purge_server(); #ifndef NO_MASTODON_API mastoapi_purge(); #endif } /** archive **/ void srv_archive(const char *direction, const char *url, xs_dict *req, const char *payload, int p_size, int status, xs_dict *headers, const char *body, int b_size) /* archives a connection */ { /* obsessive archiving */ xs *date = tid(0); xs *dir = xs_fmt("%s/archive/%s_%s", srv_basedir, date, direction); FILE *f; if (mkdirx(dir) != -1) { xs *meta_fn = xs_fmt("%s/_META", dir); if ((f = fopen(meta_fn, "w")) != NULL) { xs *j1 = xs_json_dumps(req, 4); xs *j2 = xs_json_dumps(headers, 4); fprintf(f, "dir: %s\n", direction); if (url) fprintf(f, "url: %s\n", url); fprintf(f, "req: %s\n", j1); fprintf(f, "p_size: %d\n", p_size); fprintf(f, "status: %d\n", status); fprintf(f, "response: %s\n", j2); fprintf(f, "b_size: %d\n", b_size); fclose(f); } if (p_size && payload) { xs *payload_fn = NULL; xs *payload_fn_raw = NULL; const char *v = xs_dict_get(req, "content-type"); if (v && xs_str_in(v, "json") != -1) { payload_fn = xs_fmt("%s/payload.json", dir); if ((f = fopen(payload_fn, "w")) != NULL) { xs *v1 = xs_json_loads(payload); xs *j1 = NULL; if (v1 != NULL) j1 = xs_json_dumps(v1, 4); if (j1 != NULL) fwrite(j1, strlen(j1), 1, f); else fwrite(payload, p_size, 1, f); fclose(f); } } payload_fn_raw = xs_fmt("%s/payload", dir); if ((f = fopen(payload_fn_raw, "w")) != NULL) { fwrite(payload, p_size, 1, f); fclose(f); } } if (b_size && body) { xs *body_fn = NULL; const char *v = xs_dict_get(headers, "content-type"); if (v && xs_str_in(v, "json") != -1) { body_fn = xs_fmt("%s/body.json", dir); if ((f = fopen(body_fn, "w")) != NULL) { xs *v1 = xs_json_loads(body); xs *j1 = NULL; if (v1 != NULL) j1 = xs_json_dumps(v1, 4); if (j1 != NULL) fwrite(j1, strlen(j1), 1, f); else fwrite(body, b_size, 1, f); fclose(f); } } else { body_fn = xs_fmt("%s/body", dir); if ((f = fopen(body_fn, "w")) != NULL) { fwrite(body, b_size, 1, f); fclose(f); } } } } } void srv_archive_error(const char *prefix, const xs_str *err, const xs_dict *req, const xs_val *data) /* archives an error */ { xs *ntid = tid(0); xs *fn = xs_fmt("%s/error/%s_%s", srv_basedir, ntid, prefix); FILE *f; if ((f = fopen(fn, "w")) != NULL) { fprintf(f, "Error: %s\n", err); if (req) { fprintf(f, "Request headers:\n"); xs_json_dump(req, 4, f); fprintf(f, "\n"); } if (data) { fprintf(f, "Data:\n"); if (xs_type(data) == XSTYPE_LIST || xs_type(data) == XSTYPE_DICT) { xs_json_dump(data, 4, f); } else fprintf(f, "%s", data); fprintf(f, "\n"); } fclose(f); } } void srv_archive_qitem(const char *prefix, xs_dict *q_item) /* archives a q_item in the error folder */ { xs *ntid = tid(0); xs *fn = xs_fmt("%s/error/%s_qitem_%s", srv_basedir, ntid, prefix); FILE *f; if ((f = fopen(fn, "w")) != NULL) { xs_json_dump(q_item, 4, f); fclose(f); } } t_announcement *announcement(const double after) /* returns announcement text or NULL if none exists or it is olde than "after" */ { static const long int MAX_SIZE = 2048; static t_announcement a = { .text = NULL, .timestamp = 0.0, }; static xs_str *fn = NULL; if (fn == NULL) fn = xs_fmt("%s/announcement.txt", srv_basedir); const double ts = mtime(fn); /* file does not exist or other than what was requested */ if (ts == 0.0 || ts <= after) return NULL; /* nothing changed, just return the current announcement */ if (a.text != NULL && ts <= a.timestamp) return &a; /* read and store new announcement */ FILE *f; if ((f = fopen(fn, "r")) != NULL) { fseek (f, 0, SEEK_END); const long int length = ftell(f); if (length > MAX_SIZE) { /* this is probably unintentional */ srv_log(xs_fmt("announcement.txt too big: %ld bytes, max is %ld, ignoring.", length, MAX_SIZE)); } else if (length > 0) { fseek (f, 0, SEEK_SET); char *buffer = malloc(length + 1); if (buffer) { fread(buffer, 1, length, f); buffer[length] = '\0'; free(a.text); a.text = buffer; a.timestamp = ts; } else { srv_log("Error allocating memory for announcement"); } } else { /* an empty file means no announcement */ free(a.text); a.text = NULL; a.timestamp = 0.0; } fclose (f); } if (a.text != NULL) return &a; return NULL; } xs_str *make_url(const char *href, const char *proxy, int by_token) /* makes an URL, possibly including proxying */ { xs_str *url = NULL; if (proxy && !xs_startswith(href, srv_baseurl)) { xs *p = NULL; if (by_token) { xs *tks = xs_fmt("%s:%s", srv_proxy_token_seed, proxy); xs *tk = xs_md5_hex(tks, strlen(tks)); p = xs_fmt("%s/y/%s/", proxy, tk); } else p = xs_fmt("%s/x/", proxy); url = xs_replace(href, "https:/" "/", p); } else url = xs_dup(href); return url; } /** bad login throttle **/ xs_str *_badlogin_fn(const char *addr) { xs *md5 = xs_md5_hex(addr, strlen(addr)); xs *dir = xs_fmt("%s/badlogin", srv_basedir); mkdirx(dir); return xs_fmt("%s/%s", dir, md5); } int _badlogin_read(const char *fn, int *failures) /* reads a badlogin file */ { int ok = 0; FILE *f; pthread_mutex_lock(&data_mutex); if ((f = fopen(fn, "r")) != NULL) { xs *l = xs_readline(f); fclose(f); if (sscanf(l, "%d", failures) == 1) ok = 1; } pthread_mutex_unlock(&data_mutex); return ok; } int badlogin_check(const char *user, const char *addr) /* checks if this address is authorized to try a login */ { int valid = 1; if (xs_type(addr) == XSTYPE_STRING) { xs *fn = _badlogin_fn(addr); double mt = mtime(fn); if (mt > 0) { int badlogin_expire = xs_number_get(xs_dict_get_def(srv_config, "badlogin_expire", "300")); mt += badlogin_expire; /* if file is expired, delete and give pass */ if (mt < time(NULL)) { srv_debug(1, xs_fmt("Login from %s for %s allowed again", addr, user)); unlink(fn); } else { int failures; if (_badlogin_read(fn, &failures)) { int badlogin_max = xs_number_get(xs_dict_get_def(srv_config, "badlogin_retries", "5")); if (failures >= badlogin_max) { valid = 0; xs *d = xs_str_iso_date((time_t) mt); srv_debug(1, xs_fmt("Login from %s for %s forbidden until %s", addr, user, d)); } } } } } return valid; } void badlogin_inc(const char *user, const char *addr) /* increments a bad login from this address */ { if (xs_type(addr) == XSTYPE_STRING) { int failures = 0; xs *fn = _badlogin_fn(addr); FILE *f; _badlogin_read(fn, &failures); pthread_mutex_lock(&data_mutex); if ((f = fopen(fn, "w")) != NULL) { failures++; fprintf(f, "%d %s %s\n", failures, addr, user); fclose(f); srv_log(xs_fmt("Registered %d login failure(s) from %s for %s", failures, addr, user)); } pthread_mutex_unlock(&data_mutex); } }