Ken Simpson wrote:
Hi Everyone,
This patch adds a poll() method to APR::Socket that allows you to poll
a nonblocking APR socket -- similar to the select() call in Unix-land.

Synopsis:
 use APR::Socket();
 use Apache::Connection ();
 use APR::Const -compile => qw(POLLIN);

 my $timeout = 10_000_000; # microseconds

 my $socket = $connection->client_socket();
 $socket->poll($connection->pool, $timeout, APR::POLLIN);

Thanks to Stas for showing me the ropes of how to add this support
in. mock and I will be updating Apache::TieBucketBrigade to take
advantage of this polling interface so that you can do magic things
like make Apache look just like a regular Perl socket or file handle
-- which would be handy if you were, say, writing an SMTP server that
leverages Net::Server::Mail (see Apache::SMTP). </shameless_plug>

Nice work, Ken. Though you've missed the adjusted test files. I've applied a few more tweaks and adjusted the tests. The patch is below.


One missing part is to add the API doc to docs/api/APR/Socket.pod and then it can be committed.

Another thing is the test. I'll discuss it in a separate email.

Index: xs/APR/Socket/APR__Socket.h
===================================================================
RCS file: /home/cvs/modperl-2.0/xs/APR/Socket/APR__Socket.h,v
retrieving revision 1.11
diff -u -u -r1.11 APR__Socket.h
--- xs/APR/Socket/APR__Socket.h 9 Jun 2004 14:46:22 -0000       1.11
+++ xs/APR/Socket/APR__Socket.h 31 Aug 2004 03:50:11 -0000
@@ -96,3 +96,22 @@
     MP_RUN_CROAK(apr_socket_opt_set(socket, opt, val),
                  "APR::Socket::opt_set");
 }
+
+static MP_INLINE
+apr_status_t mpxs_APR__Socket_poll(apr_socket_t *socket,
+                                   apr_pool_t *pool,
+                                   apr_interval_time_t timeout,
+                                   apr_int16_t reqevents)
+{
+    apr_pollfd_t fd;
+    apr_int32_t nsds;
+
+    /* what to poll */
+    fd.p         = pool;
+    fd.desc_type = APR_POLL_SOCKET;
+    fd.desc.s    = socket;
+    fd.reqevents = reqevents;
+    fd.rtnevents = 0; /* XXX: not really necessary to set this */
+
+    return apr_poll(&fd, 1, &nsds, timeout);
+}
Index: xs/maps/apr_functions.map
===================================================================
RCS file: /home/cvs/modperl-2.0/xs/maps/apr_functions.map,v
retrieving revision 1.85
diff -u -u -r1.85 apr_functions.map
--- xs/maps/apr_functions.map   25 Aug 2004 22:32:01 -0000      1.85
+++ xs/maps/apr_functions.map   31 Aug 2004 03:50:11 -0000
@@ -72,6 +72,8 @@
 -apr_socket_sendfile
 -apr_socket_sendv
 !apr_socket_from_file
+ mpxs_APR__Socket_poll | | apr_socket_t *:socket, apr_pool_t *:pool, \
+   apr_interval_time_t:timeout, apr_int16_t:reqevents

MODULE=APR::SockAddr
!apr_sockaddr_info_get
Index: xs/tables/current/ModPerl/FunctionTable.pm
===================================================================
RCS file: /home/cvs/modperl-2.0/xs/tables/current/ModPerl/FunctionTable.pm,v
retrieving revision 1.176
diff -u -u -r1.176 FunctionTable.pm
--- xs/tables/current/ModPerl/FunctionTable.pm 25 Aug 2004 22:32:01 -0000 1.176
+++ xs/tables/current/ModPerl/FunctionTable.pm 31 Aug 2004 03:50:11 -0000
@@ -2,7 +2,7 @@


 # !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
 # ! WARNING: generated by ModPerl::ParseSource/0.01
