Blame


1 35e1f40a 2021-03-14 op /*
2 35e1f40a 2021-03-14 op * Copyright (c) 2021 Omar Polo <op@omarpolo.com>
3 35e1f40a 2021-03-14 op *
4 35e1f40a 2021-03-14 op * Permission to use, copy, modify, and distribute this software for any
5 35e1f40a 2021-03-14 op * purpose with or without fee is hereby granted, provided that the above
6 35e1f40a 2021-03-14 op * copyright notice and this permission notice appear in all copies.
7 35e1f40a 2021-03-14 op *
8 35e1f40a 2021-03-14 op * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 35e1f40a 2021-03-14 op * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 35e1f40a 2021-03-14 op * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 35e1f40a 2021-03-14 op * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 35e1f40a 2021-03-14 op * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 35e1f40a 2021-03-14 op * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 35e1f40a 2021-03-14 op * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 35e1f40a 2021-03-14 op */
16 35e1f40a 2021-03-14 op
17 740f578b 2021-03-15 op /*
18 eb2ed626 2021-10-07 op * Handles config and runtime files
19 740f578b 2021-03-15 op */
20 786e6deb 2021-07-21 op
21 786e6deb 2021-07-21 op #include "compat.h"
22 35e1f40a 2021-03-14 op
23 6cd6a9e1 2021-03-20 op #include <sys/stat.h>
24 65124267 2021-08-13 op #include <sys/types.h>
25 6cd6a9e1 2021-03-20 op
26 65124267 2021-08-13 op #include <dirent.h>
27 35e1f40a 2021-03-14 op #include <errno.h>
28 de2a69bb 2021-05-17 op #include <fcntl.h>
29 35e1f40a 2021-03-14 op #include <limits.h>
30 eb2ed626 2021-10-07 op #include <libgen.h>
31 35e1f40a 2021-03-14 op #include <stdio.h>
32 35e1f40a 2021-03-14 op #include <stdlib.h>
33 35e1f40a 2021-03-14 op #include <string.h>
34 35e1f40a 2021-03-14 op #include <unistd.h>
35 4913b479 2021-07-12 op
36 5a824be4 2021-07-13 op #include "pages.h"
37 f63b8f73 2022-04-24 op #include "parser.h"
38 eb9cbcba 2022-01-03 op #include "session.h"
39 f63b8f73 2022-04-24 op #include "telescope.h"
40 9d65b1d9 2022-01-11 op #include "utils.h"
41 35e1f40a 2021-03-14 op
42 f63b8f73 2022-04-24 op #include "fs.h"
43 f63b8f73 2022-04-24 op
44 f63b8f73 2022-04-24 op #ifndef nitems
45 f63b8f73 2022-04-24 op #define nitems(x) (sizeof(x) / sizeof(x[0]))
46 f63b8f73 2022-04-24 op #endif
47 f63b8f73 2022-04-24 op
48 35e1f40a 2021-03-14 op static void die(void) __attribute__((__noreturn__));
49 5cbe1763 2021-08-13 op static int select_non_dot(const struct dirent *);
50 5cbe1763 2021-08-13 op static int select_non_dotdot(const struct dirent *);
51 eb2ed626 2021-10-07 op static size_t join_path(char*, const char*, const char*, size_t);
52 eb2ed626 2021-10-07 op static void getenv_default(char*, const char*, const char*, size_t);
53 eb2ed626 2021-10-07 op static void mkdirs(const char*, mode_t);
54 6c8ddace 2022-01-03 op static void init_paths(void);
55 50f03682 2022-01-03 op static void load_last_session(void);
56 9e97090d 2022-02-26 op static void load_hist(void);
57 83ed72f1 2022-01-03 op static int last_time_crashed(void);
58 f63b8f73 2022-04-24 op static void load_certs(struct ohash *);
59 35e1f40a 2021-03-14 op
60 f88fbabc 2021-11-27 op /*
61 f88fbabc 2021-11-27 op * Where to store user data. These are all equal to ~/.telescope if
62 f88fbabc 2021-11-27 op * it exists.
63 f88fbabc 2021-11-27 op */
64 fb3d08c1 2021-10-07 op char config_path_base[PATH_MAX];
65 fb3d08c1 2021-10-07 op char data_path_base[PATH_MAX];
66 fb3d08c1 2021-10-07 op char cache_path_base[PATH_MAX];
67 eb2ed626 2021-10-07 op
68 4cf6ba13 2022-02-11 op char ctlsock_path[PATH_MAX];
69 eb2ed626 2021-10-07 op char config_path[PATH_MAX];
70 fb3d08c1 2021-10-07 op char lockfile_path[PATH_MAX];
71 fb3d08c1 2021-10-07 op char bookmark_file[PATH_MAX];
72 fb3d08c1 2021-10-07 op char known_hosts_file[PATH_MAX], known_hosts_tmp[PATH_MAX];
73 fb3d08c1 2021-10-07 op char crashed_file[PATH_MAX];
74 de6a6a40 2022-04-24 op char session_file[PATH_MAX], session_file_tmp[PATH_MAX];
75 de6a6a40 2022-04-24 op char history_file[PATH_MAX], history_file_tmp[PATH_MAX];
76 740f578b 2021-03-15 op
77 35e1f40a 2021-03-14 op static void __attribute__((__noreturn__))
78 35e1f40a 2021-03-14 op die(void)
79 35e1f40a 2021-03-14 op {
80 35e1f40a 2021-03-14 op abort(); /* TODO */
81 35e1f40a 2021-03-14 op }
82 35e1f40a 2021-03-14 op
83 f63b8f73 2022-04-24 op static int
84 f63b8f73 2022-04-24 op select_non_dot(const struct dirent *d)
85 fb4dc49f 2021-08-13 op {
86 f63b8f73 2022-04-24 op return strcmp(d->d_name, ".");
87 f63b8f73 2022-04-24 op }
88 fb4dc49f 2021-08-13 op
89 f63b8f73 2022-04-24 op static int
90 f63b8f73 2022-04-24 op select_non_dotdot(const struct dirent *d)
91 f63b8f73 2022-04-24 op {
92 f63b8f73 2022-04-24 op return strcmp(d->d_name, ".") && strcmp(d->d_name, "..");
93 fb4dc49f 2021-08-13 op }
94 fb4dc49f 2021-08-13 op
95 fb4dc49f 2021-08-13 op static void
96 f63b8f73 2022-04-24 op send_dir(struct tab *tab, const char *path)
97 35e1f40a 2021-03-14 op {
98 f63b8f73 2022-04-24 op struct dirent **names;
99 f63b8f73 2022-04-24 op int (*selector)(const struct dirent *) = select_non_dot;
100 f63b8f73 2022-04-24 op int i, len;
101 35e1f40a 2021-03-14 op
102 f63b8f73 2022-04-24 op #if notyet
103 f63b8f73 2022-04-24 op /*
104 f63b8f73 2022-04-24 op * need something to fake a redirect
105 f63b8f73 2022-04-24 op */
106 35e1f40a 2021-03-14 op
107 f63b8f73 2022-04-24 op if (!has_suffix(path, "/")) {
108 f63b8f73 2022-04-24 op if (asprintf(&s, "%s/", path) == -1)
109 f63b8f73 2022-04-24 op die();
110 f63b8f73 2022-04-24 op send_hdr(peerid, 30, s);
111 f63b8f73 2022-04-24 op free(s);
112 f63b8f73 2022-04-24 op return;
113 56e7efb4 2021-07-21 op }
114 f63b8f73 2022-04-24 op #endif
115 56e7efb4 2021-07-21 op
116 f63b8f73 2022-04-24 op if (!strcmp(path, "/"))
117 f63b8f73 2022-04-24 op selector = select_non_dotdot;
118 f63b8f73 2022-04-24 op
119 f63b8f73 2022-04-24 op if ((len = scandir(path, &names, selector, alphasort)) == -1) {
120 f63b8f73 2022-04-24 op load_page_from_str(tab, "# failure reading the directory\n");
121 a2728733 2021-07-18 op return;
122 35e1f40a 2021-03-14 op }
123 a2728733 2021-07-18 op
124 f63b8f73 2022-04-24 op parser_init(tab, gemtext_initparser);
125 f63b8f73 2022-04-24 op parser_parsef(tab, "# Index of %s\n\n", path);
126 56e7efb4 2021-07-21 op
127 f63b8f73 2022-04-24 op for (i = 0; i < len; ++i) {
128 f63b8f73 2022-04-24 op const char *sufx = "";
129 fb4dc49f 2021-08-13 op
130 f63b8f73 2022-04-24 op if (names[i]->d_type == DT_DIR)
131 f63b8f73 2022-04-24 op sufx = "/";
132 f63b8f73 2022-04-24 op
133 f63b8f73 2022-04-24 op parser_parsef(tab, "=> %s%s\n", names[i]->d_name, sufx);
134 f63b8f73 2022-04-24 op }
135 f63b8f73 2022-04-24 op
136 f63b8f73 2022-04-24 op parser_free(tab);
137 f63b8f73 2022-04-24 op free(names);
138 fb4dc49f 2021-08-13 op }
139 fb4dc49f 2021-08-13 op
140 f63b8f73 2022-04-24 op static int
141 f63b8f73 2022-04-24 op is_dir(FILE *fp)
142 24a68158 2021-08-13 op {
143 f63b8f73 2022-04-24 op struct stat sb;
144 24a68158 2021-08-13 op
145 f63b8f73 2022-04-24 op if (fstat(fileno(fp), &sb) == -1)
146 f63b8f73 2022-04-24 op return 0;
147 24a68158 2021-08-13 op
148 f63b8f73 2022-04-24 op return S_ISDIR(sb.st_mode);
149 24a68158 2021-08-13 op }
150 24a68158 2021-08-13 op
151 f63b8f73 2022-04-24 op static parserinit
152 65124267 2021-08-13 op file_type(const char *path)
153 fb4dc49f 2021-08-13 op {
154 21268f10 2022-04-24 op const struct mapping {
155 fb4dc49f 2021-08-13 op const char *ext;
156 f63b8f73 2022-04-24 op parserinit fn;
157 fb4dc49f 2021-08-13 op } ms[] = {
158 f63b8f73 2022-04-24 op {"diff", textpatch_initparser},
159 f63b8f73 2022-04-24 op {"gemini", gemtext_initparser},
160 f63b8f73 2022-04-24 op {"gmi", gemtext_initparser},
161 f63b8f73 2022-04-24 op {"markdown", textplain_initparser},
162 f63b8f73 2022-04-24 op {"md", textplain_initparser},
163 f63b8f73 2022-04-24 op {"patch", gemtext_initparser},
164 fb4dc49f 2021-08-13 op {NULL, NULL},
165 65124267 2021-08-13 op }, *m;
166 21268f10 2022-04-24 op const char *dot;
167 65124267 2021-08-13 op
168 65124267 2021-08-13 op if ((dot = strrchr(path, '.')) == NULL)
169 f63b8f73 2022-04-24 op return textplain_initparser;
170 65124267 2021-08-13 op
171 65124267 2021-08-13 op dot++;
172 65124267 2021-08-13 op
173 65124267 2021-08-13 op for (m = ms; m->ext != NULL; ++m)
174 65124267 2021-08-13 op if (!strcmp(m->ext, dot))
175 f63b8f73 2022-04-24 op return m->fn;
176 65124267 2021-08-13 op
177 f63b8f73 2022-04-24 op return textplain_initparser;
178 5cbe1763 2021-08-13 op }
179 5cbe1763 2021-08-13 op
180 f63b8f73 2022-04-24 op void
181 f63b8f73 2022-04-24 op fs_load_url(struct tab *tab, const char *url)
182 5cbe1763 2021-08-13 op {
183 f63b8f73 2022-04-24 op const char *bpath = "bookmarks.gmi", *fallback = "# Not found\n";
184 f63b8f73 2022-04-24 op parserinit initfn = gemtext_initparser;
185 f63b8f73 2022-04-24 op char path[PATH_MAX];
186 f63b8f73 2022-04-24 op FILE *fp = NULL;
187 f63b8f73 2022-04-24 op size_t i;
188 f63b8f73 2022-04-24 op char buf[BUFSIZ];
189 f63b8f73 2022-04-24 op struct page {
190 f63b8f73 2022-04-24 op const char *name;
191 f63b8f73 2022-04-24 op const char *path;
192 f63b8f73 2022-04-24 op const uint8_t *data;
193 f63b8f73 2022-04-24 op size_t len;
194 f63b8f73 2022-04-24 op } pages[] = {
195 f63b8f73 2022-04-24 op {"about", NULL, about_about, about_about_len},
196 f63b8f73 2022-04-24 op {"blank", NULL, about_blank, about_blank_len},
197 f63b8f73 2022-04-24 op {"bookmarks", bpath, bookmarks, bookmarks_len},
198 f63b8f73 2022-04-24 op {"crash", NULL, about_crash, about_crash_len},
199 f63b8f73 2022-04-24 op {"help", NULL, about_help, about_help_len},
200 f63b8f73 2022-04-24 op {"license", NULL, about_license, about_license_len},
201 f63b8f73 2022-04-24 op {"new", NULL, about_new, about_new_len},
202 f63b8f73 2022-04-24 op }, *page = NULL;
203 65124267 2021-08-13 op
204 f63b8f73 2022-04-24 op if (!strncmp(url, "about:", 6)) {
205 f63b8f73 2022-04-24 op url += 6;
206 65124267 2021-08-13 op
207 f63b8f73 2022-04-24 op for (i = 0; page == NULL && i < nitems(pages); ++i) {
208 f63b8f73 2022-04-24 op if (!strcmp(url, pages[i].name))
209 f63b8f73 2022-04-24 op page = &pages[i];
210 f63b8f73 2022-04-24 op }
211 5d43215b 2021-08-13 op
212 f63b8f73 2022-04-24 op if (page == NULL)
213 f63b8f73 2022-04-24 op goto done;
214 5cbe1763 2021-08-13 op
215 f63b8f73 2022-04-24 op strlcpy(path, data_path_base, sizeof(path));
216 f63b8f73 2022-04-24 op strlcat(path, "/", sizeof(path));
217 f63b8f73 2022-04-24 op if (page->path != NULL)
218 f63b8f73 2022-04-24 op strlcat(path, page->path, sizeof(path));
219 f63b8f73 2022-04-24 op else {
220 f63b8f73 2022-04-24 op strlcat(path, "page/about_", sizeof(path));
221 f63b8f73 2022-04-24 op strlcat(path, page->name, sizeof(path));
222 f63b8f73 2022-04-24 op strlcat(path, ".gmi", sizeof(path));
223 f63b8f73 2022-04-24 op }
224 65124267 2021-08-13 op
225 f63b8f73 2022-04-24 op fallback = page->data;
226 f63b8f73 2022-04-24 op } else if (!strncmp(url, "file://", 7)) {
227 f63b8f73 2022-04-24 op url += 7;
228 f63b8f73 2022-04-24 op strlcpy(path, url, sizeof(path));
229 f63b8f73 2022-04-24 op initfn = file_type(url);
230 f63b8f73 2022-04-24 op } else
231 f63b8f73 2022-04-24 op goto done;
232 65124267 2021-08-13 op
233 f63b8f73 2022-04-24 op if ((fp = fopen(path, "r")) == NULL)
234 f63b8f73 2022-04-24 op goto done;
235 65124267 2021-08-13 op
236 f63b8f73 2022-04-24 op if (is_dir(fp)) {
237 f63b8f73 2022-04-24 op fclose(fp);
238 f63b8f73 2022-04-24 op send_dir(tab, path);
239 f63b8f73 2022-04-24 op goto done;
240 fb4dc49f 2021-08-13 op }
241 fb4dc49f 2021-08-13 op
242 f63b8f73 2022-04-24 op parser_init(tab, initfn);
243 f63b8f73 2022-04-24 op for (;;) {
244 f63b8f73 2022-04-24 op size_t r;
245 fb4dc49f 2021-08-13 op
246 f63b8f73 2022-04-24 op r = fread(buf, 1, sizeof(buf), fp);
247 f63b8f73 2022-04-24 op if (!parser_parse(tab, buf, r))
248 f63b8f73 2022-04-24 op break;
249 f63b8f73 2022-04-24 op if (r != sizeof(buf))
250 f63b8f73 2022-04-24 op break;
251 fb4dc49f 2021-08-13 op }
252 f63b8f73 2022-04-24 op parser_free(tab);
253 fb4dc49f 2021-08-13 op
254 f63b8f73 2022-04-24 op done:
255 f63b8f73 2022-04-24 op if (fp != NULL)
256 f63b8f73 2022-04-24 op fclose(fp);
257 f63b8f73 2022-04-24 op else
258 f63b8f73 2022-04-24 op load_page_from_str(tab, fallback);
259 740f578b 2021-03-15 op }
260 740f578b 2021-03-15 op
261 f63b8f73 2022-04-24 op int
262 f63b8f73 2022-04-24 op update_cert(const struct tofu_entry *e)
263 3a227e9a 2021-03-18 op {
264 288fd238 2021-04-25 op FILE *tmp, *f;
265 288fd238 2021-04-25 op char sfn[PATH_MAX], *line = NULL, *t;
266 288fd238 2021-04-25 op size_t l, linesize = 0;
267 288fd238 2021-04-25 op ssize_t linelen;
268 f63b8f73 2022-04-24 op int fd, err;
269 288fd238 2021-04-25 op
270 288fd238 2021-04-25 op strlcpy(sfn, known_hosts_tmp, sizeof(sfn));
271 288fd238 2021-04-25 op if ((fd = mkstemp(sfn)) == -1 ||
272 288fd238 2021-04-25 op (tmp = fdopen(fd, "w")) == NULL) {
273 288fd238 2021-04-25 op if (fd != -1) {
274 288fd238 2021-04-25 op unlink(sfn);
275 288fd238 2021-04-25 op close(fd);
276 288fd238 2021-04-25 op }
277 f63b8f73 2022-04-24 op return -1;
278 288fd238 2021-04-25 op }
279 288fd238 2021-04-25 op
280 288fd238 2021-04-25 op if ((f = fopen(known_hosts_file, "r")) == NULL) {
281 288fd238 2021-04-25 op unlink(sfn);
282 288fd238 2021-04-25 op fclose(tmp);
283 f63b8f73 2022-04-24 op return -1;
284 288fd238 2021-04-25 op }
285 288fd238 2021-04-25 op
286 f63b8f73 2022-04-24 op l = strlen(e->domain);
287 288fd238 2021-04-25 op while ((linelen = getline(&line, &linesize, f)) != -1) {
288 f63b8f73 2022-04-24 op if ((t = strstr(line, e->domain)) != NULL &&
289 288fd238 2021-04-25 op (line[l] == ' ' || line[l] == '\t'))
290 288fd238 2021-04-25 op continue;
291 288fd238 2021-04-25 op /* line has a trailing \n */
292 288fd238 2021-04-25 op fprintf(tmp, "%s", line);
293 288fd238 2021-04-25 op }
294 f63b8f73 2022-04-24 op fprintf(tmp, "%s %s %d\n", e->domain, e->hash, e->verified);
295 288fd238 2021-04-25 op
296 288fd238 2021-04-25 op free(line);
297 f63b8f73 2022-04-24 op err = ferror(tmp);
298 288fd238 2021-04-25 op
299 288fd238 2021-04-25 op fclose(tmp);
300 288fd238 2021-04-25 op fclose(f);
301 288fd238 2021-04-25 op
302 f63b8f73 2022-04-24 op if (err) {
303 288fd238 2021-04-25 op unlink(sfn);
304 f63b8f73 2022-04-24 op return -1;
305 288fd238 2021-04-25 op }
306 288fd238 2021-04-25 op
307 f63b8f73 2022-04-24 op if (rename(sfn, known_hosts_file))
308 f63b8f73 2022-04-24 op return -1;
309 f63b8f73 2022-04-24 op return 0;
310 35e1f40a 2021-03-14 op }
311 35e1f40a 2021-03-14 op
312 eb2ed626 2021-10-07 op static size_t
313 eb2ed626 2021-10-07 op join_path(char *buf, const char *lhs, const char *rhs, size_t buflen)
314 35e1f40a 2021-03-14 op {
315 eb2ed626 2021-10-07 op strlcpy(buf, lhs, buflen);
316 eb2ed626 2021-10-07 op return strlcat(buf, rhs, buflen);
317 eb2ed626 2021-10-07 op }
318 d0fd368a 2021-07-15 op
319 eb2ed626 2021-10-07 op static void
320 eb2ed626 2021-10-07 op getenv_default(char *buf, const char *name, const char *def, size_t buflen)
321 eb2ed626 2021-10-07 op {
322 eb2ed626 2021-10-07 op size_t ret;
323 eb2ed626 2021-10-07 op char *home, *env;
324 740f578b 2021-03-15 op
325 eb2ed626 2021-10-07 op if ((home = getenv("HOME")) == NULL)
326 eb2ed626 2021-10-07 op errx(1, "HOME is not defined");
327 3a227e9a 2021-03-18 op
328 eb2ed626 2021-10-07 op if ((env = getenv(name)) != NULL)
329 eb2ed626 2021-10-07 op ret = strlcpy(buf, env, buflen);
330 eb2ed626 2021-10-07 op else
331 eb2ed626 2021-10-07 op ret = join_path(buf, home, def, buflen);
332 288fd238 2021-04-25 op
333 eb2ed626 2021-10-07 op if (ret >= buflen)
334 eb2ed626 2021-10-07 op errx(1, "buffer too small for %s", name);
335 eb2ed626 2021-10-07 op }
336 c7107cec 2021-04-01 op
337 eb2ed626 2021-10-07 op static void
338 eb2ed626 2021-10-07 op mkdirs(const char *path, mode_t mode)
339 eb2ed626 2021-10-07 op {
340 444dad86 2021-10-07 op char copy[PATH_MAX+1], orig[PATH_MAX+1], *parent;
341 ea080950 2021-07-20 op
342 eb2ed626 2021-10-07 op strlcpy(copy, path, sizeof(copy));
343 444dad86 2021-10-07 op strlcpy(orig, path, sizeof(orig));
344 eb2ed626 2021-10-07 op parent = dirname(copy);
345 eb2ed626 2021-10-07 op if (!strcmp(parent, "/"))
346 eb2ed626 2021-10-07 op return;
347 eb2ed626 2021-10-07 op mkdirs(parent, mode);
348 eb2ed626 2021-10-07 op
349 444dad86 2021-10-07 op if (mkdir(orig, mode) != 0) {
350 eb2ed626 2021-10-07 op if (errno == EEXIST)
351 eb2ed626 2021-10-07 op return;
352 444dad86 2021-10-07 op err(1, "can't mkdir %s", orig);
353 eb2ed626 2021-10-07 op }
354 eb2ed626 2021-10-07 op }
355 eb2ed626 2021-10-07 op
356 eb2ed626 2021-10-07 op static void
357 6c8ddace 2022-01-03 op init_paths(void)
358 eb2ed626 2021-10-07 op {
359 7e60a21a 2022-01-03 op char xdg_config_base[PATH_MAX];
360 7e60a21a 2022-01-03 op char xdg_data_base[PATH_MAX];
361 7e60a21a 2022-01-03 op char xdg_cache_base[PATH_MAX];
362 7e60a21a 2022-01-03 op char old_path[PATH_MAX];
363 7e60a21a 2022-01-03 op char *home;
364 7e60a21a 2022-01-03 op struct stat info;
365 eb2ed626 2021-10-07 op
366 eb2ed626 2021-10-07 op /* old path */
367 eb2ed626 2021-10-07 op if ((home = getenv("HOME")) == NULL)
368 eb2ed626 2021-10-07 op errx(1, "HOME is not defined");
369 eb2ed626 2021-10-07 op join_path(old_path, home, "/.telescope", sizeof(old_path));
370 eb2ed626 2021-10-07 op
371 eb2ed626 2021-10-07 op /* if ~/.telescope exists, use that instead of xdg dirs */
372 eb2ed626 2021-10-07 op if (stat(old_path, &info) == 0 && S_ISDIR(info.st_mode)) {
373 eb2ed626 2021-10-07 op join_path(config_path_base, home, "/.telescope",
374 eb2ed626 2021-10-07 op sizeof(config_path_base));
375 eb2ed626 2021-10-07 op join_path(data_path_base, home, "/.telescope",
376 eb2ed626 2021-10-07 op sizeof(data_path_base));
377 eb2ed626 2021-10-07 op join_path(cache_path_base, home, "/.telescope",
378 eb2ed626 2021-10-07 op sizeof(cache_path_base));
379 eb2ed626 2021-10-07 op return;
380 eb2ed626 2021-10-07 op }
381 eb2ed626 2021-10-07 op
382 eb2ed626 2021-10-07 op /* xdg paths */
383 eb2ed626 2021-10-07 op getenv_default(xdg_config_base, "XDG_CONFIG_HOME", "/.config",
384 eb2ed626 2021-10-07 op sizeof(xdg_config_base));
385 eb2ed626 2021-10-07 op getenv_default(xdg_data_base, "XDG_DATA_HOME", "/.local/share",
386 eb2ed626 2021-10-07 op sizeof(xdg_data_base));
387 eb2ed626 2021-10-07 op getenv_default(xdg_cache_base, "XDG_CACHE_HOME", "/.cache",
388 eb2ed626 2021-10-07 op sizeof(xdg_cache_base));
389 eb2ed626 2021-10-07 op
390 eb2ed626 2021-10-07 op join_path(config_path_base, xdg_config_base, "/telescope",
391 eb2ed626 2021-10-07 op sizeof(config_path_base));
392 eb2ed626 2021-10-07 op join_path(data_path_base, xdg_data_base, "/telescope",
393 eb2ed626 2021-10-07 op sizeof(data_path_base));
394 eb2ed626 2021-10-07 op join_path(cache_path_base, xdg_cache_base, "/telescope",
395 eb2ed626 2021-10-07 op sizeof(cache_path_base));
396 eb2ed626 2021-10-07 op
397 eb2ed626 2021-10-07 op mkdirs(xdg_config_base, S_IRWXU);
398 eb2ed626 2021-10-07 op mkdirs(xdg_data_base, S_IRWXU);
399 eb2ed626 2021-10-07 op mkdirs(xdg_cache_base, S_IRWXU);
400 eb2ed626 2021-10-07 op
401 eb2ed626 2021-10-07 op mkdirs(config_path_base, S_IRWXU);
402 eb2ed626 2021-10-07 op mkdirs(data_path_base, S_IRWXU);
403 eb2ed626 2021-10-07 op mkdirs(cache_path_base, S_IRWXU);
404 eb2ed626 2021-10-07 op }
405 eb2ed626 2021-10-07 op
406 eb2ed626 2021-10-07 op int
407 eb2ed626 2021-10-07 op fs_init(void)
408 eb2ed626 2021-10-07 op {
409 6c8ddace 2022-01-03 op init_paths();
410 eb2ed626 2021-10-07 op
411 4cf6ba13 2022-02-11 op join_path(ctlsock_path, cache_path_base, "/ctl",
412 4cf6ba13 2022-02-11 op sizeof(ctlsock_path));
413 eb2ed626 2021-10-07 op join_path(config_path, config_path_base, "/config",
414 eb2ed626 2021-10-07 op sizeof(config_path));
415 eb2ed626 2021-10-07 op join_path(lockfile_path, cache_path_base, "/lock",
416 eb2ed626 2021-10-07 op sizeof(lockfile_path));
417 eb2ed626 2021-10-07 op join_path(bookmark_file, data_path_base, "/bookmarks.gmi",
418 eb2ed626 2021-10-07 op sizeof(bookmark_file));
419 eb2ed626 2021-10-07 op join_path(known_hosts_file, data_path_base, "/known_hosts",
420 eb2ed626 2021-10-07 op sizeof(known_hosts_file));
421 eb2ed626 2021-10-07 op join_path(known_hosts_tmp, cache_path_base,
422 eb2ed626 2021-10-07 op "/known_hosts.tmp.XXXXXXXXXX", sizeof(known_hosts_tmp));
423 eb2ed626 2021-10-07 op join_path(session_file, cache_path_base, "/session",
424 de6a6a40 2022-04-24 op sizeof(session_file));
425 de6a6a40 2022-04-24 op join_path(session_file_tmp, cache_path_base, "/session.XXXXXXXXXX",
426 eb2ed626 2021-10-07 op sizeof(session_file));
427 9e97090d 2022-02-26 op join_path(history_file, cache_path_base, "/history",
428 de6a6a40 2022-04-24 op sizeof(history_file));
429 de6a6a40 2022-04-24 op join_path(history_file_tmp, cache_path_base, "/history.XXXXXXXXXX",
430 9e97090d 2022-02-26 op sizeof(history_file));
431 eb2ed626 2021-10-07 op join_path(crashed_file, cache_path_base, "/crashed",
432 eb2ed626 2021-10-07 op sizeof(crashed_file));
433 eb2ed626 2021-10-07 op
434 3a227e9a 2021-03-18 op return 1;
435 3a227e9a 2021-03-18 op }
436 bb28f1c2 2021-12-30 op
437 bb28f1c2 2021-12-30 op /*
438 f63b8f73 2022-04-24 op * Parse a line of the session file and restores it. The format is:
439 bb28f1c2 2021-12-30 op *
440 bb28f1c2 2021-12-30 op * URL [flags,...] [title]\n
441 bb28f1c2 2021-12-30 op */
442 f63b8f73 2022-04-24 op static inline struct tab *
443 f63b8f73 2022-04-24 op parse_session_line(char *line, struct tab **ct)
444 bb28f1c2 2021-12-30 op {
445 f63b8f73 2022-04-24 op struct tab *tab;
446 bb28f1c2 2021-12-30 op char *s, *t, *ap;
447 f63b8f73 2022-04-24 op const char *uri, *title = "";
448 f63b8f73 2022-04-24 op int current = 0, killed = 0;
449 f63b8f73 2022-04-24 op size_t top_line = 0, current_line = 0;
450 bb28f1c2 2021-12-30 op
451 f63b8f73 2022-04-24 op uri = line;
452 bb28f1c2 2021-12-30 op if ((s = strchr(line, ' ')) == NULL)
453 f63b8f73 2022-04-24 op return NULL;
454 3a227e9a 2021-03-18 op
455 bb28f1c2 2021-12-30 op *s++ = '\0';
456 bb28f1c2 2021-12-30 op
457 bb28f1c2 2021-12-30 op if ((t = strchr(s, ' ')) != NULL) {
458 bb28f1c2 2021-12-30 op *t++ = '\0';
459 f63b8f73 2022-04-24 op title = t;
460 bb28f1c2 2021-12-30 op }
461 bb28f1c2 2021-12-30 op
462 bb28f1c2 2021-12-30 op while ((ap = strsep(&s, ",")) != NULL) {
463 bb28f1c2 2021-12-30 op if (!strcmp(ap, "current"))
464 f63b8f73 2022-04-24 op current = 1;
465 6c74799d 2022-01-05 op else if (!strcmp(ap, "killed"))
466 f63b8f73 2022-04-24 op killed = 1;
467 e795e935 2022-01-18 op else if (has_prefix(ap, "top="))
468 f63b8f73 2022-04-24 op top_line = strtonum(ap+4, 0, UINT32_MAX, NULL);
469 e795e935 2022-01-18 op else if (has_prefix(ap, "cur="))
470 f63b8f73 2022-04-24 op current_line = strtonum(ap+4, 0, UINT32_MAX, NULL);
471 bb28f1c2 2021-12-30 op }
472 bb28f1c2 2021-12-30 op
473 f63b8f73 2022-04-24 op if (top_line > current_line) {
474 f63b8f73 2022-04-24 op top_line = 0;
475 f63b8f73 2022-04-24 op current_line = 0;
476 e795e935 2022-01-18 op }
477 e795e935 2022-01-18 op
478 f63b8f73 2022-04-24 op if ((tab = new_tab(uri, NULL, NULL)) == NULL)
479 f63b8f73 2022-04-24 op die();
480 f63b8f73 2022-04-24 op tab->hist_cur->line_off = top_line;
481 f63b8f73 2022-04-24 op tab->hist_cur->current_off = current_line;
482 f63b8f73 2022-04-24 op strlcpy(tab->buffer.page.title, title, sizeof(tab->buffer.page.title));
483 f63b8f73 2022-04-24 op
484 f63b8f73 2022-04-24 op if (current)
485 f63b8f73 2022-04-24 op *ct = tab;
486 f63b8f73 2022-04-24 op else if (killed)
487 f63b8f73 2022-04-24 op kill_tab(tab, 1);
488 f63b8f73 2022-04-24 op
489 f63b8f73 2022-04-24 op return tab;
490 1040cc7f 2021-01-02 op }
491 1040cc7f 2021-01-02 op
492 1040cc7f 2021-01-02 op static inline void
493 f63b8f73 2022-04-24 op sendhist(struct tab *tab, const char *uri, int future)
494 1040cc7f 2021-01-02 op {
495 f63b8f73 2022-04-24 op struct hist *h;
496 1040cc7f 2021-01-02 op
497 f63b8f73 2022-04-24 op if ((h = calloc(1, sizeof(*h))) == NULL)
498 f63b8f73 2022-04-24 op die();
499 f63b8f73 2022-04-24 op strlcpy(h->h, uri, sizeof(h->h));
500 1040cc7f 2021-01-02 op
501 f63b8f73 2022-04-24 op if (future)
502 f63b8f73 2022-04-24 op hist_push(&tab->hist, h);
503 f63b8f73 2022-04-24 op else
504 f63b8f73 2022-04-24 op hist_add_before(&tab->hist, tab->hist_cur, h);
505 bb28f1c2 2021-12-30 op }
506 bb28f1c2 2021-12-30 op
507 bb28f1c2 2021-12-30 op static void
508 50f03682 2022-01-03 op load_last_session(void)
509 bb28f1c2 2021-12-30 op {
510 f63b8f73 2022-04-24 op struct tab *tab = NULL, *ct = NULL;
511 bb28f1c2 2021-12-30 op FILE *session;
512 bb28f1c2 2021-12-30 op size_t linesize = 0;
513 bb28f1c2 2021-12-30 op ssize_t linelen;
514 1040cc7f 2021-01-02 op int future;
515 1040cc7f 2021-01-02 op char *nl, *s, *line = NULL;
516 bb28f1c2 2021-12-30 op
517 bb28f1c2 2021-12-30 op if ((session = fopen(session_file, "r")) == NULL) {
518 f63b8f73 2022-04-24 op new_tab("about:new", NULL, NULL);
519 f63b8f73 2022-04-24 op switch_to_tab(new_tab("about:help", NULL, NULL));
520 f63b8f73 2022-04-24 op return;
521 bb28f1c2 2021-12-30 op }
522 bb28f1c2 2021-12-30 op
523 bb28f1c2 2021-12-30 op while ((linelen = getline(&line, &linesize, session)) != -1) {
524 bb28f1c2 2021-12-30 op if ((nl = strchr(line, '\n')) != NULL)
525 bb28f1c2 2021-12-30 op *nl = '\0';
526 1040cc7f 2021-01-02 op
527 1040cc7f 2021-01-02 op if (*line == '<' || *line == '>') {
528 1040cc7f 2021-01-02 op future = *line == '>';
529 1040cc7f 2021-01-02 op s = line+1;
530 f63b8f73 2022-04-24 op if (*s != ' ' || tab == NULL)
531 1040cc7f 2021-01-02 op continue;
532 f63b8f73 2022-04-24 op sendhist(tab, ++s, future);
533 1040cc7f 2021-01-02 op } else {
534 f63b8f73 2022-04-24 op tab = parse_session_line(line, &ct);
535 1040cc7f 2021-01-02 op }
536 bb28f1c2 2021-12-30 op }
537 bb28f1c2 2021-12-30 op
538 bb28f1c2 2021-12-30 op fclose(session);
539 bb28f1c2 2021-12-30 op free(line);
540 bb28f1c2 2021-12-30 op
541 f63b8f73 2022-04-24 op if (ct != NULL)
542 f63b8f73 2022-04-24 op switch_to_tab(ct);
543 bb28f1c2 2021-12-30 op
544 f63b8f73 2022-04-24 op if (last_time_crashed())
545 f63b8f73 2022-04-24 op switch_to_tab(new_tab("about:crash", NULL, NULL));
546 bb28f1c2 2021-12-30 op }
547 bb28f1c2 2021-12-30 op
548 9e97090d 2022-02-26 op static void
549 9e97090d 2022-02-26 op load_hist(void)
550 9e97090d 2022-02-26 op {
551 9e97090d 2022-02-26 op FILE *hist;
552 9e97090d 2022-02-26 op size_t linesize = 0;
553 9e97090d 2022-02-26 op ssize_t linelen;
554 9e97090d 2022-02-26 op char *nl, *spc, *line = NULL;
555 9e97090d 2022-02-26 op const char *errstr;
556 9e97090d 2022-02-26 op struct histitem hi;
557 9e97090d 2022-02-26 op
558 9e97090d 2022-02-26 op if ((hist = fopen(history_file, "r")) == NULL)
559 f63b8f73 2022-04-24 op return;
560 9e97090d 2022-02-26 op
561 9e97090d 2022-02-26 op while ((linelen = getline(&line, &linesize, hist)) != -1) {
562 9e97090d 2022-02-26 op if ((nl = strchr(line, '\n')) != NULL)
563 9e97090d 2022-02-26 op *nl = '\0';
564 9e97090d 2022-02-26 op if ((spc = strchr(line, ' ')) == NULL)
565 9e97090d 2022-02-26 op continue;
566 9e97090d 2022-02-26 op *spc = '\0';
567 9e97090d 2022-02-26 op spc++;
568 9e97090d 2022-02-26 op
569 9e97090d 2022-02-26 op memset(&hi, 0, sizeof(hi));
570 9e97090d 2022-02-26 op hi.ts = strtonum(line, INT64_MIN, INT64_MAX, &errstr);
571 9e97090d 2022-02-26 op if (errstr != NULL)
572 9e97090d 2022-02-26 op continue;
573 9e97090d 2022-02-26 op if (strlcpy(hi.uri, spc, sizeof(hi.uri)) >= sizeof(hi.uri))
574 9e97090d 2022-02-26 op continue;
575 9e97090d 2022-02-26 op
576 f63b8f73 2022-04-24 op history_push(&hi);
577 9e97090d 2022-02-26 op }
578 9e97090d 2022-02-26 op
579 9e97090d 2022-02-26 op fclose(hist);
580 9e97090d 2022-02-26 op free(line);
581 f63b8f73 2022-04-24 op
582 f63b8f73 2022-04-24 op history_sort();
583 9e97090d 2022-02-26 op }
584 9e97090d 2022-02-26 op
585 3a227e9a 2021-03-18 op int
586 f63b8f73 2022-04-24 op fs_load_state(struct ohash *certs)
587 3a227e9a 2021-03-18 op {
588 f63b8f73 2022-04-24 op load_certs(certs);
589 f63b8f73 2022-04-24 op load_hist();
590 f63b8f73 2022-04-24 op load_last_session();
591 35e1f40a 2021-03-14 op return 0;
592 35e1f40a 2021-03-14 op }
593 3a227e9a 2021-03-18 op
594 be97d6e6 2021-08-15 op /*
595 be97d6e6 2021-08-15 op * Check if the last time telescope crashed. The check is done by
596 be97d6e6 2021-08-15 op * looking at `crashed_file': if it exists then last time we crashed.
597 be97d6e6 2021-08-15 op * Then, while here, touch the file too. During IMSG_QUIT we'll
598 be97d6e6 2021-08-15 op * remove it.
599 be97d6e6 2021-08-15 op */
600 83ed72f1 2022-01-03 op static int
601 b6171794 2021-07-20 op last_time_crashed(void)
602 b6171794 2021-07-20 op {
603 be97d6e6 2021-08-15 op int fd, crashed = 1;
604 b6171794 2021-07-20 op
605 2b409042 2021-09-15 op if (safe_mode)
606 2b409042 2021-09-15 op return 0;
607 2b409042 2021-09-15 op
608 be97d6e6 2021-08-15 op if (unlink(crashed_file) == -1 && errno == ENOENT)
609 be97d6e6 2021-08-15 op crashed = 0;
610 be97d6e6 2021-08-15 op
611 be97d6e6 2021-08-15 op if ((fd = open(crashed_file, O_CREAT|O_WRONLY, 0600)) == -1)
612 be97d6e6 2021-08-15 op return crashed;
613 b6171794 2021-07-20 op close(fd);
614 be97d6e6 2021-08-15 op
615 be97d6e6 2021-08-15 op return crashed;
616 b6171794 2021-07-20 op }
617 b6171794 2021-07-20 op
618 b6171794 2021-07-20 op int
619 d0fd368a 2021-07-15 op lock_session(void)
620 d0fd368a 2021-07-15 op {
621 d0fd368a 2021-07-15 op struct flock lock;
622 d0fd368a 2021-07-15 op int fd;
623 d0fd368a 2021-07-15 op
624 d0fd368a 2021-07-15 op if ((fd = open(lockfile_path, O_WRONLY|O_CREAT, 0600)) == -1)
625 d0fd368a 2021-07-15 op return -1;
626 d0fd368a 2021-07-15 op
627 d0fd368a 2021-07-15 op lock.l_start = 0;
628 d0fd368a 2021-07-15 op lock.l_len = 0;
629 d0fd368a 2021-07-15 op lock.l_type = F_WRLCK;
630 d0fd368a 2021-07-15 op lock.l_whence = SEEK_SET;
631 c6d03cf5 2021-04-25 op
632 d0fd368a 2021-07-15 op if (fcntl(fd, F_SETLK, &lock) == -1) {
633 d0fd368a 2021-07-15 op close(fd);
634 d0fd368a 2021-07-15 op return -1;
635 d0fd368a 2021-07-15 op }
636 d0fd368a 2021-07-15 op
637 d0fd368a 2021-07-15 op return fd;
638 d0fd368a 2021-07-15 op }
639 d0fd368a 2021-07-15 op
640 6400962b 2022-01-03 op static inline int
641 c6d03cf5 2021-04-25 op parse_khost_line(char *line, char *tmp[3])
642 c6d03cf5 2021-04-25 op {
643 c6d03cf5 2021-04-25 op char **ap;
644 c6d03cf5 2021-04-25 op
645 c6d03cf5 2021-04-25 op for (ap = tmp; ap < &tmp[3] &&
646 c6d03cf5 2021-04-25 op (*ap = strsep(&line, " \t\n")) != NULL;) {
647 c6d03cf5 2021-04-25 op if (**ap != '\0')
648 c6d03cf5 2021-04-25 op ap++;
649 c6d03cf5 2021-04-25 op }
650 3a227e9a 2021-03-18 op
651 c6d03cf5 2021-04-25 op return ap == &tmp[3] && *line == '\0';
652 c6d03cf5 2021-04-25 op }
653 c6d03cf5 2021-04-25 op
654 eb722b50 2022-01-03 op static void
655 f63b8f73 2022-04-24 op load_certs(struct ohash *certs)
656 3a227e9a 2021-03-18 op {
657 c6d03cf5 2021-04-25 op char *tmp[3], *line = NULL;
658 6cd6a9e1 2021-03-20 op const char *errstr;
659 ec1fa0b0 2021-04-25 op size_t lineno = 0, linesize = 0;
660 3a227e9a 2021-03-18 op ssize_t linelen;
661 3a227e9a 2021-03-18 op FILE *f;
662 f63b8f73 2022-04-24 op struct tofu_entry *e;
663 3a227e9a 2021-03-18 op
664 3a227e9a 2021-03-18 op if ((f = fopen(known_hosts_file, "r")) == NULL)
665 eb722b50 2022-01-03 op return;
666 3a227e9a 2021-03-18 op
667 f63b8f73 2022-04-24 op if ((e = calloc(1, sizeof(*e))) == NULL) {
668 f63b8f73 2022-04-24 op fclose(f);
669 f63b8f73 2022-04-24 op return;
670 f63b8f73 2022-04-24 op }
671 f63b8f73 2022-04-24 op
672 3a227e9a 2021-03-18 op while ((linelen = getline(&line, &linesize, f)) != -1) {
673 ec1fa0b0 2021-04-25 op lineno++;
674 3a227e9a 2021-03-18 op
675 eb722b50 2022-01-03 op if (parse_khost_line(line, tmp)) {
676 f63b8f73 2022-04-24 op strlcpy(e->domain, tmp[0], sizeof(e->domain));
677 f63b8f73 2022-04-24 op strlcpy(e->hash, tmp[1], sizeof(e->hash));
678 3a227e9a 2021-03-18 op
679 f63b8f73 2022-04-24 op e->verified = strtonum(tmp[2], 0, 1, &errstr);
680 c6d03cf5 2021-04-25 op if (errstr != NULL)
681 c6d03cf5 2021-04-25 op errx(1, "%s:%zu verification for %s is %s: %s",
682 c6d03cf5 2021-04-25 op known_hosts_file, lineno,
683 f63b8f73 2022-04-24 op e->domain, errstr, tmp[2]);
684 eb722b50 2022-01-03 op
685 f63b8f73 2022-04-24 op tofu_add(certs, e);
686 c6d03cf5 2021-04-25 op } else {
687 ec1fa0b0 2021-04-25 op warnx("%s:%zu invalid entry",
688 ec1fa0b0 2021-04-25 op known_hosts_file, lineno);
689 c6d03cf5 2021-04-25 op }
690 3a227e9a 2021-03-18 op }
691 3a227e9a 2021-03-18 op
692 3a227e9a 2021-03-18 op free(line);
693 eb722b50 2022-01-03 op fclose(f);
694 eb722b50 2022-01-03 op return;
695 3a227e9a 2021-03-18 op }