package HH::Unispool::Config::Entry::Device::0; use 5.006; use base qw( HH::Unispool::Config::Entry::Device ); use strict; use warnings; use AutoLoader qw(AUTOLOAD); use Error qw(:try); use HH::Unispool::Config::ExecPri; # Used by _value_is_allowed our %ALLOW_ISA = ( 'execution_priority' => [ 'HH::Unispool::Config::ExecPri' ], ); # Used by _value_is_allowed our %ALLOW_REF = ( ); # Used by _value_is_allowed our %ALLOW_RX = ( 'device_file' => [ '^.+$' ], 'device_password' => [ '^.*$' ], 'header_name' => [ '^.*$' ], 'trailer_name' => [ '^.*$' ], ); # Used by _value_is_allowed our %ALLOW_VALUE = ( ); # Used by _initialize our %DEFAULT_VALUE = ( 'execution_priority' => HH::Unispool::Config::ExecPri->new( { execution_priority => '' } ), 'initially_spooled' => 1, 'networkwide' => 1, 'save_printfile' => 0, 'unispool_control' => 0, 'unispool_header' => 0, ); # Package version our ($VERSION) = '$Revision: 0.3 $' =~ /\$Revision:\s+([^\s]+)/; 1; __END__ =head1 NAME HH::Unispool::Config::Entry::Device::0 - UNISPOOL type 0 device =head1 SYNOPSIS # Information on device type 0 is obtained through an informal document. # I guess this device type is out of fashion... use strict; use HH::Unispool::Config; use HH::Unispool::Config::OS; use HH::Unispool::Config::Scope; use HH::Unispool::Config::Entry::System; use HH::Unispool::Config::Entry::Device::0; # Create a config from scratch with a _Local_ scope my $conf = HH::Unispool::Config->new( { scope => HH::Unispool::Config::Scope->new ( { scope => [ qw( _Local_) ], } ), } ); # Create the _Local_ system and add it to the configuration my $sysn = HH::Unispool::Config::Entry::System->new( { name => '_Local_', local_system_name => 'foo', network_name => 'foo.bar.org', os => HH::Unispool::Config::OS->new( { os => 'Solaris', } ), } ); $conf->add_system($sysn); # Add a type 0 device to _Local_ my $dev = HH::Unispool::Config::Entry::Device::0->new( { name => 'lp', filter_name => 'cat', device_file => '/dev/lp0', } ); $sysn->add_device($dev); # Write the configuration $conf->write('t/config.sample.d0.out'); =head1 ABSTRACT UNISPOOL type 0 device =head1 DESCRIPTION Class C holds information for UNISPOOL type 0 device =head1 CONSTRUCTOR =over =item new(OPT_HASH_REF) Creates a new C object. C is a hash reference used to pass initialization options. C is mandatory. On error an exception C is thrown. Options for C may include: =over =item B> Passed to L. Mandatory option. =item B> Passed to L. =item B> Passed to L. Defaults to Bnew( { execution_priority => '' } )>. =item B> Passed to L. =item B> Passed to L. Defaults to B<1>. =item B> Passed to L. Defaults to B<1>. =item B> Passed to L. Defaults to B<0>. =item B> Passed to L. =item B> Passed to L. Defaults to B<0>. =item B> Passed to L. Defaults to B<0>. =back Options for C inherited through package B> may include: =over =item B> Passed to L. Mandatory option. =back Options for C inherited through package B> may include: =over =item B> Passed to L. =item B> Passed to L. Mandatory option. =back Options for C inherited through package B> may include: =over =item B> Passed to L. Defaults to B<0>. =item B> Passed to L. =back =item new_from_tokenizer(TOKENIZER) This method is an implementation from package C. Constructs a new C object using tokens. C is an C reference. On error an exception C is thrown. =back =head1 METHODS =over =item diff(TO [, DIFF_NUMBER]) This method is an implementation from package C. Finds differences between two objects. In C terms, the object is the B object and the specified C parameter the B object. C is a reference to an identical object class. Returns an empty string if no difference found and a difference descritpion string otherwise. On error an exception C is thrown. Paremeter C if specified, overrules the value of C. =item get_description() This method is inherited from package C. Returns the description for the device. =item get_device_file() Returns the device file to which the device is connected. =item get_device_password() Returns the password required to access the device. =item get_execution_priority() Returns the execution priority of the driver process on MPE hosts. =item get_filter_name() This method is inherited from package C. Returns the name of the filter file to be used when printfiles for this device are generated. =item get_header_name() Returns the control procedure to be executed before printing the printfile. =item get_name() This method is inherited from package C. Returns the entry name. =item get_number() This method is inherited from package C. Returns the entry number. =item get_trailer_name() Returns the control procedure to be executed after printing the printfile. =item is_diff_number() This method is inherited from package C. Returns whether L should consider the C attribtutes or not. =item is_initially_spooled() Returns whether an automatic STARTSPOOL must be performed when UNISPOOL is started or not. =item is_networkwide() Returns whether the device must be made available from each node in the configuration cluster or not. =item is_save_printfile() Returns whether printfiles printed on this device should be saved or not. =item is_unispool_control() Returns whether the UNISPOOL control is on (is not available on development system used) or not. =item is_unispool_header() Returns whether standard UNISPOOL banners pages are printed initially or not. =item set_description(VALUE) This method is inherited from package C. Set the description for the device. C is the value. On error an exception C is thrown. =over =item VALUE must match regular expression: =over =item ^.*$ =back =back =item set_device_file(VALUE) Set the device file to which the device is connected. C is the value. On error an exception C is thrown. =over =item VALUE must match regular expression: =over =item ^.+$ =back =back =item set_device_password(VALUE) Set the password required to access the device. C is the value. On error an exception C is thrown. =over =item VALUE must match regular expression: =over =item ^.*$ =back =back =item set_diff_number(VALUE) This method is inherited from package C. State that L should consider the C attribtutes. C is the value. Default value at initialization is C<0>. On error an exception C is thrown. =item set_execution_priority(VALUE) Set the execution priority of the driver process on MPE hosts. C is the value. Default value at initialization is Cnew( { execution_priority => '' } )>. On error an exception C is thrown. =over =item VALUE must be a (sub)class of: =over =item HH::Unispool::Config::ExecPri =back =back =item set_filter_name(VALUE) This method is inherited from package C. Set the name of the filter file to be used when printfiles for this device are generated. C is the value. On error an exception C is thrown. =over =item VALUE must match regular expression: =over =item ^.*$ =back =back =item set_header_name(VALUE) Set the control procedure to be executed before printing the printfile. C is the value. On error an exception C is thrown. =over =item VALUE must match regular expression: =over =item ^.*$ =back =back =item set_initially_spooled(VALUE) State that an automatic STARTSPOOL must be performed when UNISPOOL is started. C is the value. Default value at initialization is C<1>. On error an exception C is thrown. =item set_name(VALUE) This method is inherited from package C. Set the entry name. C is the value. C may not be C. On error an exception C is thrown. =over =item VALUE must match regular expression: =over =item ^.+$ =back =back =item set_networkwide(VALUE) State that the device must be made available from each node in the configuration cluster. C is the value. Default value at initialization is C<1>. On error an exception C is thrown. =item set_number(VALUE) This method is inherited from package C. Set the entry number. C is the value. On error an exception C is thrown. =over =item VALUE must match regular expression: =over =item ^\d*$ =back =back =item set_save_printfile(VALUE) State that printfiles printed on this device should be saved. C is the value. Default value at initialization is C<0>. On error an exception C is thrown. =item set_trailer_name(VALUE) Set the control procedure to be executed after printing the printfile. C is the value. On error an exception C is thrown. =over =item VALUE must match regular expression: =over =item ^.*$ =back =back =item set_unispool_control(VALUE) State that the UNISPOOL control is on (is not available on development system used). C is the value. Default value at initialization is C<0>. On error an exception C is thrown. =item set_unispool_header(VALUE) State that standard UNISPOOL banners pages are printed initially. C is the value. Default value at initialization is C<0>. On error an exception C is thrown. =item write(FILE_HANDLE) This method is an implementation from package C. Writes the entry to the specified file handle. C is an C reference. On error an exception C is thrown. =back =head1 SEE ALSO L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L, L =head1 BUGS None known (yet.) =head1 HISTORY First development: February 2003 Last update: September 2003 =head1 AUTHOR Vincenzo Zocca =head1 COPYRIGHT Copyright 2003 by Vincenzo Zocca =head1 LICENSE This file is part of the C module hierarchy for Perl by Vincenzo Zocca. The HH::Unispool::Config module hierarchy is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. The HH::Unispool::Config module hierarchy is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with the HH::Unispool::Config module hierarchy; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA =cut sub new_from_tokenizer { my $class = shift; my $tokenizer = shift; # First token must be a HH::Unispool::Config::File::Token::Numbered::Device::0 my $s = $tokenizer->get(); $s->isa('HH::Unispool::Config::File::Token::Numbered::Device::0') || throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::new_from_tokenizer, expected a first token from 'TOKENIZER' of class 'HH::Unispool::Config::File::Token::Numbered::Device::0'."); # Fill the initialization option hash my %opt = (); $opt{number} = $s->get_number(); $opt{name} = $s->get_device_name(); $opt{device_password} = $s->get_device_password(); $opt{unispool_header} = $s->is_unispool_header(); $opt{initially_spooled} = $s->is_initially_spooled(); $opt{networkwide} = $s->is_networkwide(); $opt{save_printfile} = $s->is_save_printfile(); $opt{unispool_control} = $s->is_unispool_control(); $opt{execution_priority} = $s->get_execution_priority(); # Allow P, X and I tokens my $p = undef; my $x = undef; my $i = undef; while ( my $tok = $tokenizer->get() ) { if ( ! $tok->isa('HH::Unispool::Config::File::Token::Numbered') || $tok->get_number() != $s->get_number() ) { $tokenizer->unget(); last; } elsif ( $tok->isa('HH::Unispool::Config::File::Token::Numbered::Device::P') ) { defined ($p) && throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::new_from_tokenizer, multiple tokens obtained from 'TOKENIZER' for entry $opt{name}/$opt{number} from class 'HH::Unispool::Config::File::Token::Numbered::Device::P'."); $p = $tok; $opt{device_file} = $p->get_device_file(); } elsif ( $tok->isa('HH::Unispool::Config::File::Token::Numbered::X') ) { defined ($x) && throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::new_from_tokenizer, multiple tokens obtained from 'TOKENIZER' for entry $opt{name}/$opt{number} from class 'HH::Unispool::Config::File::Token::Numbered::X'."); $x = $tok; $opt{header_name} = $x->get_header_name(); $opt{trailer_name} = $x->get_trailer_name(); $opt{filter_name} = $x->get_filter_name(); } elsif ( $tok->isa('HH::Unispool::Config::File::Token::Numbered::Device::Info') ) { defined ($i) && throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::new_from_tokenizer, multiple tokens obtained from 'TOKENIZER' for entry $opt{name}/$opt{number} from class 'HH::Unispool::Config::File::Token::Numbered::Device::Info'."); $i = $tok; $opt{description} = $i->get_description(); } else { throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::new_from_tokenizer, expected tokens from 'TOKENIZER' for entry $opt{name}/$opt{number} from either class 'HH::Unispool::Config::File::Token::Numbered::Device::Info' or 'HH::Unispool::Config::File::Token::Numbered::Network'."); } } # Construct a new object and return it return( HH::Unispool::Config::Entry::Device::0->new(\%opt) ); } sub _initialize { my $self = shift; my $opt = defined($_[0]) ? shift : {}; # Check $opt ref($opt) eq 'HASH' || throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::_initialize, first argument must be 'HASH' reference."); # device_file, SINGLE, mandatory exists( $opt->{device_file} ) || throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::_initialize, option 'device_file' is mandatory."); $self->set_device_file( $opt->{device_file} ); # device_password, SINGLE exists( $opt->{device_password} ) && $self->set_device_password( $opt->{device_password} ); # execution_priority, SINGLE, with default value $self->set_execution_priority( exists( $opt->{execution_priority} ) ? $opt->{execution_priority} : $DEFAULT_VALUE{execution_priority} ); # header_name, SINGLE exists( $opt->{header_name} ) && $self->set_header_name( $opt->{header_name} ); # initially_spooled, BOOLEAN, with default value $self->set_initially_spooled( exists( $opt->{initially_spooled} ) ? $opt->{initially_spooled} : $DEFAULT_VALUE{initially_spooled} ); # networkwide, BOOLEAN, with default value $self->set_networkwide( exists( $opt->{networkwide} ) ? $opt->{networkwide} : $DEFAULT_VALUE{networkwide} ); # save_printfile, BOOLEAN, with default value $self->set_save_printfile( exists( $opt->{save_printfile} ) ? $opt->{save_printfile} : $DEFAULT_VALUE{save_printfile} ); # trailer_name, SINGLE exists( $opt->{trailer_name} ) && $self->set_trailer_name( $opt->{trailer_name} ); # unispool_control, BOOLEAN, with default value $self->set_unispool_control( exists( $opt->{unispool_control} ) ? $opt->{unispool_control} : $DEFAULT_VALUE{unispool_control} ); # unispool_header, BOOLEAN, with default value $self->set_unispool_header( exists( $opt->{unispool_header} ) ? $opt->{unispool_header} : $DEFAULT_VALUE{unispool_header} ); # Call the superclass' _initialize $self->SUPER::_initialize($opt); # Return $self return($self); } sub _value_is_allowed { my $name = shift; # Value is allowed if no ALLOW clauses exist for the named attribute if ( ! exists( $ALLOW_ISA{$name} ) && ! exists( $ALLOW_REF{$name} ) && ! exists( $ALLOW_RX{$name} ) && ! exists( $ALLOW_VALUE{$name} ) ) { return(1); } # At this point, all values in @_ must to be allowed CHECK_VALUES: foreach my $val (@_) { # Check ALLOW_ISA if ( ref($val) && exists( $ALLOW_ISA{$name} ) ) { foreach my $class ( @{ $ALLOW_ISA{$name} } ) { &UNIVERSAL::isa( $val, $class ) && next CHECK_VALUES; } } # Check ALLOW_REF if ( ref($val) && exists( $ALLOW_REF{$name} ) ) { exists( $ALLOW_REF{$name}{ ref($val) } ) && next CHECK_VALUES; } # Check ALLOW_RX if ( defined($val) && ! ref($val) && exists( $ALLOW_RX{$name} ) ) { foreach my $rx ( @{ $ALLOW_RX{$name} } ) { $val =~ /$rx/ && next CHECK_VALUES; } } # Check ALLOW_VALUE if ( ! ref($val) && exists( $ALLOW_VALUE{$name} ) ) { exists( $ALLOW_VALUE{$name}{$val} ) && next CHECK_VALUES; } # We caught a not allowed value return(0); } # OK, all values are allowed return(1); } sub diff { my $from = shift; my $to = shift; my $diff_number = shift; $diff_number = $from->is_diff_number() if ( ! defined( $diff_number ) ); # Reference types must be identical if ( ref($from) ne ref($to) ) { my $rf = ref($from); my $rt = ref($to); throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::diff, FROM ($rf) and TO ($rt) reference types differ."); } # Diff message my $diff = $from->SUPER::diff($to, $diff_number); # Diff the device file if ( $from->get_device_file() ne $to->get_device_file() ) { my $ref = ref($from); my $vf = $from->get_device_file(); my $vt = $to->get_device_file(); my $name = $from->get_name(); my $number = $from->get_number(); $diff .= "$ref/$name/$number: device file difference: $vf <-> $vt\n"; } # Diff the device password if ( $from->get_device_password() ne $to->get_device_password() ) { my $ref = ref($from); my $vf = $from->get_device_password(); my $vt = $to->get_device_password(); my $name = $from->get_name(); my $number = $from->get_number(); $diff .= "$ref/$name/$number: device password difference: $vf <-> $vt\n"; } # Diff the execution priority $diff .= $from->get_execution_priority()->diff( $to->get_execution_priority() ); # Diff the header name if ( $from->get_header_name() ne $to->get_header_name() ) { my $ref = ref($from); my $vf = $from->get_header_name(); my $vt = $to->get_header_name(); my $name = $from->get_name(); my $number = $from->get_number(); $diff .= "$ref/$name/$number: header name difference: $vf <-> $vt\n"; } # Diff the initially spooled state if ( $from->is_initially_spooled() != $to->is_initially_spooled() ) { my $ref = ref($from); my $vf = $from->is_initially_spooled(); my $vt = $to->is_initially_spooled(); my $name = $from->get_name(); my $number = $from->get_number(); $diff .= "$ref/$name/$number: initially spooled difference: $vf <-> $vt\n"; } # Diff the networkwide state if ( $from->is_networkwide() != $to->is_networkwide() ) { my $ref = ref($from); my $vf = $from->is_networkwide(); my $vt = $to->is_networkwide(); my $name = $from->get_name(); my $number = $from->get_number(); $diff .= "$ref/$name/$number: networkwide difference: $vf <-> $vt\n"; } # Diff the save printfile state if ( $from->is_save_printfile() != $to->is_save_printfile() ) { my $ref = ref($from); my $vf = $from->is_save_printfile(); my $vt = $to->is_save_printfile(); my $name = $from->get_name(); my $number = $from->get_number(); $diff .= "$ref/$name/$number: save printfile difference: $vf <-> $vt\n"; } # Diff the trailer name if ( $from->get_trailer_name() ne $to->get_trailer_name() ) { my $ref = ref($from); my $vf = $from->get_trailer_name(); my $vt = $to->get_trailer_name(); my $name = $from->get_name(); my $number = $from->get_number(); $diff .= "$ref/$name/$number: trailer name difference: $vf <-> $vt\n"; } # Diff the unispool control state if ( $from->is_unispool_control() != $to->is_unispool_control() ) { my $ref = ref($from); my $vf = $from->is_unispool_control(); my $vt = $to->is_unispool_control(); my $name = $from->get_name(); my $number = $from->get_number(); $diff .= "$ref/$name/$number: unispool control difference: $vf <-> $vt\n"; } # Diff the unispool header state if ( $from->is_unispool_header() != $to->is_unispool_header() ) { my $ref = ref($from); my $vf = $from->is_unispool_header(); my $vt = $to->is_unispool_header(); my $name = $from->get_name(); my $number = $from->get_number(); $diff .= "$ref/$name/$number: unispool header difference: $vf <-> $vt\n"; } # Return diff return($diff); } sub get_device_file { my $self = shift; return( $self->{HH_Unispool_Config_Entry_Device_0}{device_file} ); } sub get_device_password { my $self = shift; return( $self->{HH_Unispool_Config_Entry_Device_0}{device_password} ); } sub get_execution_priority { my $self = shift; return( $self->{HH_Unispool_Config_Entry_Device_0}{execution_priority} ); } sub get_header_name { my $self = shift; return( $self->{HH_Unispool_Config_Entry_Device_0}{header_name} ); } sub get_trailer_name { my $self = shift; return( $self->{HH_Unispool_Config_Entry_Device_0}{trailer_name} ); } sub is_initially_spooled { my $self = shift; if ( $self->{HH_Unispool_Config_Entry_Device_0}{initially_spooled} ) { return(1); } else { return(0); } } sub is_networkwide { my $self = shift; if ( $self->{HH_Unispool_Config_Entry_Device_0}{networkwide} ) { return(1); } else { return(0); } } sub is_save_printfile { my $self = shift; if ( $self->{HH_Unispool_Config_Entry_Device_0}{save_printfile} ) { return(1); } else { return(0); } } sub is_unispool_control { my $self = shift; if ( $self->{HH_Unispool_Config_Entry_Device_0}{unispool_control} ) { return(1); } else { return(0); } } sub is_unispool_header { my $self = shift; if ( $self->{HH_Unispool_Config_Entry_Device_0}{unispool_header} ) { return(1); } else { return(0); } } sub set_device_file { my $self = shift; my $val = shift; # Check if isa/ref/rx/value is allowed &_value_is_allowed( 'device_file', $val ) || throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::set_device_file, the specified value '$val' is not allowed."); # Assignment $self->{HH_Unispool_Config_Entry_Device_0}{device_file} = $val; } sub set_device_password { my $self = shift; my $val = shift; # Check if isa/ref/rx/value is allowed &_value_is_allowed( 'device_password', $val ) || throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::set_device_password, the specified value '$val' is not allowed."); # Assignment $self->{HH_Unispool_Config_Entry_Device_0}{device_password} = $val; } sub set_execution_priority { my $self = shift; my $val = shift; # Check if isa/ref/rx/value is allowed &_value_is_allowed( 'execution_priority', $val ) || throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::set_execution_priority, the specified value '$val' is not allowed."); # Assignment $self->{HH_Unispool_Config_Entry_Device_0}{execution_priority} = $val; } sub set_header_name { my $self = shift; my $val = shift; # Check if isa/ref/rx/value is allowed &_value_is_allowed( 'header_name', $val ) || throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::set_header_name, the specified value '$val' is not allowed."); # Assignment $self->{HH_Unispool_Config_Entry_Device_0}{header_name} = $val; } sub set_initially_spooled { my $self = shift; if (shift) { $self->{HH_Unispool_Config_Entry_Device_0}{initially_spooled} = 1; } else { $self->{HH_Unispool_Config_Entry_Device_0}{initially_spooled} = 0; } } sub set_networkwide { my $self = shift; if (shift) { $self->{HH_Unispool_Config_Entry_Device_0}{networkwide} = 1; } else { $self->{HH_Unispool_Config_Entry_Device_0}{networkwide} = 0; } } sub set_save_printfile { my $self = shift; if (shift) { $self->{HH_Unispool_Config_Entry_Device_0}{save_printfile} = 1; } else { $self->{HH_Unispool_Config_Entry_Device_0}{save_printfile} = 0; } } sub set_trailer_name { my $self = shift; my $val = shift; # Check if isa/ref/rx/value is allowed &_value_is_allowed( 'trailer_name', $val ) || throw Error::Simple("ERROR: HH::Unispool::Config::Entry::Device::0::set_trailer_name, the specified value '$val' is not allowed."); # Assignment $self->{HH_Unispool_Config_Entry_Device_0}{trailer_name} = $val; } sub set_unispool_control { my $self = shift; if (shift) { $self->{HH_Unispool_Config_Entry_Device_0}{unispool_control} = 1; } else { $self->{HH_Unispool_Config_Entry_Device_0}{unispool_control} = 0; } } sub set_unispool_header { my $self = shift; if (shift) { $self->{HH_Unispool_Config_Entry_Device_0}{unispool_header} = 1; } else { $self->{HH_Unispool_Config_Entry_Device_0}{unispool_header} = 0; } } sub write { my $self = shift; my $fh = shift; # Make the four tokens require HH::Unispool::Config::File::Token::Numbered::Device::0; my $d = HH::Unispool::Config::File::Token::Numbered::Device::0->new( { number => $self->get_number(), device_name => $self->get_name(), device_password => $self->get_device_password(), unispool_header => $self->is_unispool_header(), initially_spooled => $self->is_initially_spooled(), networkwide => $self->is_networkwide(), save_printfile => $self->is_save_printfile(), unispool_control => $self->is_unispool_control(), execution_priority => $self->get_execution_priority(), } ); require HH::Unispool::Config::File::Token::Numbered::Device::P; my $p = HH::Unispool::Config::File::Token::Numbered::Device::P->new( { number => $self->get_number(), device_file => $self->get_device_file(), } ); require HH::Unispool::Config::File::Token::Numbered::X; my $x = HH::Unispool::Config::File::Token::Numbered::X->new( { number => $self->get_number(), header_name => $self->get_header_name(), trailer_name => $self->get_trailer_name(), filter_name => $self->get_filter_name(), } ); require HH::Unispool::Config::File::Token::Numbered::Device::Info; my $i = HH::Unispool::Config::File::Token::Numbered::Device::Info->new( { number => $self->get_number(), description => $self->get_description(), } ); # Print the tokens $fh->print( $d->write_string() ); $fh->print( $p->write_string() ); $fh->print( $i->write_string() ); $fh->print( $x->write_string() ); }