format.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408
  1. /* snac - A simple, minimalistic ActivityPub instance */
  2. /* copyright (c) 2022 - 2024 grunfink et al. / MIT license */
  3. #include "xs.h"
  4. #include "xs_regex.h"
  5. #include "xs_mime.h"
  6. #include "xs_html.h"
  7. #include "xs_json.h"
  8. #include "xs_time.h"
  9. #include "snac.h"
  10. /* emoticons, people laughing and such */
  11. const char *smileys[] = {
  12. ":-)", "🙂",
  13. ":-D", "😀",
  14. "X-D", "😆",
  15. ";-)", "😉",
  16. "B-)", "😎",
  17. ">:-(", "😡",
  18. ":-(", "😞",
  19. ":-*", "😘",
  20. ":-/", "😕",
  21. "8-o", "😲",
  22. "%-)", "🤪",
  23. ":_(", "😢",
  24. ":-|", "😐",
  25. "<3", "&#10084;&#65039;",
  26. ":facepalm:", "&#129318;",
  27. ":shrug:", "&#129335;",
  28. ":shrug2:", "&#175;\\_(&#12484;)_/&#175;",
  29. ":eyeroll:", "&#128580;",
  30. ":beer:", "&#127866;",
  31. ":beers:", "&#127867;",
  32. ":munch:", "&#128561;",
  33. ":thumb:", "&#128077;",
  34. NULL, NULL
  35. };
  36. xs_dict *emojis(void)
  37. /* returns a dict with the emojis */
  38. {
  39. xs *fn = xs_fmt("%s/emojis.json", srv_basedir);
  40. FILE *f;
  41. if (mtime(fn) == 0) {
  42. /* file does not exist; create it with the defaults */
  43. xs *d = xs_dict_new();
  44. const char **emo = smileys;
  45. while (*emo) {
  46. d = xs_dict_append(d, emo[0], emo[1]);
  47. emo += 2;
  48. }
  49. if ((f = fopen(fn, "w")) != NULL) {
  50. xs_json_dump(d, 4, f);
  51. fclose(f);
  52. }
  53. else
  54. srv_log(xs_fmt("Error creating '%s'", fn));
  55. }
  56. xs_dict *d = NULL;
  57. if ((f = fopen(fn, "r")) != NULL) {
  58. d = xs_json_load(f);
  59. fclose(f);
  60. if (d == NULL)
  61. srv_log(xs_fmt("JSON parse error in '%s'", fn));
  62. }
  63. else
  64. srv_log(xs_fmt("Error opening '%s'", fn));
  65. return d;
  66. }
  67. static xs_str *format_line(const char *line, xs_list **attach)
  68. /* formats a line */
  69. {
  70. xs_str *s = xs_str_new(NULL);
  71. char *p;
  72. const char *v;
  73. /* split by markup */
  74. xs *sm = xs_regex_split(line,
  75. "("
  76. "`[^`]+`" "|"
  77. "~~[^~]+~~" "|"
  78. "\\*\\*?\\*?[^\\*]+\\*?\\*?\\*" "|"
  79. "!\\[[^]]+\\]\\([^\\)]+\\)" "|"
  80. "\\[[^]]+\\]\\([^\\)]+\\)" "|"
  81. "https?:/" "/[^[:space:]]+"
  82. ")");
  83. int n = 0;
  84. p = sm;
  85. while (xs_list_iter(&p, &v)) {
  86. if ((n & 0x1)) {
  87. /* markup */
  88. if (xs_startswith(v, "`")) {
  89. xs *s1 = xs_strip_chars_i(xs_dup(v), "`");
  90. xs *e1 = encode_html(s1);
  91. xs *s2 = xs_fmt("<code>%s</code>", e1);
  92. s = xs_str_cat(s, s2);
  93. }
  94. else
  95. if (xs_startswith(v, "***")) {
  96. xs *s1 = xs_strip_chars_i(xs_dup(v), "*");
  97. xs *s2 = xs_fmt("<b><i>%s</i></b>", s1);
  98. s = xs_str_cat(s, s2);
  99. }
  100. else
  101. if (xs_startswith(v, "**")) {
  102. xs *s1 = xs_strip_chars_i(xs_dup(v), "*");
  103. xs *s2 = xs_fmt("<b>%s</b>", s1);
  104. s = xs_str_cat(s, s2);
  105. }
  106. else
  107. if (xs_startswith(v, "*")) {
  108. xs *s1 = xs_strip_chars_i(xs_dup(v), "*");
  109. xs *s2 = xs_fmt("<i>%s</i>", s1);
  110. s = xs_str_cat(s, s2);
  111. }
  112. else
  113. if (xs_startswith(v, "~~")) {
  114. xs *s1 = xs_strip_chars_i(xs_dup(v), "~");
  115. xs *e1 = encode_html(s1);
  116. xs *s2 = xs_fmt("<s>%s</s>", e1);
  117. s = xs_str_cat(s, s2);
  118. }
  119. else
  120. if (xs_startswith(v, "http")) {
  121. xs *u = xs_replace(v, "#", "&#35;");
  122. xs *v2 = xs_strip_chars_i(xs_dup(u), ".,)");
  123. const char *mime = xs_mime_by_ext(v2);
  124. if (attach != NULL && xs_startswith(mime, "image/")) {
  125. /* if it's a link to an image, insert it as an attachment */
  126. xs *d = xs_dict_new();
  127. d = xs_dict_append(d, "mediaType", mime);
  128. d = xs_dict_append(d, "url", v2);
  129. d = xs_dict_append(d, "name", "");
  130. d = xs_dict_append(d, "type", "Image");
  131. *attach = xs_list_append(*attach, d);
  132. }
  133. else {
  134. xs *s1 = xs_fmt("<a href=\"%s\" target=\"_blank\">%s</a>", v2, u);
  135. s = xs_str_cat(s, s1);
  136. }
  137. }
  138. else
  139. if (*v == '[') {
  140. /* markdown-like links [label](url) */
  141. xs *w = xs_strip_chars_i(xs_replace(v, "#", "&#35;"), "[)");
  142. xs *l = xs_split_n(w, "](", 1);
  143. if (xs_list_len(l) == 2) {
  144. xs *link = xs_fmt("<a href=\"%s\">%s</a>",
  145. xs_list_get(l, 1), xs_list_get(l, 0));
  146. s = xs_str_cat(s, link);
  147. }
  148. else
  149. s = xs_str_cat(s, v);
  150. }
  151. else
  152. if (*v == '!') {
  153. /* markdown-like images ![alt text](url to image) */
  154. xs *w = xs_strip_chars_i(xs_replace(v, "#", "&#35;"), "![)");
  155. xs *l = xs_split_n(w, "](", 1);
  156. if (xs_list_len(l) == 2) {
  157. const char *alt_text = xs_list_get(l, 0);
  158. const char *img_url = xs_list_get(l, 1);
  159. const char *mime = xs_mime_by_ext(img_url);
  160. if (attach != NULL && xs_startswith(mime, "image/")) {
  161. xs *d = xs_dict_new();
  162. d = xs_dict_append(d, "mediaType", mime);
  163. d = xs_dict_append(d, "url", img_url);
  164. d = xs_dict_append(d, "name", alt_text);
  165. d = xs_dict_append(d, "type", "Image");
  166. *attach = xs_list_append(*attach, d);
  167. }
  168. else {
  169. xs *link = xs_fmt("<a href=\"%s\">%s</a>", img_url, alt_text);
  170. s = xs_str_cat(s, link);
  171. }
  172. }
  173. else
  174. s = xs_str_cat(s, v);
  175. }
  176. else
  177. s = xs_str_cat(s, v);
  178. }
  179. else
  180. /* surrounded text, copy directly */
  181. s = xs_str_cat(s, v);
  182. n++;
  183. }
  184. return s;
  185. }
  186. xs_str *not_really_markdown(const char *content, xs_list **attach, xs_list **tag)
  187. /* formats a content using some Markdown rules */
  188. {
  189. xs_str *s = xs_str_new(NULL);
  190. int in_pre = 0;
  191. int in_blq = 0;
  192. xs *list;
  193. char *p;
  194. const char *v;
  195. /* work by lines */
  196. list = xs_split(content, "\n");
  197. p = list;
  198. while (xs_list_iter(&p, &v)) {
  199. xs *ss = NULL;
  200. if (strcmp(v, "```") == 0) {
  201. if (!in_pre)
  202. s = xs_str_cat(s, "<pre>");
  203. else
  204. s = xs_str_cat(s, "</pre>");
  205. in_pre = !in_pre;
  206. continue;
  207. }
  208. if (in_pre) {
  209. // Encode all HTML characters when we're in pre element until we are out.
  210. ss = encode_html(v);
  211. s = xs_str_cat(s, ss);
  212. s = xs_str_cat(s, "<br>");
  213. continue;
  214. }
  215. else
  216. ss = xs_strip_i(format_line(v, attach));
  217. if (xs_startswith(ss, "---")) {
  218. /* delete the --- */
  219. ss = xs_strip_i(xs_crop_i(ss, 3, 0));
  220. s = xs_str_cat(s, "<hr>");
  221. s = xs_str_cat(s, ss);
  222. continue;
  223. }
  224. if (xs_startswith(ss, ">")) {
  225. /* delete the > and subsequent spaces */
  226. ss = xs_strip_i(xs_crop_i(ss, 1, 0));
  227. if (!in_blq) {
  228. s = xs_str_cat(s, "<blockquote>");
  229. in_blq = 1;
  230. }
  231. s = xs_str_cat(s, ss);
  232. s = xs_str_cat(s, "<br>");
  233. continue;
  234. }
  235. if (in_blq) {
  236. s = xs_str_cat(s, "</blockquote>");
  237. in_blq = 0;
  238. }
  239. s = xs_str_cat(s, ss);
  240. s = xs_str_cat(s, "<br>");
  241. }
  242. if (in_blq)
  243. s = xs_str_cat(s, "</blockquote>");
  244. if (in_pre)
  245. s = xs_str_cat(s, "</pre>");
  246. /* some beauty fixes */
  247. s = xs_replace_i(s, "<br><br><blockquote>", "<br><blockquote>");
  248. s = xs_replace_i(s, "</blockquote><br>", "</blockquote>");
  249. s = xs_replace_i(s, "</pre><br>", "</pre>");
  250. {
  251. /* traditional emoticons */
  252. xs *d = emojis();
  253. int c = 0;
  254. const char *k, *v;
  255. while (xs_dict_next(d, &k, &v, &c)) {
  256. const char *t = NULL;
  257. /* is it an URL to an image? */
  258. if (xs_startswith(v, "https:/" "/") && xs_startswith((t = xs_mime_by_ext(v)), "image/")) {
  259. if (tag && xs_str_in(s, k) != -1) {
  260. /* add the emoji to the tag list */
  261. xs *e = xs_dict_new();
  262. xs *i = xs_dict_new();
  263. xs *u = xs_str_utctime(0, ISO_DATE_SPEC);
  264. e = xs_dict_append(e, "id", v);
  265. e = xs_dict_append(e, "type", "Emoji");
  266. e = xs_dict_append(e, "name", k);
  267. e = xs_dict_append(e, "updated", u);
  268. i = xs_dict_append(i, "type", "Image");
  269. i = xs_dict_append(i, "mediaType", t);
  270. i = xs_dict_append(i, "url", v);
  271. e = xs_dict_append(e, "icon", i);
  272. *tag = xs_list_append(*tag, e);
  273. }
  274. }
  275. else
  276. s = xs_replace_i(s, k, v);
  277. }
  278. }
  279. return s;
  280. }
  281. const char *valid_tags[] = {
  282. "a", "p", "br", "br/", "blockquote", "ul", "ol", "li", "cite", "small",
  283. "span", "i", "b", "u", "s", "pre", "code", "em", "strong", "hr", "img", "del", "bdi", NULL
  284. };
  285. xs_str *sanitize(const char *content)
  286. /* cleans dangerous HTML output */
  287. {
  288. xs_str *s = xs_str_new(NULL);
  289. xs *sl;
  290. int n = 0;
  291. char *p;
  292. const char *v;
  293. sl = xs_regex_split(content, "</?[^>]+>");
  294. p = sl;
  295. n = 0;
  296. while (xs_list_iter(&p, &v)) {
  297. if (n & 0x1) {
  298. xs *s1 = xs_strip_i(xs_crop_i(xs_dup(v), v[1] == '/' ? 2 : 1, -1));
  299. xs *l1 = xs_split_n(s1, " ", 1);
  300. xs *tag = xs_tolower_i(xs_dup(xs_list_get(l1, 0)));
  301. xs *s2 = NULL;
  302. int i;
  303. /* check if it's one of the valid tags */
  304. for (i = 0; valid_tags[i]; i++) {
  305. if (strcmp(tag, valid_tags[i]) == 0)
  306. break;
  307. }
  308. if (valid_tags[i]) {
  309. /* accepted tag: rebuild it with only the accepted elements */
  310. xs *el = xs_regex_select(v, "(src|href|rel|class|target)=\"[^\"]*\"");
  311. xs *s3 = xs_join(el, " ");
  312. s2 = xs_fmt("<%s%s%s%s>",
  313. v[1] == '/' ? "/" : "", tag, xs_list_len(el) ? " " : "", s3);
  314. s = xs_str_cat(s, s2);
  315. } else {
  316. /* treat end of divs as paragraph breaks */
  317. if (strcmp(v, "</div>"))
  318. s = xs_str_cat(s, "<p>");
  319. }
  320. }
  321. else {
  322. /* non-tag */
  323. s = xs_str_cat(s, v);
  324. }
  325. n++;
  326. }
  327. return s;
  328. }
  329. xs_str *encode_html(const char *str)
  330. /* escapes html characters */
  331. {
  332. xs_str *encoded = xs_html_encode((char *)str);
  333. /* Restore only <br>. Probably safe. Let's hope nothing goes wrong with this. */
  334. encoded = xs_replace_i(encoded, "&lt;br&gt;", "<br>");
  335. return encoded;
  336. }