diff options
author | Sitaram Chamarty <sitaram@atc.tcs.com> | 2017-08-11 19:01:11 +0530 |
---|---|---|
committer | Sitaram Chamarty <sitaram@atc.tcs.com> | 2017-08-11 19:01:14 +0530 |
commit | 41b7885b77cfe992ad3c96d0b021ece51ce1b3e3 (patch) | |
tree | 19206e682a4c8855edde050e50ab020100d28639 /src | |
parent | remove a couple of redundant/useless lines (diff) | |
download | gitolite-gentoo-41b7885b77cfe992ad3c96d0b021ece51ce1b3e3.tar.gz gitolite-gentoo-41b7885b77cfe992ad3c96d0b021ece51ce1b3e3.tar.bz2 gitolite-gentoo-41b7885b77cfe992ad3c96d0b021ece51ce1b3e3.zip |
reduce stat() and other calls for huge installs
Fedora has 42000 or so repos, and a 'gitolite compile' was taking too
long.
This set of changes reduces the number of stat() and other calls from 9
to 2 per unchanged, existing repo. (The expectation is that only a few
repos' rules are being changed each time, so this helps somewhat
optimise the others not to take the same amount of time).
Diffstat (limited to 'src')
-rw-r--r-- | src/lib/Gitolite/Conf/Store.pm | 35 |
1 files changed, 28 insertions, 7 deletions
diff --git a/src/lib/Gitolite/Conf/Store.pm b/src/lib/Gitolite/Conf/Store.pm index c7f9ab5..6809f67 100644 --- a/src/lib/Gitolite/Conf/Store.pm +++ b/src/lib/Gitolite/Conf/Store.pm @@ -25,6 +25,8 @@ use Exporter 'import'; use Data::Dumper; $Data::Dumper::Indent = 1; $Data::Dumper::Sortkeys = 1; +use Fcntl; +use GDBM_File; use Gitolite::Rc; use Gitolite::Common; @@ -41,6 +43,11 @@ my %groups; my %configs; my %split_conf; +# reduce the number of unnecessary stat() calls for installations like Fedora, +# which have (as of 2017-08) about 42000 repos. Each compile will only +# *really* change a few repos so this helps us not to touch the others. +my %gl_conf_cache; + my @repolist; # current repo list; reset on each 'repo ...' line my $subconf = 'master'; my $nextseq = 0; @@ -188,10 +195,13 @@ sub new_repos { next unless $repo =~ $REPONAME_PATT; # skip repo patterns next if $repo =~ m(^\@|EXTCMD/); # skip groups and fake repos - # use gl-conf as a sentinel - hook_1($repo) if -d "$repo.git" and not -f "$repo.git/gl-conf"; + # use gl-conf as a sentinel; if it exists, all is well + next if -f "$repo.git/gl-conf"; - if ( not -d "$repo.git" ) { + if (-d "$repo.git") { + # directory exists but sentinel missing? Maybe a freshly imported repo? + hook_1($repo); + } else { push @{ $rc{NEW_REPOS_CREATED} }, $repo; trigger( 'PRE_CREATE', $repo ); new_repo($repo); @@ -237,16 +247,24 @@ sub hook_repos { sub store { trace(3); + my $dbf = "$rc{GL_ADMIN_BASE}/gl-conf.cache"; + tie(%gl_conf_cache, 'GDBM_File', $dbf, O_RDWR|O_CREAT, 0666) or _die "Tie '$dbf' failed: $!"; + # first write out the ones for the physical repos _chdir( $rc{GL_REPO_BASE} ); - my $phy_repos = list_phy_repos(1); - for my $repo ( @{$phy_repos} ) { + # list of repos (union of keys of %repos plus %configs) + my %kr_kc; + @kr_kc{ keys %repos } = (); + @kr_kc{ keys %configs } = (); + for my $repo ( keys %kr_kc ) { store_1($repo); } _chdir( $rc{GL_ADMIN_BASE} ); store_common(); + + untie %gl_conf_cache; } sub parse_done { @@ -284,7 +302,7 @@ sub store_1 { # warning: writes and *deletes* it from %repos and %configs my ($repo) = shift; trace( 3, $repo ); - return unless ( $repos{$repo} or $configs{$repo} ) and -d "$repo.git"; + return unless -d "$repo.git"; my ( %one_repo, %one_config ); @@ -301,7 +319,10 @@ sub store_1 { $dumped_data .= Data::Dumper->Dump( [ \%one_config ], [qw(*one_config)] ); } - _print( "$repo.git/gl-conf", $dumped_data ); + if ( ($gl_conf_cache{$repo} || '') ne $dumped_data ) { + _print( "$repo.git/gl-conf", $dumped_data ); + $gl_conf_cache{$repo} = $dumped_data; + } $split_conf{$repo} = 1; } |