snac.h 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450
  1. /* snac - A simple, minimalistic ActivityPub instance */
  2. /* copyright (c) 2022 - 2025 grunfink et al. / MIT license */
  3. #define VERSION "2.73"
  4. #define USER_AGENT "snac/" VERSION
  5. #define WHAT_IS_SNAC_URL "https:/" "/comam.es/what-is-snac"
  6. #define DIR_PERM 00770
  7. #define DIR_PERM_ADD 02770
  8. #define ISO_DATE_SPEC "%Y-%m-%dT%H:%M:%SZ"
  9. #ifndef MAX_THREADS
  10. #define MAX_THREADS 256
  11. #endif
  12. #ifndef MAX_JSON_DEPTH
  13. #define MAX_JSON_DEPTH 8
  14. #endif
  15. #ifndef MAX_CONVERSATION_LEVELS
  16. #define MAX_CONVERSATION_LEVELS 48
  17. #endif
  18. #define MD5_HEX_SIZE 33
  19. #define MD5_ALREADY_SEEN_MARK "00000000000000000000000000000000"
  20. extern double disk_layout;
  21. extern xs_str *srv_basedir;
  22. extern xs_dict *srv_config;
  23. extern xs_str *srv_baseurl;
  24. extern xs_str *srv_proxy_token_seed;
  25. extern xs_dict *srv_langs;
  26. extern int dbglevel;
  27. #define L(s) lang_str((s), user)
  28. #define POSTLIKE_OBJECT_TYPE "Note|Question|Page|Article|Video|Audio|Event"
  29. int mkdirx(const char *pathname);
  30. int valid_status(int status);
  31. xs_str *tid(int offset);
  32. double ftime(void);
  33. void srv_log(xs_str *str);
  34. #define srv_debug(level, str) do { if (dbglevel >= (level)) \
  35. { srv_log((str)); } } while (0)
  36. typedef struct {
  37. xs_str *uid; /* uid */
  38. xs_str *basedir; /* user base directory */
  39. xs_dict *config; /* user configuration */
  40. xs_dict *config_o; /* user configuration admin override */
  41. xs_dict *key; /* keypair */
  42. xs_dict *links; /* validated links */
  43. xs_str *actor; /* actor url */
  44. xs_str *md5; /* actor url md5 */
  45. const xs_dict *lang;/* string translation dict */
  46. } snac;
  47. typedef struct {
  48. int s_size; /* struct size (for double checking) */
  49. int srv_running; /* server running on/off */
  50. int use_fcgi; /* FastCGI use on/off */
  51. time_t srv_start_time; /* start time */
  52. int job_fifo_size; /* job fifo size */
  53. int peak_job_fifo_size; /* maximum job fifo size seen */
  54. int n_threads; /* number of configured threads */
  55. enum { THST_STOP, THST_WAIT, THST_IN, THST_QUEUE } th_state[MAX_THREADS];
  56. } srv_state;
  57. extern srv_state *p_state;
  58. void snac_log(snac *user, xs_str *str);
  59. #define snac_debug(user, level, str) do { if (dbglevel >= (level)) \
  60. { snac_log((user), (str)); } } while (0)
  61. int srv_open(const char *basedir, int auto_upgrade);
  62. void srv_free(void);
  63. void sbox_enter(const char *basedir);
  64. int user_open(snac *snac, const char *uid);
  65. void user_free(snac *snac);
  66. xs_list *user_list(void);
  67. int user_open_by_md5(snac *snac, const char *md5);
  68. int user_persist(snac *snac, int publish);
  69. int validate_uid(const char *uid);
  70. xs_str *hash_password(const char *uid, const char *passwd, const char *nonce);
  71. int check_password(const char *uid, const char *passwd, const char *hash);
  72. void srv_archive(const char *direction, const char *url, xs_dict *req,
  73. const char *payload, int p_size,
  74. int status, xs_dict *headers,
  75. const char *body, int b_size);
  76. void srv_archive_error(const char *prefix, const xs_str *err,
  77. const xs_dict *req, const xs_val *data);
  78. void srv_archive_qitem(const char *prefix, xs_dict *q_item);
  79. double mtime_nl(const char *fn, int *n_link);
  80. #define mtime(fn) mtime_nl(fn, NULL)
  81. double f_ctime(const char *fn);
  82. int index_add_md5(const char *fn, const char *md5);
  83. int index_add(const char *fn, const char *id);
  84. int index_gc(const char *fn);
  85. int index_first(const char *fn, char md5[MD5_HEX_SIZE]);
  86. int index_len(const char *fn);
  87. xs_list *index_list(const char *fn, int max);
  88. int index_desc_next(FILE *f, char md5[MD5_HEX_SIZE]);
  89. int index_desc_first(FILE *f, char md5[MD5_HEX_SIZE], int skip);
  90. int index_asc_next(FILE *f, char md5[MD5_HEX_SIZE]);
  91. int index_asc_first(FILE *f, char md5[MD5_HEX_SIZE], const char *seek_md5);
  92. xs_list *index_list_desc(const char *fn, int skip, int show);
  93. int object_add(const char *id, const xs_dict *obj);
  94. int object_add_ow(const char *id, const xs_dict *obj);
  95. int object_here_by_md5(const char *id);
  96. int object_here(const char *id);
  97. int object_get_by_md5(const char *md5, xs_dict **obj);
  98. int object_get(const char *id, xs_dict **obj);
  99. int object_del(const char *id);
  100. int object_del_if_unref(const char *id);
  101. double object_ctime_by_md5(const char *md5);
  102. double object_ctime(const char *id);
  103. double object_mtime_by_md5(const char *md5);
  104. double object_mtime(const char *id);
  105. void object_touch(const char *id);
  106. int object_admire(const char *id, const char *actor, int like);
  107. int object_unadmire(const char *id, const char *actor, int like);
  108. int object_likes_len(const char *id);
  109. int object_announces_len(const char *id);
  110. xs_list *object_children(const char *id);
  111. xs_list *object_likes(const char *id);
  112. xs_list *object_announces(const char *id);
  113. int object_parent(const char *md5, char parent[MD5_HEX_SIZE]);
  114. int object_user_cache_add(snac *snac, const char *id, const char *cachedir);
  115. int object_user_cache_del(snac *snac, const char *id, const char *cachedir);
  116. int follower_add(snac *snac, const char *actor);
  117. int follower_del(snac *snac, const char *actor);
  118. int follower_check(snac *snac, const char *actor);
  119. xs_list *follower_list(snac *snac);
  120. int pending_add(snac *user, const char *actor, const xs_dict *msg);
  121. int pending_check(snac *user, const char *actor);
  122. xs_dict *pending_get(snac *user, const char *actor);
  123. void pending_del(snac *user, const char *actor);
  124. xs_list *pending_list(snac *user);
  125. double timeline_mtime(snac *snac);
  126. int timeline_touch(snac *snac);
  127. int timeline_here(snac *snac, const char *md5);
  128. int timeline_get_by_md5(snac *snac, const char *md5, xs_dict **msg);
  129. int timeline_del(snac *snac, const char *id);
  130. xs_str *user_index_fn(snac *user, const char *idx_name);
  131. xs_list *timeline_simple_list(snac *user, const char *idx_name, int skip, int show, int *more);
  132. xs_list *timeline_list(snac *snac, const char *idx_name, int skip, int show, int *more);
  133. int timeline_add(snac *snac, const char *id, const xs_dict *o_msg);
  134. int timeline_admire(snac *snac, const char *id, const char *admirer, int like);
  135. xs_list *timeline_top_level(snac *snac, const xs_list *list);
  136. void timeline_add_mark(snac *user);
  137. xs_list *local_list(snac *snac, int max);
  138. xs_str *instance_index_fn(void);
  139. xs_list *timeline_instance_list(int skip, int show);
  140. int following_add(snac *snac, const char *actor, const xs_dict *msg);
  141. int following_del(snac *snac, const char *actor);
  142. int following_check(snac *snac, const char *actor);
  143. int following_get(snac *snac, const char *actor, xs_dict **data);
  144. xs_list *following_list(snac *snac);
  145. void mute(snac *snac, const char *actor);
  146. void unmute(snac *snac, const char *actor);
  147. int is_muted(snac *snac, const char *actor);
  148. xs_list *muted_list(snac *user);
  149. int is_bookmarked(snac *user, const char *id);
  150. int bookmark(snac *user, const char *id);
  151. int unbookmark(snac *user, const char *id);
  152. xs_list *bookmark_list(snac *user);
  153. xs_str *bookmark_index_fn(snac *user);
  154. int pin(snac *user, const char *id);
  155. int unpin(snac *user, const char *id);
  156. int is_pinned(snac *user, const char *id);
  157. int is_pinned_by_md5(snac *user, const char *md5);
  158. xs_list *pinned_list(snac *user);
  159. int is_draft(snac *user, const char *id);
  160. void draft_del(snac *user, const char *id);
  161. void draft_add(snac *user, const char *id, const xs_dict *msg);
  162. xs_list *draft_list(snac *user);
  163. int limited(snac *user, const char *id, int cmd);
  164. #define is_limited(user, id) limited((user), (id), 0)
  165. #define limit(user, id) limited((user), (id), 1)
  166. #define unlimit(user, id) limited((user), (id), 2)
  167. void hide(snac *snac, const char *id);
  168. int is_hidden(snac *snac, const char *id);
  169. void tag_index(const char *id, const xs_dict *obj);
  170. xs_str *tag_fn(const char *tag);
  171. xs_list *tag_search(const char *tag, int skip, int show);
  172. xs_val *list_maint(snac *user, const char *list, int op);
  173. xs_str *list_timeline_fn(snac *user, const char *list);
  174. xs_list *list_timeline(snac *user, const char *list, int skip, int show);
  175. xs_val *list_content(snac *user, const char *list_id, const char *actor_md5, int op);
  176. void list_distribute(snac *user, const char *who, const xs_dict *post);
  177. int actor_add(const char *actor, const xs_dict *msg);
  178. int actor_get(const char *actor, xs_dict **data);
  179. int actor_get_refresh(snac *user, const char *actor, xs_dict **data);
  180. int static_get(snac *snac, const char *id, xs_val **data, int *size, const char *inm, xs_str **etag);
  181. void static_put(snac *snac, const char *id, const char *data, int size);
  182. void static_put_meta(snac *snac, const char *id, const char *str);
  183. xs_str *static_get_meta(snac *snac, const char *id);
  184. double history_mtime(snac *snac, const char *id);
  185. void history_add(snac *snac, const char *id, const char *content, int size,
  186. xs_str **etag);
  187. int history_get(snac *snac, const char *id, xs_str **content, int *size,
  188. const char *inm, xs_str **etag);
  189. int history_del(snac *snac, const char *id);
  190. xs_list *history_list(snac *snac);
  191. void lastlog_write(snac *snac, const char *source);
  192. xs_str *notify_check_time(snac *snac, int reset);
  193. void notify_add(snac *snac, const char *type, const char *utype,
  194. const char *actor, const char *objid, const xs_dict *msg);
  195. xs_dict *notify_get(snac *snac, const char *id);
  196. int notify_new_num(snac *snac);
  197. xs_list *notify_list(snac *snac, int skip, int show);
  198. void notify_clear(snac *snac);
  199. xs_dict *markers_get(snac *snac, const xs_list *markers);
  200. xs_dict *markers_set(snac *snac, const char *home_marker, const char *notify_marker);
  201. void inbox_add(const char *inbox);
  202. void inbox_add_by_actor(const xs_dict *actor);
  203. xs_list *inbox_list(void);
  204. int is_instance_blocked(const char *instance);
  205. int instance_block(const char *instance);
  206. int instance_unblock(const char *instance);
  207. int content_match(const char *file, const xs_dict *msg);
  208. xs_list *content_search(snac *user, const char *regex,
  209. int priv, int skip, int show, int max_secs, int *timeout);
  210. void enqueue_input(snac *snac, const xs_dict *msg, const xs_dict *req, int retries);
  211. void enqueue_shared_input(const xs_dict *msg, const xs_dict *req, int retries);
  212. void enqueue_output_raw(const char *keyid, const char *seckey,
  213. const xs_dict *msg, const xs_str *inbox,
  214. int retries, int p_status);
  215. void enqueue_output(snac *snac, const xs_dict *msg,
  216. const xs_str *inbox, int retries, int p_status);
  217. void enqueue_output_by_actor(snac *snac, const xs_dict *msg,
  218. const xs_str *actor, int retries);
  219. void enqueue_email(const xs_str *msg, int retries);
  220. void enqueue_telegram(const xs_str *msg, const char *bot, const char *chat_id);
  221. void enqueue_ntfy(const xs_str *msg, const char *ntfy_server, const char *ntfy_token);
  222. void enqueue_message(snac *snac, const xs_dict *msg);
  223. void enqueue_close_question(snac *user, const char *id, int end_secs);
  224. void enqueue_object_request(snac *user, const char *id, int forward_secs);
  225. void enqueue_verify_links(snac *user);
  226. void enqueue_actor_refresh(snac *user, const char *actor, int forward_secs);
  227. int was_question_voted(snac *user, const char *id);
  228. xs_list *user_queue(snac *snac);
  229. xs_list *queue(void);
  230. xs_dict *queue_get(const char *fn);
  231. xs_dict *dequeue(const char *fn);
  232. void purge(snac *snac);
  233. void purge_all(void);
  234. xs_dict *http_signed_request_raw(const char *keyid, const char *seckey,
  235. const char *method, const char *url,
  236. const xs_dict *headers,
  237. const char *body, int b_size,
  238. int *status, xs_str **payload, int *p_size,
  239. int timeout);
  240. xs_dict *http_signed_request(snac *snac, const char *method, const char *url,
  241. const xs_dict *headers,
  242. const char *body, int b_size,
  243. int *status, xs_str **payload, int *p_size,
  244. int timeout);
  245. int check_signature(const xs_dict *req, xs_str **err);
  246. srv_state *srv_state_op(xs_str **fname, int op);
  247. void httpd(void);
  248. int webfinger_request_signed(snac *snac, const char *qs, xs_str **actor, xs_str **user);
  249. int webfinger_request(const char *qs, xs_str **actor, xs_str **user);
  250. int webfinger_request_fake(const char *qs, xs_str **actor, xs_str **user);
  251. int webfinger_get_handler(xs_dict *req, const char *q_path,
  252. xs_val **body, int *b_size, char **ctype);
  253. const char *default_avatar_base64(void);
  254. xs_str *process_tags(snac *snac, const char *content, xs_list **tag);
  255. const char *get_atto(const xs_dict *msg);
  256. const char *get_in_reply_to(const xs_dict *msg);
  257. xs_list *get_attachments(const xs_dict *msg);
  258. xs_dict *msg_admiration(snac *snac, const char *object, const char *type);
  259. xs_dict *msg_repulsion(snac *user, const char *id, const char *type);
  260. xs_dict *msg_create(snac *snac, const xs_dict *object);
  261. xs_dict *msg_follow(snac *snac, const char *actor);
  262. xs_dict *msg_note(snac *snac, const xs_str *content, const xs_val *rcpts,
  263. const xs_str *in_reply_to, const xs_list *attach,
  264. int scope, const char *lang_str, const char *msg_date);
  265. xs_dict *msg_undo(snac *snac, const xs_val *object);
  266. xs_dict *msg_delete(snac *snac, const char *id);
  267. xs_dict *msg_actor(snac *snac);
  268. xs_dict *msg_update(snac *snac, const xs_dict *object);
  269. xs_dict *msg_ping(snac *user, const char *rcpt);
  270. xs_dict *msg_pong(snac *user, const char *rcpt, const char *object);
  271. xs_dict *msg_move(snac *user, const char *new_account);
  272. xs_dict *msg_accept(snac *snac, const xs_val *object, const char *to);
  273. xs_dict *msg_question(snac *user, const char *content, xs_list *attach,
  274. const xs_list *opts, int multiple, int end_secs);
  275. int activitypub_request(snac *snac, const char *url, xs_dict **data);
  276. int actor_request(snac *user, const char *actor, xs_dict **data);
  277. int send_to_inbox_raw(const char *keyid, const char *seckey,
  278. const xs_str *inbox, const xs_dict *msg,
  279. xs_val **payload, int *p_size, int timeout);
  280. int send_to_inbox(snac *snac, const xs_str *inbox, const xs_dict *msg,
  281. xs_val **payload, int *p_size, int timeout);
  282. xs_str *get_actor_inbox(const char *actor, int shared);
  283. int send_to_actor(snac *snac, const char *actor, const xs_dict *msg,
  284. xs_val **payload, int *p_size, int timeout);
  285. int is_msg_public(const xs_dict *msg);
  286. int is_msg_from_private_user(const xs_dict *msg);
  287. int is_msg_for_me(snac *snac, const xs_dict *msg);
  288. int process_user_queue(snac *snac);
  289. void process_queue_item(xs_dict *q_item);
  290. int process_queue(void);
  291. int activitypub_get_handler(const xs_dict *req, const char *q_path,
  292. char **body, int *b_size, char **ctype);
  293. int activitypub_post_handler(const xs_dict *req, const char *q_path,
  294. char *payload, int p_size,
  295. char **body, int *b_size, char **ctype);
  296. xs_dict *emojis(void);
  297. xs_str *not_really_markdown(const char *content, xs_list **attach, xs_list **tag);
  298. xs_str *sanitize(const char *content);
  299. xs_str *encode_html(const char *str);
  300. xs_str *html_timeline(snac *user, const xs_list *list, int read_only,
  301. int skip, int show, int show_more,
  302. const char *title, const char *page, int utl, const char *error);
  303. int html_get_handler(const xs_dict *req, const char *q_path,
  304. char **body, int *b_size, char **ctype,
  305. xs_str **etag, xs_str **last_modified);
  306. int html_post_handler(const xs_dict *req, const char *q_path,
  307. char *payload, int p_size,
  308. char **body, int *b_size, char **ctype);
  309. xs_str *timeline_to_rss(snac *user, const xs_list *timeline,
  310. const char *title, const char *link, const char *desc);
  311. int write_default_css(void);
  312. int snac_init(const char *_basedir);
  313. int adduser(const char *uid);
  314. int resetpwd(snac *snac);
  315. int deluser(snac *user);
  316. extern const char *snac_blurb;
  317. void job_post(const xs_val *job, int urgent);
  318. void job_wait(xs_val **job);
  319. int oauth_get_handler(const xs_dict *req, const char *q_path,
  320. char **body, int *b_size, char **ctype);
  321. int oauth_post_handler(const xs_dict *req, const char *q_path,
  322. const char *payload, int p_size,
  323. char **body, int *b_size, char **ctype);
  324. int mastoapi_get_handler(const xs_dict *req, const char *q_path,
  325. char **body, int *b_size, char **ctype, xs_str **link);
  326. int mastoapi_post_handler(const xs_dict *req, const char *q_path,
  327. const char *payload, int p_size,
  328. char **body, int *b_size, char **ctype);
  329. int mastoapi_delete_handler(const xs_dict *req, const char *q_path,
  330. const char *payload, int p_size,
  331. char **body, int *b_size, char **ctype);
  332. int mastoapi_put_handler(const xs_dict *req, const char *q_path,
  333. const char *payload, int p_size,
  334. char **body, int *b_size, char **ctype);
  335. void persist_image(const char *key, const xs_val *data, const char *payload, snac *snac);
  336. int mastoapi_patch_handler(const xs_dict *req, const char *q_path,
  337. const char *payload, int p_size,
  338. char **body, int *b_size, char **ctype);
  339. void mastoapi_purge(void);
  340. void verify_links(snac *user);
  341. void export_csv(snac *user);
  342. int migrate_account(snac *user);
  343. void import_blocked_accounts_csv(snac *user, const char *fn);
  344. void import_following_accounts_csv(snac *user, const char *fn);
  345. void import_list_csv(snac *user, const char *fn);
  346. void import_csv(snac *user);
  347. typedef enum {
  348. #define HTTP_STATUS(code, name, text) HTTP_STATUS_ ## name = code,
  349. #include "http_codes.h"
  350. #undef HTTP_STATUS
  351. } http_status;
  352. const char *http_status_text(int status);
  353. typedef struct {
  354. double timestamp;
  355. char *text;
  356. } t_announcement;
  357. t_announcement *announcement(double after);
  358. xs_str *make_url(const char *href, const char *proxy, int by_token);
  359. int badlogin_check(const char *user, const char *addr);
  360. void badlogin_inc(const char *user, const char *addr);
  361. const char *lang_str(const char *str, const snac *user);