| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package Solaris::SMF::Service; |
|
2
|
|
|
|
|
|
|
BEGIN { |
|
3
|
3
|
|
|
3
|
|
11
|
eval { |
|
4
|
3
|
|
|
|
|
79
|
require Data::Dumper; |
|
5
|
|
|
|
|
|
|
} |
|
6
|
|
|
|
|
|
|
}; |
|
7
|
|
|
|
|
|
|
|
|
8
|
3
|
|
|
3
|
|
15
|
use warnings; |
|
|
3
|
|
|
|
|
6
|
|
|
|
3
|
|
|
|
|
75
|
|
|
9
|
3
|
|
|
3
|
|
16
|
use strict; |
|
|
3
|
|
|
|
|
4
|
|
|
|
3
|
|
|
|
|
97
|
|
|
10
|
3
|
|
|
3
|
|
16
|
use Params::Validate qw( validate validate_pos :types ); |
|
|
3
|
|
|
|
|
5
|
|
|
|
3
|
|
|
|
|
572
|
|
|
11
|
3
|
|
|
3
|
|
17
|
use Carp; |
|
|
3
|
|
|
|
|
6
|
|
|
|
3
|
|
|
|
|
4761
|
|
|
12
|
|
|
|
|
|
|
|
|
13
|
|
|
|
|
|
|
my $debug = $ENV{RELEASE_TESTING} ? $ENV{RELEASE_TESTING} : 0; |
|
14
|
|
|
|
|
|
|
|
|
15
|
|
|
|
|
|
|
=head1 NAME |
|
16
|
|
|
|
|
|
|
|
|
17
|
|
|
|
|
|
|
Solaris::SMF::Service - Encapsulate Solaris 10 services in Perl |
|
18
|
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
=head1 VERSION |
|
20
|
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
Version 0.02 |
|
22
|
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
=cut |
|
24
|
|
|
|
|
|
|
|
|
25
|
|
|
|
|
|
|
our $VERSION = '0.02'; |
|
26
|
|
|
|
|
|
|
|
|
27
|
|
|
|
|
|
|
=head1 SYNOPSIS |
|
28
|
|
|
|
|
|
|
|
|
29
|
|
|
|
|
|
|
Interface to Sun's Service Management Facility in Solaris 10. This module provides |
|
30
|
|
|
|
|
|
|
a wrapper around 'svcs', 'svcadm' and 'svccfg'. |
|
31
|
|
|
|
|
|
|
|
|
32
|
|
|
|
|
|
|
The SMF in Solaris is a replacement for inetd as well as the runlevel-based stopping |
|
33
|
|
|
|
|
|
|
and starting of daemons. Service definitions are stored in an XML database. |
|
34
|
|
|
|
|
|
|
|
|
35
|
|
|
|
|
|
|
The biggest advantages in using SMF are the resiliency support, consistent interface and |
|
36
|
|
|
|
|
|
|
inter-service dependencies it offers. Services that die for any reason can be automatically |
|
37
|
|
|
|
|
|
|
restarted by the operating system; all services can be enabled or disabled using the same |
|
38
|
|
|
|
|
|
|
commands; and services can be started as soon as all the services they depend upon have |
|
39
|
|
|
|
|
|
|
been started, rather than at a fixed point in the boot process. |
|
40
|
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
=head1 METHODS |
|
42
|
|
|
|
|
|
|
|
|
43
|
|
|
|
|
|
|
=cut |
|
44
|
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
sub _svcs { |
|
46
|
0
|
|
|
0
|
|
|
my $self = shift; |
|
47
|
0
|
|
|
|
|
|
local $ENV{PATH} = '/bin:/usr/bin:/sbin:/usr/sbin'; |
|
48
|
0
|
0
|
|
|
|
|
open my $svc_list, '-|', " svcs -aH '$self->{FMRI}' 2>/dev/null" |
|
49
|
|
|
|
|
|
|
or croak 'Unable to query SMF services'; |
|
50
|
0
|
|
|
|
|
|
while ( my $svc_line = <$svc_list> ) { |
|
51
|
0
|
|
|
|
|
|
my ( $state, $date, $FMRI ) = ( |
|
52
|
|
|
|
|
|
|
$svc_line =~ m/ |
|
53
|
|
|
|
|
|
|
^ |
|
54
|
|
|
|
|
|
|
([^\s]+) # Current state |
|
55
|
|
|
|
|
|
|
[\s]+ |
|
56
|
|
|
|
|
|
|
([^\s]+) # Date this state was set |
|
57
|
|
|
|
|
|
|
[\s]+ |
|
58
|
|
|
|
|
|
|
( (?: svc: | lrc: ) [^\s]+ ) # FMRI |
|
59
|
|
|
|
|
|
|
\n? |
|
60
|
|
|
|
|
|
|
$ |
|
61
|
|
|
|
|
|
|
/xms |
|
62
|
|
|
|
|
|
|
); |
|
63
|
0
|
0
|
|
|
|
|
if ($FMRI) { |
|
64
|
0
|
|
|
|
|
|
close $svc_list; |
|
65
|
0
|
|
|
|
|
|
return ( $state, $date ); |
|
66
|
|
|
|
|
|
|
} |
|
67
|
|
|
|
|
|
|
} |
|
68
|
0
|
|
|
|
|
|
croak "Unable to determine status of $self->{FMRI}"; |
|
69
|
|
|
|
|
|
|
} |
|
70
|
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
sub _svcprop { |
|
72
|
0
|
0
|
|
0
|
|
|
$debug && warn( '_svcprop ' . join( ',', @_ ) ); |
|
73
|
0
|
|
|
|
|
|
my $self = shift; |
|
74
|
0
|
|
|
|
|
|
local $ENV{PATH} = '/bin:/usr/bin:/sbin:/usr/sbin'; |
|
75
|
0
|
0
|
|
|
|
|
open my $svcprop_list, '-|', " svcprop '$self->{FMRI}' 2>/dev/null" |
|
76
|
|
|
|
|
|
|
or croak 'Unable to query SMF service properties'; |
|
77
|
0
|
|
|
|
|
|
my %properties; |
|
78
|
0
|
|
|
|
|
|
while ( my $svcprop_line = <$svcprop_list> ) { |
|
79
|
0
|
|
|
|
|
|
my ( $name, $type, $value ) = ( |
|
80
|
|
|
|
|
|
|
$svcprop_line =~ m/ |
|
81
|
|
|
|
|
|
|
^ |
|
82
|
|
|
|
|
|
|
([^\s]+) # Property name |
|
83
|
|
|
|
|
|
|
[\s]+ |
|
84
|
|
|
|
|
|
|
([^\s]+) # Type of property |
|
85
|
|
|
|
|
|
|
[\s]+ |
|
86
|
|
|
|
|
|
|
([^\s]*[^\n]*) # Value of property |
|
87
|
|
|
|
|
|
|
$ |
|
88
|
|
|
|
|
|
|
/xms |
|
89
|
|
|
|
|
|
|
); |
|
90
|
0
|
0
|
|
|
|
|
if ($name) { |
|
91
|
0
|
|
|
|
|
|
$properties{$name}{type} = $type; |
|
92
|
0
|
|
|
|
|
|
$properties{$name}{value} = $value; |
|
93
|
|
|
|
|
|
|
} |
|
94
|
0
|
0
|
|
|
|
|
$debug && print STDERR Data::Dumper->Dump( [$name, $type, $value], [qw($name $type $value)] ); |
|
95
|
|
|
|
|
|
|
} |
|
96
|
0
|
0
|
|
|
|
|
$debug && print STDERR Data::Dumper->Dump( [\%properties], [qw(%properties)] ); |
|
97
|
0
|
|
|
|
|
|
return \%properties; |
|
98
|
|
|
|
|
|
|
} |
|
99
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
sub _svcadm { |
|
101
|
0
|
0
|
|
0
|
|
|
$debug && warn( '_svcadm ' . join( ',', @_ ) ); |
|
102
|
0
|
|
|
|
|
|
my $self = shift; |
|
103
|
0
|
|
|
|
|
|
my $svcadm_action = shift; |
|
104
|
0
|
|
|
|
|
|
local $ENV{PATH} = '/bin:/usr/bin:/sbin:/usr/sbin'; |
|
105
|
0
|
0
|
|
|
|
|
open my $svc_adm, '-|', " svcadm $svcadm_action '$self->{FMRI}' 2>&1" |
|
106
|
|
|
|
|
|
|
or croak 'Unable to administer SMF services'; |
|
107
|
0
|
|
|
|
|
|
close $svc_adm; |
|
108
|
|
|
|
|
|
|
} |
|
109
|
|
|
|
|
|
|
|
|
110
|
|
|
|
|
|
|
=head2 new |
|
111
|
|
|
|
|
|
|
|
|
112
|
|
|
|
|
|
|
Create a new Service object. The parameter must be a valid, unique FMRI. |
|
113
|
|
|
|
|
|
|
|
|
114
|
|
|
|
|
|
|
=cut |
|
115
|
|
|
|
|
|
|
|
|
116
|
|
|
|
|
|
|
sub new { |
|
117
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'new ' . join( ',', @_ ) ); |
|
118
|
0
|
|
|
|
|
|
my $class = shift; |
|
119
|
0
|
|
|
|
|
|
my $FMRI = shift; |
|
120
|
0
|
|
|
|
|
|
my $service = bless {}, __PACKAGE__; |
|
121
|
0
|
|
|
|
|
|
$service->{FMRI} = $FMRI; |
|
122
|
0
|
|
|
|
|
|
return $service; |
|
123
|
|
|
|
|
|
|
} |
|
124
|
|
|
|
|
|
|
|
|
125
|
|
|
|
|
|
|
=head2 status |
|
126
|
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
Get the current status of this service. Returns a string, 'disabled', 'enabled', 'offline'. |
|
128
|
|
|
|
|
|
|
|
|
129
|
|
|
|
|
|
|
=cut |
|
130
|
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
sub status { |
|
132
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'status ' . join( ',', @_ ) ); |
|
133
|
0
|
|
|
|
|
|
my $self = shift; |
|
134
|
0
|
|
|
|
|
|
my ( $status, $date ) = $self->_svcs(); |
|
135
|
0
|
0
|
|
|
|
|
$debug |
|
136
|
|
|
|
|
|
|
&& warn( Data::Dumper->Dump( [ $status, $date ], [qw($status $date)] ) ); |
|
137
|
0
|
|
|
|
|
|
return $status; |
|
138
|
|
|
|
|
|
|
} |
|
139
|
|
|
|
|
|
|
|
|
140
|
|
|
|
|
|
|
=head2 FMRI |
|
141
|
|
|
|
|
|
|
|
|
142
|
|
|
|
|
|
|
Returns the Fault Managed Resource Identifier for this service. |
|
143
|
|
|
|
|
|
|
|
|
144
|
|
|
|
|
|
|
=cut |
|
145
|
|
|
|
|
|
|
|
|
146
|
|
|
|
|
|
|
sub FMRI { |
|
147
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'FMRI ' . join( ',', @_ ) ); |
|
148
|
0
|
|
|
|
|
|
my $self = shift; |
|
149
|
0
|
|
|
|
|
|
return $self->{FMRI}; |
|
150
|
|
|
|
|
|
|
} |
|
151
|
|
|
|
|
|
|
|
|
152
|
|
|
|
|
|
|
=head2 properties |
|
153
|
|
|
|
|
|
|
|
|
154
|
|
|
|
|
|
|
Returns all or some properties for this service. |
|
155
|
|
|
|
|
|
|
|
|
156
|
|
|
|
|
|
|
=cut |
|
157
|
|
|
|
|
|
|
|
|
158
|
|
|
|
|
|
|
sub properties { |
|
159
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'properties ' . join( ',', @_ ) ); |
|
160
|
0
|
|
|
|
|
|
my $self = shift; |
|
161
|
0
|
|
|
|
|
|
my $properties = $self->_svcprop(); |
|
162
|
0
|
|
|
|
|
|
return %{$properties}; |
|
|
0
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
} |
|
164
|
|
|
|
|
|
|
|
|
165
|
|
|
|
|
|
|
=head2 property |
|
166
|
|
|
|
|
|
|
|
|
167
|
|
|
|
|
|
|
Returns the value of a single property of this service. |
|
168
|
|
|
|
|
|
|
|
|
169
|
|
|
|
|
|
|
=cut |
|
170
|
|
|
|
|
|
|
|
|
171
|
|
|
|
|
|
|
sub property { |
|
172
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'property ' . join( ',', @_ ) ); |
|
173
|
0
|
|
|
|
|
|
my $self = shift; |
|
174
|
0
|
|
|
|
|
|
my $p = validate_pos( @_, { type => SCALAR } ); |
|
175
|
0
|
|
|
|
|
|
my ($property_name) = @{$p}; |
|
|
0
|
|
|
|
|
|
|
|
176
|
|
|
|
|
|
|
|
|
177
|
0
|
|
|
|
|
|
my $properties = $self->_svcprop(); |
|
178
|
0
|
0
|
|
|
|
|
$debug && warn( Data::Dumper->Dump( [$properties], [qw($properties)] ) ); |
|
179
|
0
|
0
|
|
|
|
|
if ( defined $properties->{$property_name} ) { |
|
180
|
0
|
|
|
|
|
|
return $properties->{$property_name}{value}; |
|
181
|
|
|
|
|
|
|
} |
|
182
|
|
|
|
|
|
|
else { |
|
183
|
0
|
|
|
|
|
|
carp "Unable to find property '$property_name' for " . $self->{FMRI}; |
|
184
|
0
|
|
|
|
|
|
return undef; |
|
185
|
|
|
|
|
|
|
} |
|
186
|
|
|
|
|
|
|
} |
|
187
|
|
|
|
|
|
|
|
|
188
|
|
|
|
|
|
|
=head2 property_type |
|
189
|
|
|
|
|
|
|
|
|
190
|
|
|
|
|
|
|
Returns the type of a single property of this service. |
|
191
|
|
|
|
|
|
|
|
|
192
|
|
|
|
|
|
|
=cut |
|
193
|
|
|
|
|
|
|
|
|
194
|
|
|
|
|
|
|
sub property_type { |
|
195
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'property_type ' . join( ',', @_ ) ); |
|
196
|
0
|
|
|
|
|
|
my $self = shift; |
|
197
|
0
|
|
|
|
|
|
my $p = validate_pos( @_, { type => SCALAR } ); |
|
198
|
0
|
|
|
|
|
|
my ($property_name) = @{$p}; |
|
|
0
|
|
|
|
|
|
|
|
199
|
|
|
|
|
|
|
|
|
200
|
0
|
|
|
|
|
|
my $properties = $self->_svcprop(); |
|
201
|
0
|
0
|
|
|
|
|
$debug && warn( Data::Dumper->Dump([$properties], [qw($properties)]) ); |
|
202
|
0
|
0
|
|
|
|
|
if ( defined $properties->{$property_name} ) { |
|
203
|
0
|
|
|
|
|
|
return $properties->{$property_name}{type}; |
|
204
|
|
|
|
|
|
|
} |
|
205
|
|
|
|
|
|
|
else { |
|
206
|
0
|
|
|
|
|
|
carp "Unable to find property '$property_name' for " . $self->{FMRI}; |
|
207
|
0
|
|
|
|
|
|
return undef; |
|
208
|
|
|
|
|
|
|
} |
|
209
|
|
|
|
|
|
|
} |
|
210
|
|
|
|
|
|
|
|
|
211
|
|
|
|
|
|
|
=head2 disable |
|
212
|
|
|
|
|
|
|
|
|
213
|
|
|
|
|
|
|
This instructs SMF to disable the service permanently. To disable temporarily, |
|
214
|
|
|
|
|
|
|
that is until the next time the server is rebooted, use the 'stop' method. |
|
215
|
|
|
|
|
|
|
|
|
216
|
|
|
|
|
|
|
=cut |
|
217
|
|
|
|
|
|
|
sub disable { |
|
218
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'disable ' . join( ',', @_ ) ); |
|
219
|
0
|
|
|
|
|
|
my $self = shift; |
|
220
|
0
|
|
|
|
|
|
return $self->_svcadm('disable'); |
|
221
|
|
|
|
|
|
|
} |
|
222
|
|
|
|
|
|
|
|
|
223
|
|
|
|
|
|
|
=head2 stop |
|
224
|
|
|
|
|
|
|
|
|
225
|
|
|
|
|
|
|
This instructs SMF to stop the service. It uses the -t flag to svcadm, so that |
|
226
|
|
|
|
|
|
|
using this call will not prevent the service from starting the next time the |
|
227
|
|
|
|
|
|
|
server reboots. |
|
228
|
|
|
|
|
|
|
|
|
229
|
|
|
|
|
|
|
=cut |
|
230
|
|
|
|
|
|
|
sub stop { |
|
231
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'stop ' . join( ',', @_ ) ); |
|
232
|
0
|
|
|
|
|
|
my $self = shift; |
|
233
|
0
|
|
|
|
|
|
return $self->_svcadm('disable -t'); |
|
234
|
|
|
|
|
|
|
} |
|
235
|
|
|
|
|
|
|
|
|
236
|
|
|
|
|
|
|
=head2 enable |
|
237
|
|
|
|
|
|
|
|
|
238
|
|
|
|
|
|
|
This instructs SMF to enable the service permanently. To enable temporarily, |
|
239
|
|
|
|
|
|
|
that is until the next time the server is rebooted, see the 'start' method. |
|
240
|
|
|
|
|
|
|
|
|
241
|
|
|
|
|
|
|
=cut |
|
242
|
|
|
|
|
|
|
sub enable { |
|
243
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'enable ' . join( ',', @_ ) ); |
|
244
|
0
|
|
|
|
|
|
my $self = shift; |
|
245
|
0
|
|
|
|
|
|
return $self->_svcadm('enable'); |
|
246
|
|
|
|
|
|
|
} |
|
247
|
|
|
|
|
|
|
|
|
248
|
|
|
|
|
|
|
=head2 start |
|
249
|
|
|
|
|
|
|
|
|
250
|
|
|
|
|
|
|
This instructs SMF to start the service. This change is not made persistent |
|
251
|
|
|
|
|
|
|
unless you use the 'enable' method. |
|
252
|
|
|
|
|
|
|
|
|
253
|
|
|
|
|
|
|
=cut |
|
254
|
|
|
|
|
|
|
sub start { |
|
255
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'start ' . join( ',', @_ ) ); |
|
256
|
0
|
|
|
|
|
|
my $self = shift; |
|
257
|
0
|
|
|
|
|
|
return $self->_svcadm('enable -t'); |
|
258
|
|
|
|
|
|
|
} |
|
259
|
|
|
|
|
|
|
|
|
260
|
|
|
|
|
|
|
=head2 refresh |
|
261
|
|
|
|
|
|
|
|
|
262
|
|
|
|
|
|
|
This instructs SMF to refresh the service. Needed whenever alterations are |
|
263
|
|
|
|
|
|
|
made to a service's properties. It acts as the analogue of a SQL 'commit'. |
|
264
|
|
|
|
|
|
|
|
|
265
|
|
|
|
|
|
|
=cut |
|
266
|
|
|
|
|
|
|
sub refresh { |
|
267
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'refresh ' . join( ',', @_ ) ); |
|
268
|
0
|
|
|
|
|
|
my $self = shift; |
|
269
|
0
|
|
|
|
|
|
return $self->_svcadm('refresh'); |
|
270
|
|
|
|
|
|
|
} |
|
271
|
|
|
|
|
|
|
|
|
272
|
|
|
|
|
|
|
=head2 clear |
|
273
|
|
|
|
|
|
|
|
|
274
|
|
|
|
|
|
|
This instructs SMF to clear the service's state, that is, to remove the |
|
275
|
|
|
|
|
|
|
'failed' marker from it. This is needed prior to starting a failed service. |
|
276
|
|
|
|
|
|
|
|
|
277
|
|
|
|
|
|
|
=cut |
|
278
|
|
|
|
|
|
|
sub clear { |
|
279
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'clear ' . join( ',', @_ ) ); |
|
280
|
0
|
|
|
|
|
|
my $self = shift; |
|
281
|
0
|
|
|
|
|
|
return $self->_svcadm('clear'); |
|
282
|
|
|
|
|
|
|
} |
|
283
|
|
|
|
|
|
|
|
|
284
|
|
|
|
|
|
|
=head2 mark |
|
285
|
|
|
|
|
|
|
|
|
286
|
|
|
|
|
|
|
This instructs SMF to mark the service as failed. |
|
287
|
|
|
|
|
|
|
|
|
288
|
|
|
|
|
|
|
=cut |
|
289
|
|
|
|
|
|
|
sub mark { |
|
290
|
0
|
0
|
|
0
|
1
|
|
$debug && warn( 'mark ' . join( ',', @_ ) ); |
|
291
|
0
|
|
|
|
|
|
my $self = shift; |
|
292
|
0
|
|
|
|
|
|
return $self->_svcadm('mark'); |
|
293
|
|
|
|
|
|
|
} |
|
294
|
|
|
|
|
|
|
|
|
295
|
|
|
|
|
|
|
=head1 AUTHOR |
|
296
|
|
|
|
|
|
|
|
|
297
|
|
|
|
|
|
|
Brad Macpherson, C<< >> |
|
298
|
|
|
|
|
|
|
|
|
299
|
|
|
|
|
|
|
=head1 BUGS |
|
300
|
|
|
|
|
|
|
|
|
301
|
|
|
|
|
|
|
Please report any bugs or feature requests to C, or through |
|
302
|
|
|
|
|
|
|
the web interface at L. I will be notified, and then you'll |
|
303
|
|
|
|
|
|
|
automatically be notified of progress on your bug as I make changes. |
|
304
|
|
|
|
|
|
|
|
|
305
|
|
|
|
|
|
|
|
|
306
|
|
|
|
|
|
|
|
|
307
|
|
|
|
|
|
|
|
|
308
|
|
|
|
|
|
|
=head1 SUPPORT |
|
309
|
|
|
|
|
|
|
|
|
310
|
|
|
|
|
|
|
You can find documentation for this module with the perldoc command. |
|
311
|
|
|
|
|
|
|
|
|
312
|
|
|
|
|
|
|
perldoc Solaris::SMF::Service |
|
313
|
|
|
|
|
|
|
|
|
314
|
|
|
|
|
|
|
|
|
315
|
|
|
|
|
|
|
You can also look for information at: |
|
316
|
|
|
|
|
|
|
|
|
317
|
|
|
|
|
|
|
=over 4 |
|
318
|
|
|
|
|
|
|
|
|
319
|
|
|
|
|
|
|
=item * RT: CPAN's request tracker |
|
320
|
|
|
|
|
|
|
|
|
321
|
|
|
|
|
|
|
L |
|
322
|
|
|
|
|
|
|
|
|
323
|
|
|
|
|
|
|
=item * AnnoCPAN: Annotated CPAN documentation |
|
324
|
|
|
|
|
|
|
|
|
325
|
|
|
|
|
|
|
L |
|
326
|
|
|
|
|
|
|
|
|
327
|
|
|
|
|
|
|
=item * CPAN Ratings |
|
328
|
|
|
|
|
|
|
|
|
329
|
|
|
|
|
|
|
L |
|
330
|
|
|
|
|
|
|
|
|
331
|
|
|
|
|
|
|
=item * Search CPAN |
|
332
|
|
|
|
|
|
|
|
|
333
|
|
|
|
|
|
|
L |
|
334
|
|
|
|
|
|
|
|
|
335
|
|
|
|
|
|
|
=back |
|
336
|
|
|
|
|
|
|
|
|
337
|
|
|
|
|
|
|
|
|
338
|
|
|
|
|
|
|
=head1 ACKNOWLEDGEMENTS |
|
339
|
|
|
|
|
|
|
|
|
340
|
|
|
|
|
|
|
|
|
341
|
|
|
|
|
|
|
=head1 COPYRIGHT & LICENCE |
|
342
|
|
|
|
|
|
|
|
|
343
|
|
|
|
|
|
|
Copyright 2009 Brad Macpherson. |
|
344
|
|
|
|
|
|
|
|
|
345
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify it |
|
346
|
|
|
|
|
|
|
under the terms of either: the GNU General Public Licence as published |
|
347
|
|
|
|
|
|
|
by the Free Software Foundation; or the Artistic Licence. |
|
348
|
|
|
|
|
|
|
|
|
349
|
|
|
|
|
|
|
See http://dev.perl.org/licenses/ for more information. |
|
350
|
|
|
|
|
|
|
|
|
351
|
|
|
|
|
|
|
|
|
352
|
|
|
|
|
|
|
=cut |
|
353
|
|
|
|
|
|
|
|
|
354
|
|
|
|
|
|
|
1; # End of Solaris::SMF::Service |