# Copyright 1995,1996,1997,1998 Spider Boardman. # All rights reserved. # # Automatic licensing for this software is available. This software # can be copied and used under the terms of the GNU Public License, # version 1 or (at your option) any later version, or under the # terms of the Artistic license. Both of these can be found with # the Perl distribution, which this software is intended to augment. # # THIS PACKAGE IS PROVIDED "AS IS" AND WITHOUT ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED # WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE. package Net::UNIX; use 5.004_05; # new minimum Perl version for this package use strict; #use Carp; sub carp { require Carp; goto &Carp::carp; } sub croak { require Carp; goto &Carp::croak; } use vars qw($VERSION @ISA @EXPORT @EXPORT_OK %EXPORT_TAGS $AUTOLOAD); my $myclass; BEGIN { $myclass = __PACKAGE__; $VERSION = '0.86'; } sub Version { "$myclass v$VERSION" } use AutoLoader; #use Exporter (); use Net::Gen 0.85 qw(/pack_sockaddr$/); use Socket qw(!pack_sockaddr_un !unpack_sockaddr_un !SOMAXCONN); BEGIN { @ISA = qw(Net::Gen); # Items to export into callers namespace by default. # (Move infrequently used names to @EXPORT_OK below.) @EXPORT = qw( ); @EXPORT_OK = qw( pack_sockaddr_un unpack_sockaddr_un ); %EXPORT_TAGS = ( routines => [qw(pack_sockaddr_un unpack_sockaddr_un)], ALL => [@EXPORT, @EXPORT_OK], ); } ;# sub AUTOLOAD inherited from Net::Gen ;# since 5.003_96 will break simple subroutines with inherited autoload, cheat sub AUTOLOAD { $Net::Gen::AUTOLOAD = $AUTOLOAD; goto &Net::Gen::AUTOLOAD; } # Preloaded methods go here. Autoload methods go after __END__, and are # processed by the autosplit program. ;# No additional sockopts for UNIX-domain sockets (?) my $sun_path_len = length(Socket::unpack_sockaddr_un(Socket::pack_sockaddr_un(''))); sub _canonpath ($) # $path; returns NUL-padded $path for sun_addr { my $path = shift; my $ix; # extend to proper length $ix = index($path, "\0"); if ($ix >= 0) { substr($path,$ix) = "\0" x ($sun_path_len - $ix) if $ix < $sun_path_len; } else { $ix = length($path); if ($ix < $sun_path_len) { $path .= "\0" x ($sun_path_len - $ix); } else { $path .= "\0"; } } $path; } sub pack_sockaddr_un ($;$) # [$family,] $path { my(@args) = @_; unshift(@args, AF_UNIX) if @args == 1; pack_sockaddr($args[0], _canonpath($args[1])); } sub unpack_sockaddr_un ($) # $sockaddr_un; returns [$fam,] $path { my $addr = shift; my ($fam,$path) = unpack_sockaddr($addr); my $nul = index($path, "\0"); if ($nul >= 0) { substr($path, $nul) = ''; } $fam ||= AF_UNIX; wantarray ? ($fam, $path) : $path; } my $debug = 0; sub _debug # $this, [$newval] ; returns oldval { use attrs 'locked'; my ($this,$newval) = @_; return $this->debug($newval) if ref $this; my $prev = $debug; $debug = 0+$newval if defined $newval; $prev; } my %keyhandlers = (thispath => \&_setbindpath, destpath => \&_setconnpath, ); my @Keys = qw(); sub new # $class, [\%params] { my $whoami = $_[0]->_trace(\@_,1); my($class,@Args,$self) = @_; $self = $class->SUPER::new(@Args); $class = ref $class if ref $class; ($self || $class)->_trace(\@_,2,", self" . (defined $self ? "=$self" : " undefined") . " after sub-new"); if ($self) { # register our keys and their handlers $self->registerParamKeys(\@Keys) if @Keys; $self->registerParamHandlers(\%keyhandlers); # register our socket options # none for AF_UNIX? # set our expected parameters $self->setparams({PF => PF_UNIX, AF => AF_UNIX, type => SOCK_DGRAM}, -1); if ($class eq $myclass) { unless ($self->init(@Args)) { local $!; # protect errno undef $self; # from the side-effects of this undef $self; # another statement needed for unwinding } } } if ($debug) { if ($self) { print STDERR "${myclass}::new returning self=$self\n"; } else { print STDERR "${myclass}::new returning undef\n"; } } $self; } sub _setbindpath # $self, 'thispath', $path { my($self,$what,$path) = @_; my $ix; if (!defined($path)) { # removing, so cooperate delete ${*$self}{Parms}{srcaddrlist}; return ''; } # canonicalize the path to be of the right length, if possible $path = _canonpath($path); $ix = index($path, "\0"); # check for NUL-termination if (!$ix) { # empty path is not a bind delete ${*$self}{Parms}{srcaddrlist}; $_[2] = undef; } else { ${*$self}{Parms}{srcaddrlist} = [pack_sockaddr_un($self->getparam('AF',AF_UNIX,1), $path)]; } ''; } sub _setconnpath # $self, 'destpath', $path { my($self,$what,$path) = @_; my $ix; if (!defined($path)) { # removing, so cooperate delete ${*$self}{Parms}{dstaddrlist}; return ''; } # canonicalize the path to be of the right length, if possible $path = _canonpath($path); $ix = index($path, "\0"); # check for NUL-termination if (!$ix) { # empty path? "$what parameter has no path: $path"; } else { # just try it here ${*$self}{Parms}{dstaddrlist} = [pack_sockaddr_un($self->getparam('AF',AF_UNIX,1), $path)]; ''; } } sub _init # $self, whatpath[, $path][, \%params] { use attrs 'locked', 'method'; my ($self,$what,@args,$path,$parms) = @_; if (@args == 1 or @args == 2) { $parms = $args[-1]; $parms = undef unless $parms and ref($parms) eq 'HASH'; $path = $args[0]; $path = undef if defined($path) and ref($path); } croak("Invalid call to ${myclass}::init(@_)") if @args == 2 and !$parms or @args > 2 or !$what; $parms ||= {}; $$parms{$what} = $path if defined $path; return undef unless $self->SUPER::init($parms); if (scalar %$parms) { return undef unless $self->setparams($parms); } if ($self->getparams([qw(srcaddr srcaddrlist dstaddr dstaddrlist)],1) >0) { return undef unless $self->isopen or $self->open; if ($self->getparams([qw(srcaddr srcaddrlist)],1) > 0) { return undef unless $self->isbound or $self->bind; } if ($self->getparams([qw(dstaddr dstaddrlist)],1) > 0) { return undef unless $self->isconnected or $self->connect or $self->isconnecting and !$self->blocking; } } $self; } sub init # $self [, $destpath][, \%params] { my ($self,@args) = @_; $self->_init('destpath',@args); } sub connect # $self [, $destpath] [, \%newparams] { use attrs 'locked', 'method'; my($self,$path,$parms) = @_; if (@_ > 3 or @_ == 3 and (!ref($parms) or ref($path))) { croak("Invalid arguments to ${myclass}::connect(@_), called"); } if (@_ == 2 and ref $path) { $parms = $path; undef $path; } else { $parms ||= {}; } if (defined $path) { $$parms{destpath} = $path; } if (@_ > 1) { return unless $self->setparams($parms); } $self->SUPER::connect; } sub format_addr # ($class|$obj) , $sockaddr { my ($this,$addr) = @_; my ($fam,$sdata) = unpack_sockaddr($addr); if ($fam == AF_UNIX) { $sdata = unpack_sockaddr_un($addr); } else { $sdata = $this->SUPER::format_addr($addr); } $sdata; } 1; # autoloaded methods go after the END token (& pod) below __END__ =head1 NAME Net::UNIX - UNIX-domain sockets interface module =head1 SYNOPSIS use Socket; # optional use Net::Gen; # optional use Net::UNIX; =head1 DESCRIPTION The C module provides services for UNIX-domain socket communication. It is layered atop the C module, which is part of the same distribution. =head2 Public Methods The following methods are provided by the C module itself, rather than just being inherited from C. =over =item new Usage: $obj = new Net::UNIX; $obj = new Net::UNIX $pathname; $obj = new Net::UNIX \%parameters; $obj = new Net::UNIX $pathname, \%parameters; $obj = 'Net::UNIX'->new(); $obj = 'Net::UNIX'->new($pathname); $obj = 'Net::UNIX'->new(\%parameters); $obj = 'Net::UNIX'->new($pathname, \%parameters); Returns a newly-initialised object of the given class. If called for a derived class, no validation of the supplied parameters will be performed. (This is so that the derived class can add the parameter validation it needs to the object before allowing the validation.) Otherwise, it will cause the parameters to be validated by calling its C method. In particular, this means that if a pathname is given, an object will be returned only if a connect() call was successful. The examples above show the indirect object syntax which many prefer, as well as the guaranteed-to-be-safe static method call. There are occasional problems with the indirect object syntax, which tend to be rather obscure when encountered. See FURL:http://www.rosat.mpe-garching.mpg.de/mailing-lists/perl-porters/1998-01/msg01674.htmlE> for details. =item init Usage: return undef unless $self = $self->init; return undef unless $self = $self->init(\%parameters); return undef unless $self = $self->init($pathname); return undef unless $self = $self->init($pathname, \%parameters); Verifies that all previous parameter assignments are valid (via C). Returns the incoming object on success, and C on failure. Usually called only via a derived class's C method or its own C call. =item bind Usage: $ok = $obj->bind; $ok = $obj->bind($pathname); $ok = $obj->bind($pathname,\%newparameters); Updates the object with the supplied new parameters (if supplied), then sets up the C object parameter with the specified $pathname argument (if supplied), and then returns the value from the inherited C method. Example: $ok = $obj->bind('/tmp/.fnord'); # start a service on /tmp/.fnord =item connect Usage: $ok = $obj->connect; $ok = $obj->connect($pathname); $ok = $obj->connect($pathname,\%newparameters); Attempts to establish a connection for the object. If the C argument is specified, it will be used to update the object parameters. Then, if the $pathname argument is specified, it will be used to set the C object parameter. Finally, the result of a call to the inherited C method will be returned. =item format_addr Usage: $string = $obj->format_addr($sockaddr); $string = format_addr Module $sockaddr; Returns a formatted representation of the socket address. This is normally just a pathname, or the constant string C<''>. =back =head2 Protected Methods [See the description in L for my definition of protected methods in Perl.] None. =head2 Known Socket Options There are no socket options known to the C module itself. =head2 Known Object Parameters There are no object parameters registered by the C module itself. =head2 TIESCALAR Tieing of scalars to a UNIX-domain handle is supported by inheritance from the C method of C. That method only succeeds if a call to a C method results in an object for which the C method returns a true result. Thus, for C, C will not succeed unless the C argument is given. Each assignment to the tied scalar is really a call to the C method (via the C method), and each read from the tied scalar is really a call to the C method (via the C method). =head2 Non-Method Subroutines =over =item pack_sockaddr_un Usage: $connect_address = pack_sockaddr_un($family, $pathname); $connect_address = pack_sockaddr_un($pathname); Returns the packed C corresponding to the provided $family and $pathname arguments. The $family argument as assumed to be C if it is missing. This is otherwise the same as the pack_sockaddr_un() routine in the C module. =item unpack_sockaddr_un Usage: ($family, $pathname) = unpack_sockaddr_un($connected_address); $pathname = unpack_sockaddr_un($connected_address); Returns the address family and pathname (if known) from the supplied packed C. This is the inverse of pack_sockaddr_un(). It differs from the implementation in the C module in its return of the C<$family> value, and in that it trims the returned pathname at the first null character. =back =head2 Exports =over =item default None. =item exportable C C =item tags The following I<:tags> are available for grouping exportable items: =over =item :routines C C =item :ALL All of the above exportable items. =back Z<> =back =head1 AUTHOR Spider Boardman Fspider@Orb.Nashua.NH.USE> =cut #other sections should be added, sigh. #any real autoloaded methods go after this line sub setdebug # $this, [bool, [norecurse]] { use attrs 'locked'; my $this = shift; $this->_debug($_[0]) . ((@_ > 1 && $_[1]) ? '' : $this->SUPER::setdebug(@_)); } sub bind # $self [, $destpath] [, \%newparams] { use attrs 'locked', 'method'; my($self,$path,$parms) = @_; if (@_ > 3 or @_ == 3 and (!ref($parms) or ref($path))) { my $whoami = $self->_trace; croak("Invalid arguments to ${whoami}(@_), called"); } if (@_ == 2 and ref $path) { $parms = $path; undef $path; } else { $parms ||= {}; } if (defined $path) { $$parms{thispath} = $path; } if (@_ > 1) { return unless $self->setparams($parms); } $self->SUPER::bind; }