X-Git-Url: http://www.git.stargrave.org/?a=blobdiff_plain;f=lib%2FPublicInbox%2FIMAP.pm;h=2282e3ce78c6be097b289d04f68ce9c9b0e0842b;hb=eeeae20893a2595631359e719eadac9686572734;hp=0852ffab868dc61bfba42d9ca8f1858911ddcfd0;hpb=1804e2961f3a28045b601a982f44ff61ea33a2fe;p=public-inbox.git diff --git a/lib/PublicInbox/IMAP.pm b/lib/PublicInbox/IMAP.pm index 0852ffab..2282e3ce 100644 --- a/lib/PublicInbox/IMAP.pm +++ b/lib/PublicInbox/IMAP.pm @@ -21,8 +21,10 @@ use PublicInbox::Eml; use PublicInbox::EmlContentFoo qw(parse_content_disposition); use PublicInbox::DS qw(now); use PublicInbox::Syscall qw(EPOLLIN EPOLLONESHOT); +use PublicInbox::GitAsyncCat; use Text::ParseWords qw(parse_line); use Errno qw(EAGAIN); + my $Address; for my $mod (qw(Email::Address::XS Mail::Address)) { eval "require $mod" or next; @@ -367,14 +369,30 @@ EOF \$ret; } -sub uid_fetch_cb { # called by git->cat_async +sub requeue_once ($) { + my ($self) = @_; + # COMPRESS users all share the same DEFLATE context. + # Flush it here to ensure clients don't see + # each other's data + $self->zflush; + + # no recursion, schedule another call ASAP, + # but only after all pending writes are done. + # autovivify wbuf: + my $new_size = push(@{$self->{wbuf}}, \&long_step); + + # wbuf may be populated by $cb, no need to rearm if so: + $self->requeue if $new_size == 1; +} + +sub uid_fetch_cb { # called by git->cat_async via git_async_msg my ($bref, $oid, $type, $size, $fetch_m_arg) = @_; my ($self, undef, $ibx, undef, undef, $msgs, $want) = @$fetch_m_arg; my $smsg = shift @$msgs or die 'BUG: no smsg'; if (!defined($oid)) { # it's possible to have TOCTOU if an admin runs # public-inbox-(edit|purge), just move onto the next message - return unless defined $want->{-seqno}; + return requeue_once($self) unless defined $want->{-seqno}; $bref = dummy_message($smsg->{num}, $ibx); } else { $smsg->{blob} eq $oid or die "BUG: $smsg->{blob} != $oid"; @@ -420,6 +438,7 @@ sub uid_fetch_cb { # called by git->cat_async partial_emit($self, $partial, $eml); } $self->msg_more(")\r\n"); + requeue_once($self); } sub uid_fetch_m { # long_response @@ -432,11 +451,7 @@ sub uid_fetch_m { # long_response } $$beg = $msgs->[-1]->{num} + 1; } - my $git = $ibx->git; - $git->cat_async_begin; # TODO: actually make async - $git->cat_async($msgs->[0]->{blob}, \&uid_fetch_cb, \@_); - $git->cat_async_wait; - 1; + git_async_msg($ibx, $msgs->[0], \&uid_fetch_cb, \@_); } sub cmd_status ($$$;@) { @@ -544,25 +559,23 @@ sub hdrs_regexp ($) { # BODY[($SECTION_IDX.)?HEADER.FIELDS.NOT ($HDRS)]<$offset.$bytes> sub partial_hdr_not { - my ($eml, $section_idx, $hdrs) = @_; + my ($eml, $section_idx, $hdrs_re) = @_; if (defined $section_idx) { $eml = eml_body_idx($eml, $section_idx) or return; } my $str = $eml->header_obj->as_string; - my $re = hdrs_regexp($hdrs); - $str =~ s/$re//g; + $str =~ s/$hdrs_re//g; $str .= "\r\n"; } # BODY[($SECTION_IDX.)?HEADER.FIELDS ($HDRS)]<$offset.$bytes> sub partial_hdr_get { - my ($eml, $section_idx, $hdrs) = @_; + my ($eml, $section_idx, $hdrs_re) = @_; if (defined $section_idx) { $eml = eml_body_idx($eml, $section_idx) or return; } my $str = $eml->header_obj->as_string; - my $re = hdrs_regexp($hdrs); - join('', ($str =~ m/($re)/g), "\r\n"); + join('', ($str =~ m/($hdrs_re)/g), "\r\n"); } sub partial_prepare ($$$) { @@ -583,9 +596,10 @@ sub partial_prepare ($$$) { (?:HEADER\.FIELDS(\.NOT)?)\x20 # 2 \(([A-Z0-9\-\x20]+)\) # 3 - hdrs \](?:<([0-9]+)(?:\.([0-9]+))?>)?\z/sx) { # 4 5 - $partial->{$att} = [ $2 ? \&partial_hdr_not - : \&partial_hdr_get, - $1, $3, $4, $5 ]; + my $tmp = $partial->{$att} = [ $2 ? \&partial_hdr_not + : \&partial_hdr_get, + $1, undef, $4, $5 ]; + $tmp->[2] = hdrs_regexp($3); } else { undef; } @@ -678,20 +692,17 @@ sub seq_fetch_m { # long_response my $seq = $want->{-seqno}++; my $cur_num = $msgs->[0]->{num}; if ($cur_num == $seq) { # as expected - my $git = $ibx->git; - $git->cat_async_begin; # TODO: actually make async - $git->cat_async($msgs->[0]->{blob}, \&uid_fetch_cb, \@_); - $git->cat_async_wait; + git_async_msg($ibx, $msgs->[0], \&uid_fetch_cb, \@_); } elsif ($cur_num > $seq) { # send dummy messages until $seq catches up to $cur_num my $smsg = bless { num => $seq, ts => 0 }, 'PublicInbox::Smsg'; unshift @$msgs, $smsg; my $bref = dummy_message($seq, $ibx); uid_fetch_cb($bref, undef, undef, undef, \@_); + $smsg; # blessed response since uid_fetch_cb requeues } else { # should not happen die "BUG: cur_num=$cur_num < seq=$seq"; } - 1; # more messages on the way } sub cmd_fetch ($$$;@) { @@ -811,17 +822,8 @@ sub long_step { } elsif ($more) { # $self->{wbuf}: $self->update_idle_time; - # COMPRESS users all share the same DEFLATE context. - # Flush it here to ensure clients don't see - # each other's data - $self->zflush; - - # no recursion, schedule another call ASAP, but only after - # all pending writes are done. autovivify wbuf: - my $new_size = push(@{$self->{wbuf}}, \&long_step); - - # wbuf may be populated by $cb, no need to rearm if so: - $self->requeue if $new_size == 1; + # control passed to $more may be a GitAsyncCat object + requeue_once($self) if !ref($more); } else { # all done! delete $self->{long_cb}; my $elapsed = now() - $t0;