You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
116 lines
2.3 KiB
116 lines
2.3 KiB
#include <string.h> |
|
#include <sys/types.h> |
|
#include <sys/socket.h> |
|
|
|
#include "rsh.h" |
|
#include "quote.h" |
|
#include "cache.h" |
|
|
|
#define COMMAND_SIZE 4096 |
|
|
|
/* |
|
* Append a string to a string buffer, with or without shell quoting. |
|
* Return true if the buffer overflowed. |
|
*/ |
|
static int add_to_string(char **ptrp, int *sizep, const char *str, int quote) |
|
{ |
|
char *p = *ptrp; |
|
int size = *sizep; |
|
int oc; |
|
int err = 0; |
|
|
|
if ( quote ) { |
|
oc = sq_quote_buf(p, size, str); |
|
} else { |
|
oc = strlen(str); |
|
memcpy(p, str, (oc >= size) ? size-1 : oc); |
|
} |
|
|
|
if ( oc >= size ) { |
|
err = 1; |
|
oc = size-1; |
|
} |
|
|
|
*ptrp += oc; |
|
**ptrp = '\0'; |
|
*sizep -= oc; |
|
return err; |
|
} |
|
|
|
int setup_connection(int *fd_in, int *fd_out, const char *remote_prog, |
|
char *url, int rmt_argc, char **rmt_argv) |
|
{ |
|
char *host; |
|
char *path; |
|
int sv[2]; |
|
char command[COMMAND_SIZE]; |
|
char *posn; |
|
int sizen; |
|
int of; |
|
int i; |
|
pid_t pid; |
|
|
|
if (!strcmp(url, "-")) { |
|
*fd_in = 0; |
|
*fd_out = 1; |
|
return 0; |
|
} |
|
|
|
host = strstr(url, "//"); |
|
if (host) { |
|
host += 2; |
|
path = strchr(host, '/'); |
|
} else { |
|
host = url; |
|
path = strchr(host, ':'); |
|
if (path) |
|
*(path++) = '\0'; |
|
} |
|
if (!path) { |
|
return error("Bad URL: %s", url); |
|
} |
|
/* $GIT_RSH <host> "env GIT_DIR=<path> <remote_prog> <args...>" */ |
|
sizen = COMMAND_SIZE; |
|
posn = command; |
|
of = 0; |
|
of |= add_to_string(&posn, &sizen, "env ", 0); |
|
of |= add_to_string(&posn, &sizen, GIT_DIR_ENVIRONMENT "=", 0); |
|
of |= add_to_string(&posn, &sizen, path, 1); |
|
of |= add_to_string(&posn, &sizen, " ", 0); |
|
of |= add_to_string(&posn, &sizen, remote_prog, 1); |
|
|
|
for ( i = 0 ; i < rmt_argc ; i++ ) { |
|
of |= add_to_string(&posn, &sizen, " ", 0); |
|
of |= add_to_string(&posn, &sizen, rmt_argv[i], 1); |
|
} |
|
|
|
of |= add_to_string(&posn, &sizen, " -", 0); |
|
|
|
if ( of ) |
|
return error("Command line too long"); |
|
|
|
if (socketpair(AF_UNIX, SOCK_STREAM, 0, sv)) |
|
return error("Couldn't create socket"); |
|
|
|
pid = fork(); |
|
if (pid < 0) |
|
return error("Couldn't fork"); |
|
if (!pid) { |
|
const char *ssh, *ssh_basename; |
|
ssh = getenv("GIT_SSH"); |
|
if (!ssh) ssh = "ssh"; |
|
ssh_basename = strrchr(ssh, '/'); |
|
if (!ssh_basename) |
|
ssh_basename = ssh; |
|
else |
|
ssh_basename++; |
|
close(sv[1]); |
|
dup2(sv[0], 0); |
|
dup2(sv[0], 1); |
|
execlp(ssh, ssh_basename, host, command, NULL); |
|
} |
|
close(sv[0]); |
|
*fd_in = sv[1]; |
|
*fd_out = sv[1]; |
|
return 0; |
|
}
|
|
|