html.c 71 KB

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