html.c 65 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175
  1. /* snac - A simple, minimalistic ActivityPub instance */
  2. /* copyright (c) 2022 - 2023 grunfink / MIT license */
  3. #include "xs.h"
  4. #include "xs_io.h"
  5. #include "xs_json.h"
  6. #include "xs_regex.h"
  7. #include "xs_set.h"
  8. #include "xs_openssl.h"
  9. #include "xs_time.h"
  10. #include "xs_mime.h"
  11. #include "snac.h"
  12. int login(snac *snac, const xs_dict *headers)
  13. /* tries a login */
  14. {
  15. int logged_in = 0;
  16. const char *auth = xs_dict_get(headers, "authorization");
  17. if (auth && xs_startswith(auth, "Basic ")) {
  18. int sz;
  19. xs *s1 = xs_crop_i(xs_dup(auth), 6, 0);
  20. xs *s2 = xs_base64_dec(s1, &sz);
  21. xs *l1 = xs_split_n(s2, ":", 1);
  22. if (xs_list_len(l1) == 2) {
  23. logged_in = check_password(
  24. xs_list_get(l1, 0), xs_list_get(l1, 1),
  25. xs_dict_get(snac->config, "passwd"));
  26. }
  27. }
  28. if (logged_in)
  29. lastlog_write(snac, "web");
  30. return logged_in;
  31. }
  32. xs_str *actor_name(xs_dict *actor)
  33. /* gets the actor name */
  34. {
  35. xs_list *p;
  36. char *v;
  37. xs_str *name;
  38. if (xs_is_null((v = xs_dict_get(actor, "name"))) || *v == '\0') {
  39. if (xs_is_null(v = xs_dict_get(actor, "preferredUsername")) || *v == '\0') {
  40. v = "anonymous";
  41. }
  42. }
  43. name = xs_dup(v);
  44. /* replace the :shortnames: */
  45. if (!xs_is_null(p = xs_dict_get(actor, "tag"))) {
  46. /* iterate the tags */
  47. while (xs_list_iter(&p, &v)) {
  48. char *t = xs_dict_get(v, "type");
  49. if (t && strcmp(t, "Emoji") == 0) {
  50. char *n = xs_dict_get(v, "name");
  51. char *i = xs_dict_get(v, "icon");
  52. if (n && i) {
  53. char *u = xs_dict_get(i, "url");
  54. xs *img = xs_fmt("<img src=\"%s\" style=\"height: 1em\" loading=\"lazy\"/>", u);
  55. name = xs_replace_i(name, n, img);
  56. }
  57. }
  58. }
  59. }
  60. return name;
  61. }
  62. xs_str *html_actor_icon(xs_str *os, char *actor,
  63. const char *date, const char *udate, const char *url, int priv)
  64. {
  65. xs *s = xs_str_new(NULL);
  66. xs *avatar = NULL;
  67. char *v;
  68. xs *name = actor_name(actor);
  69. /* get the avatar */
  70. if ((v = xs_dict_get(actor, "icon")) != NULL &&
  71. (v = xs_dict_get(v, "url")) != NULL) {
  72. avatar = xs_dup(v);
  73. }
  74. if (avatar == NULL)
  75. avatar = xs_fmt("data:image/png;base64, %s", default_avatar_base64());
  76. {
  77. xs *s1 = xs_fmt("<p><img class=\"snac-avatar\" src=\"%s\" alt=\"\" "
  78. "loading=\"lazy\"/>\n", avatar);
  79. s = xs_str_cat(s, s1);
  80. }
  81. {
  82. xs *s1 = xs_fmt("<a href=\"%s\" class=\"p-author h-card snac-author\">%s</a>",
  83. xs_dict_get(actor, "id"), name);
  84. s = xs_str_cat(s, s1);
  85. }
  86. if (!xs_is_null(url)) {
  87. xs *s1 = xs_fmt(" <a href=\"%s\">»</a>", url);
  88. s = xs_str_cat(s, s1);
  89. }
  90. if (priv)
  91. s = xs_str_cat(s, " <span title=\"private\">&#128274;</span>");
  92. if (strcmp(xs_dict_get(actor, "type"), "Service") == 0)
  93. s = xs_str_cat(s, " <span title=\"bot\">&#129302;</span>");
  94. if (xs_is_null(date)) {
  95. s = xs_str_cat(s, "\n&nbsp;\n");
  96. }
  97. else {
  98. xs *date_label = xs_crop_i(xs_dup(date), 0, 10);
  99. xs *date_title = xs_dup(date);
  100. if (!xs_is_null(udate)) {
  101. xs *sd = xs_crop_i(xs_dup(udate), 0, 10);
  102. date_label = xs_str_cat(date_label, " / ");
  103. date_label = xs_str_cat(date_label, sd);
  104. date_title = xs_str_cat(date_title, " / ");
  105. date_title = xs_str_cat(date_title, udate);
  106. }
  107. xs *s1 = xs_fmt(
  108. "\n<time class=\"dt-published snac-pubdate\" title=\"%s\">%s</time>\n",
  109. date_title, date_label);
  110. s = xs_str_cat(s, s1);
  111. }
  112. {
  113. char *username, *id;
  114. xs *s1;
  115. if (xs_is_null(username = xs_dict_get(actor, "preferredUsername")) || *username == '\0') {
  116. /* This should never be reached */
  117. username = "anonymous";
  118. }
  119. if (xs_is_null(id = xs_dict_get(actor, "id")) || *id == '\0') {
  120. /* This should never be reached */
  121. id = "https://social.example.org/anonymous";
  122. }
  123. /* "LIKE AN ANIMAL" */
  124. xs *domain = xs_split(id, "/");
  125. xs *user = xs_fmt("@%s@%s", username, xs_list_get(domain, 2));
  126. s1 = xs_fmt(
  127. "<br><a href=\"%s\" class=\"p-author-tag h-card snac-author-tag\">%s</a>",
  128. xs_dict_get(actor, "id"), user);
  129. s = xs_str_cat(s, s1);
  130. }
  131. return xs_str_cat(os, s);
  132. }
  133. xs_str *html_msg_icon(snac *snac, xs_str *os, const xs_dict *msg)
  134. {
  135. char *actor_id;
  136. xs *actor = NULL;
  137. if ((actor_id = xs_dict_get(msg, "attributedTo")) == NULL)
  138. actor_id = xs_dict_get(msg, "actor");
  139. if (actor_id && valid_status(actor_get(snac, actor_id, &actor))) {
  140. char *date = NULL;
  141. char *udate = NULL;
  142. char *url = NULL;
  143. int priv = 0;
  144. if (strcmp(xs_dict_get(msg, "type"), "Note") == 0)
  145. url = xs_dict_get(msg, "id");
  146. priv = !is_msg_public(snac, msg);
  147. date = xs_dict_get(msg, "published");
  148. udate = xs_dict_get(msg, "updated");
  149. os = html_actor_icon(os, actor, date, udate, url, priv);
  150. }
  151. return os;
  152. }
  153. d_char *html_user_header(snac *snac, d_char *s, int local)
  154. /* creates the HTML header */
  155. {
  156. char *p, *v;
  157. s = xs_str_cat(s, "<!DOCTYPE html>\n<html>\n<head>\n");
  158. s = xs_str_cat(s, "<meta name=\"viewport\" "
  159. "content=\"width=device-width, initial-scale=1\"/>\n");
  160. s = xs_str_cat(s, "<meta name=\"generator\" "
  161. "content=\"" USER_AGENT "\"/>\n");
  162. /* add server CSS */
  163. p = xs_dict_get(srv_config, "cssurls");
  164. while (xs_list_iter(&p, &v)) {
  165. xs *s1 = xs_fmt("<link rel=\"stylesheet\" type=\"text/css\" href=\"%s\"/>\n", v);
  166. s = xs_str_cat(s, s1);
  167. }
  168. /* add the user CSS */
  169. {
  170. xs *css = NULL;
  171. int size;
  172. /* try to open the user css */
  173. if (!valid_status(static_get(snac, "style.css", &css, &size))) {
  174. /* it's not there; try to open the server-wide css */
  175. FILE *f;
  176. xs *g_css_fn = xs_fmt("%s/style.css", srv_basedir);
  177. if ((f = fopen(g_css_fn, "r")) != NULL) {
  178. css = xs_readall(f);
  179. fclose(f);
  180. }
  181. }
  182. if (css != NULL) {
  183. xs *s1 = xs_fmt("<style>%s</style>\n", css);
  184. s = xs_str_cat(s, s1);
  185. }
  186. }
  187. {
  188. xs *s1 = xs_fmt("<title>%s (@%s@%s)</title>\n",
  189. xs_dict_get(snac->config, "name"),
  190. snac->uid,
  191. xs_dict_get(srv_config, "host"));
  192. s = xs_str_cat(s, s1);
  193. }
  194. xs *avatar = xs_dup(xs_dict_get(snac->config, "avatar"));
  195. if (avatar == NULL || *avatar == '\0') {
  196. xs_free(avatar);
  197. avatar = xs_fmt("data:image/png;base64, %s", default_avatar_base64());
  198. }
  199. {
  200. xs *s_bio = xs_dup(xs_dict_get(snac->config, "bio"));
  201. int n;
  202. /* shorten the bio */
  203. for (n = 0; s_bio[n] && s_bio[n] != '&' && s_bio[n] != '.' &&
  204. s_bio[n] != '\r' && s_bio[n] != '\n' && n < 128; n++);
  205. s_bio[n] = '\0';
  206. xs *s_avatar = xs_dup(avatar);
  207. /* don't inline an empty avatar: create a real link */
  208. if (xs_startswith(s_avatar, "data:")) {
  209. xs_free(s_avatar);
  210. s_avatar = xs_fmt("%s/susie.png", srv_baseurl);
  211. }
  212. /* og properties */
  213. xs *s1 = xs_fmt(
  214. "<meta property=\"og:site_name\" content=\"%s\"/>\n"
  215. "<meta property=\"og:title\" content=\"%s (@%s@%s)\"/>\n"
  216. "<meta property=\"og:description\" content=\"%s\"/>\n"
  217. "<meta property=\"og:image\" content=\"%s\"/>\n"
  218. "<meta property=\"og:image:width\" content=\"300\"/>\n"
  219. "<meta property=\"og:image:height\" content=\"300\"/>\n",
  220. xs_dict_get(srv_config, "host"),
  221. xs_dict_get(snac->config, "name"),
  222. snac->uid,
  223. xs_dict_get(srv_config, "host"),
  224. s_bio,
  225. s_avatar);
  226. s = xs_str_cat(s, s1);
  227. }
  228. {
  229. xs *s1 = xs_fmt("<link rel=\"alternate\" type=\"application/rss+xml\" "
  230. "title=\"RSS\" href=\"%s.rss\" />\n", snac->actor);
  231. s = xs_str_cat(s, s1);
  232. }
  233. s = xs_str_cat(s, "</head>\n<body>\n");
  234. /* top nav */
  235. s = xs_str_cat(s, "<nav class=\"snac-top-nav\">");
  236. {
  237. xs *s1;
  238. s1 = xs_fmt("<img src=\"%s\" class=\"snac-avatar\" alt=\"\"/>&nbsp;", avatar);
  239. s = xs_str_cat(s, s1);
  240. }
  241. {
  242. xs *s1;
  243. if (local)
  244. s1 = xs_fmt(
  245. "<a href=\"%s.rss\">%s</a> - "
  246. "<a href=\"%s/admin\" rel=\"nofollow\">%s</a></nav>\n",
  247. snac->actor, L("RSS"),
  248. snac->actor, L("private"));
  249. else {
  250. xs *n_list = notify_list(snac, 1);
  251. int n_len = xs_list_len(n_list);
  252. xs *n_str = NULL;
  253. /* show the number of new notifications, if there are any */
  254. if (n_len)
  255. n_str = xs_fmt("<sup style=\"background-color: red; "
  256. "color: white;\"> %d </sup> ", n_len);
  257. else
  258. n_str = xs_str_new("");
  259. s1 = xs_fmt(
  260. "<a href=\"%s\">%s</a> - "
  261. "<a href=\"%s/admin\">%s</a> - "
  262. "<a href=\"%s/notifications\">%s</a>%s - "
  263. "<a href=\"%s/people\">%s</a></nav>\n",
  264. snac->actor, L("public"),
  265. snac->actor, L("private"),
  266. snac->actor, L("notifications"), n_str,
  267. snac->actor, L("people"));
  268. }
  269. s = xs_str_cat(s, s1);
  270. }
  271. /* user info */
  272. {
  273. char *_tmpl =
  274. "<div class=\"h-card snac-top-user\">\n"
  275. "<p class=\"p-name snac-top-user-name\">%s</p>\n"
  276. "<p class=\"snac-top-user-id\">@%s@%s</p>\n";
  277. xs *s1 = xs_fmt(_tmpl,
  278. xs_dict_get(snac->config, "name"),
  279. xs_dict_get(snac->config, "uid"), xs_dict_get(srv_config, "host")
  280. );
  281. s = xs_str_cat(s, s1);
  282. if (local) {
  283. xs *bio = not_really_markdown(xs_dict_get(snac->config, "bio"), NULL);
  284. xs *s1 = xs_fmt("<div class=\"p-note snac-top-user-bio\">%s</div>\n", bio);
  285. s = xs_str_cat(s, s1);
  286. }
  287. s = xs_str_cat(s, "</div>\n");
  288. }
  289. return s;
  290. }
  291. d_char *html_top_controls(snac *snac, d_char *s)
  292. /* generates the top controls */
  293. {
  294. char *_tmpl =
  295. "<div class=\"snac-top-controls\">\n"
  296. "<div class=\"snac-note\">\n"
  297. "<form autocomplete=\"off\" method=\"post\" "
  298. "action=\"%s/admin/note\" enctype=\"multipart/form-data\">\n"
  299. "<textarea class=\"snac-textarea\" name=\"content\" "
  300. "rows=\"8\" wrap=\"virtual\" required=\"required\"></textarea>\n"
  301. "<input type=\"hidden\" name=\"in_reply_to\" value=\"\">\n"
  302. "<p>%s: <input type=\"checkbox\" name=\"sensitive\">\n"
  303. "<p>%s: <input type=\"checkbox\" name=\"mentioned_only\">\n"
  304. "<details><summary>%s</summary>\n" /** attach **/
  305. "<p>%s: <input type=\"file\" name=\"attach\">\n"
  306. "<p>%s: <input type=\"text\" name=\"alt_text\">\n"
  307. "</details>\n"
  308. "<p>"
  309. "<details><summary>%s</summary>\n" /** poll **/
  310. "<p>%s:<br>\n"
  311. "<textarea class=\"snac-textarea\" name=\"poll_options\" "
  312. "rows=\"6\" wrap=\"virtual\"></textarea>\n"
  313. "<p><select name=\"poll_multiple\">\n"
  314. "<option value=\"off\">%s</option>\n"
  315. "<option value=\"on\">%s</option>\n"
  316. "</select>\n"
  317. "<select name=\"poll_end_secs\" id=\"poll_end_secs\">\n"
  318. "<option value=\"300\">%s</option>\n"
  319. "<option value=\"3600\">%s</option>\n"
  320. "<option value=\"86400\">%s</option>\n"
  321. "</select>\n"
  322. "</details>\n"
  323. "<p><input type=\"submit\" class=\"button\" value=\"%s\">\n"
  324. "</form><p>\n"
  325. "</div>\n"
  326. "<div class=\"snac-top-controls-more\">\n"
  327. "<details><summary>%s</summary>\n"
  328. "<form autocomplete=\"off\" method=\"post\" action=\"%s/admin/action\">\n" /** follow **/
  329. "<input type=\"text\" name=\"actor\" required=\"required\">\n"
  330. "<input type=\"submit\" name=\"action\" value=\"%s\"> %s\n"
  331. "</form><p>\n"
  332. "<form autocomplete=\"off\" method=\"post\" action=\"%s/admin/action\">\n" /** boost **/
  333. "<input type=\"text\" name=\"id\" required=\"required\">\n"
  334. "<input type=\"submit\" name=\"action\" value=\"%s\"> %s\n"
  335. "</form><p>\n"
  336. "<details><summary>%s</summary>\n"
  337. "<div class=\"snac-user-setup\">\n" /** user setup **/
  338. "<form autocomplete=\"off\" method=\"post\" "
  339. "action=\"%s/admin/user-setup\" enctype=\"multipart/form-data\">\n"
  340. "<p>%s:<br>\n"
  341. "<input type=\"text\" name=\"name\" value=\"%s\"></p>\n"
  342. "<p>%s: <input type=\"file\" name=\"avatar_file\"></p>\n"
  343. "<p>%s:<br>\n"
  344. "<textarea name=\"bio\" cols=\"40\" rows=\"4\">%s</textarea></p>\n"
  345. "<p><input type=\"checkbox\" name=\"cw\" id=\"cw\" %s>\n"
  346. "<label for=\"cw\">%s</label></p>\n"
  347. "<p>%s:<br>\n"
  348. "<input type=\"text\" name=\"email\" value=\"%s\"></p>\n"
  349. "<p>%s:<br>\n"
  350. "<input type=\"text\" name=\"telegram_bot\" placeholder=\"Bot API key\" value=\"%s\"> "
  351. "<input type=\"text\" name=\"telegram_chat_id\" placeholder=\"Chat id\" value=\"%s\"></p>\n"
  352. "<p>%s:<br>\n"
  353. "<input type=\"number\" name=\"purge_days\" value=\"%s\"></p>\n"
  354. "<p><input type=\"checkbox\" name=\"drop_dm_from_unknown\" id=\"drop_dm_from_unknown\" %s>\n"
  355. "<label for=\"drop_dm_from_unknown\">%s</label></p>\n"
  356. "<p><input type=\"checkbox\" name=\"bot\" id=\"bot\" %s>\n"
  357. "<label for=\"bot\">%s</label></p>\n"
  358. "<p>%s:<br>\n"
  359. "<input type=\"password\" name=\"passwd1\" value=\"\"></p>\n"
  360. "<p>%s:<br>\n"
  361. "<input type=\"password\" name=\"passwd2\" value=\"\"></p>\n"
  362. "<input type=\"submit\" class=\"button\" value=\"%s\">\n"
  363. "</form>\n"
  364. "</div>\n"
  365. "</details>\n"
  366. "</details>\n"
  367. "</div>\n"
  368. "</div>\n";
  369. const char *email = "[disabled by admin]";
  370. if (xs_type(xs_dict_get(srv_config, "disable_email_notifications")) != XSTYPE_TRUE) {
  371. email = xs_dict_get(snac->config_o, "email");
  372. if (xs_is_null(email)) {
  373. email = xs_dict_get(snac->config, "email");
  374. if (xs_is_null(email))
  375. email = "";
  376. }
  377. }
  378. char *cw = xs_dict_get(snac->config, "cw");
  379. if (xs_is_null(cw))
  380. cw = "";
  381. char *telegram_bot = xs_dict_get(snac->config, "telegram_bot");
  382. if (xs_is_null(telegram_bot))
  383. telegram_bot = "";
  384. char *telegram_chat_id = xs_dict_get(snac->config, "telegram_chat_id");
  385. if (xs_is_null(telegram_chat_id))
  386. telegram_chat_id = "";
  387. const char *purge_days = xs_dict_get(snac->config, "purge_days");
  388. if (!xs_is_null(purge_days) && xs_type(purge_days) == XSTYPE_NUMBER)
  389. purge_days = xs_number_str(purge_days);
  390. else
  391. purge_days = "0";
  392. const char *d_dm_f_u = xs_dict_get(snac->config, "drop_dm_from_unknown");
  393. const char *bot = xs_dict_get(snac->config, "bot");
  394. xs *s1 = xs_fmt(_tmpl,
  395. snac->actor,
  396. L("Sensitive content"),
  397. L("Only for mentioned people"),
  398. L("Attach..."),
  399. L("File"),
  400. L("File description"),
  401. L("Poll..."),
  402. L("Poll options (one per line, up to 8)"),
  403. L("One choice"),
  404. L("Multiple choices"),
  405. L("End in 5 minutes"),
  406. L("End in 1 hour"),
  407. L("End in 1 day"),
  408. L("Post"),
  409. L("More options..."),
  410. snac->actor,
  411. L("Follow"), L("(by URL or user@host)"),
  412. snac->actor,
  413. L("Boost"), L("(by URL)"),
  414. L("User setup..."),
  415. snac->actor,
  416. L("User name"),
  417. xs_dict_get(snac->config, "name"),
  418. L("Avatar"),
  419. L("Bio"),
  420. xs_dict_get(snac->config, "bio"),
  421. strcmp(cw, "open") == 0 ? "checked" : "",
  422. L("Always show sensitive content"),
  423. L("Email address for notifications"),
  424. email,
  425. L("Telegram notifications (bot key and chat id)"),
  426. telegram_bot,
  427. telegram_chat_id,
  428. L("Maximum days to keep posts (0: server settings)"),
  429. purge_days,
  430. xs_type(d_dm_f_u) == XSTYPE_TRUE ? "checked" : "",
  431. L("Drop direct messages from people you don't follow"),
  432. xs_type(bot) == XSTYPE_TRUE ? "checked" : "",
  433. L("This account is a bot"),
  434. L("Password (only to change it)"),
  435. L("Repeat Password"),
  436. L("Update user info")
  437. );
  438. s = xs_str_cat(s, s1);
  439. return s;
  440. }
  441. d_char *html_button(d_char *s, char *clss, char *label)
  442. {
  443. xs *s1 = xs_fmt(
  444. "<input type=\"submit\" name=\"action\" "
  445. "class=\"snac-btn-%s\" value=\"%s\">\n",
  446. clss, label);
  447. return xs_str_cat(s, s1);
  448. }
  449. xs_str *build_mentions(snac *snac, const xs_dict *msg)
  450. /* returns a string with the mentions in msg */
  451. {
  452. xs_str *s = xs_str_new(NULL);
  453. char *list = xs_dict_get(msg, "tag");
  454. char *v;
  455. while (xs_list_iter(&list, &v)) {
  456. char *type = xs_dict_get(v, "type");
  457. char *href = xs_dict_get(v, "href");
  458. char *name = xs_dict_get(v, "name");
  459. if (type && strcmp(type, "Mention") == 0 &&
  460. href && strcmp(href, snac->actor) != 0 && name) {
  461. xs *s1 = NULL;
  462. if (name[0] != '@') {
  463. s1 = xs_fmt("@%s", name);
  464. name = s1;
  465. }
  466. xs *l = xs_split(name, "@");
  467. /* is it a name without a host? */
  468. if (xs_list_len(l) < 3) {
  469. /* split the href and pick the host name LIKE AN ANIMAL */
  470. /* would be better to query the webfinger but *won't do that* here */
  471. xs *l2 = xs_split(href, "/");
  472. if (xs_list_len(l2) >= 3) {
  473. xs *s1 = xs_fmt("%s@%s ", name, xs_list_get(l2, 2));
  474. s = xs_str_cat(s, s1);
  475. }
  476. }
  477. else {
  478. s = xs_str_cat(s, name);
  479. s = xs_str_cat(s, " ");
  480. }
  481. }
  482. }
  483. if (*s) {
  484. xs *s1 = s;
  485. s = xs_fmt("\n\n\nCC: %s", s1);
  486. }
  487. return s;
  488. }
  489. xs_str *html_entry_controls(snac *snac, xs_str *os, const xs_dict *msg, const char *md5)
  490. {
  491. char *id = xs_dict_get(msg, "id");
  492. char *actor = xs_dict_get(msg, "attributedTo");
  493. xs *likes = object_likes(id);
  494. xs *boosts = object_announces(id);
  495. xs *s = xs_str_new(NULL);
  496. s = xs_str_cat(s, "<div class=\"snac-controls\">\n");
  497. {
  498. xs *s1 = xs_fmt(
  499. "<form autocomplete=\"off\" method=\"post\" action=\"%s/admin/action\">\n"
  500. "<input type=\"hidden\" name=\"id\" value=\"%s\">\n"
  501. "<input type=\"hidden\" name=\"actor\" value=\"%s\">\n"
  502. "<input type=\"hidden\" name=\"redir\" value=\"%s_entry\">\n"
  503. "\n",
  504. snac->actor, id, actor, md5
  505. );
  506. s = xs_str_cat(s, s1);
  507. }
  508. if (xs_list_in(likes, snac->md5) == -1) {
  509. /* not already liked; add button */
  510. s = html_button(s, "like", L("Like"));
  511. }
  512. if (is_msg_public(snac, msg)) {
  513. if (strcmp(actor, snac->actor) == 0 || xs_list_in(boosts, snac->md5) == -1) {
  514. /* not already boosted or us; add button */
  515. s = html_button(s, "boost", L("Boost"));
  516. }
  517. }
  518. if (strcmp(actor, snac->actor) != 0) {
  519. /* controls for other actors than this one */
  520. if (following_check(snac, actor)) {
  521. s = html_button(s, "unfollow", L("Unfollow"));
  522. }
  523. else {
  524. s = html_button(s, "follow", L("Follow"));
  525. }
  526. s = html_button(s, "mute", L("MUTE"));
  527. }
  528. s = html_button(s, "delete", L("Delete"));
  529. s = html_button(s, "hide", L("Hide"));
  530. s = xs_str_cat(s, "</form>\n");
  531. const char *prev_src1 = xs_dict_get(msg, "sourceContent");
  532. if (!xs_is_null(prev_src1) && strcmp(actor, snac->actor) == 0) {
  533. xs *prev_src = xs_replace(prev_src1, "<", "&lt;");
  534. /* post can be edited */
  535. xs *s1 = xs_fmt(
  536. "<p><details><summary>%s</summary>\n"
  537. "<p><div class=\"snac-note\" id=\"%s_edit\">\n"
  538. "<form autocomplete=\"off\" method=\"post\" action=\"%s/admin/note\" "
  539. "enctype=\"multipart/form-data\" id=\"%s_edit_form\">\n"
  540. "<textarea class=\"snac-textarea\" name=\"content\" "
  541. "rows=\"4\" wrap=\"virtual\" required=\"required\">%s</textarea>\n"
  542. "<input type=\"hidden\" name=\"edit_id\" value=\"%s\">\n"
  543. "<p>%s: <input type=\"checkbox\" name=\"sensitive\">\n"
  544. "<p>%s: <input type=\"checkbox\" name=\"mentioned_only\">\n"
  545. "<details><summary>%s</summary>\n"
  546. "<p>%s: <input type=\"file\" name=\"attach\">\n"
  547. "<p>%s: <input type=\"text\" name=\"alt_text\">\n"
  548. "</details>\n"
  549. "<input type=\"hidden\" name=\"redir\" value=\"%s_entry\">\n"
  550. "<p><input type=\"submit\" class=\"button\" value=\"%s\">\n"
  551. "</form><p></div>\n"
  552. "</details><p>"
  553. "\n",
  554. L("Edit..."),
  555. md5,
  556. snac->actor, md5,
  557. prev_src,
  558. id,
  559. L("Sensitive content"),
  560. L("Only for mentioned people"),
  561. L("Attach..."),
  562. L("File"),
  563. L("File description"),
  564. md5,
  565. L("Post")
  566. );
  567. s = xs_str_cat(s, s1);
  568. }
  569. {
  570. /* the post textarea */
  571. xs *ct = build_mentions(snac, msg);
  572. xs *s1 = xs_fmt(
  573. "<p><details><summary>%s</summary>\n"
  574. "<p><div class=\"snac-note\" id=\"%s_reply\">\n"
  575. "<form autocomplete=\"off\" method=\"post\" action=\"%s/admin/note\" "
  576. "enctype=\"multipart/form-data\" id=\"%s_reply_form\">\n"
  577. "<textarea class=\"snac-textarea\" name=\"content\" "
  578. "rows=\"4\" wrap=\"virtual\" required=\"required\">%s</textarea>\n"
  579. "<input type=\"hidden\" name=\"in_reply_to\" value=\"%s\">\n"
  580. "<p>%s: <input type=\"checkbox\" name=\"sensitive\">\n"
  581. "<p>%s: <input type=\"checkbox\" name=\"mentioned_only\">\n"
  582. "<details><summary>%s</summary>\n"
  583. "<p>%s: <input type=\"file\" name=\"attach\">\n"
  584. "<p>%s: <input type=\"text\" name=\"alt_text\">\n"
  585. "</details>\n"
  586. "<input type=\"hidden\" name=\"redir\" value=\"%s_entry\">\n"
  587. "<p><input type=\"submit\" class=\"button\" value=\"%s\">\n"
  588. "</form><p></div>\n"
  589. "</details><p>"
  590. "\n",
  591. L("Reply..."),
  592. md5,
  593. snac->actor, md5,
  594. ct,
  595. id,
  596. L("Sensitive content"),
  597. L("Only for mentioned people"),
  598. L("Attach..."),
  599. L("File"),
  600. L("File description"),
  601. md5,
  602. L("Post")
  603. );
  604. s = xs_str_cat(s, s1);
  605. }
  606. s = xs_str_cat(s, "</div>\n");
  607. return xs_str_cat(os, s);
  608. }
  609. xs_str *html_entry(snac *snac, xs_str *os, const xs_dict *msg, int local,
  610. int level, const char *md5, int hide_children)
  611. {
  612. char *id = xs_dict_get(msg, "id");
  613. char *type = xs_dict_get(msg, "type");
  614. char *actor;
  615. int sensitive = 0;
  616. char *v;
  617. xs *boosts = NULL;
  618. /* do not show non-public messages in the public timeline */
  619. if (local && !is_msg_public(snac, msg))
  620. return os;
  621. /* hidden? do nothing more for this conversation */
  622. if (is_hidden(snac, id))
  623. return os;
  624. xs *s = xs_str_new("<div>\n");
  625. {
  626. xs *s1 = xs_fmt("<a name=\"%s_entry\"></a>\n", md5);
  627. s = xs_str_cat(s, s1);
  628. }
  629. if (strcmp(type, "Follow") == 0) {
  630. s = xs_str_cat(s, "<div class=\"snac-post\">\n");
  631. xs *s1 = xs_fmt("<div class=\"snac-origin\">%s</div>\n", L("follows you"));
  632. s = xs_str_cat(s, s1);
  633. s = html_msg_icon(snac, s, msg);
  634. s = xs_str_cat(s, "</div>\n");
  635. return xs_str_cat(os, s);
  636. }
  637. else
  638. if (strcmp(type, "Note") != 0 && strcmp(type, "Question") != 0) {
  639. /* skip oddities */
  640. return os;
  641. }
  642. /* ignore notes with "name", as they are votes to Questions */
  643. if (strcmp(type, "Note") == 0 && !xs_is_null(xs_dict_get(msg, "name")))
  644. return os;
  645. /* bring the main actor */
  646. if ((actor = xs_dict_get(msg, "attributedTo")) == NULL)
  647. return os;
  648. /* ignore muted morons immediately */
  649. if (is_muted(snac, actor))
  650. return os;
  651. if (strcmp(actor, snac->actor) != 0 && !valid_status(actor_get(snac, actor, NULL)))
  652. return os;
  653. s = xs_str_cat(s, "<div class=\"snac-score\">"); /** **/
  654. if (strcmp(type, "Question") == 0) {
  655. /* add the ballot box emoji */
  656. xs *f = xs_fmt("<span title=\"%s\"> &#128499; </span>", L("Poll"));
  657. s = xs_str_cat(s, f);
  658. if (was_question_voted(snac, id)) {
  659. /* add a check to show this poll was voted */
  660. xs *f2 = xs_fmt("<span title=\"%s\"> &#10003; </span>", L("Voted"));
  661. s = xs_str_cat(s, f2);
  662. }
  663. }
  664. /* if this is our post, add the score */
  665. if (xs_startswith(id, snac->actor)) {
  666. int n_likes = object_likes_len(id);
  667. int n_boosts = object_announces_len(id);
  668. /* alternate emojis: %d &#128077; %d &#128257; */
  669. xs *s1 = xs_fmt("%d &#9733; %d &#8634;\n", n_likes, n_boosts);
  670. s = xs_str_cat(s, s1);
  671. }
  672. s = xs_str_cat(s, "</div>\n");
  673. if (level == 0)
  674. s = xs_str_cat(s, "<div class=\"snac-post\">\n"); /** **/
  675. else
  676. s = xs_str_cat(s, "<div class=\"snac-child\">\n"); /** **/
  677. if (boosts == NULL)
  678. boosts = object_announces(id);
  679. if (xs_list_len(boosts)) {
  680. /* if somebody boosted this, show as origin */
  681. char *p = xs_list_get(boosts, -1);
  682. xs *actor_r = NULL;
  683. if (xs_list_in(boosts, snac->md5) != -1) {
  684. /* we boosted this */
  685. xs *s1 = xs_fmt(
  686. "<div class=\"snac-origin\">"
  687. "<a href=\"%s\">%s</a> %s</a></div>",
  688. snac->actor, xs_dict_get(snac->config, "name"), L("boosted")
  689. );
  690. s = xs_str_cat(s, s1);
  691. }
  692. else
  693. if (valid_status(object_get_by_md5(p, &actor_r))) {
  694. xs *name = actor_name(actor_r);
  695. if (!xs_is_null(name)) {
  696. xs *s1 = xs_fmt(
  697. "<div class=\"snac-origin\">"
  698. "<a href=\"%s\">%s</a> %s</div>\n",
  699. xs_dict_get(actor_r, "id"),
  700. name,
  701. L("boosted")
  702. );
  703. s = xs_str_cat(s, s1);
  704. }
  705. }
  706. }
  707. else
  708. if (strcmp(type, "Note") == 0) {
  709. if (level == 0) {
  710. /* is the parent not here? */
  711. char *parent = xs_dict_get(msg, "inReplyTo");
  712. if (!xs_is_null(parent) && *parent && !timeline_here(snac, parent)) {
  713. xs *s1 = xs_fmt(
  714. "<div class=\"snac-origin\">%s "
  715. "<a href=\"%s\">»</a></div>\n",
  716. L("in reply to"), parent
  717. );
  718. s = xs_str_cat(s, s1);
  719. }
  720. }
  721. }
  722. s = html_msg_icon(snac, s, msg);
  723. /* add the content */
  724. s = xs_str_cat(s, "<div class=\"e-content snac-content\">\n"); /** **/
  725. /* is it sensitive? */
  726. if (!xs_is_null(v = xs_dict_get(msg, "sensitive")) && xs_type(v) == XSTYPE_TRUE) {
  727. if (xs_is_null(v = xs_dict_get(msg, "summary")) || *v == '\0')
  728. v = "...";
  729. /* only show it when not in the public timeline and the config setting is "open" */
  730. char *cw = xs_dict_get(snac->config, "cw");
  731. if (xs_is_null(cw) || local)
  732. cw = "";
  733. xs *s1 = xs_fmt("<details %s><summary>%s [%s]</summary>\n", cw, v, L("SENSITIVE CONTENT"));
  734. s = xs_str_cat(s, s1);
  735. sensitive = 1;
  736. }
  737. #if 0
  738. {
  739. xs *md5 = xs_md5_hex(id, strlen(id));
  740. xs *s1 = xs_fmt("<p><code>%s</code></p>\n", md5);
  741. s = xs_str_cat(s, s1);
  742. }
  743. #endif
  744. {
  745. const char *content = xs_dict_get(msg, "content");
  746. xs *c = sanitize(xs_is_null(content) ? "" : content);
  747. char *p, *v;
  748. /* do some tweaks to the content */
  749. c = xs_replace_i(c, "\r", "");
  750. while (xs_endswith(c, "<br><br>"))
  751. c = xs_crop_i(c, 0, -4);
  752. c = xs_replace_i(c, "<br><br>", "<p>");
  753. if (!xs_startswith(c, "<p>")) {
  754. xs *s1 = c;
  755. c = xs_fmt("<p>%s</p>", s1);
  756. }
  757. /* replace the :shortnames: */
  758. if (!xs_is_null(p = xs_dict_get(msg, "tag"))) {
  759. /* iterate the tags */
  760. while (xs_list_iter(&p, &v)) {
  761. char *t = xs_dict_get(v, "type");
  762. if (t && strcmp(t, "Emoji") == 0) {
  763. char *n = xs_dict_get(v, "name");
  764. char *i = xs_dict_get(v, "icon");
  765. if (n && i) {
  766. char *u = xs_dict_get(i, "url");
  767. xs *img = xs_fmt("<img src=\"%s\" style=\"height: 1em\" "
  768. "loading=\"lazy\" title=\"%s\"/>", u, n);
  769. c = xs_replace_i(c, n, img);
  770. }
  771. }
  772. }
  773. }
  774. if (strcmp(type, "Question") == 0) { /** question content **/
  775. xs_list *oo = xs_dict_get(msg, "oneOf");
  776. xs_list *ao = xs_dict_get(msg, "anyOf");
  777. xs_list *p;
  778. xs_dict *v;
  779. int closed = 0;
  780. if (xs_dict_get(msg, "closed"))
  781. closed = 2;
  782. else
  783. if (xs_startswith(id, snac->actor))
  784. closed = 1; /* we questioned; closed for us */
  785. else
  786. if (was_question_voted(snac, id))
  787. closed = 1; /* we already voted; closed for us */
  788. /* get the appropriate list of options */
  789. p = oo != NULL ? oo : ao;
  790. if (closed) {
  791. /* closed poll */
  792. c = xs_str_cat(c, "<table class=\"snac-poll-result\">\n");
  793. while (xs_list_iter(&p, &v)) {
  794. const char *name = xs_dict_get(v, "name");
  795. const xs_dict *replies = xs_dict_get(v, "replies");
  796. if (name && replies) {
  797. int nr = xs_number_get(xs_dict_get(replies, "totalItems"));
  798. xs *l = xs_fmt("<tr><td>%s:</td><td>%d</td></tr>\n", name, nr);
  799. c = xs_str_cat(c, l);
  800. }
  801. }
  802. c = xs_str_cat(c, "</table>\n");
  803. /* if it's *really* closed, say it */
  804. if (closed == 2) {
  805. xs *s1 = xs_fmt("<p>%s</p>\n", L("Closed"));
  806. c = xs_str_cat(c, s1);
  807. }
  808. }
  809. else {
  810. /* poll still active */
  811. xs *s1 = xs_fmt("<div class=\"snac-poll-form\">\n"
  812. "<form autocomplete=\"off\" "
  813. "method=\"post\" action=\"%s/admin/vote\">\n"
  814. "<input type=\"hidden\" name=\"actor\" value= \"%s\">\n"
  815. "<input type=\"hidden\" name=\"irt\" value=\"%s\">\n",
  816. snac->actor, actor, id);
  817. while (xs_list_iter(&p, &v)) {
  818. const char *name = xs_dict_get(v, "name");
  819. if (name) {
  820. xs *opt = xs_fmt("<input type=\"%s\""
  821. " id=\"%s\" value=\"%s\" name=\"question\"> %s<br>\n",
  822. !xs_is_null(oo) ? "radio" : "checkbox",
  823. name, name, name);
  824. s1 = xs_str_cat(s1, opt);
  825. }
  826. }
  827. xs *s2 = xs_fmt("<p><input type=\"submit\" "
  828. "class=\"button\" value=\"%s\">\n</form>\n</div>\n\n", L("Vote"));
  829. s1 = xs_str_cat(s1, s2);
  830. c = xs_str_cat(c, s1);
  831. }
  832. }
  833. s = xs_str_cat(s, c);
  834. }
  835. s = xs_str_cat(s, "\n");
  836. /* add the attachments */
  837. char *attach;
  838. if ((attach = xs_dict_get(msg, "attachment")) != NULL) { /** **/
  839. char *v;
  840. /* make custom css for attachments easier */
  841. s = xs_str_cat(s, "<p class=\"snac-content-attachments\">\n");
  842. while (xs_list_iter(&attach, &v)) {
  843. char *t = xs_dict_get(v, "mediaType");
  844. if (xs_is_null(t))
  845. continue;
  846. if (xs_startswith(t, "image/")) {
  847. char *url = xs_dict_get(v, "url");
  848. char *name = xs_dict_get(v, "name");
  849. if (url != NULL) {
  850. if (xs_is_null(name))
  851. name = "";
  852. xs *s1 = xs_fmt(
  853. "<a href=\"%s\" target=\"_blank\"><img src=\"%s\" alt=\"%s\" title=\"%s\" loading=\"lazy\"/></a>\n",
  854. url, url, name, name);
  855. s = xs_str_cat(s, s1);
  856. }
  857. }
  858. else
  859. if (xs_startswith(t, "video/")) {
  860. char *url = xs_dict_get(v, "url");
  861. if (url != NULL) {
  862. xs *s1 = xs_fmt("<object data=\"%s\"></object>\n", url);
  863. s = xs_str_cat(s, s1);
  864. }
  865. }
  866. }
  867. s = xs_str_cat(s, "</p>\n");
  868. }
  869. if (sensitive)
  870. s = xs_str_cat(s, "</details><p>\n");
  871. s = xs_str_cat(s, "</div>\n");
  872. /** controls **/
  873. if (!local)
  874. s = html_entry_controls(snac, s, msg, md5);
  875. /** children **/
  876. if (!hide_children) {
  877. xs *children = object_children(id);
  878. int left = xs_list_len(children);
  879. if (left) {
  880. char *p, *cmd5;
  881. int older_open = 0;
  882. xs *ss = xs_str_new(NULL);
  883. int n_children = 0;
  884. ss = xs_str_cat(ss, "<details open><summary>...</summary><p>\n");
  885. if (level < 4)
  886. ss = xs_str_cat(ss, "<div class=\"snac-children\">\n");
  887. else
  888. ss = xs_str_cat(ss, "<div>\n");
  889. if (left > 3) {
  890. xs *s1 = xs_fmt("<details><summary>%s</summary>\n", L("Older..."));
  891. ss = xs_str_cat(ss, s1);
  892. older_open = 1;
  893. }
  894. p = children;
  895. while (xs_list_iter(&p, &cmd5)) {
  896. xs *chd = NULL;
  897. timeline_get_by_md5(snac, cmd5, &chd);
  898. if (older_open && left <= 3) {
  899. ss = xs_str_cat(ss, "</details>\n");
  900. older_open = 0;
  901. }
  902. if (chd != NULL && xs_is_null(xs_dict_get(chd, "name"))) {
  903. ss = html_entry(snac, ss, chd, local, level + 1, cmd5, hide_children);
  904. n_children++;
  905. }
  906. else
  907. snac_debug(snac, 2, xs_fmt("cannot read from timeline child %s", cmd5));
  908. left--;
  909. }
  910. if (older_open)
  911. ss = xs_str_cat(ss, "</details>\n");
  912. ss = xs_str_cat(ss, "</div>\n");
  913. ss = xs_str_cat(ss, "</details>\n");
  914. if (n_children)
  915. s = xs_str_cat(s, ss);
  916. }
  917. }
  918. s = xs_str_cat(s, "</div>\n</div>\n");
  919. return xs_str_cat(os, s);
  920. }
  921. xs_str *html_user_footer(xs_str *s)
  922. {
  923. xs *s1 = xs_fmt(
  924. "<div class=\"snac-footer\">\n"
  925. "<a href=\"%s\">%s</a> - "
  926. "powered by <a href=\"%s\""
  927. "<abbr title=\"Social Networks Are Crap\">snac</abbr></div></a>\n",
  928. srv_baseurl,
  929. L("about this site"),
  930. WHAT_IS_SNAC_URL
  931. );
  932. return xs_str_cat(s, s1);
  933. }
  934. xs_str *html_timeline(snac *snac, const xs_list *list, int local, int skip, int show, int show_more)
  935. /* returns the HTML for the timeline */
  936. {
  937. xs_str *s = xs_str_new(NULL);
  938. xs_list *p = (xs_list *)list;
  939. char *v;
  940. double t = ftime();
  941. s = html_user_header(snac, s, local);
  942. if (!local)
  943. s = html_top_controls(snac, s);
  944. s = xs_str_cat(s, "<a name=\"snac-posts\"></a>\n");
  945. s = xs_str_cat(s, "<div class=\"snac-posts\">\n");
  946. while (xs_list_iter(&p, &v)) {
  947. xs *msg = NULL;
  948. if (!valid_status(timeline_get_by_md5(snac, v, &msg)))
  949. continue;
  950. s = html_entry(snac, s, msg, local, 0, v, 0);
  951. }
  952. s = xs_str_cat(s, "</div>\n");
  953. if (local) {
  954. xs *s1 = xs_fmt(
  955. "<div class=\"snac-history\">\n"
  956. "<p class=\"snac-history-title\">%s</p><ul>\n",
  957. L("History")
  958. );
  959. s = xs_str_cat(s, s1);
  960. xs *list = history_list(snac);
  961. char *p, *v;
  962. p = list;
  963. while (xs_list_iter(&p, &v)) {
  964. xs *fn = xs_replace(v, ".html", "");
  965. xs *s1 = xs_fmt(
  966. "<li><a href=\"%s/h/%s\">%s</a></li>\n",
  967. snac->actor, v, fn);
  968. s = xs_str_cat(s, s1);
  969. }
  970. s = xs_str_cat(s, "</ul></div>\n");
  971. }
  972. {
  973. xs *s1 = xs_fmt("<!-- %lf seconds -->\n", ftime() - t);
  974. s = xs_str_cat(s, s1);
  975. }
  976. if (show_more) {
  977. xs *s1 = xs_fmt(
  978. "<p>"
  979. "<a href=\"%s%s?skip=%d&show=%d\" name=\"snac-more\">%s</a>"
  980. "</p>\n", snac->actor, local ? "" : "/admin", skip + show, show, L("Load more..."));
  981. s = xs_str_cat(s, s1);
  982. }
  983. s = html_user_footer(s);
  984. s = xs_str_cat(s, "</body>\n</html>\n");
  985. return s;
  986. }
  987. d_char *html_people_list(snac *snac, d_char *os, d_char *list, const char *header, const char *t)
  988. {
  989. xs *s = xs_str_new(NULL);
  990. xs *h = xs_fmt("<h2>%s</h2>\n", header);
  991. char *p, *actor_id;
  992. s = xs_str_cat(s, h);
  993. p = list;
  994. while (xs_list_iter(&p, &actor_id)) {
  995. xs *md5 = xs_md5_hex(actor_id, strlen(actor_id));
  996. xs *actor = NULL;
  997. if (valid_status(actor_get(snac, actor_id, &actor))) {
  998. s = xs_str_cat(s, "<div class=\"snac-post\">\n");
  999. s = html_actor_icon(s, actor, xs_dict_get(actor, "published"), NULL, NULL, 0);
  1000. /* content (user bio) */
  1001. char *c = xs_dict_get(actor, "summary");
  1002. if (!xs_is_null(c)) {
  1003. s = xs_str_cat(s, "<div class=\"snac-content\">\n");
  1004. xs *sc = sanitize(c);
  1005. if (xs_startswith(sc, "<p>"))
  1006. s = xs_str_cat(s, sc);
  1007. else {
  1008. xs *s1 = xs_fmt("<p>%s</p>", sc);
  1009. s = xs_str_cat(s, s1);
  1010. }
  1011. s = xs_str_cat(s, "</div>\n");
  1012. }
  1013. /* buttons */
  1014. s = xs_str_cat(s, "<div class=\"snac-controls\">\n");
  1015. xs *s1 = xs_fmt(
  1016. "<p><form autocomplete=\"off\" method=\"post\" action=\"%s/admin/action\">\n"
  1017. "<input type=\"hidden\" name=\"actor\" value=\"%s\">\n"
  1018. "<input type=\"hidden\" name=\"actor-form\" value=\"yes\">\n",
  1019. snac->actor, actor_id
  1020. );
  1021. s = xs_str_cat(s, s1);
  1022. if (following_check(snac, actor_id))
  1023. s = html_button(s, "unfollow", L("Unfollow"));
  1024. else {
  1025. s = html_button(s, "follow", L("Follow"));
  1026. if (follower_check(snac, actor_id))
  1027. s = html_button(s, "delete", L("Delete"));
  1028. }
  1029. if (is_muted(snac, actor_id))
  1030. s = html_button(s, "unmute", L("Unmute"));
  1031. else
  1032. s = html_button(s, "mute", L("MUTE"));
  1033. s = xs_str_cat(s, "</form>\n");
  1034. /* the post textarea */
  1035. xs *s2 = xs_fmt(
  1036. "<p><details><summary>%s</summary>\n"
  1037. "<p><div class=\"snac-note\" id=\"%s_%s_dm\">\n"
  1038. "<form autocomplete=\"off\" method=\"post\" action=\"%s/admin/note\" "
  1039. "enctype=\"multipart/form-data\" id=\"%s_reply_form\">\n"
  1040. "<textarea class=\"snac-textarea\" name=\"content\" "
  1041. "rows=\"4\" wrap=\"virtual\" required=\"required\"></textarea>\n"
  1042. "<input type=\"hidden\" name=\"to\" value=\"%s\">\n"
  1043. "<p><input type=\"file\" name=\"attach\">\n"
  1044. "<p><input type=\"submit\" class=\"button\" value=\"%s\">\n"
  1045. "</form><p></div>\n"
  1046. "</details><p>\n",
  1047. L("Direct Message..."),
  1048. md5, t,
  1049. snac->actor, md5,
  1050. actor_id,
  1051. L("Post")
  1052. );
  1053. s = xs_str_cat(s, s2);
  1054. s = xs_str_cat(s, "</div>\n");
  1055. s = xs_str_cat(s, "</div>\n");
  1056. }
  1057. }
  1058. return xs_str_cat(os, s);
  1059. }
  1060. d_char *html_people(snac *snac)
  1061. {
  1062. d_char *s = xs_str_new(NULL);
  1063. xs *wing = following_list(snac);
  1064. xs *wers = follower_list(snac);
  1065. s = html_user_header(snac, s, 0);
  1066. s = html_people_list(snac, s, wing, L("People you follow"), "i");
  1067. s = html_people_list(snac, s, wers, L("People that follow you"), "e");
  1068. s = html_user_footer(s);
  1069. s = xs_str_cat(s, "</body>\n</html>\n");
  1070. return s;
  1071. }
  1072. xs_str *html_notifications(snac *snac)
  1073. {
  1074. xs_str *s = xs_str_new(NULL);
  1075. xs *n_list = notify_list(snac, 0);
  1076. xs *n_time = notify_check_time(snac, 0);
  1077. xs_list *p = n_list;
  1078. xs_str *v;
  1079. enum { NHDR_NONE, NHDR_NEW, NHDR_OLD } stage = NHDR_NONE;
  1080. s = html_user_header(snac, s, 0);
  1081. xs *s1 = xs_fmt(
  1082. "<form autocomplete=\"off\" "
  1083. "method=\"post\" action=\"%s/admin/clear-notifications\" id=\"clear\">\n"
  1084. "<input type=\"submit\" class=\"snac-btn-like\" value=\"%s\">\n"
  1085. "</form><p>", snac->actor, L("Clear all"));
  1086. s = xs_str_cat(s, s1);
  1087. while (xs_list_iter(&p, &v)) {
  1088. xs *noti = notify_get(snac, v);
  1089. if (noti == NULL)
  1090. continue;
  1091. xs *obj = NULL;
  1092. const char *type = xs_dict_get(noti, "type");
  1093. const char *utype = xs_dict_get(noti, "utype");
  1094. const char *id = xs_dict_get(noti, "objid");
  1095. if (xs_is_null(id) || !valid_status(object_get(id, &obj)))
  1096. continue;
  1097. if (is_hidden(snac, id))
  1098. continue;
  1099. const char *actor_id = xs_dict_get(noti, "actor");
  1100. xs *actor = NULL;
  1101. if (!valid_status(actor_get(snac, actor_id, &actor)))
  1102. continue;
  1103. xs *a_name = actor_name(actor);
  1104. if (strcmp(v, n_time) > 0) {
  1105. /* unseen notification */
  1106. if (stage == NHDR_NONE) {
  1107. xs *s1 = xs_fmt("<h2>%s</h2>\n", L("New"));
  1108. s = xs_str_cat(s, s1);
  1109. stage = NHDR_NEW;
  1110. }
  1111. }
  1112. else {
  1113. /* already seen notification */
  1114. if (stage != NHDR_OLD) {
  1115. xs *s1 = xs_fmt("<h2>%s</h2>\n", L("Already seen"));
  1116. s = xs_str_cat(s, s1);
  1117. stage = NHDR_OLD;
  1118. }
  1119. }
  1120. s = xs_str_cat(s, "<div>\n");
  1121. const char *label = type;
  1122. if (strcmp(type, "Create") == 0)
  1123. label = L("Mention");
  1124. else
  1125. if (strcmp(type, "Update") == 0 && strcmp(utype, "Question") == 0)
  1126. label = L("Finished poll");
  1127. xs *s1 = xs_fmt("<p><b>%s by <a href=\"%s\">%s</a></b>:</p>\n",
  1128. label, actor_id, a_name);
  1129. s = xs_str_cat(s, s1);
  1130. if (strcmp(type, "Follow") == 0) {
  1131. s = xs_str_cat(s, "<div class=\"snac-post\">\n");
  1132. s = html_msg_icon(snac, s, obj);
  1133. s = xs_str_cat(s, "</div>\n");
  1134. }
  1135. else {
  1136. xs *md5 = xs_md5_hex(id, strlen(id));
  1137. s = html_entry(snac, s, obj, 0, 0, md5, 1);
  1138. }
  1139. s = xs_str_cat(s, "</div>\n");
  1140. }
  1141. if (stage == NHDR_NONE) {
  1142. xs *s1 = xs_fmt("<h2>%s</h2>\n", L("None"));
  1143. s = xs_str_cat(s, s1);
  1144. }
  1145. s = html_user_footer(s);
  1146. s = xs_str_cat(s, "</body>\n</html>\n");
  1147. /* set the check time to now */
  1148. xs *dummy = notify_check_time(snac, 1);
  1149. dummy = xs_free(dummy);
  1150. timeline_touch(snac);
  1151. return s;
  1152. }
  1153. int html_get_handler(const xs_dict *req, const char *q_path,
  1154. char **body, int *b_size, char **ctype)
  1155. {
  1156. char *accept = xs_dict_get(req, "accept");
  1157. int status = 404;
  1158. snac snac;
  1159. xs *uid = NULL;
  1160. char *p_path;
  1161. int cache = 1;
  1162. int save = 1;
  1163. char *v;
  1164. xs *l = xs_split_n(q_path, "/", 2);
  1165. v = xs_list_get(l, 1);
  1166. if (xs_is_null(v)) {
  1167. srv_log(xs_fmt("html_get_handler bad query '%s'", q_path));
  1168. return 404;
  1169. }
  1170. uid = xs_dup(v);
  1171. /* rss extension? */
  1172. if (xs_endswith(uid, ".rss")) {
  1173. uid = xs_crop_i(uid, 0, -4);
  1174. p_path = ".rss";
  1175. }
  1176. else
  1177. p_path = xs_list_get(l, 2);
  1178. if (!uid || !user_open(&snac, uid)) {
  1179. /* invalid user */
  1180. srv_debug(1, xs_fmt("html_get_handler bad user %s", uid));
  1181. return 404;
  1182. }
  1183. /* return the RSS if requested by Accept header */
  1184. if (accept != NULL) {
  1185. if (xs_str_in(accept, "text/xml") != -1 ||
  1186. xs_str_in(accept, "application/rss+xml") != -1)
  1187. p_path = ".rss";
  1188. }
  1189. /* check if server config variable 'disable_cache' is set */
  1190. if ((v = xs_dict_get(srv_config, "disable_cache")) && xs_type(v) == XSTYPE_TRUE)
  1191. cache = 0;
  1192. int skip = 0;
  1193. int show = xs_number_get(xs_dict_get(srv_config, "max_timeline_entries"));
  1194. char *q_vars = xs_dict_get(req, "q_vars");
  1195. if ((v = xs_dict_get(q_vars, "skip")) != NULL)
  1196. skip = atoi(v), cache = 0, save = 0;
  1197. if ((v = xs_dict_get(q_vars, "show")) != NULL)
  1198. show = atoi(v), cache = 0, save = 0;
  1199. if (p_path == NULL) { /** public timeline **/
  1200. xs *h = xs_str_localtime(0, "%Y-%m.html");
  1201. if (cache && history_mtime(&snac, h) > timeline_mtime(&snac)) {
  1202. snac_debug(&snac, 1, xs_fmt("serving cached local timeline"));
  1203. *body = history_get(&snac, h);
  1204. *b_size = strlen(*body);
  1205. status = 200;
  1206. }
  1207. else {
  1208. xs *list = timeline_list(&snac, "public", skip, show);
  1209. xs *next = timeline_list(&snac, "public", skip + show, 1);
  1210. *body = html_timeline(&snac, list, 1, skip, show, xs_list_len(next));
  1211. *b_size = strlen(*body);
  1212. status = 200;
  1213. if (save)
  1214. history_add(&snac, h, *body, *b_size);
  1215. }
  1216. }
  1217. else
  1218. if (strcmp(p_path, "admin") == 0) { /** private timeline **/
  1219. if (!login(&snac, req)) {
  1220. *body = xs_dup(uid);
  1221. status = 401;
  1222. }
  1223. else {
  1224. if (cache && history_mtime(&snac, "timeline.html_") > timeline_mtime(&snac)) {
  1225. snac_debug(&snac, 1, xs_fmt("serving cached timeline"));
  1226. *body = history_get(&snac, "timeline.html_");
  1227. *b_size = strlen(*body);
  1228. status = 200;
  1229. }
  1230. else {
  1231. snac_debug(&snac, 1, xs_fmt("building timeline"));
  1232. xs *list = timeline_list(&snac, "private", skip, show);
  1233. xs *next = timeline_list(&snac, "private", skip + show, 1);
  1234. *body = html_timeline(&snac, list, 0, skip, show, xs_list_len(next));
  1235. *b_size = strlen(*body);
  1236. status = 200;
  1237. if (save)
  1238. history_add(&snac, "timeline.html_", *body, *b_size);
  1239. }
  1240. }
  1241. }
  1242. else
  1243. if (strcmp(p_path, "people") == 0) { /** the list of people **/
  1244. if (!login(&snac, req)) {
  1245. *body = xs_dup(uid);
  1246. status = 401;
  1247. }
  1248. else {
  1249. *body = html_people(&snac);
  1250. *b_size = strlen(*body);
  1251. status = 200;
  1252. }
  1253. }
  1254. else
  1255. if (strcmp(p_path, "notifications") == 0) { /** the list of notifications **/
  1256. if (!login(&snac, req)) {
  1257. *body = xs_dup(uid);
  1258. status = 401;
  1259. }
  1260. else {
  1261. *body = html_notifications(&snac);
  1262. *b_size = strlen(*body);
  1263. status = 200;
  1264. }
  1265. }
  1266. else
  1267. if (xs_startswith(p_path, "p/")) { /** a timeline with just one entry **/
  1268. xs *id = xs_fmt("%s/%s", snac.actor, p_path);
  1269. xs *msg = NULL;
  1270. if (valid_status(object_get(id, &msg))) {
  1271. xs *md5 = xs_md5_hex(id, strlen(id));
  1272. xs *list = xs_list_new();
  1273. list = xs_list_append(list, md5);
  1274. *body = html_timeline(&snac, list, 1, 0, 0, 0);
  1275. *b_size = strlen(*body);
  1276. status = 200;
  1277. }
  1278. }
  1279. else
  1280. if (xs_startswith(p_path, "s/")) { /** a static file **/
  1281. xs *l = xs_split(p_path, "/");
  1282. char *id = xs_list_get(l, 1);
  1283. int sz;
  1284. if (valid_status(static_get(&snac, id, body, &sz))) {
  1285. *b_size = sz;
  1286. *ctype = xs_mime_by_ext(id);
  1287. status = 200;
  1288. }
  1289. }
  1290. else
  1291. if (xs_startswith(p_path, "h/")) { /** an entry from the history **/
  1292. xs *l = xs_split(p_path, "/");
  1293. char *id = xs_list_get(l, 1);
  1294. if ((*body = history_get(&snac, id)) != NULL) {
  1295. *b_size = strlen(*body);
  1296. status = 200;
  1297. }
  1298. }
  1299. else
  1300. if (strcmp(p_path, ".rss") == 0) { /** public timeline in RSS format **/
  1301. d_char *rss;
  1302. xs *elems = timeline_simple_list(&snac, "public", 0, 20);
  1303. xs *bio = not_really_markdown(xs_dict_get(snac.config, "bio"), NULL);
  1304. char *p, *v;
  1305. /* escape tags */
  1306. bio = xs_replace_i(bio, "<", "&lt;");
  1307. bio = xs_replace_i(bio, ">", "&gt;");
  1308. rss = xs_fmt(
  1309. "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
  1310. "<rss version=\"0.91\">\n"
  1311. "<channel>\n"
  1312. "<title>%s (@%s@%s)</title>\n"
  1313. "<language>en</language>\n"
  1314. "<link>%s.rss</link>\n"
  1315. "<description>%s</description>\n",
  1316. xs_dict_get(snac.config, "name"),
  1317. snac.uid,
  1318. xs_dict_get(srv_config, "host"),
  1319. snac.actor,
  1320. bio
  1321. );
  1322. p = elems;
  1323. while (xs_list_iter(&p, &v)) {
  1324. xs *msg = NULL;
  1325. if (!valid_status(timeline_get_by_md5(&snac, v, &msg)))
  1326. continue;
  1327. char *id = xs_dict_get(msg, "id");
  1328. if (!xs_startswith(id, snac.actor))
  1329. continue;
  1330. xs *content = sanitize(xs_dict_get(msg, "content"));
  1331. xs *title = xs_str_new(NULL);
  1332. int i;
  1333. /* escape tags */
  1334. content = xs_replace_i(content, "<", "&lt;");
  1335. content = xs_replace_i(content, ">", "&gt;");
  1336. for (i = 0; content[i] && content[i] != '<' && content[i] != '&' && i < 40; i++)
  1337. title = xs_append_m(title, &content[i], 1);
  1338. xs *s = xs_fmt(
  1339. "<item>\n"
  1340. "<title>%s...</title>\n"
  1341. "<link>%s</link>\n"
  1342. "<description>%s</description>\n"
  1343. "</item>\n",
  1344. title, id, content
  1345. );
  1346. rss = xs_str_cat(rss, s);
  1347. }
  1348. rss = xs_str_cat(rss, "</channel>\n</rss>\n");
  1349. *body = rss;
  1350. *b_size = strlen(rss);
  1351. *ctype = "application/rss+xml; charset=utf-8";
  1352. status = 200;
  1353. snac_debug(&snac, 1, xs_fmt("serving RSS"));
  1354. }
  1355. else
  1356. status = 404;
  1357. user_free(&snac);
  1358. if (valid_status(status) && *ctype == NULL) {
  1359. *ctype = "text/html; charset=utf-8";
  1360. }
  1361. return status;
  1362. }
  1363. int html_post_handler(const xs_dict *req, const char *q_path,
  1364. char *payload, int p_size,
  1365. char **body, int *b_size, char **ctype)
  1366. {
  1367. (void)p_size;
  1368. (void)ctype;
  1369. int status = 0;
  1370. snac snac;
  1371. char *uid, *p_path;
  1372. xs_dict *p_vars;
  1373. xs *l = xs_split_n(q_path, "/", 2);
  1374. uid = xs_list_get(l, 1);
  1375. if (!uid || !user_open(&snac, uid)) {
  1376. /* invalid user */
  1377. srv_debug(1, xs_fmt("html_post_handler bad user %s", uid));
  1378. return 404;
  1379. }
  1380. p_path = xs_list_get(l, 2);
  1381. /* all posts must be authenticated */
  1382. if (!login(&snac, req)) {
  1383. user_free(&snac);
  1384. *body = xs_dup(uid);
  1385. return 401;
  1386. }
  1387. p_vars = xs_dict_get(req, "p_vars");
  1388. #if 0
  1389. {
  1390. xs *j1 = xs_json_dumps_pp(p_vars, 4);
  1391. printf("%s\n", j1);
  1392. }
  1393. #endif
  1394. if (p_path && strcmp(p_path, "admin/note") == 0) { /** **/
  1395. /* post note */
  1396. xs_str *content = xs_dict_get(p_vars, "content");
  1397. xs_str *in_reply_to = xs_dict_get(p_vars, "in_reply_to");
  1398. xs_str *attach_url = xs_dict_get(p_vars, "attach_url");
  1399. xs_list *attach_file = xs_dict_get(p_vars, "attach");
  1400. xs_str *to = xs_dict_get(p_vars, "to");
  1401. xs_str *sensitive = xs_dict_get(p_vars, "sensitive");
  1402. xs_str *edit_id = xs_dict_get(p_vars, "edit_id");
  1403. xs_str *alt_text = xs_dict_get(p_vars, "alt_text");
  1404. int priv = !xs_is_null(xs_dict_get(p_vars, "mentioned_only"));
  1405. xs *attach_list = xs_list_new();
  1406. /* default alt text */
  1407. if (xs_is_null(alt_text))
  1408. alt_text = "";
  1409. /* is attach_url set? */
  1410. if (!xs_is_null(attach_url) && *attach_url != '\0') {
  1411. xs *l = xs_list_new();
  1412. l = xs_list_append(l, attach_url);
  1413. l = xs_list_append(l, alt_text);
  1414. attach_list = xs_list_append(attach_list, l);
  1415. }
  1416. /* is attach_file set? */
  1417. if (!xs_is_null(attach_file) && xs_type(attach_file) == XSTYPE_LIST) {
  1418. char *fn = xs_list_get(attach_file, 0);
  1419. if (*fn != '\0') {
  1420. char *ext = strrchr(fn, '.');
  1421. xs *hash = xs_md5_hex(fn, strlen(fn));
  1422. xs *id = xs_fmt("%s%s", hash, ext);
  1423. xs *url = xs_fmt("%s/s/%s", snac.actor, id);
  1424. int fo = xs_number_get(xs_list_get(attach_file, 1));
  1425. int fs = xs_number_get(xs_list_get(attach_file, 2));
  1426. /* store */
  1427. static_put(&snac, id, payload + fo, fs);
  1428. xs *l = xs_list_new();
  1429. l = xs_list_append(l, url);
  1430. l = xs_list_append(l, alt_text);
  1431. attach_list = xs_list_append(attach_list, l);
  1432. }
  1433. }
  1434. if (content != NULL) {
  1435. xs *msg = NULL;
  1436. xs *c_msg = NULL;
  1437. xs *content_2 = xs_replace(content, "\r", "");
  1438. xs *poll_opts = NULL;
  1439. /* is there a valid set of poll options? */
  1440. const char *v = xs_dict_get(p_vars, "poll_options");
  1441. if (!xs_is_null(v) && *v) {
  1442. xs *v2 = xs_strip_i(xs_replace(v, "\r", ""));
  1443. poll_opts = xs_split(v2, "\n");
  1444. }
  1445. if (!xs_is_null(poll_opts) && xs_list_len(poll_opts)) {
  1446. /* get the rest of poll configuration */
  1447. const char *p_multiple = xs_dict_get(p_vars, "poll_multiple");
  1448. const char *p_end_secs = xs_dict_get(p_vars, "poll_end_secs");
  1449. int multiple = 0;
  1450. int end_secs = atoi(!xs_is_null(p_end_secs) ? p_end_secs : "60");
  1451. if (!xs_is_null(p_multiple) && strcmp(p_multiple, "on") == 0)
  1452. multiple = 1;
  1453. msg = msg_question(&snac, content_2, attach_list,
  1454. poll_opts, multiple, end_secs);
  1455. enqueue_close_question(&snac, xs_dict_get(msg, "id"), end_secs);
  1456. }
  1457. else
  1458. msg = msg_note(&snac, content_2, to, in_reply_to, attach_list, priv);
  1459. if (sensitive != NULL) {
  1460. msg = xs_dict_set(msg, "sensitive", xs_stock_true);
  1461. msg = xs_dict_set(msg, "summary", "...");
  1462. }
  1463. if (xs_is_null(edit_id)) {
  1464. /* new message */
  1465. c_msg = msg_create(&snac, msg);
  1466. timeline_add(&snac, xs_dict_get(msg, "id"), msg);
  1467. }
  1468. else {
  1469. /* an edition of a previous message */
  1470. xs *p_msg = NULL;
  1471. if (valid_status(object_get(edit_id, &p_msg))) {
  1472. /* copy relevant fields from previous version */
  1473. char *fields[] = { "id", "context", "url", "published",
  1474. "to", "inReplyTo", NULL };
  1475. int n;
  1476. for (n = 0; fields[n]; n++) {
  1477. char *v = xs_dict_get(p_msg, fields[n]);
  1478. msg = xs_dict_set(msg, fields[n], v);
  1479. }
  1480. /* set the updated field */
  1481. xs *updated = xs_str_utctime(0, ISO_DATE_SPEC);
  1482. msg = xs_dict_set(msg, "updated", updated);
  1483. /* overwrite object, not updating the indexes */
  1484. object_add_ow(edit_id, msg);
  1485. /* update message */
  1486. c_msg = msg_update(&snac, msg);
  1487. }
  1488. else
  1489. snac_log(&snac, xs_fmt("cannot get object '%s' for editing", edit_id));
  1490. }
  1491. if (c_msg != NULL)
  1492. enqueue_message(&snac, c_msg);
  1493. history_del(&snac, "timeline.html_");
  1494. }
  1495. status = 303;
  1496. }
  1497. else
  1498. if (p_path && strcmp(p_path, "admin/action") == 0) { /** **/
  1499. /* action on an entry */
  1500. char *id = xs_dict_get(p_vars, "id");
  1501. char *actor = xs_dict_get(p_vars, "actor");
  1502. char *action = xs_dict_get(p_vars, "action");
  1503. if (action == NULL)
  1504. return 404;
  1505. snac_debug(&snac, 1, xs_fmt("web action '%s' received", action));
  1506. status = 303;
  1507. if (strcmp(action, L("Like")) == 0) { /** **/
  1508. xs *msg = msg_admiration(&snac, id, "Like");
  1509. if (msg != NULL) {
  1510. enqueue_message(&snac, msg);
  1511. timeline_admire(&snac, xs_dict_get(msg, "object"), snac.actor, 1);
  1512. }
  1513. }
  1514. else
  1515. if (strcmp(action, L("Boost")) == 0) { /** **/
  1516. xs *msg = msg_admiration(&snac, id, "Announce");
  1517. if (msg != NULL) {
  1518. enqueue_message(&snac, msg);
  1519. timeline_admire(&snac, xs_dict_get(msg, "object"), snac.actor, 0);
  1520. }
  1521. }
  1522. else
  1523. if (strcmp(action, L("MUTE")) == 0) { /** **/
  1524. mute(&snac, actor);
  1525. }
  1526. else
  1527. if (strcmp(action, L("Unmute")) == 0) { /** **/
  1528. unmute(&snac, actor);
  1529. }
  1530. else
  1531. if (strcmp(action, L("Hide")) == 0) { /** **/
  1532. hide(&snac, id);
  1533. }
  1534. else
  1535. if (strcmp(action, L("Follow")) == 0) { /** **/
  1536. xs *msg = msg_follow(&snac, actor);
  1537. if (msg != NULL) {
  1538. /* reload the actor from the message, in may be different */
  1539. actor = xs_dict_get(msg, "object");
  1540. following_add(&snac, actor, msg);
  1541. enqueue_output_by_actor(&snac, msg, actor, 0);
  1542. }
  1543. }
  1544. else
  1545. if (strcmp(action, L("Unfollow")) == 0) { /** **/
  1546. /* get the following object */
  1547. xs *object = NULL;
  1548. if (valid_status(following_get(&snac, actor, &object))) {
  1549. xs *msg = msg_undo(&snac, xs_dict_get(object, "object"));
  1550. following_del(&snac, actor);
  1551. enqueue_output_by_actor(&snac, msg, actor, 0);
  1552. snac_log(&snac, xs_fmt("unfollowed actor %s", actor));
  1553. }
  1554. else
  1555. snac_log(&snac, xs_fmt("actor is not being followed %s", actor));
  1556. }
  1557. else
  1558. if (strcmp(action, L("Delete")) == 0) { /** **/
  1559. char *actor_form = xs_dict_get(p_vars, "actor-form");
  1560. if (actor_form != NULL) {
  1561. /* delete follower */
  1562. if (valid_status(follower_del(&snac, actor)))
  1563. snac_log(&snac, xs_fmt("deleted follower %s", actor));
  1564. else
  1565. snac_log(&snac, xs_fmt("error deleting follower %s", actor));
  1566. }
  1567. else {
  1568. /* delete an entry */
  1569. if (xs_startswith(id, snac.actor)) {
  1570. /* it's a post by us: generate a delete */
  1571. xs *msg = msg_delete(&snac, id);
  1572. enqueue_message(&snac, msg);
  1573. snac_log(&snac, xs_fmt("posted tombstone for %s", id));
  1574. }
  1575. timeline_del(&snac, id);
  1576. snac_log(&snac, xs_fmt("deleted entry %s", id));
  1577. }
  1578. }
  1579. else
  1580. status = 404;
  1581. /* delete the cached timeline */
  1582. if (status == 303)
  1583. history_del(&snac, "timeline.html_");
  1584. }
  1585. else
  1586. if (p_path && strcmp(p_path, "admin/user-setup") == 0) { /** **/
  1587. /* change of user data */
  1588. char *v;
  1589. char *p1, *p2;
  1590. if ((v = xs_dict_get(p_vars, "name")) != NULL)
  1591. snac.config = xs_dict_set(snac.config, "name", v);
  1592. if ((v = xs_dict_get(p_vars, "avatar")) != NULL)
  1593. snac.config = xs_dict_set(snac.config, "avatar", v);
  1594. if ((v = xs_dict_get(p_vars, "bio")) != NULL)
  1595. snac.config = xs_dict_set(snac.config, "bio", v);
  1596. if ((v = xs_dict_get(p_vars, "cw")) != NULL &&
  1597. strcmp(v, "on") == 0) {
  1598. snac.config = xs_dict_set(snac.config, "cw", "open");
  1599. } else { /* if the checkbox is not set, the parameter is missing */
  1600. snac.config = xs_dict_set(snac.config, "cw", "");
  1601. }
  1602. if ((v = xs_dict_get(p_vars, "email")) != NULL)
  1603. snac.config = xs_dict_set(snac.config, "email", v);
  1604. if ((v = xs_dict_get(p_vars, "telegram_bot")) != NULL)
  1605. snac.config = xs_dict_set(snac.config, "telegram_bot", v);
  1606. if ((v = xs_dict_get(p_vars, "telegram_chat_id")) != NULL)
  1607. snac.config = xs_dict_set(snac.config, "telegram_chat_id", v);
  1608. if ((v = xs_dict_get(p_vars, "purge_days")) != NULL) {
  1609. xs *days = xs_number_new(atof(v));
  1610. snac.config = xs_dict_set(snac.config, "purge_days", days);
  1611. }
  1612. if ((v = xs_dict_get(p_vars, "drop_dm_from_unknown")) != NULL && strcmp(v, "on") == 0)
  1613. snac.config = xs_dict_set(snac.config, "drop_dm_from_unknown", xs_stock_true);
  1614. else
  1615. snac.config = xs_dict_set(snac.config, "drop_dm_from_unknown", xs_stock_false);
  1616. if ((v = xs_dict_get(p_vars, "bot")) != NULL && strcmp(v, "on") == 0)
  1617. snac.config = xs_dict_set(snac.config, "bot", xs_stock_true);
  1618. else
  1619. snac.config = xs_dict_set(snac.config, "bot", xs_stock_false);
  1620. /* avatar upload */
  1621. xs_list *avatar_file = xs_dict_get(p_vars, "avatar_file");
  1622. if (!xs_is_null(avatar_file) && xs_type(avatar_file) == XSTYPE_LIST) {
  1623. char *fn = xs_list_get(avatar_file, 0);
  1624. if (*fn != '\0') {
  1625. char *ext = strrchr(fn, '.');
  1626. xs *id = xs_fmt("avatar%s", ext);
  1627. xs *url = xs_fmt("%s/s/%s", snac.actor, id);
  1628. int fo = xs_number_get(xs_list_get(avatar_file, 1));
  1629. int fs = xs_number_get(xs_list_get(avatar_file, 2));
  1630. /* store */
  1631. static_put(&snac, id, payload + fo, fs);
  1632. snac.config = xs_dict_set(snac.config, "avatar", url);
  1633. }
  1634. }
  1635. /* password change? */
  1636. if ((p1 = xs_dict_get(p_vars, "passwd1")) != NULL &&
  1637. (p2 = xs_dict_get(p_vars, "passwd2")) != NULL &&
  1638. *p1 && strcmp(p1, p2) == 0) {
  1639. xs *pw = hash_password(snac.uid, p1, NULL);
  1640. snac.config = xs_dict_set(snac.config, "passwd", pw);
  1641. }
  1642. xs *fn = xs_fmt("%s/user.json", snac.basedir);
  1643. xs *bfn = xs_fmt("%s.bak", fn);
  1644. FILE *f;
  1645. rename(fn, bfn);
  1646. if ((f = fopen(fn, "w")) != NULL) {
  1647. xs *j = xs_json_dumps_pp(snac.config, 4);
  1648. fwrite(j, strlen(j), 1, f);
  1649. fclose(f);
  1650. }
  1651. else
  1652. rename(bfn, fn);
  1653. history_del(&snac, "timeline.html_");
  1654. xs *a_msg = msg_actor(&snac);
  1655. xs *u_msg = msg_update(&snac, a_msg);
  1656. enqueue_message(&snac, u_msg);
  1657. status = 303;
  1658. }
  1659. else
  1660. if (p_path && strcmp(p_path, "admin/clear-notifications") == 0) { /** **/
  1661. notify_clear(&snac);
  1662. timeline_touch(&snac);
  1663. status = 303;
  1664. }
  1665. else
  1666. if (p_path && strcmp(p_path, "admin/vote") == 0) { /** **/
  1667. char *irt = xs_dict_get(p_vars, "irt");
  1668. const char *opt = xs_dict_get(p_vars, "question");
  1669. const char *actor = xs_dict_get(p_vars, "actor");
  1670. xs *ls = NULL;
  1671. /* multiple choices? */
  1672. if (xs_type(opt) == XSTYPE_LIST)
  1673. ls = xs_dup(opt);
  1674. else {
  1675. ls = xs_list_new();
  1676. ls = xs_list_append(ls, opt);
  1677. }
  1678. xs_list *p = ls;
  1679. xs_str *v;
  1680. while (xs_list_iter(&p, &v)) {
  1681. xs *msg = msg_note(&snac, "", actor, irt, NULL, 1);
  1682. /* set the option */
  1683. msg = xs_dict_append(msg, "name", v);
  1684. xs *c_msg = msg_create(&snac, msg);
  1685. enqueue_message(&snac, c_msg);
  1686. timeline_add(&snac, xs_dict_get(msg, "id"), msg);
  1687. }
  1688. status = 303;
  1689. }
  1690. if (status == 303) {
  1691. char *redir = xs_dict_get(p_vars, "redir");
  1692. if (xs_is_null(redir))
  1693. redir = "snac-posts";
  1694. *body = xs_fmt("%s/admin#%s", snac.actor, redir);
  1695. *b_size = strlen(*body);
  1696. }
  1697. user_free(&snac);
  1698. return status;
  1699. }