public-inbox.git  about / heads / tags
an "archives first" approach to mailing lists
blob 07ff7dcb60dd4d6dcd21ad49da48b4c3ebb1c9dd 2037 bytes (raw)
$ git show HEAD:lib/PublicInbox/Gcf2Client.pm	# shows this blob on the CLI

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
 
# Copyright (C) all contributors <meta@public-inbox.org>
# License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt>

# connects public-inbox processes to PublicInbox::Gcf2::loop()
package PublicInbox::Gcf2Client;
use v5.12;
use parent qw(PublicInbox::DS);
use PublicInbox::Git;
use PublicInbox::Gcf2; # fails if Inline::C or libgit2-dev isn't available
use PublicInbox::Spawn qw(spawn);
use Socket qw(AF_UNIX SOCK_STREAM);
use PublicInbox::Syscall qw(EPOLLIN);
use PublicInbox::IO;
use autodie qw(socketpair);

# fields:
#	sock => socket to Gcf2::loop
# The rest of these fields are compatible with what PublicInbox::Git
# uses code-sharing
#	pid => PID of Gcf2::loop process
#	pid.owner => process which spawned {pid}
#	in => same as {sock}, for compatibility with PublicInbox::Git
#	inflight => array (see PublicInbox::Git)
sub new  {
	my ($opt) = @_;
	my $self = bless {}, __PACKAGE__;
	# ensure the child process has the same @INC we do:
	my $env = { PERL5LIB => join(':', @INC) };
	socketpair(my $s1, my $s2, AF_UNIX, SOCK_STREAM, 0);
	$s1->blocking(0);
	$opt->{0} = $opt->{1} = $s2;
	my $cmd = [$^X, $^W ? ('-w') : (),
			qw[-MPublicInbox::Gcf2 -e PublicInbox::Gcf2::loop]];
	$self->{inflight} = [];
	PublicInbox::IO::attach_pid($s1, spawn($cmd, $env, $opt),
			\&PublicInbox::Git::gcf_drain, $self->{inflight});
	$self->{epwatch} = \undef; # for Git->cleanup
	$self->SUPER::new($s1, EPOLLIN);
}

sub gcf2_async ($$$;$) {
	my ($self, $req, $cb, $arg) = @_;
	my $inflight = $self->gcf_inflight or return;
	PublicInbox::Git::write_all($self, $req, \&cat_async_step, $inflight);
	push @$inflight, \$req, $cb, $arg; # ref prevents Git.pm retries
}

# ensure PublicInbox::Git::cat_async_step never calls cat_async_retry
sub alternates_changed {}

no warnings 'once';

*gcf_inflight = \&PublicInbox::Git::gcf_inflight; # for event_step
*cat_async_step = \&PublicInbox::Git::cat_async_step; # for event_step
*event_step = \&PublicInbox::Git::event_step;
*fail = \&PublicInbox::Git::fail;
*DESTROY = \&PublicInbox::Git::DESTROY;

1;

git clone https://public-inbox.org/public-inbox.git
git clone http://7fh6tueqddpjyxjmgtdiueylzoqt6pt7hec3pukyptlmohoowvhde4yd.onion/public-inbox.git