code: plan9front

Download patch

ref: 5a34cc86a807c4629491b38f0f401e885956edc0
parent: 4dd461e4400b20fbb218304e53a84d3704ae8304
author: cinap_lenrek <cinap_lenrek@felloff.net>
date: Fri Jun 4 13:47:26 EDT 2021

git: allow local repository directories as remote uri's

This is implemented by checking first if the uri is
a directory containing the .git/ subdirectory.
If this is the case, we fork git/serve serving the
repository on a pipe.

--- a/sys/src/cmd/git/proto.c
+++ b/sys/src/cmd/git/proto.c
@@ -285,19 +285,37 @@
 		snprint(cmd, sizeof(cmd), "git-%s-pack", direction);
 		dprint(1, "exec ssh '%s' '%s' %s\n", host, cmd, path);
 		execl("/bin/ssh", "ssh", host, cmd, path, nil);
-	}else{
-		close(pfd[0]);
-		c->type = ConnSsh;
-		c->rfd = pfd[1];
-		c->wfd = dup(pfd[1], -1);
+		sysfatal("exec: %r");
 	}
+	close(pfd[0]);
+	c->type = ConnSsh;
+	c->rfd = pfd[1];
+	c->wfd = dup(pfd[1], -1);
 	return 0;
 }
 
 static int
+githandshake(Conn *c, char *host, char *path, char *direction)
+{
+	char *p, *e, cmd[512];
+
+	p = cmd;
+	e = cmd + sizeof(cmd);
+	p = seprint(p, e - 1, "git-%s-pack %s", direction, path);
+	if(host != nil)
+		p = seprint(p + 1, e, "host=%s", host);
+	if(writepkt(c, cmd, p - cmd + 1) == -1){
+		fprint(2, "failed to write message\n");
+		closeconn(c);
+		return -1;
+	}
+	return 0;
+}
+
+static int
 dialhjgit(Conn *c, char *host, char *port, char *path, char *direction, int auth)
 {
-	char *ds, *p, *e, cmd[512];
+	char *ds;
 	int pid, pfd[2];
 
 	if((ds = netmkaddr(host, "tcp", port)) == nil)
@@ -317,30 +335,26 @@
 		else
 			execl("/bin/tlsclient", "tlsclient", ds, nil);
 		sysfatal("exec: %r");
-	}else{
-		close(pfd[0]);
-		p = cmd;
-		e = cmd + sizeof(cmd);
-		p = seprint(p, e - 1, "git-%s-pack %s", direction, path);
-		p = seprint(p + 1, e, "host=%s", host);
-		c->type = ConnGit9;
-		c->rfd = pfd[1];
-		c->wfd = dup(pfd[1], -1);
-		if(writepkt(c, cmd, p - cmd + 1) == -1){
-			fprint(2, "failed to write message\n");
-			close(c->rfd);
-			close(c->wfd);
-			return -1;
-		}
 	}
-	return 0;
+	close(pfd[0]);
+	c->type = ConnGit9;
+	c->rfd = pfd[1];
+	c->wfd = dup(pfd[1], -1);
+	return githandshake(c, host, path, direction);
 }
 
+void
+initconn(Conn *c, int rd, int wr)
+{
+	c->type = ConnGit;
+	c->rfd = rd;
+	c->wfd = wr;
+}
 
 static int
 dialgit(Conn *c, char *host, char *port, char *path, char *direction)
 {
-	char *ds, *p, *e, cmd[512];
+	char *ds;
 	int fd;
 
 	if((ds = netmkaddr(host, "tcp", port)) == nil)
@@ -349,29 +363,53 @@
 	fd = dial(ds, nil, nil, nil);
 	if(fd == -1)
 		return -1;
-	p = cmd;
-	e = cmd + sizeof(cmd);
-	p = seprint(p, e - 1, "git-%s-pack %s", direction, path);
-	p = seprint(p + 1, e, "host=%s", host);
 	c->type = ConnGit;
 	c->rfd = fd;
 	c->wfd = dup(fd, -1);
-	if(writepkt(c, cmd, p - cmd + 1) == -1){
-		fprint(2, "failed to write message\n");
-		close(fd);
-		return -1;
-	}
-	return 0;
+	return githandshake(c, host, path, direction);
 }
 
-void
-initconn(Conn *c, int rd, int wr)
+static int
+servelocal(Conn *c, char *path, char *direction)
 {
+	int pid, pfd[2];
+
+	if(pipe(pfd) == -1)
+		sysfatal("unable to open pipe: %r");
+	pid = fork();
+	if(pid == -1)
+		sysfatal("unable to fork");
+	if(pid == 0){
+		close(pfd[1]);
+		dup(pfd[0], 0);
+		dup(pfd[0], 1);
+		execl("/bin/git/serve", "serve", "-w", nil);
+		sysfatal("exec: %r");
+	}
+	close(pfd[0]);
 	c->type = ConnGit;
-	c->rfd = rd;
-	c->wfd = wr;
+	c->rfd = pfd[1];
+	c->wfd = dup(pfd[1], -1);
+	return githandshake(c, nil, path, direction);
 }
 
+static int
+localrepo(char *uri, char *path, int npath)
+{
+	int fd;
+
+	snprint(path, npath, "%s/.git/../", uri);
+	fd = open(path, OREAD);
+	if(fd < 0)
+		return -1;
+	if(fd2path(fd, path, npath) != 0){
+		close(fd);
+		return -1;
+	}
+	close(fd);
+	return 0;
+}
+
 int
 gitconnect(Conn *c, char *uri, char *direction)
 {
@@ -378,12 +416,16 @@
 	char proto[Nproto], host[Nhost], port[Nport];
 	char repo[Nrepo], path[Npath];
 
+	memset(c, 0, sizeof(Conn));
+	c->rfd = c->wfd = c->cfd = -1;
+
+	if(localrepo(uri, path, sizeof(path)) == 0)
+		return servelocal(c, path, direction);
+
 	if(parseuri(uri, proto, host, port, path, repo) == -1){
 		werrstr("bad uri %s", uri);
 		return -1;
 	}
-
-	memset(c, 0, sizeof(Conn));
 	if(strcmp(proto, "ssh") == 0)
 		return dialssh(c, host, port, path, direction);
 	else if(strcmp(proto, "git") == 0)