############################################################################# ## Name: select.pm ## Purpose: Safe::World::select ## Author: Graciliano M. P. ## Modified by: ## Created: 08/09/2003 ## RCS-ID: ## Copyright: (c) 2003 Graciliano M. P. ## Licence: This program is free software; you can redistribute it and/or ## modify it under the same terms as Perl itself ############################################################################# package Safe::World::select ; use strict qw(vars); use vars qw($VERSION @ISA) ; $VERSION = '0.02' ; no warnings ; ########## # SCOPES # ########## use Safe::World::Scope ; my $SCOPE_Safe_World = new Safe::World::Scope('Safe::World',undef,1) ; use vars qw($Safe_World_NOW $Safe_World_EVALX) ; *Safe_World_NOW = \$Safe::World::NOW ; *Safe_World_EVALX = \$Safe::World::EVALX ; ####### # NEW # ####### sub new { ## my @call = caller(4) ; print main::STDOUT "SELECT NEW>> $_[1] [$Safe_World_NOW][$Safe_World_NOW->{SELECT}] @call\n" ; return undef if $_[1]->{DESTROIED} ; my $eval_err = $@ ; my $this = bless({} , __PACKAGE__) ; $this->{PREVWORLD} = $Safe_World_NOW ; $Safe_World_NOW = $this->{WORLD} = $_[1] ; $this->{WORLD}->{SELECT} = {} if !$this->{WORLD}->{SELECT} ; $this->{WORLD}->{SHARING} = {} if !$this->{WORLD}->{SHARING} ; my $prevstdout = &Safe::World::SELECT( "$this->{WORLD}->{ROOT}\::STDOUT" ) ; $this->{WORLD}->{SELECT}{PREVSTDOUT} = $this->{PREVSTDOUT} = [$prevstdout , \*{$prevstdout}] ; $this->{WORLD}->{SELECT}{PREVSTDERR} = $this->{PREVSTDERR} = *main::STDERR{IO} ; $this->{WORLD}->{SELECT}{PREVSUBWARN} = $this->{PREVSUBWARN} = $SIG{__WARN__} ; $this->{WORLD}->{SELECT}{PREVSUBDIE} = $this->{PREVSUBDIE} = $SIG{__DIE__} ; open (STDERR,">&$this->{WORLD}->{ROOT}::STDERR") ; $SIG{__WARN__} = \&print_stderr ; $SIG{__DIE__} = \&handle_die ; foreach my $var ( keys %{ $this->{WORLD}->{SHARING} } ) { $this->{WORLD}->{SHARING}{$var}{OUT} = &out_get_ref_copy($var) ; if ( $this->{WORLD}->{SHARING}{$var}{IN} ) { &out_set($var , $this->{WORLD}->{SHARING}{$var}{IN}) ; $this->{WORLD}->{SHARING}{$var}{IN} = undef ; } } if ( $this->{WORLD}->{TIESTDOUT} && $this->{WORLD}->{TIESTDOUT}->{AUTO_FLUSH} ) { $| = 1 ;} $this->{WORLD}->set('$SAFEWORLD', $this->{WORLD} , 1 ) if !$this->{WORLD}->{NO_SET_SAFEWORLD} ; if ( $this->{WORLD}->{ONSELECT} ) { my $sub = $this->{WORLD}->{ONSELECT} ; &$sub($this->{WORLD}) ; } $SCOPE_Safe_World->call('sync_evalx') ; ## Safe::World::sync_evalx() ; $@ = $eval_err ; return $this ; } ########### # DESTROY # ########### sub DESTROY { my $this = shift ; ##print main::STDOUT "SELECT DESTROY>> $this\n" ; my $eval_err = $@ ; %{$this->{WORLD}->{SELECT}} = () ; $this->{WORLD}->set('$SAFEWORLD', \undef) if !$this->{WORLD}->{NO_SET_SAFEWORLD} ; if ( $this->{WORLD}->{ONUNSELECT} ) { my $sub = $this->{WORLD}->{ONUNSELECT} ; &$sub($this->{WORLD}) ; } *main::STDERR = $this->{PREVSTDERR} ; $SIG{__WARN__} = $this->{PREVSUBWARN} ; $SIG{__DIE__} = $this->{PREVSUBDIE} ; foreach my $var ( keys %{ $this->{WORLD}->{SHARING} } ) { $this->{WORLD}->{SHARING}{$var}{IN} = &out_get_ref_copy($var) ; if ( $this->{WORLD}->{SHARING}{$var}{OUT} ) { &out_set($var , $this->{WORLD}->{SHARING}{$var}{OUT}) ; $this->{WORLD}->{SHARING}{$var}{OUT} = undef ; } } &Safe::World::SELECT($this->{PREVSTDOUT}) ; $Safe_World_NOW = (ref($this->{PREVWORLD}) eq 'Safe::World') ? $this->{PREVWORLD} : undef ; $SCOPE_Safe_World->call('sync_evalx') ; ## Safe::World::sync_evalx() ; $@ = $eval_err ; return ; } #################### # OUT_GET_REF_COPY # #################### sub out_get_ref_copy { my ( $varfull ) = @_ ; my ($var_tp,$var) = ( $varfull =~ /([\$\@\%\*])(\S+)/ ) ; $var =~ s/^{'(\S+)'}$/$1/ ; $var =~ s/^main::// ; if ($var_tp eq '$') { return ${'main::'.$var} ;} elsif ($var_tp eq '@') { return [@{'main::'.$var}] ;} elsif ($var_tp eq '%') { return {%{'main::'.$var}} ;} elsif ($var_tp eq '*') { return \*{'main::'.$var} ;} else { ++$Safe_World_EVALX ; return eval("package main ; \\$varfull") ;} } ########### # OUT_SET # ########### sub out_set { my ( $var , $val ) = @_ ; my ($var_tp,$name) = ( $var =~ /([\$\@\%\*])(\S+)/ ); $name =~ s/^{'(\S+)'}$/$1/ ; $name =~ s/^main::// ; if ($var_tp eq '$') { ${'main::'.$name} = $val ;} elsif ($var_tp eq '@') { @{'main::'.$name} = @{$val} ;} elsif ($var_tp eq '%') { %{'main::'.$name} = %{$val} ;} elsif ($var_tp eq '*') { *{'main::'.$name} = $val ;} else { ++$Safe_World_EVALX ; eval("$var = \$val ;") ;} } ################ # PRINT_STDERR # ################ sub print_stderr { $Safe_World_NOW->print_stderr(@_) ; return ; } ############## # HANDLE_DIE # ############## sub handle_die { my $core_exit = ($_[0] =~ /#CORE::GLOBAL::exit#/) ? 1 : undef ; $Safe_World_NOW->{EXIT} = 1 if $core_exit ; $Safe_World_NOW->print_stderr(@_) if !$core_exit ; $Safe_World_NOW->close if $core_exit ; $@ = undef if $core_exit ; return ; } ####### # END # ####### 1;