6 #define COMMAND_DIR "git-shell-commands"
8 static int do_generic_cmd(const char *me, char *arg)
10 const char *my_argv[4];
13 if (!arg || !(arg = sq_dequote(arg)))
15 if (prefixcmp(me, "git-"))
22 return execv_git_cmd(my_argv);
25 static int do_cvs_cmd(const char *me, char *arg)
27 const char *cvsserver_argv[3] = {
28 "cvsserver", "server", NULL
31 if (!arg || strcmp(arg, "server"))
32 die("git-cvsserver only handles server: %s", arg);
35 return execv_git_cmd(cvsserver_argv);
38 static int is_valid_cmd_name(const char *cmd)
40 /* Test command contains no . or / characters */
41 return cmd[strcspn(cmd, "./")] == '\0';
44 static char *make_cmd(const char *prog)
46 char *prefix = xmalloc((strlen(prog) + strlen(COMMAND_DIR) + 2));
47 strcpy(prefix, COMMAND_DIR);
53 static void cd_to_homedir(void)
55 const char *home = getenv("HOME");
57 die("could not determine user's home directory; HOME is unset");
58 if (chdir(home) == -1)
59 die("could not chdir to user's home directory");
62 static struct commands {
64 int (*exec)(const char *me, char *arg);
66 { "git-receive-pack", do_generic_cmd },
67 { "git-upload-pack", do_generic_cmd },
68 { "git-upload-archive", do_generic_cmd },
69 { "cvs", do_cvs_cmd },
73 int main(int argc, char **argv)
76 const char **user_argv;
81 * Always open file descriptors 0/1/2 to avoid clobbering files
82 * in die(). It also avoids not messing up when the pipes are
83 * dup'ed onto stdin/stdout/stderr in the child processes we spawn.
85 devnull_fd = open("/dev/null", O_RDWR);
86 while (devnull_fd >= 0 && devnull_fd <= 2)
87 devnull_fd = dup(devnull_fd);
89 die_errno("opening /dev/null failed");
93 * Special hack to pretend to be a CVS server
95 if (argc == 2 && !strcmp(argv[1], "cvs server"))
99 * We do not accept anything but "-c" followed by "cmd arg",
100 * where "cmd" is a very limited subset of git commands.
102 else if (argc != 3 || strcmp(argv[1], "-c"))
103 die("What do you think I am? A shell?");
105 prog = xstrdup(argv[2]);
106 if (!strncmp(prog, "git", 3) && isspace(prog[3]))
107 /* Accept "git foo" as if the caller said "git-foo". */
110 for (cmd = cmd_list ; cmd->name ; cmd++) {
111 int len = strlen(cmd->name);
113 if (strncmp(cmd->name, prog, len))
121 arg = prog + len + 1;
126 exit(cmd->exec(cmd->name, arg));
130 if (split_cmdline(prog, &user_argv) != -1) {
131 if (is_valid_cmd_name(user_argv[0])) {
132 prog = make_cmd(user_argv[0]);
134 execv(user_argv[0], (char *const *) user_argv);
138 die("unrecognized command '%s'", argv[2]);
141 die("invalid command format '%s'", argv[2]);