From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on dcvr.yhbt.net X-Spam-Level: X-Spam-ASN: X-Spam-Status: No, score=-4.2 required=3.0 tests=ALL_TRUSTED,BAYES_00, DKIM_SIGNED,DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF shortcircuit=no autolearn=ham autolearn_force=no version=3.4.6 Received: from localhost (dcvr.yhbt.net [127.0.0.1]) by dcvr.yhbt.net (Postfix) with ESMTP id 9BFFB1F406 for ; Mon, 16 Oct 2023 11:09:38 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=80x24.org; s=selector1; t=1697454578; bh=jlALfTxw+TSXD1Vzk45gEHy4qYdJyUrIUR7RtgCNoU0=; h=From:To:Subject:Date:From; b=UMC5PyEkIf8Jfj7hh1RhX0EADfYD2wAoTPKSIzeG/7RKJpKX9YA6TpjBmpu4lYsPa XYhoBIctNyVr3sxkJMDExtD5MIB3FLG8c0vUP6OzIaFuHgqohon6cQB4YU+Gqki4Y8 j9eSp9cH4lWj7EcKD/AClRav3U2Nbh6JgLMlROCs= From: Eric Wong To: spew@80x24.org Subject: [PATCH] input_pipe: error handling + TTY support Date: Mon, 16 Oct 2023 11:09:38 +0000 Message-ID: <20231016110938.1206501-1-e@80x24.org> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit List-Id: --- lib/PublicInbox/InputPipe.pm | 69 +++++++++++++++++++++++++++--------- 1 file changed, 53 insertions(+), 16 deletions(-) diff --git a/lib/PublicInbox/InputPipe.pm b/lib/PublicInbox/InputPipe.pm index 60a9f01f..5f109886 100644 --- a/lib/PublicInbox/InputPipe.pm +++ b/lib/PublicInbox/InputPipe.pm @@ -5,31 +5,68 @@ package PublicInbox::InputPipe; use v5.12; use parent qw(PublicInbox::DS); -use PublicInbox::Syscall qw(EPOLLIN EPOLLET); +use PublicInbox::Syscall qw(EPOLLIN); +use POSIX (); + +sub unblock_tty ($) { + my ($self) = @_; + my $t = POSIX::Termios->new; + my $fd = fileno(my $in = $self->{sock}); + $t->getattr($fd) or croak("tcgetattr($in/$fd): $!"); + my $vmin = $t->getcc(POSIX::VMIN); + my $vtime = $t->getcc(POSIX::VTIME); + return if $vmin == 1 && $vtime == 0; + $t->setcc(POSIX::VMIN, 1); + $t->setcc(POSIX::VTIME, 0); + $t->setattr($fd, POSIX::TCSANOW) or croak("tcsetattr($in/$fd): $!"); + $t->setcc(POSIX::VMIN, $vmin); + $t->setcc(POSIX::VTIME, $vtime); + $self->{termios} = $t; +} sub consume { my ($in, $cb, @args) = @_; my $self = bless { cb => $cb, args => \@args }, __PACKAGE__; - eval { $self->SUPER::new($in, EPOLLIN|EPOLLET) }; - return $self->requeue if $@; # regular file - $in->blocking(0); # pipe or socket + eval { $self->SUPER::new($in, EPOLLIN) }; + if ($@) { + $self->{-need_rq} = 1; + $self->requeue; + } elsif (-p $in || -s _) { + $in->blocking(0); + } elsif (-t _) { # not sure I know what I'm doing w/ terminals + unblock_tty($self); + } +} + +sub close { + my ($self) = @_; + if (my $t = delete($self->{termios})) { + my $fd = fileno(my $in = $self->{sock} // return); + $t->setattr($fd, POSIX::TCSANOW) or + croak("tcsetattr($in/$fd): $!"); + } + $self->{-need_rq} ? delete($self->{sock}) : $self->SUPER::close } sub event_step { my ($self) = @_; my $r = sysread($self->{sock} // return, my $rbuf, 65536); - if ($r) { - $self->{cb}->(@{$self->{args}}, $rbuf); - return $self->requeue; # may be regular file or pipe - } - if (defined($r)) { # EOF - $self->{cb}->(@{$self->{args}}, ''); - } elsif ($!{EAGAIN}) { - return; - } else { # another error - $self->{cb}->(@{$self->{args}}, undef) - } - $self->{sock}->blocking ? delete($self->{sock}) : $self->close + eval { + if ($r) { + $self->{cb}->(@{$self->{args}}, $rbuf); + $self->requeue if $self->{-need_rq}; + } elsif (defined($r)) { # EOF + $self->{cb}->(@{$self->{args}}, ''); + $self->close + } elsif ($!{EAGAIN}) { # rely on EPOLLIN + } elsif ($!{EINTR}) { # rely on EPOLLIN for sockets/pipes/tty + $self->requeue if $self->{-need_rq}; + } else { # another error + $self->{cb}->(@{$self->{args}}, undef); + $self->close; + } + }; + $self->close if $@; } 1;