Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2006 Franck Bui-Huu |
| 3 | */ |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 4 | #include "cache.h" |
| 5 | #include "builtin.h" |
| 6 | #include "archive.h" |
| 7 | #include "pkt-line.h" |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 8 | #include "sideband.h" |
Jeff King | 1bc01ef | 2011-11-19 02:40:04 -0500 | [diff] [blame] | 9 | #include "run-command.h" |
Jeff King | 090fd4f | 2013-02-20 15:01:26 -0500 | [diff] [blame] | 10 | #include "argv-array.h" |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 11 | |
| 12 | static const char upload_archive_usage[] = |
Stephan Beyer | 1b1dd23 | 2008-07-13 15:36:15 +0200 | [diff] [blame] | 13 | "git upload-archive <repo>"; |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 14 | |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 15 | static const char deadchild[] = |
Stephan Beyer | 1b1dd23 | 2008-07-13 15:36:15 +0200 | [diff] [blame] | 16 | "git upload-archive: archiver died with error"; |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 17 | |
Rene Scharfe | 7f4d051 | 2008-07-25 12:41:23 +0200 | [diff] [blame] | 18 | #define MAX_ARGS (64) |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 19 | |
Jeff King | 1bc01ef | 2011-11-19 02:40:04 -0500 | [diff] [blame] | 20 | int cmd_upload_archive_writer(int argc, const char **argv, const char *prefix) |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 21 | { |
Jeff King | 090fd4f | 2013-02-20 15:01:26 -0500 | [diff] [blame] | 22 | struct argv_array sent_argv = ARGV_ARRAY_INIT; |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 23 | const char *arg_cmd = "argument "; |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 24 | |
Jeff King | 619b6c1 | 2017-05-30 01:13:43 -0400 | [diff] [blame] | 25 | if (argc != 2 || !strcmp(argv[1], "-h")) |
Junio C Hamano | f0c7fd4 | 2011-11-15 15:39:33 -0800 | [diff] [blame] | 26 | usage(upload_archive_usage); |
| 27 | |
Jeff King | 6379dd0 | 2013-02-20 15:00:59 -0500 | [diff] [blame] | 28 | if (!enter_repo(argv[1], 0)) |
| 29 | die("'%s' does not appear to be a git repository", argv[1]); |
Junio C Hamano | f0c7fd4 | 2011-11-15 15:39:33 -0800 | [diff] [blame] | 30 | |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 31 | /* put received options in sent_argv[] */ |
Jeff King | 090fd4f | 2013-02-20 15:01:26 -0500 | [diff] [blame] | 32 | argv_array_push(&sent_argv, "git-upload-archive"); |
| 33 | for (;;) { |
Jeff King | 74543a0 | 2013-02-20 15:02:57 -0500 | [diff] [blame] | 34 | char *buf = packet_read_line(0, NULL); |
| 35 | if (!buf) |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 36 | break; /* got a flush */ |
Jeff King | 090fd4f | 2013-02-20 15:01:26 -0500 | [diff] [blame] | 37 | if (sent_argv.argc > MAX_ARGS) |
| 38 | die("Too many options (>%d)", MAX_ARGS - 1); |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 39 | |
Christian Couder | 5955654 | 2013-11-30 21:55:40 +0100 | [diff] [blame] | 40 | if (!starts_with(buf, arg_cmd)) |
Jeff King | 090fd4f | 2013-02-20 15:01:26 -0500 | [diff] [blame] | 41 | die("'argument' token or flush expected"); |
| 42 | argv_array_push(&sent_argv, buf + strlen(arg_cmd)); |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 43 | } |
Junio C Hamano | f0c7fd4 | 2011-11-15 15:39:33 -0800 | [diff] [blame] | 44 | |
| 45 | /* parse all options sent by the client */ |
Nguyễn Thái Ngọc Duy | b612ee2 | 2018-08-13 18:14:35 +0200 | [diff] [blame] | 46 | return write_archive(sent_argv.argc, sent_argv.argv, prefix, |
| 47 | the_repository, NULL, 1); |
Franck Bui-Huu | 39345a2 | 2006-09-07 15:12:05 +0200 | [diff] [blame] | 48 | } |
| 49 | |
Tarmigan Casebolt | 28bea9e | 2009-11-14 13:33:13 -0800 | [diff] [blame] | 50 | __attribute__((format (printf, 1, 2))) |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 51 | static void error_clnt(const char *fmt, ...) |
| 52 | { |
Jeff King | 0cb9d6d | 2015-09-24 17:07:25 -0400 | [diff] [blame] | 53 | struct strbuf buf = STRBUF_INIT; |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 54 | va_list params; |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 55 | |
| 56 | va_start(params, fmt); |
Jeff King | 0cb9d6d | 2015-09-24 17:07:25 -0400 | [diff] [blame] | 57 | strbuf_vaddf(&buf, fmt, params); |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 58 | va_end(params); |
Jeff King | 0cb9d6d | 2015-09-24 17:07:25 -0400 | [diff] [blame] | 59 | send_sideband(1, 3, buf.buf, buf.len, LARGE_PACKET_MAX); |
| 60 | die("sent error to the client: %s", buf.buf); |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 61 | } |
| 62 | |
René Scharfe | 1b19fa4 | 2009-06-17 12:11:10 +0200 | [diff] [blame] | 63 | static ssize_t process_input(int child_fd, int band) |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 64 | { |
| 65 | char buf[16384]; |
| 66 | ssize_t sz = read(child_fd, buf, sizeof(buf)); |
| 67 | if (sz < 0) { |
Andy Whitcroft | 93d26e4 | 2007-01-08 15:58:08 +0000 | [diff] [blame] | 68 | if (errno != EAGAIN && errno != EINTR) |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 69 | error_clnt("read error: %s\n", strerror(errno)); |
René Scharfe | 1b19fa4 | 2009-06-17 12:11:10 +0200 | [diff] [blame] | 70 | return sz; |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 71 | } |
| 72 | send_sideband(1, band, buf, sz, LARGE_PACKET_MAX); |
René Scharfe | 1b19fa4 | 2009-06-17 12:11:10 +0200 | [diff] [blame] | 73 | return sz; |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 74 | } |
| 75 | |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 76 | int cmd_upload_archive(int argc, const char **argv, const char *prefix) |
| 77 | { |
Jeff King | 1bc01ef | 2011-11-19 02:40:04 -0500 | [diff] [blame] | 78 | struct child_process writer = { argv }; |
| 79 | |
Jeff King | 619b6c1 | 2017-05-30 01:13:43 -0400 | [diff] [blame] | 80 | if (argc == 2 && !strcmp(argv[1], "-h")) |
| 81 | usage(upload_archive_usage); |
| 82 | |
Junio C Hamano | f0c7fd4 | 2011-11-15 15:39:33 -0800 | [diff] [blame] | 83 | /* |
| 84 | * Set up sideband subprocess. |
| 85 | * |
| 86 | * We (parent) monitor and read from child, sending its fd#1 and fd#2 |
| 87 | * multiplexed out to our fd#1. If the child dies, we tell the other |
| 88 | * end over channel #3. |
| 89 | */ |
Jeff King | 1bc01ef | 2011-11-19 02:40:04 -0500 | [diff] [blame] | 90 | argv[0] = "upload-archive--writer"; |
| 91 | writer.out = writer.err = -1; |
| 92 | writer.git_cmd = 1; |
| 93 | if (start_command(&writer)) { |
Junio C Hamano | f0c7fd4 | 2011-11-15 15:39:33 -0800 | [diff] [blame] | 94 | int err = errno; |
Lars Schneider | 81c634e | 2016-10-16 16:20:29 -0700 | [diff] [blame] | 95 | packet_write_fmt(1, "NACK unable to spawn subprocess\n"); |
Junio C Hamano | f0c7fd4 | 2011-11-15 15:39:33 -0800 | [diff] [blame] | 96 | die("upload-archive: %s", strerror(err)); |
| 97 | } |
Junio C Hamano | f0c7fd4 | 2011-11-15 15:39:33 -0800 | [diff] [blame] | 98 | |
Lars Schneider | 81c634e | 2016-10-16 16:20:29 -0700 | [diff] [blame] | 99 | packet_write_fmt(1, "ACK\n"); |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 100 | packet_flush(1); |
| 101 | |
| 102 | while (1) { |
| 103 | struct pollfd pfd[2]; |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 104 | |
Jeff King | 1bc01ef | 2011-11-19 02:40:04 -0500 | [diff] [blame] | 105 | pfd[0].fd = writer.out; |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 106 | pfd[0].events = POLLIN; |
Jeff King | 1bc01ef | 2011-11-19 02:40:04 -0500 | [diff] [blame] | 107 | pfd[1].fd = writer.err; |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 108 | pfd[1].events = POLLIN; |
| 109 | if (poll(pfd, 2, -1) < 0) { |
| 110 | if (errno != EINTR) { |
Nguyễn Thái Ngọc Duy | 17bef17 | 2016-05-08 16:47:33 +0700 | [diff] [blame] | 111 | error_errno("poll failed resuming"); |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 112 | sleep(1); |
| 113 | } |
| 114 | continue; |
| 115 | } |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 116 | if (pfd[1].revents & POLLIN) |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 117 | /* Status stream ready */ |
Nicolas Pitre | 6b59f51 | 2009-11-11 17:24:42 -0500 | [diff] [blame] | 118 | if (process_input(pfd[1].fd, 2)) |
| 119 | continue; |
| 120 | if (pfd[0].revents & POLLIN) |
| 121 | /* Data stream ready */ |
| 122 | if (process_input(pfd[0].fd, 1)) |
| 123 | continue; |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 124 | |
Jeff King | 1bc01ef | 2011-11-19 02:40:04 -0500 | [diff] [blame] | 125 | if (finish_command(&writer)) |
Junio C Hamano | d3788e1 | 2006-09-12 00:26:57 -0700 | [diff] [blame] | 126 | error_clnt("%s", deadchild); |
Junio C Hamano | 23d6d11 | 2006-09-10 03:33:34 -0700 | [diff] [blame] | 127 | packet_flush(1); |
| 128 | break; |
| 129 | } |
| 130 | return 0; |
| 131 | } |