2 * Copyright (c) 2022 Stefan Sperling <stsp@openbsd.org>
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.
17 #include "got_compat.h"
19 #define GOTD_UNIX_SOCKET "/var/run/gotd.sock"
20 #define GOTD_UNIX_SOCKET_BACKLOG 10
21 #define GOTD_USER "_gotd"
22 #define GOTD_CONF_PATH "/etc/gotd.conf"
23 #ifndef GOTD_EMPTY_PATH
24 #define GOTD_EMPTY_PATH "/var/empty"
27 #ifndef GOT_LIBEXECDIR
28 #define GOT_LIBEXECDIR /usr/libexec
31 #define GOTD_STRINGIFY(x) #x
32 #define GOTD_STRINGVAL(x) GOTD_STRINGIFY(x)
34 #define GOTD_PROG_NOTIFY_EMAIL got-notify-email
35 #define GOTD_PROG_NOTIFY_HTTP got-notify-http
37 #define GOTD_PATH_PROG_NOTIFY_EMAIL \
38 GOTD_STRINGVAL(GOT_LIBEXECDIR) "/" \
39 GOTD_STRINGVAL(GOTD_PROG_NOTIFY_EMAIL)
40 #define GOTD_PATH_PROG_NOTIFY_HTTP \
41 GOTD_STRINGVAL(GOT_LIBEXECDIR) "/" \
42 GOTD_STRINGVAL(GOTD_PROG_NOTIFY_HTTP)
44 #define GOTD_MAXCLIENTS 1024
45 #define GOTD_MAX_CONN_PER_UID 4
46 #define GOTD_FD_RESERVE 5
47 #define GOTD_FD_NEEDED 6
48 #define GOTD_FILENO_MSG_PIPE 3
50 #define GOTD_DEFAULT_REQUEST_TIMEOUT 3600
52 /* Client hash tables need some extra room. */
53 #define GOTD_CLIENT_TABLE_SIZE (GOTD_MAXCLIENTS * 4)
70 void (*handler
)(int, short, void *);
77 GOTD_ACCESS_PERMITTED
= 1,
81 struct gotd_access_rule
{
82 STAILQ_ENTRY(gotd_access_rule
) entry
;
84 enum gotd_access access
;
87 #define GOTD_AUTH_READ 0x1
88 #define GOTD_AUTH_WRITE 0x2
92 STAILQ_HEAD(gotd_access_rule_list
, gotd_access_rule
);
94 enum gotd_notification_target_type
{
95 GOTD_NOTIFICATION_VIA_EMAIL
,
96 GOTD_NOTIFICATION_VIA_HTTP
99 struct gotd_notification_target
{
100 STAILQ_ENTRY(gotd_notification_target
) entry
;
102 enum gotd_notification_target_type type
;
121 STAILQ_HEAD(gotd_notification_targets
, gotd_notification_target
);
124 TAILQ_ENTRY(gotd_repo
) entry
;
129 struct gotd_access_rule_list rules
;
130 struct got_pathlist_head protected_tag_namespaces
;
131 struct got_pathlist_head protected_branch_namespaces
;
132 struct got_pathlist_head protected_branches
;
134 struct got_pathlist_head notification_refs
;
135 struct got_pathlist_head notification_ref_namespaces
;
136 struct gotd_notification_targets notification_targets
;
138 TAILQ_HEAD(gotd_repolist
, gotd_repo
);
140 struct gotd_client_capability
{
145 struct gotd_object_id_array
{
146 struct got_object_id
**ids
;
151 struct gotd_uid_connection_limit
{
156 struct gotd_child_proc
;
160 char unix_socket_path
[PATH_MAX
];
162 struct gotd_repolist repos
;
164 struct gotd_child_proc
*listen_proc
;
165 struct gotd_child_proc
*notify_proc
;
166 int notifications_enabled
;
167 struct timeval request_timeout
;
168 struct timeval auth_timeout
;
169 struct gotd_uid_connection_limit
*connection_limits
;
170 size_t nconnection_limits
;
173 const char *confpath
;
178 enum gotd_imsg_type
{
179 /* An error occured while processing a request. */
182 /* Commands used by gotctl(8). */
185 GOTD_IMSG_INFO_CLIENT
,
188 /* Request a list of references. */
190 GOTD_IMSG_LIST_REFS_INTERNAL
,
197 /* Git protocol capabilities. */
198 GOTD_IMSG_CAPABILITIES
,
199 GOTD_IMSG_CAPABILITY
,
201 /* Git protocol chatter. */
202 GOTD_IMSG_WANT
, /* The client wants an object. */
203 GOTD_IMSG_HAVE
, /* The client has an object. */
204 GOTD_IMSG_ACK
, /* The server has an object or a reference. */
205 GOTD_IMSG_NAK
, /* The server does not have an object/ref. */
206 GOTD_IMSG_REF_UPDATE
, /* The client wants to update a reference. */
207 GOTD_IMSG_REF_DELETE
, /* The client wants to delete a reference. */
208 GOTD_IMSG_FLUSH
, /* The client sent a flush packet. */
209 GOTD_IMSG_DONE
, /* The client is done chatting. */
211 /* Sending or receiving a pack file. */
212 GOTD_IMSG_SEND_PACKFILE
, /* The server is sending a pack file. */
213 GOTD_IMSG_RECV_PACKFILE
, /* The server is receiving a pack file. */
214 GOTD_IMSG_PACKIDX_FILE
, /* Temporary file handle for new pack index. */
215 GOTD_IMSG_PACKFILE_PIPE
, /* Pipe to send/receive a pack file stream. */
216 GOTD_IMSG_PACKFILE_PROGRESS
, /* Progress reporting. */
217 GOTD_IMSG_PACKFILE_READY
, /* Pack file is ready to be sent. */
218 GOTD_IMSG_PACKFILE_STATUS
, /* Received pack success/failure status. */
219 GOTD_IMSG_PACKFILE_INSTALL
, /* Received pack file can be installed. */
220 GOTD_IMSG_PACKFILE_DONE
, /* Pack file has been sent/received. */
222 /* Reference updates. */
223 GOTD_IMSG_REF_UPDATES_START
, /* Ref updates starting. */
224 GOTD_IMSG_REF_UPDATE_OK
, /* Update went OK. */
225 GOTD_IMSG_REF_UPDATE_NG
, /* Update was not good. */
226 GOTD_IMSG_REFS_UPDATED
, /* The server proccessed all ref updates. */
228 /* Client connections. */
229 GOTD_IMSG_DISCONNECT
,
232 /* Child process management. */
233 GOTD_IMSG_CLIENT_SESSION_READY
,
234 GOTD_IMSG_REPO_CHILD_READY
,
235 GOTD_IMSG_CONNECT_REPO_CHILD
,
237 /* Auth child process. */
238 GOTD_IMSG_AUTHENTICATE
,
239 GOTD_IMSG_ACCESS_GRANTED
,
241 /* Notify child process. */
242 GOTD_IMSG_CONNECT_NOTIFIER
,
243 GOTD_IMSG_CONNECT_SESSION
,
245 GOTD_IMSG_NOTIFICATION_SENT
248 /* Structure for GOTD_IMSG_ERROR. */
249 struct gotd_imsg_error
{
250 int code
; /* an error code from got_error.h */
251 int errno_code
; /* in case code equals GOT_ERR_ERRNO */
253 char msg
[GOT_ERR_MAX_MSG_SIZE
];
254 } __attribute__((__packed__
));
256 /* Structure for GOTD_IMSG_INFO. */
257 struct gotd_imsg_info
{
263 /* Followed by nrepos GOTD_IMSG_INFO_REPO messages. */
264 /* Followed by nclients GOTD_IMSG_INFO_CLIENT messages. */
267 /* Structure for GOTD_IMSG_INFO_REPO. */
268 struct gotd_imsg_info_repo
{
269 char repo_name
[NAME_MAX
];
270 char repo_path
[PATH_MAX
];
273 /* Structure for GOTD_IMSG_INFO_CLIENT */
274 struct gotd_imsg_info_client
{
277 char repo_name
[NAME_MAX
];
279 pid_t session_child_pid
;
280 pid_t repo_child_pid
;
283 /* Structure for GOTD_IMSG_LIST_REFS. */
284 struct gotd_imsg_list_refs
{
285 char repo_name
[NAME_MAX
];
286 int client_is_reading
; /* 1 if reading, 0 if writing */
289 /* Structure for GOTD_IMSG_REFLIST. */
290 struct gotd_imsg_reflist
{
293 /* Followed by nrefs times of gotd_imsg_ref/gotd_imsg_symref data. */
294 } __attribute__((__packed__
));
296 /* Structure for GOTD_IMSG_REF data. */
297 struct gotd_imsg_ref
{
298 uint8_t id
[SHA1_DIGEST_LENGTH
];
300 /* Followed by name_len data bytes. */
301 } __attribute__((__packed__
));
303 /* Structure for GOTD_IMSG_SYMREF data. */
304 struct gotd_imsg_symref
{
307 uint8_t target_id
[SHA1_DIGEST_LENGTH
];
310 * Followed by name_len + target_len data bytes.
312 } __attribute__((__packed__
));
314 /* Structure for GOTD_IMSG_CAPABILITIES data. */
315 struct gotd_imsg_capabilities
{
316 size_t ncapabilities
;
319 * Followed by ncapabilities * GOTD_IMSG_CAPABILITY.
321 } __attribute__((__packed__
));
323 /* Structure for GOTD_IMSG_CAPABILITY data. */
324 struct gotd_imsg_capability
{
329 * Followed by key_len + value_len data bytes.
331 } __attribute__((__packed__
));
333 /* Structure for GOTD_IMSG_WANT data. */
334 struct gotd_imsg_want
{
335 uint8_t object_id
[SHA1_DIGEST_LENGTH
];
336 } __attribute__((__packed__
));
338 /* Structure for GOTD_IMSG_HAVE data. */
339 struct gotd_imsg_have
{
340 uint8_t object_id
[SHA1_DIGEST_LENGTH
];
341 } __attribute__((__packed__
));
343 /* Structure for GOTD_IMSG_ACK data. */
344 struct gotd_imsg_ack
{
345 uint8_t object_id
[SHA1_DIGEST_LENGTH
];
346 } __attribute__((__packed__
));
348 /* Structure for GOTD_IMSG_NAK data. */
349 struct gotd_imsg_nak
{
350 uint8_t object_id
[SHA1_DIGEST_LENGTH
];
351 } __attribute__((__packed__
));
353 /* Structure for GOTD_IMSG_PACKFILE_STATUS data. */
354 struct gotd_imsg_packfile_status
{
357 /* Followed by reason_len data bytes. */
358 } __attribute__((__packed__
));
361 /* Structure for GOTD_IMSG_REF_UPDATE data. */
362 struct gotd_imsg_ref_update
{
363 uint8_t old_id
[SHA1_DIGEST_LENGTH
];
364 uint8_t new_id
[SHA1_DIGEST_LENGTH
];
369 /* Followed by name_len data bytes. */
370 } __attribute__((__packed__
));
372 /* Structure for GOTD_IMSG_REF_UPDATES_START data. */
373 struct gotd_imsg_ref_updates_start
{
376 /* Followed by nref_updates GOT_IMSG_REF_UPDATE_OK/NG messages. */
379 /* Structure for GOTD_IMSG_REF_UPDATE_OK data. */
380 struct gotd_imsg_ref_update_ok
{
381 uint8_t old_id
[SHA1_DIGEST_LENGTH
];
382 uint8_t new_id
[SHA1_DIGEST_LENGTH
];
386 /* Followed by name_len data bytes. */
387 } __attribute__((__packed__
));
389 /* Structure for GOTD_IMSG_REF_UPDATE_NG data. */
390 struct gotd_imsg_ref_update_ng
{
391 uint8_t old_id
[SHA1_DIGEST_LENGTH
];
392 uint8_t new_id
[SHA1_DIGEST_LENGTH
];
396 /* Followed by name_len + reason_len data bytes. */
397 } __attribute__((__packed__
));
399 /* Structure for GOTD_IMSG_SEND_PACKFILE data. */
400 struct gotd_imsg_send_packfile
{
403 /* delta cache file is sent as a file descriptor */
405 /* followed by two GOTD_IMSG_PACKFILE_PIPE messages */
408 /* Structure for GOTD_IMSG_RECV_PACKFILE data. */
409 struct gotd_imsg_recv_packfile
{
412 /* pack destination temp file is sent as a file descriptor */
416 * Structure for GOTD_IMSG_PACKFILE_PROGRESS and
417 * GOTD_IMSG_PACKFILE_READY data.
419 struct gotd_imsg_packfile_progress
{
430 /* Structure for GOTD_IMSG_PACKFILE_INSTALL. */
431 struct gotd_imsg_packfile_install
{
432 uint8_t pack_sha1
[SHA1_DIGEST_LENGTH
];
435 /* Structure for GOTD_IMSG_DISCONNECT data. */
436 struct gotd_imsg_disconnect
{
440 /* Structure for GOTD_IMSG_CONNECT. */
441 struct gotd_imsg_connect
{
447 /* Followed by username_len data bytes. */
450 /* Structure for GOTD_IMSG_CONNECT_REPO_CHILD. */
451 struct gotd_imsg_connect_repo_child
{
452 enum gotd_procid proc_id
;
454 /* repo child imsg pipe is passed via imsg fd */
457 /* Structure for GOTD_IMSG_AUTHENTICATE. */
458 struct gotd_imsg_auth
{
465 /* Structures for GOTD_IMSG_NOTIFY. */
466 enum gotd_notification_action
{
467 GOTD_NOTIF_ACTION_CREATED
,
468 GOTD_NOTIF_ACTION_REMOVED
,
469 GOTD_NOTIF_ACTION_CHANGED
471 /* IMSG_NOTIFY session <-> repo_write */
472 struct gotd_imsg_notification_content
{
473 enum gotd_notification_action action
;
474 uint8_t old_id
[SHA1_DIGEST_LENGTH
];
475 uint8_t new_id
[SHA1_DIGEST_LENGTH
];
477 /* Followed by refname_len data bytes. */
479 /* IMSG_NOTIFY session -> notify*/
480 struct gotd_imsg_notify
{
481 char repo_name
[NAME_MAX
];
482 char subject_line
[64];
485 int enter_chroot(const char *);
486 int parse_config(const char *, enum gotd_procid
, struct gotd
*);
487 struct gotd_repo
*gotd_find_repo_by_name(const char *, struct gotd_repolist
*);
488 struct gotd_repo
*gotd_find_repo_by_path(const char *, struct gotd
*);
489 struct gotd_uid_connection_limit
*gotd_find_uid_connection_limit(
490 struct gotd_uid_connection_limit
*limits
, size_t nlimits
, uid_t uid
);
491 int gotd_parseuid(const char *s
, uid_t
*uid
);
492 const struct got_error
*gotd_parse_url(char **, char **, char **,
493 char **, const char *);
496 const struct got_error
*gotd_imsg_flush(struct imsgbuf
*);
497 const struct got_error
*gotd_imsg_recv(struct imsg
*, struct imsgbuf
*, size_t);
498 const struct got_error
*gotd_imsg_poll_recv(struct imsg
*, struct imsgbuf
*,
500 const struct got_error
*gotd_imsg_recv_error(uint32_t *client_id
,
502 int gotd_imsg_send_error(struct imsgbuf
*ibuf
, uint32_t, uint32_t,
503 const struct got_error
*);
504 int gotd_imsg_send_error_event(struct gotd_imsgev
*, uint32_t, uint32_t,
505 const struct got_error
*);
506 void gotd_imsg_event_add(struct gotd_imsgev
*);
507 int gotd_imsg_compose_event(struct gotd_imsgev
*, uint16_t, uint32_t, int,
509 int gotd_imsg_forward(struct gotd_imsgev
*, struct imsg
*, int);
511 void gotd_imsg_send_ack(struct got_object_id
*, struct imsgbuf
*,
513 void gotd_imsg_send_nak(struct got_object_id
*, struct imsgbuf
*,