2 * Copyright (c) 2021, 2022, 2024 Omar Polo <op@omarpolo.com>
4 * Permission to use, copy, modify, and distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
30 tofu_init(struct ohash *h, unsigned int sz, ptrdiff_t ko)
32 struct ohash_info info = {
34 .calloc = hash_calloc,
39 ohash_init(h, sz, &info);
43 tofu_lookup(struct ohash *h, const char *domain, const char *port)
45 char buf[TOFU_URL_MAX_LEN];
48 strlcpy(buf, domain, sizeof(buf));
49 if (port != NULL && *port != '\0' && strcmp(port, "1965")) {
50 strlcat(buf, ":", sizeof(buf));
51 strlcat(buf, port, sizeof(buf));
54 slot = ohash_qlookup(h, buf);
55 return ohash_find(h, slot);
59 tofu_add(struct ohash *h, struct tofu_entry *e)
63 slot = ohash_qlookup(h, e->domain);
64 ohash_insert(h, slot, e);
68 tofu_save(struct ohash *h, struct tofu_entry *e)
74 if ((fp = fopen(known_hosts_file, "a")) == NULL)
76 fprintf(fp, "%s %s %d\n", e->domain, e->hash, e->verified);
82 tofu_update(struct ohash *h, struct tofu_entry *e)
86 if ((t = tofu_lookup(h, e->domain, NULL)) == NULL)
89 strlcpy(t->hash, e->hash, sizeof(t->hash));
90 t->verified = e->verified;
96 tofu_update_persist(struct ohash *h, struct tofu_entry *e)
99 char sfn[PATH_MAX], *line = NULL;
100 size_t l, linesize = 0;
106 strlcpy(sfn, known_hosts_tmp, sizeof(sfn));
107 if ((fd = mkstemp(sfn)) == -1 ||
108 (tmp = fdopen(fd, "w")) == NULL) {
116 if ((fp = fopen(known_hosts_file, "r")) == NULL) {
122 l = strlen(e->domain);
123 while ((linelen = getline(&line, &linesize, fp)) != -1) {
124 if (!strncmp(line, e->domain, l))
126 if (linesize > 0 && line[linesize-1] == '\n')
127 line[linesize-1] = '\0';
128 fprintf(tmp, "%s\n", line);
130 fprintf(tmp, "%s %s %d\n", e->domain, e->hash, e->verified);
142 if (rename(sfn, known_hosts_file) == -1)
148 tofu_temp_trust(struct ohash *h, const char *host, const char *port,
151 struct tofu_entry *e;
153 if ((e = calloc(1, sizeof(*e))) == NULL)
156 strlcpy(e->domain, host, sizeof(e->domain));
157 if (*port != '\0' && strcmp(port, "1965")) {
158 strlcat(e->domain, ":", sizeof(e->domain));
159 strlcat(e->domain, port, sizeof(e->domain));
161 strlcpy(e->hash, hash, sizeof(e->hash));