2 257add31 2020-09-09 stsp * Copyright (c) 2020 Stefan Sperling <stsp@openbsd.org>
4 257add31 2020-09-09 stsp * Permission to use, copy, modify, and distribute this software for any
5 257add31 2020-09-09 stsp * purpose with or without fee is hereby granted, provided that the above
6 257add31 2020-09-09 stsp * copyright notice and this permission notice appear in all copies.
8 257add31 2020-09-09 stsp * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 257add31 2020-09-09 stsp * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 257add31 2020-09-09 stsp * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 257add31 2020-09-09 stsp * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 257add31 2020-09-09 stsp * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 257add31 2020-09-09 stsp * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 257add31 2020-09-09 stsp * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17 257add31 2020-09-09 stsp #include <sys/types.h>
18 257add31 2020-09-09 stsp #include <sys/queue.h>
19 257add31 2020-09-09 stsp #include <sys/uio.h>
20 257add31 2020-09-09 stsp #include <sys/time.h>
22 257add31 2020-09-09 stsp #include <stdint.h>
23 257add31 2020-09-09 stsp #include <imsg.h>
24 257add31 2020-09-09 stsp #include <limits.h>
25 257add31 2020-09-09 stsp #include <signal.h>
26 257add31 2020-09-09 stsp #include <stdio.h>
27 257add31 2020-09-09 stsp #include <stdlib.h>
28 257add31 2020-09-09 stsp #include <string.h>
29 257add31 2020-09-09 stsp #include <sha1.h>
30 e12e0e21 2020-09-14 naddy #include <unistd.h>
31 257add31 2020-09-09 stsp #include <zlib.h>
33 257add31 2020-09-09 stsp #include "got_error.h"
34 257add31 2020-09-09 stsp #include "got_object.h"
35 5e082626 2020-09-24 stsp #include "got_path.h"
36 257add31 2020-09-09 stsp #include "got_repository.h"
38 257add31 2020-09-09 stsp #include "got_lib_delta.h"
39 257add31 2020-09-09 stsp #include "got_lib_object.h"
40 257add31 2020-09-09 stsp #include "got_lib_privsep.h"
42 257add31 2020-09-09 stsp #include "gotconfig.h"
44 257add31 2020-09-09 stsp /* parse.y */
45 257add31 2020-09-09 stsp static volatile sig_atomic_t sigint_received;
48 257add31 2020-09-09 stsp catch_sigint(int signo)
50 257add31 2020-09-09 stsp sigint_received = 1;
53 257add31 2020-09-09 stsp static const struct got_error *
54 257add31 2020-09-09 stsp make_repo_url(char **url, struct gotconfig_remote_repo *repo)
56 257add31 2020-09-09 stsp const struct got_error *err = NULL;
57 257add31 2020-09-09 stsp char *s = NULL, *p = NULL;
59 257add31 2020-09-09 stsp *url = NULL;
61 257add31 2020-09-09 stsp if (asprintf(&s, "%s://", repo->protocol) == -1)
62 257add31 2020-09-09 stsp return got_error_from_errno("asprintf");
64 257add31 2020-09-09 stsp if (repo->server) {
67 257add31 2020-09-09 stsp if (asprintf(&s, "%s%s", p, repo->server) == -1) {
68 257add31 2020-09-09 stsp err = got_error_from_errno("asprintf");
75 257add31 2020-09-09 stsp if (repo->port) {
78 257add31 2020-09-09 stsp if (asprintf(&s, "%s:%d", p, repo->port) == -1) {
79 257add31 2020-09-09 stsp err = got_error_from_errno("asprintf");
86 257add31 2020-09-09 stsp if (repo->repository) {
87 5e082626 2020-09-24 stsp char *repo_path = repo->repository;
88 5e082626 2020-09-24 stsp while (repo_path[0] == '/')
89 5e082626 2020-09-24 stsp repo_path++;
92 5e082626 2020-09-24 stsp if (asprintf(&s, "%s/%s", p, repo_path) == -1) {
93 257add31 2020-09-09 stsp err = got_error_from_errno("asprintf");
100 5e082626 2020-09-24 stsp got_path_strip_trailing_slashes(s);
107 257add31 2020-09-09 stsp return err;
110 257add31 2020-09-09 stsp static const struct got_error *
111 257add31 2020-09-09 stsp send_gotconfig_str(struct imsgbuf *ibuf, const char *value)
113 be96c417 2020-09-17 stsp size_t len = value ? strlen(value) : 0;
115 257add31 2020-09-09 stsp if (imsg_compose(ibuf, GOT_IMSG_GOTCONFIG_STR_VAL, 0, 0, -1,
116 257add31 2020-09-09 stsp value, len) == -1)
117 257add31 2020-09-09 stsp return got_error_from_errno("imsg_compose GOTCONFIG_STR_VAL");
119 257add31 2020-09-09 stsp return got_privsep_flush_imsg(ibuf);
122 257add31 2020-09-09 stsp static const struct got_error *
123 257add31 2020-09-09 stsp send_gotconfig_remotes(struct imsgbuf *ibuf,
124 257add31 2020-09-09 stsp struct gotconfig_remote_repo_list *remotes, int nremotes)
126 257add31 2020-09-09 stsp const struct got_error *err = NULL;
127 257add31 2020-09-09 stsp struct got_imsg_remotes iremotes;
128 257add31 2020-09-09 stsp struct gotconfig_remote_repo *repo;
129 257add31 2020-09-09 stsp char *url = NULL;
131 257add31 2020-09-09 stsp iremotes.nremotes = nremotes;
132 257add31 2020-09-09 stsp if (imsg_compose(ibuf, GOT_IMSG_GOTCONFIG_REMOTES, 0, 0, -1,
133 257add31 2020-09-09 stsp &iremotes, sizeof(iremotes)) == -1)
134 257add31 2020-09-09 stsp return got_error_from_errno("imsg_compose GOTCONFIG_REMOTES");
136 257add31 2020-09-09 stsp err = got_privsep_flush_imsg(ibuf);
137 257add31 2020-09-09 stsp imsg_clear(ibuf);
139 257add31 2020-09-09 stsp return err;
141 257add31 2020-09-09 stsp TAILQ_FOREACH(repo, remotes, entry) {
142 257add31 2020-09-09 stsp struct got_imsg_remote iremote;
143 257add31 2020-09-09 stsp size_t len = sizeof(iremote);
144 257add31 2020-09-09 stsp struct ibuf *wbuf;
145 b8adfa55 2020-09-25 stsp struct node_branch *branch;
146 b8adfa55 2020-09-25 stsp int nbranches = 0;
148 b8adfa55 2020-09-25 stsp branch = repo->branch;
149 b8adfa55 2020-09-25 stsp while (branch) {
150 b8adfa55 2020-09-25 stsp branch = branch->next;
151 b8adfa55 2020-09-25 stsp nbranches++;
154 b8adfa55 2020-09-25 stsp iremote.nbranches = nbranches;
155 257add31 2020-09-09 stsp iremote.mirror_references = repo->mirror_references;
156 0c8b29c5 2021-01-05 stsp iremote.fetch_all_branches = repo->fetch_all_branches;
158 257add31 2020-09-09 stsp iremote.name_len = strlen(repo->name);
159 257add31 2020-09-09 stsp len += iremote.name_len;
161 257add31 2020-09-09 stsp err = make_repo_url(&url, repo);
164 257add31 2020-09-09 stsp iremote.url_len = strlen(url);
165 257add31 2020-09-09 stsp len += iremote.url_len;
167 257add31 2020-09-09 stsp wbuf = imsg_create(ibuf, GOT_IMSG_GOTCONFIG_REMOTE, 0, 0, len);
168 257add31 2020-09-09 stsp if (wbuf == NULL) {
169 257add31 2020-09-09 stsp err = got_error_from_errno(
170 257add31 2020-09-09 stsp "imsg_create GOTCONFIG_REMOTE");
174 257add31 2020-09-09 stsp if (imsg_add(wbuf, &iremote, sizeof(iremote)) == -1) {
175 257add31 2020-09-09 stsp err = got_error_from_errno(
176 257add31 2020-09-09 stsp "imsg_add GOTCONFIG_REMOTE");
177 257add31 2020-09-09 stsp ibuf_free(wbuf);
181 257add31 2020-09-09 stsp if (imsg_add(wbuf, repo->name, iremote.name_len) == -1) {
182 257add31 2020-09-09 stsp err = got_error_from_errno(
183 257add31 2020-09-09 stsp "imsg_add GOTCONFIG_REMOTE");
184 257add31 2020-09-09 stsp ibuf_free(wbuf);
187 257add31 2020-09-09 stsp if (imsg_add(wbuf, url, iremote.url_len) == -1) {
188 257add31 2020-09-09 stsp err = got_error_from_errno(
189 257add31 2020-09-09 stsp "imsg_add GOTCONFIG_REMOTE");
190 257add31 2020-09-09 stsp ibuf_free(wbuf);
194 257add31 2020-09-09 stsp wbuf->fd = -1;
195 257add31 2020-09-09 stsp imsg_close(ibuf, wbuf);
196 257add31 2020-09-09 stsp err = got_privsep_flush_imsg(ibuf);
201 257add31 2020-09-09 stsp url = NULL;
203 b8adfa55 2020-09-25 stsp branch = repo->branch;
204 b8adfa55 2020-09-25 stsp while (branch) {
205 b8adfa55 2020-09-25 stsp err = send_gotconfig_str(ibuf, branch->branch_name);
208 b8adfa55 2020-09-25 stsp branch = branch->next;
213 257add31 2020-09-09 stsp return err;
216 257add31 2020-09-09 stsp static const struct got_error *
217 257add31 2020-09-09 stsp validate_config(struct gotconfig *gotconfig)
219 257add31 2020-09-09 stsp struct gotconfig_remote_repo *repo, *repo2;
220 257add31 2020-09-09 stsp static char msg[512];
222 257add31 2020-09-09 stsp TAILQ_FOREACH(repo, &gotconfig->remotes, entry) {
223 257add31 2020-09-09 stsp if (repo->name == NULL) {
224 257add31 2020-09-09 stsp return got_error_msg(GOT_ERR_PARSE_CONFIG,
225 257add31 2020-09-09 stsp "name required for remote repository");
228 257add31 2020-09-09 stsp TAILQ_FOREACH(repo2, &gotconfig->remotes, entry) {
229 257add31 2020-09-09 stsp if (repo == repo2 ||
230 257add31 2020-09-09 stsp strcmp(repo->name, repo2->name) != 0)
232 257add31 2020-09-09 stsp snprintf(msg, sizeof(msg),
233 257add31 2020-09-09 stsp "duplicate remote repository name '%s'",
234 257add31 2020-09-09 stsp repo->name);
235 257add31 2020-09-09 stsp return got_error_msg(GOT_ERR_PARSE_CONFIG, msg);
238 257add31 2020-09-09 stsp if (repo->server == NULL) {
239 257add31 2020-09-09 stsp snprintf(msg, sizeof(msg),
240 257add31 2020-09-09 stsp "server required for remote repository \"%s\"",
241 257add31 2020-09-09 stsp repo->name);
242 257add31 2020-09-09 stsp return got_error_msg(GOT_ERR_PARSE_CONFIG, msg);
245 257add31 2020-09-09 stsp if (repo->protocol == NULL) {
246 257add31 2020-09-09 stsp snprintf(msg, sizeof(msg),
247 257add31 2020-09-09 stsp "protocol required for remote repository \"%s\"",
248 257add31 2020-09-09 stsp repo->name);
249 257add31 2020-09-09 stsp return got_error_msg(GOT_ERR_PARSE_CONFIG, msg);
251 257add31 2020-09-09 stsp if (strcmp(repo->protocol, "ssh") != 0 &&
252 257add31 2020-09-09 stsp strcmp(repo->protocol, "git+ssh") != 0 &&
253 257add31 2020-09-09 stsp strcmp(repo->protocol, "git") != 0) {
254 257add31 2020-09-09 stsp snprintf(msg, sizeof(msg),"unknown protocol \"%s\" "
255 257add31 2020-09-09 stsp "for remote repository \"%s\"", repo->protocol,
256 257add31 2020-09-09 stsp repo->name);
257 257add31 2020-09-09 stsp return got_error_msg(GOT_ERR_PARSE_CONFIG, msg);
260 257add31 2020-09-09 stsp if (repo->repository == NULL) {
261 257add31 2020-09-09 stsp snprintf(msg, sizeof(msg),
262 257add31 2020-09-09 stsp "repository path required for remote "
263 257add31 2020-09-09 stsp "repository \"%s\"", repo->name);
264 257add31 2020-09-09 stsp return got_error_msg(GOT_ERR_PARSE_CONFIG, msg);
268 257add31 2020-09-09 stsp return NULL;
272 257add31 2020-09-09 stsp main(int argc, char *argv[])
274 257add31 2020-09-09 stsp const struct got_error *err = NULL;
275 257add31 2020-09-09 stsp struct imsgbuf ibuf;
276 53dfa00d 2020-09-10 stsp struct gotconfig *gotconfig = NULL;
277 257add31 2020-09-09 stsp size_t datalen;
278 257add31 2020-09-09 stsp const char *filename = "got.conf";
280 257add31 2020-09-09 stsp static int attached;
282 257add31 2020-09-09 stsp while (!attached)
285 257add31 2020-09-09 stsp signal(SIGINT, catch_sigint);
287 257add31 2020-09-09 stsp imsg_init(&ibuf, GOT_IMSG_FD_CHILD);
289 257add31 2020-09-09 stsp #ifndef PROFILE
290 257add31 2020-09-09 stsp /* revoke access to most system calls */
291 257add31 2020-09-09 stsp if (pledge("stdio recvfd", NULL) == -1) {
292 257add31 2020-09-09 stsp err = got_error_from_errno("pledge");
293 257add31 2020-09-09 stsp got_privsep_send_error(&ibuf, err);
298 257add31 2020-09-09 stsp if (argc > 1)
299 257add31 2020-09-09 stsp filename = argv[1];
302 257add31 2020-09-09 stsp struct imsg imsg;
304 257add31 2020-09-09 stsp memset(&imsg, 0, sizeof(imsg));
305 257add31 2020-09-09 stsp imsg.fd = -1;
307 257add31 2020-09-09 stsp if (sigint_received) {
308 257add31 2020-09-09 stsp err = got_error(GOT_ERR_CANCELLED);
312 257add31 2020-09-09 stsp err = got_privsep_recv_imsg(&imsg, &ibuf, 0);
314 257add31 2020-09-09 stsp if (err->code == GOT_ERR_PRIVSEP_PIPE)
315 257add31 2020-09-09 stsp err = NULL;
319 257add31 2020-09-09 stsp if (imsg.hdr.type == GOT_IMSG_STOP)
322 257add31 2020-09-09 stsp switch (imsg.hdr.type) {
323 257add31 2020-09-09 stsp case GOT_IMSG_GOTCONFIG_PARSE_REQUEST:
324 257add31 2020-09-09 stsp datalen = imsg.hdr.len - IMSG_HEADER_SIZE;
325 257add31 2020-09-09 stsp if (datalen != 0) {
326 257add31 2020-09-09 stsp err = got_error(GOT_ERR_PRIVSEP_LEN);
329 257add31 2020-09-09 stsp if (imsg.fd == -1){
330 257add31 2020-09-09 stsp err = got_error(GOT_ERR_PRIVSEP_NO_FD);
334 257add31 2020-09-09 stsp if (gotconfig)
335 257add31 2020-09-09 stsp gotconfig_free(gotconfig);
336 257add31 2020-09-09 stsp err = gotconfig_parse(&gotconfig, filename, &imsg.fd);
339 257add31 2020-09-09 stsp err = validate_config(gotconfig);
341 257add31 2020-09-09 stsp case GOT_IMSG_GOTCONFIG_AUTHOR_REQUEST:
342 257add31 2020-09-09 stsp if (gotconfig == NULL) {
343 257add31 2020-09-09 stsp err = got_error(GOT_ERR_PRIVSEP_MSG);
346 257add31 2020-09-09 stsp err = send_gotconfig_str(&ibuf,
347 257add31 2020-09-09 stsp gotconfig->author ? gotconfig->author : "");
349 257add31 2020-09-09 stsp case GOT_IMSG_GOTCONFIG_REMOTES_REQUEST:
350 257add31 2020-09-09 stsp if (gotconfig == NULL) {
351 257add31 2020-09-09 stsp err = got_error(GOT_ERR_PRIVSEP_MSG);
354 257add31 2020-09-09 stsp err = send_gotconfig_remotes(&ibuf,
355 257add31 2020-09-09 stsp &gotconfig->remotes, gotconfig->nremotes);
358 257add31 2020-09-09 stsp err = got_error(GOT_ERR_PRIVSEP_MSG);
362 257add31 2020-09-09 stsp if (imsg.fd != -1) {
363 257add31 2020-09-09 stsp if (close(imsg.fd) == -1 && err == NULL)
364 257add31 2020-09-09 stsp err = got_error_from_errno("close");
367 257add31 2020-09-09 stsp imsg_free(&imsg);
372 257add31 2020-09-09 stsp imsg_clear(&ibuf);
374 257add31 2020-09-09 stsp if (!sigint_received && err->code != GOT_ERR_PRIVSEP_PIPE) {
375 257add31 2020-09-09 stsp fprintf(stderr, "%s: %s\n", getprogname(), err->msg);
376 257add31 2020-09-09 stsp got_privsep_send_error(&ibuf, err);
379 257add31 2020-09-09 stsp if (close(GOT_IMSG_FD_CHILD) != 0 && err == NULL)
380 257add31 2020-09-09 stsp err = got_error_from_errno("close");
381 257add31 2020-09-09 stsp return err ? 1 : 0;