-# !          Wed Aug 25 14:56:13 2004
+# !          Mon Aug 30 22:40:23 2004
 # !          do NOT edit, any changes will be lost !
 # !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!

@@ -5574,6 +5574,28 @@
       {
         'type' => 'apr_int32_t',
         'name' => 'val'
+      }
+    ]
+  },
+  {
+    'return_type' => 'apr_status_t',
+    'name' => 'mpxs_APR__Socket_poll',
+    'args' => [
+      {
+        'type' => 'apr_socket_t *',
+        'name' => 'socket'
+      },
+      {
+        'type' => 'apr_pool_t *',
+        'name' => 'pool'
+      },
+      {
+        'type' => 'apr_interval_time_t',
+        'name' => 'timeout'
+      },
+      {
+        'type' => 'apr_int16_t',
+        'name' => 'reqevents'
       }
     ]
   },

--- /dev/null 1969-12-31 19:00:00.000000000 -0500
+++ t/protocol/echo_nonblock.t 2004-08-30 23:49:48.873451131 -0400
@@ -0,0 +1,29 @@
+use strict;
+use warnings FATAL => 'all';
+
+use Test;
+use Apache::TestUtil;
+use Apache::TestRequest ();
+
+plan tests => 3;
+
+my $socket = Apache::TestRequest::vhost_socket('TestProtocol::echo_nonblock');
+
+ok $socket;
+
+my $received;
+my $expected;
+
+$expected = "nonblocking";
+print $socket "$expected\n";
+chomp($received = <$socket> || '');
+ok t_cmp $received, $expected, "no timeout";
+
+# Wait two seconds so that the server will time out.
+sleep 2;
+
+$expected = "TIMEUP";
+print $socket "should timeout\n";
+chomp($received = <$socket> || '');
+ok t_cmp $received, $expected, "timed out";
+


--- /dev/null 1969-12-31 19:00:00.000000000 -0500
+++ t/protocol/TestProtocol/echo_nonblock.pm 2004-08-30 22:41:59.940665923 -0400
@@ -0,0 +1,47 @@
+package TestProtocol::echo_nonblock;
+
+# this test reads from/writes to the socket doing nonblocking IO
+
+use strict;
+use warnings FATAL => 'all';
+
+use Apache::Connection ();
+use APR::Socket ();
+
+use Apache::TestTrace;
+
+use Apache::Const -compile => 'OK';
+use APR::Const -compile => qw(SO_NONBLOCK TIMEUP SUCCESS POLLIN);
+
+use constant BUFF_LEN => 1024;
+
+sub handler {
+ my $c = shift;
+ my $socket = $c->client_socket;
+
+ $socket->opt_set(APR::SO_NONBLOCK => 1);
+ while (1) {
+ # Wait up to one second for data to arrive.
+ my $rc = $socket->poll($c->pool, 1_000_000, APR::POLLIN);
+ if ($rc == APR::SUCCESS) {
+ if ($socket->recv(my $buf, BUFF_LEN)) {
+ debug "no timeout";
+ $socket->send($buf);
+ }
+ else {
+ last;
+ }
+ }
+ elsif ($rc == APR::TIMEUP) {
+ debug "timeout";
+ $socket->send("TIMEUP\n");
+ }
+ else {
+ die "poll error: $rc: " . APR::Error::strerror($rc);
+ }
+ }
+
+ Apache::OK;
+}
+
+1;




--
__________________________________________________________________
Stas Bekman            JAm_pH ------> Just Another mod_perl Hacker
http://stason.org/     mod_perl Guide ---> http://perl.apache.org
mailto:[EMAIL PROTECTED] http://use.perl.org http://apacheweek.com
http://modperlbook.org http://apache.org   http://ticketmaster.com

---------------------------------------------------------------------
To unsubscribe, e-mail: [EMAIL PROTECTED]
For additional commands, e-mail: [EMAIL PROTECTED]



Reply via email to