utils.c 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914
  1. /* snac - A simple, minimalistic ActivityPub instance */
  2. /* copyright (c) 2022 - 2025 grunfink et al. / MIT license */
  3. #include "xs.h"
  4. #include "xs_io.h"
  5. #include "xs_json.h"
  6. #include "xs_time.h"
  7. #include "xs_openssl.h"
  8. #include "xs_random.h"
  9. #include "xs_glob.h"
  10. #include "xs_curl.h"
  11. #include "xs_regex.h"
  12. #include "snac.h"
  13. #include <sys/stat.h>
  14. #include <stdlib.h>
  15. static const char *default_srv_config = "{"
  16. "\"host\": \"\","
  17. "\"prefix\": \"\","
  18. "\"address\": \"127.0.0.1\","
  19. "\"port\": 8001,"
  20. "\"layout\": 0.0,"
  21. "\"dbglevel\": 0,"
  22. "\"queue_retry_minutes\": 2,"
  23. "\"queue_retry_max\": 10,"
  24. "\"queue_timeout\": 6,"
  25. "\"queue_timeout_2\": 8,"
  26. "\"cssurls\": [\"\"],"
  27. "\"def_timeline_entries\": 50,"
  28. "\"max_timeline_entries\": 50,"
  29. "\"timeline_purge_days\": 120,"
  30. "\"local_purge_days\": 0,"
  31. "\"min_account_age\": 0,"
  32. "\"admin_email\": \"\","
  33. "\"admin_account\": \"\","
  34. "\"title\": \"\","
  35. "\"short_description\": \"\","
  36. "\"short_description_raw\": false,"
  37. "\"protocol\": \"https\","
  38. "\"fastcgi\": false"
  39. "}";
  40. static const char *default_css =
  41. "body { max-width: 48em; margin: auto; line-height: 1.5; padding: 0.8em; word-wrap: break-word; }\n"
  42. "pre { overflow-x: scroll; }\n"
  43. ".snac-embedded-video, img { max-width: 100% }\n"
  44. ".snac-origin { font-size: 85% }\n"
  45. ".snac-score { float: right; font-size: 85% }\n"
  46. ".snac-top-user { text-align: center; padding-bottom: 2em }\n"
  47. ".snac-top-user-name { font-size: 200% }\n"
  48. ".snac-top-user-id { font-size: 150% }\n"
  49. ".snac-announcement { border: black 1px solid; padding: 0.5em }\n"
  50. ".snac-avatar { float: left; height: 2.5em; width: 2.5em; padding: 0.25em }\n"
  51. ".snac-author { font-size: 90%; text-decoration: none }\n"
  52. ".snac-author-tag { font-size: 80% }\n"
  53. ".snac-pubdate { color: #a0a0a0; font-size: 90% }\n"
  54. ".snac-top-controls { padding-bottom: 1.5em }\n"
  55. ".snac-post { border-top: 1px solid #a0a0a0; padding-top: 0.5em; padding-bottom: 0.5em; }\n"
  56. ".snac-children { padding-left: 1em; border-left: 1px solid #a0a0a0; }\n"
  57. ".snac-thread-cont { border-top: 1px dashed #a0a0a0; }\n"
  58. ".snac-textarea { font-family: inherit; width: 100% }\n"
  59. ".snac-history { border: 1px solid #606060; border-radius: 3px; margin: 2.5em 0; padding: 0 2em }\n"
  60. ".snac-btn-mute { float: right; margin-left: 0.5em }\n"
  61. ".snac-btn-unmute { float: right; margin-left: 0.5em }\n"
  62. ".snac-btn-follow { float: right; margin-left: 0.5em }\n"
  63. ".snac-btn-unfollow { float: right; margin-left: 0.5em }\n"
  64. ".snac-btn-hide { float: right; margin-left: 0.5em }\n"
  65. ".snac-btn-delete { float: right; margin-left: 0.5em }\n"
  66. ".snac-btn-limit { float: right; margin-left: 0.5em }\n"
  67. ".snac-btn-unlimit { float: right; margin-left: 0.5em }\n"
  68. ".snac-footer { margin-top: 2em; font-size: 75% }\n"
  69. ".snac-poll-result { margin-left: auto; margin-right: auto; }\n"
  70. ".snac-list-of-lists { padding-left: 0; }\n"
  71. ".snac-list-of-lists li { display: inline; border: 1px solid #a0a0a0; border-radius: 25px;\n"
  72. " margin-right: 0.5em; padding-left: 0.5em; padding-right: 0.5em; }\n"
  73. ".snac-no-more-unseen-posts { border-top: 1px solid #a0a0a0; border-bottom: 1px solid #a0a0a0; padding: 0.5em 0; margin: 1em 0; }\n"
  74. "@media (prefers-color-scheme: dark) { \n"
  75. " body, input, textarea { background-color: #000; color: #fff; }\n"
  76. " a { color: #7799dd }\n"
  77. " a:visited { color: #aa99dd }\n"
  78. "}\n"
  79. ;
  80. const char *snac_blurb =
  81. "<p><b>%host%</b> is a <a href=\"https:/"
  82. "/en.wikipedia.org/wiki/Fediverse\">Fediverse</a> "
  83. "instance that uses the <a href=\"https:/"
  84. "/en.wikipedia.org/wiki/ActivityPub\">ActivityPub</a> "
  85. "protocol. In other words, users at this host can communicate with people "
  86. "that use software like Mastodon, Pleroma, Friendica, etc. "
  87. "all around the world.</p>\n"
  88. "<p>This server runs the "
  89. "<a href=\"" WHAT_IS_SNAC_URL "\">snac</a> software and there is no "
  90. "automatic sign-up process.</p>\n"
  91. ;
  92. static const char *greeting_html =
  93. "<!DOCTYPE html>\n"
  94. "<html><head>\n"
  95. "<meta name=\"viewport\" content=\"width=device-width, initial-scale=1\"/>\n"
  96. "<link rel=\"icon\" type=\"image/x-icon\" href=\"https://%host%/favicon.ico\"/>\n"
  97. "<title>Welcome to %host%</title>\n</head>\n"
  98. "<body style=\"margin: auto; max-width: 50em\">\n"
  99. "%blurb%"
  100. "<p>The following users are part of this community:</p>\n"
  101. "\n"
  102. "%userlist%\n"
  103. "\n"
  104. "<p>This site is powered by <abbr title=\"Social Networks Are Crap\">snac</abbr>.</p>\n"
  105. "</body></html>\n";
  106. int write_default_css(void)
  107. {
  108. FILE *f;
  109. xs *sfn = xs_fmt("%s/style.css", srv_basedir);
  110. if ((f = fopen(sfn, "w")) == NULL)
  111. return 1;
  112. fwrite(default_css, strlen(default_css), 1, f);
  113. fclose(f);
  114. return 0;
  115. }
  116. int snac_init(const char *basedir)
  117. {
  118. FILE *f;
  119. if (basedir == NULL) {
  120. printf("Base directory: "); fflush(stdout);
  121. srv_basedir = xs_strip_i(xs_readline(stdin));
  122. }
  123. else
  124. srv_basedir = xs_str_new(basedir);
  125. if (srv_basedir == NULL || *srv_basedir == '\0')
  126. return 1;
  127. if (xs_endswith(srv_basedir, "/"))
  128. srv_basedir = xs_crop_i(srv_basedir, 0, -1);
  129. if (mtime(srv_basedir) != 0.0) {
  130. printf("ERROR: directory '%s' must not exist.\n", srv_basedir);
  131. return 1;
  132. }
  133. srv_config = xs_json_loads(default_srv_config);
  134. xs *layout = xs_number_new(disk_layout);
  135. srv_config = xs_dict_set(srv_config, "layout", layout);
  136. int is_unix_socket = 0;
  137. printf("Network address or full path to unix socket [%s]: ", xs_dict_get(srv_config, "address")); fflush(stdout);
  138. {
  139. xs *i = xs_strip_i(xs_readline(stdin));
  140. if (*i) {
  141. srv_config = xs_dict_set(srv_config, "address", i);
  142. if (*i == '/')
  143. is_unix_socket = 1;
  144. }
  145. }
  146. if (!is_unix_socket) {
  147. printf("Network port [%d]: ", (int)xs_number_get(xs_dict_get(srv_config, "port"))); fflush(stdout);
  148. {
  149. xs *i = xs_strip_i(xs_readline(stdin));
  150. if (*i) {
  151. xs *n = xs_number_new(atoi(i));
  152. srv_config = xs_dict_set(srv_config, "port", n);
  153. }
  154. }
  155. }
  156. else {
  157. xs *n = xs_number_new(0);
  158. srv_config = xs_dict_set(srv_config, "port", n);
  159. }
  160. printf("Host name: "); fflush(stdout);
  161. {
  162. xs *i = xs_strip_i(xs_readline(stdin));
  163. if (*i == '\0')
  164. return 1;
  165. srv_config = xs_dict_set(srv_config, "host", i);
  166. }
  167. printf("URL prefix: "); fflush(stdout);
  168. {
  169. xs *i = xs_strip_i(xs_readline(stdin));
  170. if (*i) {
  171. if (xs_endswith(i, "/"))
  172. i = xs_crop_i(i, 0, -1);
  173. srv_config = xs_dict_set(srv_config, "prefix", i);
  174. }
  175. }
  176. printf("Admin email address (optional): "); fflush(stdout);
  177. {
  178. xs *i = xs_strip_i(xs_readline(stdin));
  179. srv_config = xs_dict_set(srv_config, "admin_email", i);
  180. }
  181. if (mkdirx(srv_basedir) == -1) {
  182. printf("ERROR: cannot create directory '%s'\n", srv_basedir);
  183. return 1;
  184. }
  185. xs *udir = xs_fmt("%s/user", srv_basedir);
  186. mkdirx(udir);
  187. xs *odir = xs_fmt("%s/object", srv_basedir);
  188. mkdirx(odir);
  189. xs *qdir = xs_fmt("%s/queue", srv_basedir);
  190. mkdirx(qdir);
  191. xs *ibdir = xs_fmt("%s/inbox", srv_basedir);
  192. mkdirx(ibdir);
  193. xs *gfn = xs_fmt("%s/greeting.html", srv_basedir);
  194. if ((f = fopen(gfn, "w")) == NULL) {
  195. printf("ERROR: cannot create '%s'\n", gfn);
  196. return 1;
  197. }
  198. xs *gh = xs_replace(greeting_html, "%blurb%", snac_blurb);
  199. fwrite(gh, strlen(gh), 1, f);
  200. fclose(f);
  201. if (write_default_css()) {
  202. printf("ERROR: cannot create style.css\n");
  203. return 1;
  204. }
  205. xs *cfn = xs_fmt("%s/server.json", srv_basedir);
  206. if ((f = fopen(cfn, "w")) == NULL) {
  207. printf("ERROR: cannot create '%s'\n", cfn);
  208. return 1;
  209. }
  210. xs_json_dump(srv_config, 4, f);
  211. fclose(f);
  212. printf("Done.\n");
  213. return 0;
  214. }
  215. void new_password(const char *uid, xs_str **clear_pwd, xs_str **hashed_pwd)
  216. /* creates a random password */
  217. {
  218. int rndbuf[3];
  219. xs_rnd_buf(rndbuf, sizeof(rndbuf));
  220. *clear_pwd = xs_base64_enc((char *)rndbuf, sizeof(rndbuf));
  221. *hashed_pwd = hash_password(uid, *clear_pwd, NULL);
  222. }
  223. int adduser(const char *uid)
  224. /* creates a new user */
  225. {
  226. snac snac;
  227. xs *config = xs_dict_new();
  228. xs *date = xs_str_utctime(0, ISO_DATE_SPEC);
  229. xs *pwd = NULL;
  230. xs *pwd_f = NULL;
  231. xs *key = NULL;
  232. FILE *f;
  233. if (uid == NULL) {
  234. printf("Username: "); fflush(stdout);
  235. uid = xs_strip_i(xs_readline(stdin));
  236. }
  237. if (!validate_uid(uid)) {
  238. printf("ERROR: only alphanumeric characters and _ are allowed in user ids.\n");
  239. return 1;
  240. }
  241. if (user_open(&snac, uid)) {
  242. printf("ERROR: user '%s' already exists\n", snac.uid);
  243. return 1;
  244. }
  245. new_password(uid, &pwd, &pwd_f);
  246. config = xs_dict_append(config, "uid", uid);
  247. config = xs_dict_append(config, "name", uid);
  248. config = xs_dict_append(config, "avatar", "");
  249. config = xs_dict_append(config, "bio", "");
  250. config = xs_dict_append(config, "cw", "");
  251. config = xs_dict_append(config, "published", date);
  252. config = xs_dict_append(config, "passwd", pwd_f);
  253. xs *basedir = xs_fmt("%s/user/%s", srv_basedir, uid);
  254. if (mkdirx(basedir) == -1) {
  255. printf("ERROR: cannot create directory '%s'\n", basedir);
  256. return 0;
  257. }
  258. const char *dirs[] = {
  259. "followers", "following", "muted", "hidden",
  260. "public", "private", "queue", "history",
  261. "static", NULL };
  262. int n;
  263. for (n = 0; dirs[n]; n++) {
  264. xs *d = xs_fmt("%s/%s", basedir, dirs[n]);
  265. mkdirx(d);
  266. }
  267. /* add a specially short data retention time for the relay */
  268. if (strcmp(uid, "relay") == 0)
  269. config = xs_dict_set(config, "purge_days", xs_stock(1));
  270. xs *cfn = xs_fmt("%s/user.json", basedir);
  271. if ((f = fopen(cfn, "w")) == NULL) {
  272. printf("ERROR: cannot create '%s'\n", cfn);
  273. return 1;
  274. }
  275. else {
  276. xs_json_dump(config, 4, f);
  277. fclose(f);
  278. }
  279. printf("\nCreating RSA key...\n");
  280. key = xs_evp_genkey(2048);
  281. printf("Done.\n");
  282. xs *kfn = xs_fmt("%s/key.json", basedir);
  283. if ((f = fopen(kfn, "w")) == NULL) {
  284. printf("ERROR: cannot create '%s'\n", kfn);
  285. return 1;
  286. }
  287. else {
  288. xs_json_dump(key, 4, f);
  289. fclose(f);
  290. }
  291. printf("\nUser password is %s\n", pwd);
  292. printf("\nGo to %s/%s and continue configuring your user there.\n", srv_baseurl, uid);
  293. return 0;
  294. }
  295. int resetpwd(snac *snac)
  296. /* creates a new password for the user */
  297. {
  298. xs *clear_pwd = NULL;
  299. xs *hashed_pwd = NULL;
  300. xs *fn = xs_fmt("%s/user.json", snac->basedir);
  301. FILE *f;
  302. int ret = 0;
  303. new_password(snac->uid, &clear_pwd, &hashed_pwd);
  304. snac->config = xs_dict_set(snac->config, "passwd", hashed_pwd);
  305. if ((f = fopen(fn, "w")) != NULL) {
  306. xs_json_dump(snac->config, 4, f);
  307. fclose(f);
  308. printf("New password for user %s is %s\n", snac->uid, clear_pwd);
  309. }
  310. else {
  311. printf("ERROR: cannot write to %s\n", fn);
  312. ret = 1;
  313. }
  314. return ret;
  315. }
  316. void rm_rf(const char *dir)
  317. /* does an rm -rf (yes, I'm also scared) */
  318. {
  319. xs *d = xs_str_cat(xs_dup(dir), "/" "*");
  320. xs *l = xs_glob(d, 0, 0);
  321. xs_list *p = l;
  322. const xs_str *v;
  323. if (dbglevel >= 1)
  324. printf("Deleting directory %s\n", dir);
  325. while (xs_list_iter(&p, &v)) {
  326. struct stat st;
  327. if (stat(v, &st) != -1) {
  328. if (st.st_mode & S_IFDIR) {
  329. rm_rf(v);
  330. }
  331. else {
  332. if (dbglevel >= 1)
  333. printf("Deleting file %s\n", v);
  334. if (unlink(v) == -1)
  335. printf("ERROR: cannot delete file %s\n", v);
  336. }
  337. }
  338. else
  339. printf("ERROR: stat() fail for %s\n", v);
  340. }
  341. if (rmdir(dir) == -1)
  342. printf("ERROR: cannot delete directory %s\n", dir);
  343. }
  344. int deluser(snac *user)
  345. /* deletes a user */
  346. {
  347. int ret = 0;
  348. xs *fwers = following_list(user);
  349. xs_list *p = fwers;
  350. const xs_str *v;
  351. while (xs_list_iter(&p, &v)) {
  352. xs *object = NULL;
  353. if (valid_status(following_get(user, v, &object))) {
  354. xs *msg = msg_undo(user, xs_dict_get(object, "object"));
  355. following_del(user, v);
  356. enqueue_output_by_actor(user, msg, v, 0);
  357. printf("Unfollowing actor %s\n", v);
  358. }
  359. }
  360. rm_rf(user->basedir);
  361. return ret;
  362. }
  363. void verify_links(snac *user)
  364. /* verifies a user's links */
  365. {
  366. xs *metadata = NULL;
  367. const xs_dict *md = xs_dict_get(user->config, "metadata");
  368. const char *k, *v;
  369. int changed = 0;
  370. xs *headers = xs_dict_new();
  371. headers = xs_dict_append(headers, "accept", "text/html");
  372. headers = xs_dict_append(headers, "user-agent", USER_AGENT " (link verify)");
  373. if (xs_type(md) == XSTYPE_DICT)
  374. metadata = xs_dup(md);
  375. else
  376. if (xs_type(md) == XSTYPE_STRING) {
  377. /* convert to dict for easier iteration */
  378. metadata = xs_dict_new();
  379. xs *l = xs_split(md, "\n");
  380. const char *ll;
  381. xs_list_foreach(l, ll) {
  382. xs *kv = xs_split_n(ll, "=", 1);
  383. const char *k = xs_list_get(kv, 0);
  384. const char *v = xs_list_get(kv, 1);
  385. if (k && v) {
  386. xs *kk = xs_strip_i(xs_dup(k));
  387. xs *vv = xs_strip_i(xs_dup(v));
  388. metadata = xs_dict_set(metadata, kk, vv);
  389. }
  390. }
  391. }
  392. int c = 0;
  393. while (metadata && xs_dict_next(metadata, &k, &v, &c)) {
  394. /* not an https link? skip */
  395. if (!xs_startswith(v, "https:/" "/"))
  396. continue;
  397. int status;
  398. xs *req = NULL;
  399. xs *payload = NULL;
  400. int p_size = 0;
  401. req = xs_http_request("GET", v, headers, NULL, 0, &status,
  402. &payload, &p_size, 0);
  403. if (!valid_status(status)) {
  404. snac_log(user, xs_fmt("link %s verify error %d", v, status));
  405. continue;
  406. }
  407. /* extract the links */
  408. xs *ls = xs_regex_select(payload, "< *(a|link) +[^>]+>");
  409. xs_list *lp = ls;
  410. const char *ll;
  411. int vfied = 0;
  412. while (!vfied && xs_list_iter(&lp, &ll)) {
  413. /* extract href and rel */
  414. xs *r = xs_regex_select(ll, "(href|rel) *= *(\"[^\"]*\"|'[^']*')");
  415. /* must have both attributes */
  416. if (xs_list_len(r) != 2)
  417. continue;
  418. xs *href = NULL;
  419. int is_rel_me = 0;
  420. xs_list *pr = r;
  421. const char *ar;
  422. while (xs_list_iter(&pr, &ar)) {
  423. xs *nq = xs_dup(ar);
  424. nq = xs_replace_i(nq, "\"", "");
  425. nq = xs_replace_i(nq, "'", "");
  426. xs *r2 = xs_split_n(nq, "=", 1);
  427. if (xs_list_len(r2) != 2)
  428. continue;
  429. xs *ak = xs_strip_i(xs_dup(xs_list_get(r2, 0)));
  430. xs *av = xs_strip_i(xs_dup(xs_list_get(r2, 1)));
  431. if (strcmp(ak, "href") == 0)
  432. href = xs_dup(av);
  433. else
  434. if (strcmp(ak, "rel") == 0) {
  435. /* split the value by spaces */
  436. xs *vbs = xs_split(av, " ");
  437. /* is any of it "me"? */
  438. if (xs_list_in(vbs, "me") != -1)
  439. is_rel_me = 1;
  440. }
  441. }
  442. /* after all this acrobatics, do we have an href and a rel="me"? */
  443. if (href != NULL && is_rel_me) {
  444. /* is it the same as the actor? */
  445. if (strcmp(href, user->actor) == 0) {
  446. /* got it! */
  447. xs *verified_time = xs_number_new((double)time(NULL));
  448. if (user->links == NULL)
  449. user->links = xs_dict_new();
  450. user->links = xs_dict_set(user->links, v, verified_time);
  451. vfied = 1;
  452. }
  453. else
  454. snac_debug(user, 1,
  455. xs_fmt("verify link %s rel='me' found but not related (%s)", v, href));
  456. }
  457. }
  458. if (vfied) {
  459. changed++;
  460. snac_log(user, xs_fmt("link %s verified", v));
  461. }
  462. else {
  463. snac_log(user, xs_fmt("link %s not verified (rel='me' not found)", v));
  464. }
  465. }
  466. if (changed) {
  467. FILE *f;
  468. /* update the links.json file */
  469. xs *fn = xs_fmt("%s/links.json", user->basedir);
  470. xs *bfn = xs_fmt("%s.bak", fn);
  471. rename(fn, bfn);
  472. if ((f = fopen(fn, "w")) != NULL) {
  473. xs_json_dump(user->links, 4, f);
  474. fclose(f);
  475. }
  476. else
  477. rename(bfn, fn);
  478. }
  479. }
  480. void export_csv(snac *user)
  481. /* exports user data to current directory in a way that pleases Mastodon */
  482. {
  483. FILE *f;
  484. xs *fn = NULL;
  485. fn = xs_fmt("%s/export/bookmarks.csv", user->basedir);
  486. if ((f = fopen(fn, "w")) != NULL) {
  487. snac_log(user, xs_fmt("Creating %s...", fn));
  488. xs *l = bookmark_list(user);
  489. const char *md5;
  490. xs_list_foreach(l, md5) {
  491. xs *post = NULL;
  492. if (valid_status(object_get_by_md5(md5, &post))) {
  493. const char *id = xs_dict_get(post, "id");
  494. if (xs_type(id) == XSTYPE_STRING)
  495. fprintf(f, "%s\n", id);
  496. }
  497. }
  498. fclose(f);
  499. }
  500. else
  501. snac_log(user, xs_fmt("Cannot create file %s", fn));
  502. xs_free(fn);
  503. fn = xs_fmt("%s/export/blocked_accounts.csv", user->basedir);
  504. if ((f = fopen(fn, "w")) != NULL) {
  505. snac_log(user, xs_fmt("Creating %s...", fn));
  506. xs *l = muted_list(user);
  507. const char *actor;
  508. xs_list_foreach(l, actor) {
  509. xs *uid = NULL;
  510. webfinger_request_fake(actor, NULL, &uid);
  511. fprintf(f, "%s\n", uid);
  512. }
  513. fclose(f);
  514. }
  515. else
  516. snac_log(user, xs_fmt("Cannot create file %s", fn));
  517. xs_free(fn);
  518. fn = xs_fmt("%s/export/lists.csv", user->basedir);
  519. if ((f = fopen(fn, "w")) != NULL) {
  520. snac_log(user, xs_fmt("Creating %s...", fn));
  521. xs *lol = list_maint(user, NULL, 0);
  522. const xs_list *li;
  523. xs_list_foreach(lol, li) {
  524. const char *lid = xs_list_get(li, 0);
  525. const char *ltitle = xs_list_get(li, 1);
  526. xs *actors = list_content(user, lid, NULL, 0);
  527. const char *md5;
  528. xs_list_foreach(actors, md5) {
  529. xs *actor = NULL;
  530. if (valid_status(object_get_by_md5(md5, &actor))) {
  531. const char *id = xs_dict_get(actor, "id");
  532. xs *uid = NULL;
  533. webfinger_request_fake(id, NULL, &uid);
  534. fprintf(f, "%s,%s\n", ltitle, uid);
  535. }
  536. }
  537. }
  538. fclose(f);
  539. }
  540. else
  541. snac_log(user, xs_fmt("Cannot create file %s", fn));
  542. xs_free(fn);
  543. fn = xs_fmt("%s/export/following_accounts.csv", user->basedir);
  544. if ((f = fopen(fn, "w")) != NULL) {
  545. snac_log(user, xs_fmt("Creating %s...", fn));
  546. fprintf(f, "Account address,Show boosts,Notify on new posts,Languages\n");
  547. xs *fwing = following_list(user);
  548. const char *actor;
  549. xs_list_foreach(fwing, actor) {
  550. xs *uid = NULL;
  551. webfinger_request_fake(actor, NULL, &uid);
  552. fprintf(f, "%s,%s,false,\n", uid, limited(user, actor, 0) ? "false" : "true");
  553. }
  554. fclose(f);
  555. }
  556. else
  557. snac_log(user, xs_fmt("Cannot create file %s", fn));
  558. }
  559. void import_blocked_accounts_csv(snac *user, const char *ifn)
  560. /* imports a Mastodon CSV file of blocked accounts */
  561. {
  562. FILE *f;
  563. xs *l = xs_split(ifn, "/");
  564. xs *fn = xs_fmt("%s/import/%s", user->basedir, xs_list_get(l, -1));
  565. if ((f = fopen(fn, "r")) != NULL) {
  566. snac_log(user, xs_fmt("Importing from %s...", fn));
  567. while (!feof(f)) {
  568. xs *l = xs_strip_i(xs_readline(f));
  569. if (*l && strchr(l, '@') != NULL) {
  570. xs *url = NULL;
  571. xs *uid = NULL;
  572. if (valid_status(webfinger_request(l, &url, &uid))) {
  573. if (is_muted(user, url))
  574. snac_log(user, xs_fmt("Actor %s already MUTEd", url));
  575. else {
  576. mute(user, url);
  577. snac_log(user, xs_fmt("MUTEd actor %s", url));
  578. }
  579. }
  580. else
  581. snac_log(user, xs_fmt("Webfinger error for account %s", l));
  582. }
  583. }
  584. fclose(f);
  585. }
  586. else
  587. snac_log(user, xs_fmt("Cannot open file %s", fn));
  588. }
  589. void import_following_accounts_csv(snac *user, const char *ifn)
  590. /* imports a Mastodon CSV file of accounts to follow */
  591. {
  592. FILE *f;
  593. xs *l = xs_split(ifn, "/");
  594. xs *fn = xs_fmt("%s/import/%s", user->basedir, xs_list_get(l, -1));
  595. if ((f = fopen(fn, "r")) != NULL) {
  596. snac_log(user, xs_fmt("Importing from %s...", fn));
  597. while (!feof(f)) {
  598. xs *l = xs_strip_i(xs_readline(f));
  599. if (*l) {
  600. xs *l2 = xs_split(l, ",");
  601. const char *acct = xs_list_get(l2, 0);
  602. const char *show = xs_list_get(l2, 1);
  603. if (acct) {
  604. /* not a valid account? skip (probably the CSV header) */
  605. if (strchr(acct, '@') == NULL)
  606. continue;
  607. xs *msg = msg_follow(user, acct);
  608. if (msg == NULL) {
  609. snac_log(user, xs_fmt("Cannot follow %s -- server down?", acct));
  610. continue;
  611. }
  612. const char *actor = xs_dict_get(msg, "object");
  613. if (following_check(user, actor))
  614. snac_log(user, xs_fmt("Actor %s already followed", actor));
  615. else {
  616. following_add(user, actor, msg);
  617. enqueue_output_by_actor(user, msg, actor, 0);
  618. snac_log(user, xs_fmt("Following %s", actor));
  619. }
  620. if (show && strcmp(show, "false") == 0) {
  621. limit(user, actor);
  622. snac_log(user, xs_fmt("Limiting boosts from actor %s", actor));
  623. }
  624. else {
  625. unlimit(user, actor);
  626. snac_log(user, xs_fmt("Unlimiting boosts from actor %s", actor));
  627. }
  628. }
  629. }
  630. }
  631. fclose(f);
  632. }
  633. else
  634. snac_log(user, xs_fmt("Cannot open file %s", fn));
  635. }
  636. void import_list_csv(snac *user, const char *ifn)
  637. /* imports a Mastodon CSV file list */
  638. {
  639. FILE *f;
  640. xs *l = xs_split(ifn, "/");
  641. xs *fn = xs_fmt("%s/import/%s", user->basedir, xs_list_get(l, -1));
  642. if ((f = fopen(fn, "r")) != NULL) {
  643. snac_log(user, xs_fmt("Importing from %s...", fn));
  644. while (!feof(f)) {
  645. xs *l = xs_strip_i(xs_readline(f));
  646. if (*l) {
  647. xs *l2 = xs_split(l, ",");
  648. const char *lname = xs_list_get(l2, 0);
  649. const char *acct = xs_list_get(l2, 1);
  650. if (lname && acct) {
  651. /* create the list */
  652. xs *list_id = list_maint(user, lname, 1);
  653. xs *url = NULL;
  654. xs *uid = NULL;
  655. if (valid_status(webfinger_request(acct, &url, &uid))) {
  656. xs *actor_md5 = xs_md5_hex(url, strlen(url));
  657. list_content(user, list_id, actor_md5, 1);
  658. snac_log(user, xs_fmt("Added %s to list %s", url, lname));
  659. if (!following_check(user, url)) {
  660. xs *msg = msg_follow(user, url);
  661. if (msg == NULL) {
  662. snac_log(user, xs_fmt("Cannot follow %s -- server down?", acct));
  663. continue;
  664. }
  665. following_add(user, url, msg);
  666. enqueue_output_by_actor(user, msg, url, 0);
  667. snac_log(user, xs_fmt("Following %s", url));
  668. }
  669. }
  670. else
  671. snac_log(user, xs_fmt("Webfinger error while adding %s to list %s", acct, lname));
  672. }
  673. }
  674. }
  675. fclose(f);
  676. }
  677. else
  678. snac_log(user, xs_fmt("Cannot open file %s", fn));
  679. }
  680. void import_csv(snac *user)
  681. /* import CSV files from Mastodon */
  682. {
  683. FILE *f;
  684. import_blocked_accounts_csv(user, "blocked_accounts.csv");
  685. import_following_accounts_csv(user, "following_accounts.csv");
  686. import_list_csv(user, "lists.csv");
  687. xs *fn = xs_fmt("%s/import/bookmarks.csv", user->basedir);
  688. if ((f = fopen(fn, "r")) != NULL) {
  689. snac_log(user, xs_fmt("Importing from %s...", fn));
  690. while (!feof(f)) {
  691. xs *l = xs_strip_i(xs_readline(f));
  692. if (*l) {
  693. xs *post = NULL;
  694. if (!valid_status(object_get(l, &post))) {
  695. if (!valid_status(activitypub_request(user, l, &post))) {
  696. snac_log(user, xs_fmt("Error getting object %s for bookmarking", l));
  697. continue;
  698. }
  699. }
  700. if (post == NULL)
  701. continue;
  702. /* request the actor that created the post */
  703. const char *actor = get_atto(post);
  704. if (xs_type(actor) == XSTYPE_STRING)
  705. actor_request(user, actor, NULL);
  706. object_add_ow(l, post);
  707. timeline_add(user, l, post);
  708. bookmark(user, l);
  709. snac_log(user, xs_fmt("Bookmarked %s", l));
  710. }
  711. }
  712. fclose(f);
  713. }
  714. else
  715. snac_log(user, xs_fmt("Cannot open file %s", fn));
  716. }