html.c 31 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103
  1. /* snac - A simple, minimalistic ActivityPub instance */
  2. /* copyright (c) 2022 grunfink - MIT license */
  3. #include "xs.h"
  4. #include "xs_io.h"
  5. #include "xs_encdec.h"
  6. #include "xs_json.h"
  7. #include "xs_regex.h"
  8. #include "xs_set.h"
  9. #include "xs_openssl.h"
  10. #include "xs_time.h"
  11. #include "xs_mime.h"
  12. #include "snac.h"
  13. int login(snac *snac, char *headers)
  14. /* tries a login */
  15. {
  16. int logged_in = 0;
  17. char *auth = xs_dict_get(headers, "authorization");
  18. if (auth && xs_startswith(auth, "Basic ")) {
  19. int sz;
  20. xs *s1 = xs_crop(xs_dup(auth), 6, 0);
  21. xs *s2 = xs_base64_dec(s1, &sz);
  22. xs *l1 = xs_split_n(s2, ":", 1);
  23. if (xs_list_len(l1) == 2) {
  24. logged_in = check_password(
  25. xs_list_get(l1, 0), xs_list_get(l1, 1),
  26. xs_dict_get(snac->config, "passwd"));
  27. }
  28. }
  29. return logged_in;
  30. }
  31. d_char *html_msg_icon(snac *snac, d_char *os, char *msg)
  32. {
  33. char *actor_id;
  34. xs *actor = NULL;
  35. xs *s = xs_str_new(NULL);
  36. if ((actor_id = xs_dict_get(msg, "attributedTo")) == NULL)
  37. actor_id = xs_dict_get(msg, "actor");
  38. if (actor_id && valid_status(actor_get(snac, actor_id, &actor))) {
  39. xs *name = NULL;
  40. xs *avatar = NULL;
  41. char *p, *v;
  42. /* get the name */
  43. if (xs_is_null((v = xs_dict_get(actor, "name"))) || *v == '\0') {
  44. if (xs_is_null(v = xs_dict_get(actor, "preferredUsername"))) {
  45. v = "user";
  46. }
  47. }
  48. name = xs_dup(v);
  49. /* replace the :shortnames: */
  50. if (!xs_is_null(p = xs_dict_get(actor, "tag"))) {
  51. /* iterate the tags */
  52. while (xs_list_iter(&p, &v)) {
  53. char *t = xs_dict_get(v, "type");
  54. if (t && strcmp(t, "Emoji") == 0) {
  55. char *n = xs_dict_get(v, "name");
  56. char *i = xs_dict_get(v, "icon");
  57. if (n && i) {
  58. char *u = xs_dict_get(i, "url");
  59. xs *img = xs_fmt("<img src=\"%s\" style=\"height: 1em\"/>", u);
  60. name = xs_replace_i(name, n, img);
  61. }
  62. }
  63. }
  64. }
  65. /* get the avatar */
  66. if ((v = xs_dict_get(actor, "icon")) != NULL &&
  67. (v = xs_dict_get(v, "url")) != NULL) {
  68. avatar = xs_dup(v);
  69. }
  70. if (avatar == NULL)
  71. avatar = xs_fmt("data:image/png;base64, %s", susie);
  72. {
  73. xs *s1 = xs_fmt("<p><img class=\"snac-avatar\" src=\"%s\" alt=\"\"/>\n", avatar);
  74. s = xs_str_cat(s, s1);
  75. }
  76. {
  77. xs *s1 = xs_fmt("<a href=\"%s\" class=\"p-author h-card snac-author\">%s</a>",
  78. actor_id, name);
  79. s = xs_str_cat(s, s1);
  80. }
  81. if (strcmp(xs_dict_get(msg, "type"), "Note") == 0) {
  82. xs *s1 = xs_fmt(" <a href=\"%s\">»</a>", xs_dict_get(msg, "id"));
  83. s = xs_str_cat(s, s1);
  84. }
  85. if (!is_msg_public(snac, msg))
  86. s = xs_str_cat(s, " <span title=\"private\">&#128274;</span>");
  87. if ((v = xs_dict_get(msg, "published")) == NULL) {
  88. s = xs_str_cat(s, "<br>\n<time>&nbsp;</time>\n");
  89. }
  90. else {
  91. xs *sd = xs_crop(xs_dup(v), 0, 10);
  92. xs *s1 = xs_fmt(
  93. "<br>\n<time class=\"dt-published snac-pubdate\">%s</time>\n", sd);
  94. s = xs_str_cat(s, s1);
  95. }
  96. }
  97. return xs_str_cat(os, s);
  98. }
  99. d_char *html_user_header(snac *snac, d_char *s, int local)
  100. /* creates the HTML header */
  101. {
  102. char *p, *v;
  103. s = xs_str_cat(s, "<!DOCTYPE html>\n<html>\n<head>\n");
  104. s = xs_str_cat(s, "<meta name=\"viewport\" "
  105. "content=\"width=device-width, initial-scale=1\"/>\n");
  106. s = xs_str_cat(s, "<meta name=\"generator\" "
  107. "content=\"" USER_AGENT "\"/>\n");
  108. /* add server CSS */
  109. p = xs_dict_get(srv_config, "cssurls");
  110. while (xs_list_iter(&p, &v)) {
  111. xs *s1 = xs_fmt("<link rel=\"stylesheet\" type=\"text/css\" href=\"%s\"/>\n", v);
  112. s = xs_str_cat(s, s1);
  113. }
  114. /* add the user CSS */
  115. {
  116. xs *css = NULL;
  117. int size;
  118. if (valid_status(static_get(snac, "style.css", &css, &size))) {
  119. xs *s1 = xs_fmt("<style>%s</style>\n", css);
  120. s = xs_str_cat(s, s1);
  121. }
  122. }
  123. {
  124. xs *s1 = xs_fmt("<title>%s</title>\n", xs_dict_get(snac->config, "name"));
  125. s = xs_str_cat(s, s1);
  126. }
  127. s = xs_str_cat(s, "</head>\n<body>\n");
  128. /* top nav */
  129. s = xs_str_cat(s, "<nav class=\"snac-top-nav\">");
  130. {
  131. xs *s1;
  132. if (local)
  133. s1 = xs_fmt("<a href=\"%s/admin\">%s</a></nav>\n", snac->actor, L("admin"));
  134. else
  135. s1 = xs_fmt("<a href=\"%s\">%s</a></nav>\n", snac->actor, L("public"));
  136. s = xs_str_cat(s, s1);
  137. }
  138. /* user info */
  139. {
  140. xs *bio = NULL;
  141. char *_tmpl =
  142. "<div class=\"h-card snac-top-user\">\n"
  143. "<p class=\"p-name snac-top-user-name\">%s</p>\n"
  144. "<p class=\"snac-top-user-id\">@%s@%s</p>\n"
  145. "<div class=\"p-note snac-top-user-bio\">%s</div>\n"
  146. "</div>\n";
  147. not_really_markdown(xs_dict_get(snac->config, "bio"), &bio);
  148. xs *s1 = xs_fmt(_tmpl,
  149. xs_dict_get(snac->config, "name"),
  150. xs_dict_get(snac->config, "uid"), xs_dict_get(srv_config, "host"),
  151. bio
  152. );
  153. s = xs_str_cat(s, s1);
  154. }
  155. return s;
  156. }
  157. d_char *html_top_controls(snac *snac, d_char *s)
  158. /* generates the top controls */
  159. {
  160. char *_tmpl =
  161. "<div class=\"snac-top-controls\">\n"
  162. "<div class=\"snac-note\">\n"
  163. "<form method=\"post\" action=\"%s/admin/note\" enctype=\"multipart/form-data\">\n"
  164. "<textarea class=\"snac-textarea\" name=\"content\" "
  165. "rows=\"8\" wrap=\"virtual\" required=\"required\"></textarea>\n"
  166. "<input type=\"hidden\" name=\"in_reply_to\" value=\"\">\n"
  167. "<p><input type=\"file\" name=\"attach\">\n"
  168. "<p><input type=\"submit\" class=\"button\" value=\"%s\">\n"
  169. "</form><p>\n"
  170. "</div>\n"
  171. "<div class=\"snac-top-controls-more\">\n"
  172. "<details><summary>%s</summary>\n"
  173. "<form method=\"post\" action=\"%s/admin/action\">\n"
  174. "<input type=\"text\" name=\"actor\" required=\"required\">\n"
  175. "<input type=\"submit\" name=\"action\" value=\"%s\"> %s\n"
  176. "</form><p>\n"
  177. "<form method=\"post\" action=\"%s/admin/action\">\n"
  178. "<input type=\"text\" name=\"id\" required=\"required\">\n"
  179. "<input type=\"submit\" name=\"action\" value=\"%s\"> %s\n"
  180. "</form><p>\n"
  181. "<details><summary>%s</summary>\n"
  182. "<div class=\"snac-user-setup\">\n"
  183. "<form method=\"post\" action=\"%s/admin/user-setup\">\n"
  184. "<p>%s:<br>\n"
  185. "<input type=\"text\" name=\"name\" value=\"%s\"></p>\n"
  186. "<p>%s:<br>\n"
  187. "<input type=\"text\" name=\"avatar\" value=\"%s\"></p>\n"
  188. "<p>%s:<br>\n"
  189. "<textarea name=\"bio\" cols=\"40\" rows=\"4\">%s</textarea></p>\n"
  190. "<p>%s:<br>\n"
  191. "<input type=\"text\" name=\"email\" value=\"%s\"></p>\n"
  192. "<p>%s:<br>\n"
  193. "<input type=\"password\" name=\"passwd1\" value=\"\"></p>\n"
  194. "<p>%s:<br>\n"
  195. "<input type=\"password\" name=\"passwd2\" value=\"\"></p>\n"
  196. "<input type=\"submit\" class=\"button\" value=\"%s\">\n"
  197. "</form>\n"
  198. "</div>\n"
  199. "</details>\n"
  200. "</details>\n"
  201. "</div>\n"
  202. "</div>\n";
  203. char *email = xs_dict_get(snac->config, "email");
  204. if (xs_is_null(email))
  205. email = "";
  206. xs *s1 = xs_fmt(_tmpl,
  207. snac->actor,
  208. L("Post"),
  209. L("More options..."),
  210. snac->actor,
  211. L("Follow"), L("(by URL or user@host)"),
  212. snac->actor,
  213. L("Boost"), L("(by URL)"),
  214. L("User setup..."),
  215. snac->actor,
  216. L("User name"),
  217. xs_dict_get(snac->config, "name"),
  218. L("Avatar URL"),
  219. xs_dict_get(snac->config, "avatar"),
  220. L("Bio"),
  221. xs_dict_get(snac->config, "bio"),
  222. L("Email address for notifications"),
  223. email,
  224. L("Password (only to change it)"),
  225. L("Repeat Password"),
  226. L("Update user info")
  227. );
  228. s = xs_str_cat(s, s1);
  229. return s;
  230. }
  231. d_char *html_button(d_char *s, char *clss, char *label)
  232. {
  233. xs *s1 = xs_fmt(
  234. "<input type=\"submit\" name=\"action\" "
  235. "class=\"snac-btn-%s\" value=\"%s\">\n",
  236. clss, label);
  237. return xs_str_cat(s, s1);
  238. }
  239. d_char *build_mentions(snac *snac, char *msg)
  240. /* returns a string with the mentions in msg */
  241. {
  242. d_char *s = xs_str_new(NULL);
  243. char *list = xs_dict_get(msg, "tag");
  244. char *v;
  245. while (xs_list_iter(&list, &v)) {
  246. char *type = xs_dict_get(v, "type");
  247. char *href = xs_dict_get(v, "href");
  248. char *name = xs_dict_get(v, "name");
  249. if (type && strcmp(type, "Mention") == 0 &&
  250. href && strcmp(href, snac->actor) != 0 && name) {
  251. xs *l = xs_split(name, "@");
  252. /* is it a name without a host? */
  253. if (xs_list_len(l) < 3) {
  254. /* split the href and pick the host name LIKE AN ANIMAL */
  255. /* would be better to query the webfinger but *won't do that* here */
  256. xs *l2 = xs_split(href, "/");
  257. if (xs_list_len(l2) >= 3) {
  258. xs *s1 = xs_fmt("%s@%s ", name, xs_list_get(l2, 2));
  259. s = xs_str_cat(s, s1);
  260. }
  261. }
  262. else {
  263. s = xs_str_cat(s, name);
  264. s = xs_str_cat(s, " ");
  265. }
  266. }
  267. }
  268. return s;
  269. }
  270. d_char *html_entry_controls(snac *snac, d_char *os, char *msg)
  271. {
  272. char *id = xs_dict_get(msg, "id");
  273. char *actor = xs_dict_get(msg, "attributedTo");
  274. char *meta = xs_dict_get(msg, "_snac");
  275. xs *s = xs_str_new(NULL);
  276. xs *md5 = xs_md5_hex(id, strlen(id));
  277. s = xs_str_cat(s, "<div class=\"snac-controls\">\n");
  278. {
  279. xs *s1 = xs_fmt(
  280. "<form method=\"post\" action=\"%s/admin/action\">\n"
  281. "<input type=\"hidden\" name=\"id\" value=\"%s\">\n"
  282. "<input type=\"hidden\" name=\"actor\" value=\"%s\">\n"
  283. "<input type=\"button\" name=\"action\" "
  284. "value=\"%s\" onclick=\""
  285. "x = document.getElementById('%s_reply'); "
  286. "if (x.style.display == 'block') "
  287. " x.style.display = 'none'; else "
  288. " x.style.display = 'block';"
  289. "\">\n",
  290. snac->actor, id, actor,
  291. L("Reply"),
  292. md5
  293. );
  294. s = xs_str_cat(s, s1);
  295. }
  296. if (strcmp(actor, snac->actor) != 0) {
  297. /* controls for other actors than this one */
  298. char *l;
  299. l = xs_dict_get(meta, "liked_by");
  300. if (xs_list_in(l, snac->actor) == -1) {
  301. /* not already liked; add button */
  302. s = html_button(s, "like", L("Like"));
  303. }
  304. l = xs_dict_get(meta, "announced_by");
  305. if (xs_list_in(l, snac->actor) == -1) {
  306. /* not already boosted; add button */
  307. s = html_button(s, "boost", L("Boost"));
  308. }
  309. if (following_check(snac, actor)) {
  310. s = html_button(s, "unfollow", L("Unfollow"));
  311. }
  312. else {
  313. s = html_button(s, "follow", L("Follow"));
  314. s = html_button(s, "mute", L("MUTE"));
  315. }
  316. }
  317. s = html_button(s, "delete", L("Delete"));
  318. s = xs_str_cat(s, "</form>\n");
  319. {
  320. /* the post textarea */
  321. xs *ct = build_mentions(snac, msg);
  322. xs *s1 = xs_fmt(
  323. "<p><div class=\"snac-note\" style=\"display: none\" id=\"%s_reply\">\n"
  324. "<form method=\"post\" action=\"%s/admin/note\" "
  325. "enctype=\"multipart/form-data\" id=\"%s_reply_form\">\n"
  326. "<textarea class=\"snac-textarea\" name=\"content\" "
  327. "rows=\"4\" wrap=\"virtual\" required=\"required\">%s</textarea>\n"
  328. "<input type=\"hidden\" name=\"in_reply_to\" value=\"%s\">\n"
  329. "<p><input type=\"file\" name=\"attach\">\n"
  330. "<p><input type=\"submit\" class=\"button\" value=\"%s\">\n"
  331. "</form><p></div>\n",
  332. md5,
  333. snac->actor, md5,
  334. ct,
  335. id,
  336. L("Post")
  337. );
  338. s = xs_str_cat(s, s1);
  339. }
  340. s = xs_str_cat(s, "</div>\n");
  341. return xs_str_cat(os, s);
  342. }
  343. d_char *html_entry(snac *snac, d_char *os, char *msg, xs_set *seen, int local, int level)
  344. {
  345. char *id = xs_dict_get(msg, "id");
  346. char *type = xs_dict_get(msg, "type");
  347. char *meta = xs_dict_get(msg, "_snac");
  348. char *actor;
  349. /* do not show non-public messages in the public timeline */
  350. if (local && !is_msg_public(snac, msg))
  351. return os;
  352. /* return if already seen */
  353. if (xs_set_add(seen, id) == 0)
  354. return os;
  355. xs *s = xs_str_new(NULL);
  356. if (strcmp(type, "Follow") == 0) {
  357. s = xs_str_cat(s, "<div class=\"snac-post\">\n");
  358. xs *s1 = xs_fmt("<div class=\"snac-origin\">%s</div>\n", L("follows you"));
  359. s = xs_str_cat(s, s1);
  360. s = html_msg_icon(snac, s, msg);
  361. s = xs_str_cat(s, "</div>\n");
  362. return xs_str_cat(os, s);
  363. }
  364. /* bring the main actor */
  365. if ((actor = xs_dict_get(msg, "attributedTo")) == NULL)
  366. return os;
  367. /* ignore muted morons immediately */
  368. if (is_muted(snac, actor))
  369. return os;
  370. if (strcmp(actor, snac->actor) != 0 && !valid_status(actor_get(snac, actor, NULL)))
  371. return os;
  372. /* if this is our post, add the score */
  373. if (xs_startswith(id, snac->actor)) {
  374. int likes = xs_list_len(xs_dict_get(meta, "liked_by"));
  375. int boosts = xs_list_len(xs_dict_get(meta, "announced_by"));
  376. /* alternate emojis: %d &#128077; %d &#128257; */
  377. xs *s1 = xs_fmt(
  378. "<div class=\"snac-score\">%d &#9733; %d &#8634;</div>\n",
  379. likes, boosts);
  380. s = xs_str_cat(s, s1);
  381. }
  382. if (level == 0) {
  383. char *p;
  384. s = xs_str_cat(s, "<div class=\"snac-post\">\n");
  385. /* print the origin of the post, if any */
  386. if (!xs_is_null(p = xs_dict_get(meta, "referrer"))) {
  387. xs *actor_r = NULL;
  388. if (valid_status(actor_get(snac, p, &actor_r))) {
  389. char *name;
  390. if ((name = xs_dict_get(actor_r, "name")) == NULL)
  391. name = xs_dict_get(actor_r, "preferredUsername");
  392. xs *s1 = xs_fmt(
  393. "<div class=\"snac-origin\">"
  394. "<a href=\"%s\">%s</a> %s</div>\n",
  395. xs_dict_get(actor_r, "id"),
  396. name,
  397. L("boosted")
  398. );
  399. s = xs_str_cat(s, s1);
  400. }
  401. }
  402. else
  403. if (!xs_is_null((p = xs_dict_get(meta, "parent"))) && *p) {
  404. /* this may happen if any of the autor or the parent aren't here */
  405. xs *s1 = xs_fmt(
  406. "<div class=\"snac-origin\">%s "
  407. "<a href=\"%s\">»</a></div>\n",
  408. L("in reply to"), p
  409. );
  410. s = xs_str_cat(s, s1);
  411. }
  412. else
  413. if (!xs_is_null((p = xs_dict_get(meta, "announced_by"))) &&
  414. xs_list_in(p, snac->actor) != -1) {
  415. /* we boosted this */
  416. xs *s1 = xs_fmt(
  417. "<div class=\"snac-origin\">"
  418. "<a href=\"%s\">%s</a> %s</a></div>",
  419. snac->actor, xs_dict_get(snac->config, "name"), L("liked")
  420. );
  421. s = xs_str_cat(s, s1);
  422. }
  423. else
  424. if (!xs_is_null((p = xs_dict_get(meta, "liked_by"))) &&
  425. xs_list_in(p, snac->actor) != -1) {
  426. /* we liked this */
  427. xs *s1 = xs_fmt(
  428. "<div class=\"snac-origin\">"
  429. "<a href=\"%s\">%s</a> %s</a></div>",
  430. snac->actor, xs_dict_get(snac->config, "name"), L("liked")
  431. );
  432. s = xs_str_cat(s, s1);
  433. }
  434. }
  435. else
  436. s = xs_str_cat(s, "<div class=\"snac-child\">\n");
  437. s = html_msg_icon(snac, s, msg);
  438. /* add the content */
  439. s = xs_str_cat(s, "<div class=\"e-content snac-content\">\n");
  440. {
  441. xs *c = xs_dup(xs_dict_get(msg, "content"));
  442. char *p, *v;
  443. /* do some tweaks to the content */
  444. c = xs_replace_i(c, "\r", "");
  445. while (xs_endswith(c, "<br><br>"))
  446. c = xs_crop(c, 0, -4);
  447. c = xs_replace_i(c, "<br><br>", "<p>");
  448. if (!xs_startswith(c, "<p>")) {
  449. xs *s1 = c;
  450. c = xs_fmt("<p>%s</p>", s1);
  451. }
  452. /* replace the :shortnames: */
  453. if (!xs_is_null(p = xs_dict_get(msg, "tag"))) {
  454. /* iterate the tags */
  455. while (xs_list_iter(&p, &v)) {
  456. char *t = xs_dict_get(v, "type");
  457. if (t && strcmp(t, "Emoji") == 0) {
  458. char *n = xs_dict_get(v, "name");
  459. char *i = xs_dict_get(v, "icon");
  460. if (n && i) {
  461. char *u = xs_dict_get(i, "url");
  462. xs *img = xs_fmt("<img src=\"%s\" style=\"height: 1em\"/>", u);
  463. c = xs_replace_i(c, n, img);
  464. }
  465. }
  466. }
  467. }
  468. s = xs_str_cat(s, c);
  469. }
  470. s = xs_str_cat(s, "\n");
  471. /* add the attachments */
  472. char *attach;
  473. if ((attach = xs_dict_get(msg, "attachment")) != NULL) {
  474. char *v;
  475. while (xs_list_iter(&attach, &v)) {
  476. char *t = xs_dict_get(v, "mediaType");
  477. if (xs_is_null(t))
  478. continue;
  479. if (xs_startswith(t, "image/")) {
  480. char *url = xs_dict_get(v, "url");
  481. char *name = xs_dict_get(v, "name");
  482. if (url != NULL) {
  483. xs *s1 = xs_fmt("<p><img src=\"%s\" alt=\"%s\"/></p>\n",
  484. url, xs_is_null(name) ? "" : name);
  485. s = xs_str_cat(s, s1);
  486. }
  487. }
  488. else
  489. if (xs_startswith(t, "video/")) {
  490. char *url = xs_dict_get(v, "url");
  491. if (url != NULL) {
  492. xs *s1 = xs_fmt("<p><object data=\"%s\"></object></p>\n", url);
  493. s = xs_str_cat(s, s1);
  494. }
  495. }
  496. }
  497. }
  498. s = xs_str_cat(s, "</div>\n");
  499. /** controls **/
  500. if (!local)
  501. s = html_entry_controls(snac, s, msg);
  502. /** children **/
  503. char *children = xs_dict_get(meta, "children");
  504. if (xs_list_len(children)) {
  505. int left = xs_list_len(children);
  506. char *id;
  507. if (level < 4)
  508. s = xs_str_cat(s, "<div class=\"snac-children\">\n");
  509. else
  510. s = xs_str_cat(s, "<div>\n");
  511. if (left > 3)
  512. s = xs_str_cat(s, "<details><summary>...</summary>\n");
  513. while (xs_list_iter(&children, &id)) {
  514. xs *chd = timeline_find(snac, id);
  515. if (left == 3)
  516. s = xs_str_cat(s, "</details>\n");
  517. if (chd != NULL)
  518. s = html_entry(snac, s, chd, seen, local, level + 1);
  519. else
  520. snac_debug(snac, 2, xs_fmt("cannot read from timeline child %s", id));
  521. left--;
  522. }
  523. s = xs_str_cat(s, "</div>\n");
  524. }
  525. s = xs_str_cat(s, "</div>\n");
  526. return xs_str_cat(os, s);
  527. }
  528. d_char *html_user_footer(snac *snac, d_char *s)
  529. {
  530. xs *s1 = xs_fmt(
  531. "<div class=\"snac-footer\">\n"
  532. "<a href=\"%s\">%s</a> - "
  533. "powered by <abbr title=\"Social Networks Are Crap\">snac</abbr></div>\n",
  534. srv_baseurl,
  535. L("about this site")
  536. );
  537. return xs_str_cat(s, s1);
  538. }
  539. d_char *html_timeline(snac *snac, char *list, int local)
  540. /* returns the HTML for the timeline */
  541. {
  542. d_char *s = xs_str_new(NULL);
  543. xs_set *seen = xs_set_new(4096);
  544. char *v;
  545. double t = ftime();
  546. s = html_user_header(snac, s, local);
  547. if (!local)
  548. s = html_top_controls(snac, s);
  549. s = xs_str_cat(s, "<a name=\"snac-posts\"></a>\n");
  550. s = xs_str_cat(s, "<div class=\"snac-posts\">\n");
  551. while (xs_list_iter(&list, &v)) {
  552. xs *msg = timeline_get(snac, v);
  553. s = html_entry(snac, s, msg, seen, local, 0);
  554. }
  555. s = xs_str_cat(s, "</div>\n");
  556. if (local) {
  557. xs *s1 = xs_fmt(
  558. "<div class=\"snac-history\">\n"
  559. "<p class=\"snac-history-title\">%s</p><ul>\n",
  560. L("History")
  561. );
  562. s = xs_str_cat(s, s1);
  563. xs *list = history_list(snac);
  564. char *p, *v;
  565. p = list;
  566. while (xs_list_iter(&p, &v)) {
  567. xs *fn = xs_replace(v, ".html", "");
  568. xs *s1 = xs_fmt(
  569. "<li><a href=\"%s/h/%s\">%s</li>\n",
  570. snac->actor, v, fn);
  571. s = xs_str_cat(s, s1);
  572. }
  573. s = xs_str_cat(s, "</ul></div>\n");
  574. }
  575. s = html_user_footer(snac, s);
  576. {
  577. xs *s1 = xs_fmt("<!-- %lf seconds -->\n", ftime() - t);
  578. s = xs_str_cat(s, s1);
  579. }
  580. s = xs_str_cat(s, "</body>\n</html>\n");
  581. xs_set_free(seen);
  582. return s;
  583. }
  584. int html_get_handler(d_char *req, char *q_path, char **body, int *b_size, char **ctype)
  585. {
  586. int status = 404;
  587. snac snac;
  588. char *uid, *p_path;
  589. int cache = 1;
  590. char *v;
  591. xs *l = xs_split_n(q_path, "/", 2);
  592. uid = xs_list_get(l, 1);
  593. if (!uid || !user_open(&snac, uid)) {
  594. /* invalid user */
  595. srv_log(xs_fmt("html_get_handler bad user %s", uid));
  596. return 404;
  597. }
  598. /* check if server config variable 'disable_cache' is set */
  599. if ((v = xs_dict_get(srv_config, "disable_cache")) && xs_type(v) == XSTYPE_TRUE)
  600. cache = 0;
  601. p_path = xs_list_get(l, 2);
  602. if (p_path == NULL) {
  603. /* public timeline */
  604. xs *h = xs_str_localtime(0, "%Y-%m.html");
  605. if (cache && history_mtime(&snac, h) > timeline_mtime(&snac)) {
  606. snac_debug(&snac, 1, xs_fmt("serving cached local timeline"));
  607. *body = history_get(&snac, h);
  608. *b_size = strlen(*body);
  609. status = 200;
  610. }
  611. else {
  612. xs *list = local_list(&snac, 0xfffffff);
  613. *body = html_timeline(&snac, list, 1);
  614. *b_size = strlen(*body);
  615. status = 200;
  616. history_add(&snac, h, *body, *b_size);
  617. }
  618. }
  619. else
  620. if (strcmp(p_path, "admin") == 0) {
  621. /* private timeline */
  622. if (!login(&snac, req))
  623. status = 401;
  624. else {
  625. if (cache && history_mtime(&snac, "timeline.html_") > timeline_mtime(&snac)) {
  626. snac_debug(&snac, 1, xs_fmt("serving cached timeline"));
  627. *body = history_get(&snac, "timeline.html_");
  628. *b_size = strlen(*body);
  629. status = 200;
  630. }
  631. else {
  632. snac_debug(&snac, 1, xs_fmt("building timeline"));
  633. xs *list = timeline_list(&snac, 0xfffffff);
  634. *body = html_timeline(&snac, list, 0);
  635. *b_size = strlen(*body);
  636. status = 200;
  637. history_add(&snac, "timeline.html_", *body, *b_size);
  638. }
  639. }
  640. }
  641. else
  642. if (xs_startswith(p_path, "p/")) {
  643. /* a timeline with just one entry */
  644. xs *id = xs_fmt("%s/%s", snac.actor, p_path);
  645. xs *fn = _timeline_find_fn(&snac, id);
  646. if (fn != NULL) {
  647. xs *list = xs_list_new();
  648. list = xs_list_append(list, fn);
  649. *body = html_timeline(&snac, list, 1);
  650. *b_size = strlen(*body);
  651. status = 200;
  652. }
  653. }
  654. else
  655. if (xs_startswith(p_path, "s/")) {
  656. /* a static file */
  657. xs *l = xs_split(p_path, "/");
  658. char *id = xs_list_get(l, 1);
  659. int sz;
  660. if (valid_status(static_get(&snac, id, body, &sz))) {
  661. *b_size = sz;
  662. *ctype = xs_mime_by_ext(id);
  663. status = 200;
  664. }
  665. }
  666. else
  667. if (xs_startswith(p_path, "h/")) {
  668. /* an entry from the history */
  669. xs *l = xs_split(p_path, "/");
  670. char *id = xs_list_get(l, 1);
  671. if ((*body = history_get(&snac, id)) != NULL) {
  672. *b_size = strlen(*body);
  673. status = 200;
  674. }
  675. }
  676. else
  677. status = 404;
  678. user_free(&snac);
  679. if (valid_status(status) && *ctype == NULL) {
  680. *ctype = "text/html; charset=utf-8";
  681. }
  682. return status;
  683. }
  684. int html_post_handler(d_char *req, char *q_path, d_char *payload, int p_size,
  685. char **body, int *b_size, char **ctype)
  686. {
  687. int status = 0;
  688. snac snac;
  689. char *uid, *p_path;
  690. char *p_vars;
  691. xs *l = xs_split_n(q_path, "/", 2);
  692. uid = xs_list_get(l, 1);
  693. if (!uid || !user_open(&snac, uid)) {
  694. /* invalid user */
  695. srv_log(xs_fmt("html_get_handler bad user %s", uid));
  696. return 404;
  697. }
  698. p_path = xs_list_get(l, 2);
  699. /* all posts must be authenticated */
  700. if (!login(&snac, req))
  701. return 401;
  702. p_vars = xs_dict_get(req, "p_vars");
  703. #if 0
  704. {
  705. xs *j1 = xs_json_dumps_pp(p_vars, 4);
  706. printf("%s\n", j1);
  707. }
  708. #endif
  709. if (p_path && strcmp(p_path, "admin/note") == 0) {
  710. /* post note */
  711. char *content = xs_dict_get(p_vars, "content");
  712. char *in_reply_to = xs_dict_get(p_vars, "in_reply_to");
  713. char *attach_url = xs_dict_get(p_vars, "attach_url");
  714. char *attach_file = xs_dict_get(p_vars, "attach");
  715. xs *attach_list = xs_list_new();
  716. /* is attach_url set? */
  717. if (!xs_is_null(attach_url) && *attach_url != '\0')
  718. attach_list = xs_list_append(attach_list, attach_url);
  719. /* is attach_file set? */
  720. if (!xs_is_null(attach_file) && xs_type(attach_file) == XSTYPE_LIST) {
  721. char *fn = xs_list_get(attach_file, 0);
  722. if (*fn != '\0') {
  723. char *ext = strrchr(fn, '.');
  724. xs *ntid = tid(0);
  725. xs *id = xs_fmt("%s%s", ntid, ext);
  726. xs *url = xs_fmt("%s/s/%s", snac.actor, id);
  727. int fo = xs_number_get(xs_list_get(attach_file, 1));
  728. int fs = xs_number_get(xs_list_get(attach_file, 2));
  729. /* store */
  730. static_put(&snac, id, payload + fo, fs);
  731. attach_list = xs_list_append(attach_list, url);
  732. }
  733. }
  734. if (content != NULL) {
  735. xs *msg = NULL;
  736. xs *c_msg = NULL;
  737. msg = msg_note(&snac, content, NULL, in_reply_to, attach_list);
  738. c_msg = msg_create(&snac, msg);
  739. post(&snac, c_msg);
  740. timeline_add(&snac, xs_dict_get(msg, "id"), msg, in_reply_to, NULL);
  741. }
  742. status = 303;
  743. }
  744. else
  745. if (p_path && strcmp(p_path, "admin/action") == 0) {
  746. /* action on an entry */
  747. char *id = xs_dict_get(p_vars, "id");
  748. char *actor = xs_dict_get(p_vars, "actor");
  749. char *action = xs_dict_get(p_vars, "action");
  750. if (action == NULL)
  751. return 404;
  752. snac_debug(&snac, 1, xs_fmt("web action '%s' received", action));
  753. status = 303;
  754. if (strcmp(action, L("Like")) == 0) {
  755. xs *msg = msg_admiration(&snac, id, "Like");
  756. post(&snac, msg);
  757. timeline_admire(&snac, id, snac.actor, 1);
  758. }
  759. else
  760. if (strcmp(action, L("Boost")) == 0) {
  761. xs *msg = msg_admiration(&snac, id, "Announce");
  762. post(&snac, msg);
  763. timeline_admire(&snac, id, snac.actor, 0);
  764. }
  765. else
  766. if (strcmp(action, L("MUTE")) == 0) {
  767. mute(&snac, actor);
  768. }
  769. else
  770. if (strcmp(action, L("Follow")) == 0) {
  771. xs *msg = msg_follow(&snac, actor);
  772. /* reload the actor from the message, in may be different */
  773. actor = xs_dict_get(msg, "object");
  774. following_add(&snac, actor, msg);
  775. enqueue_output(&snac, msg, actor, 0);
  776. }
  777. else
  778. if (strcmp(action, L("Unfollow")) == 0) {
  779. /* get the following object */
  780. xs *object = NULL;
  781. if (valid_status(following_get(&snac, actor, &object))) {
  782. xs *msg = msg_undo(&snac, xs_dict_get(object, "object"));
  783. following_del(&snac, actor);
  784. enqueue_output(&snac, msg, actor, 0);
  785. snac_log(&snac, xs_fmt("unfollowed actor %s", actor));
  786. }
  787. else
  788. snac_log(&snac, xs_fmt("actor is not being followed %s", actor));
  789. }
  790. else
  791. if (strcmp(action, L("Delete")) == 0) {
  792. /* delete an entry */
  793. if (xs_startswith(id, snac.actor)) {
  794. /* it's a post by us: generate a delete */
  795. xs *msg = msg_delete(&snac, id);
  796. post(&snac, msg);
  797. snac_log(&snac, xs_fmt("posted tombstone for %s", id));
  798. }
  799. timeline_del(&snac, id);
  800. snac_log(&snac, xs_fmt("deleted entry %s", id));
  801. }
  802. else
  803. status = 404;
  804. /* delete the cached timeline */
  805. if (status == 303)
  806. history_del(&snac, "timeline.html_");
  807. }
  808. else
  809. if (p_path && strcmp(p_path, "admin/user-setup") == 0) {
  810. /* change of user data */
  811. char *v;
  812. char *p1, *p2;
  813. if ((v = xs_dict_get(p_vars, "name")) != NULL)
  814. snac.config = xs_dict_set(snac.config, "name", v);
  815. if ((v = xs_dict_get(p_vars, "avatar")) != NULL)
  816. snac.config = xs_dict_set(snac.config, "avatar", v);
  817. if ((v = xs_dict_get(p_vars, "bio")) != NULL)
  818. snac.config = xs_dict_set(snac.config, "bio", v);
  819. if ((v = xs_dict_get(p_vars, "email")) != NULL)
  820. snac.config = xs_dict_set(snac.config, "email", v);
  821. /* password change? */
  822. if ((p1 = xs_dict_get(p_vars, "passwd1")) != NULL &&
  823. (p2 = xs_dict_get(p_vars, "passwd2")) != NULL &&
  824. *p1 && strcmp(p1, p2) == 0) {
  825. xs *pw = hash_password(snac.uid, p1, NULL);
  826. snac.config = xs_dict_set(snac.config, "passwd", pw);
  827. }
  828. xs *fn = xs_fmt("%s/user.json", snac.basedir);
  829. xs *bfn = xs_fmt("%s.bak", fn);
  830. FILE *f;
  831. rename(fn, bfn);
  832. if ((f = fopen(fn, "w")) != NULL) {
  833. xs *j = xs_json_dumps_pp(snac.config, 4);
  834. fwrite(j, strlen(j), 1, f);
  835. fclose(f);
  836. }
  837. else
  838. rename(bfn, fn);
  839. history_del(&snac, "timeline.html_");
  840. xs *a_msg = msg_actor(&snac);
  841. xs *u_msg = msg_update(&snac, a_msg);
  842. post(&snac, u_msg);
  843. status = 303;
  844. }
  845. if (status == 303) {
  846. *body = xs_fmt("%s/admin#snac-posts", snac.actor);
  847. *b_size = strlen(*body);
  848. }
  849. return status;
  850. }