about summary refs log tree commit homepage
path: root/lib/PublicInbox/Git.pm
diff options
context:
space:
mode:
authorEric Wong <e@80x24.org>2021-09-29 21:25:20 +0000
committerEric Wong <e@80x24.org>2021-09-29 21:37:46 +0000
commit56ab6cba3ee89fdf47aa4e088d3c8a2d0bdab8d7 (patch)
treeb66c91757d62abee75a70f2cdf2c03bd85aea913 /lib/PublicInbox/Git.pm
parentf7e29ec64a0cbba63c4df244e916848f75bc9346 (diff)
downloadpublic-inbox-56ab6cba3ee89fdf47aa4e088d3c8a2d0bdab8d7.tar.gz
Long pathnames are difficult to read and distinguish in ps(1)
output.  Deep paths can also slow down pathname resolution
when dealing with loose objects, so we put "cat-file --batch"
deeper into the directory tree.

Since v2 processes are in the form of $INBOXDIR/all.git, keep
the basename of $INBOXDIR in --git-dir= so it's easy to
distinguish between processes just by looking at ps(1).

While "git -C" also exists, it's only present in git 1.8.5+.
We also need to keep in mind the "directory" pointed to by
--git-dir= need not be a directory (nor a symlink pointing
to one).

This reduces pathname resolution overhead for v1 and v2 inbox
git processes, but unfortunately not for extindex since that
needs to store alternates as absolute paths.
Diffstat (limited to 'lib/PublicInbox/Git.pm')
-rw-r--r--lib/PublicInbox/Git.pm18
1 files changed, 12 insertions, 6 deletions
diff --git a/lib/PublicInbox/Git.pm b/lib/PublicInbox/Git.pm
index d5b1d39d..97c39aad 100644
--- a/lib/PublicInbox/Git.pm
+++ b/lib/PublicInbox/Git.pm
@@ -62,6 +62,8 @@ sub git_quote ($) {
 
 sub new {
         my ($class, $git_dir) = @_;
+        $git_dir =~ tr!/!/!s;
+        $git_dir =~ s!/*\z!!s;
         # may contain {-tmp} field for File::Temp::Dir
         bless { git_dir => $git_dir, alt_st => '', -git_path => {} }, $class
 }
@@ -110,18 +112,22 @@ sub _bidi_pipe {
                 }
                 return;
         }
-        my ($out_r, $out_w);
-        pipe($out_r, $out_w) or $self->fail("pipe failed: $!");
-        my @cmd = (qw(git), "--git-dir=$self->{git_dir}",
+        pipe(my ($out_r, $out_w)) or $self->fail("pipe failed: $!");
+        my $rdr = { 0 => $out_r };
+        my $gd = $self->{git_dir};
+        if ($gd =~ s!/([^/]+/[^/]+)\z!/!) {
+                $rdr->{-C} = $gd;
+                $gd = $1;
+        }
+        my @cmd = (qw(git), "--git-dir=$gd",
                         qw(-c core.abbrev=40 cat-file), $batch);
-        my $redir = { 0 => $out_r };
         if ($err) {
                 my $id = "git.$self->{git_dir}$batch.err";
                 my $fh = tmpfile($id) or $self->fail("tmpfile($id): $!");
                 $self->{$err} = $fh;
-                $redir->{2} = $fh;
+                $rdr->{2} = $fh;
         }
-        my ($in_r, $p) = popen_rd(\@cmd, undef, $redir);
+        my ($in_r, $p) = popen_rd(\@cmd, undef, $rdr);
         $self->{$pid} = $p;
         $self->{"$pid.owner"} = $$;
         $out_w->autoflush(1);