appending stdout / stderr to output
[app-cpan2pkg.git] / lib / App / CPAN2Pkg / Module.pm
blobb0acab32c0d6c6b969e1239b21c24179423b3255
2 # This file is part of App::CPAN2Pkg.
3 # Copyright (c) 2009 Jerome Quelin, all rights reserved.
5 # This program is free software; you can redistribute it and/or modify
6 # it under the same terms as Perl itself.
10 package App::CPAN2Pkg::Module;
12 use strict;
13 use warnings;
15 use Class::XSAccessor
16 constructor => '_new',
17 accessors => {
18 name => 'name',
19 shortname => 'shortname',
20 _output => '_output',
21 _prereqs => '_prereqs',
22 _wheels => '_wheels',
24 use POE;
25 use POE::Filter::Line;
26 use POE::Wheel::Run;
30 # if ( not available in cooker ) is_in_dist
31 # then
32 # compute dependencies find_prereqs
33 # repeat with each dep
34 # cpan2dist cpan2dist
35 # install local install_from_local
36 # while ( not available locally ) is_installed
37 # do
38 # prompt user to fix manually
39 # done
40 # import import_local_to_dist
41 # submit (included above)
42 # ack available (manual?)
44 # else
45 # urpmi --auto perl(module::to::install) install_from_dist
46 # fi
48 # on debian / ubuntu
49 # $ apt-file find Audio/MPD.pm
50 # libaudio-mpd-perl: /usr/share/perl5/Audio/MPD.pm
51 # status:
52 # - find dist hosting module
53 # - computing dependencies
54 # - installing dependencies
55 # - check cooker availability
56 # - cpan2dist
57 # - install local
58 # - check local availability
59 # - mdvsys import
60 # - mdvsys submit
61 # - wait for kenobi build
67 #--
68 # CONSTRUCTOR
70 sub spawn {
71 my ($class, $module) = @_;
73 # creating the object
74 my $short = $module;
75 $short =~ s/::/:/g;
76 $short =~ s/[[:lower:]]//g;
77 my $obj = App::CPAN2Pkg::Module->_new(
78 name => $module,
79 shortname => $short,
80 _prereqs => {},
81 _wheels => {},
84 # spawning the session
85 my $session = POE::Session->create(
86 inline_states => {
87 # public events
88 find_prereqs => \&find_prereqs,
89 is_in_dist => \&is_in_dist,
90 # private events
91 _find_prereqs => \&_find_prereqs,
92 _stderr => \&_stderr,
93 _stdout => \&_stdout,
94 # poe inline states
95 _start => \&_start,
96 _stop => sub { warn "stop"; },
98 heap => $obj,
100 return $session->ID;
105 # SUBS
107 # -- public events
109 sub find_prereqs {
110 my ($k, $self) = @_[KERNEL, HEAP];
112 # preparing command
113 my $module = $self->name;
114 my $cmd = "cpanp /prereqs show $module";
115 $self->_log_new_step($k, 'Finding module prereqs',
116 "Running command: $cmd" );
118 # running command
119 $self->_output('');
120 $ENV{LC_ALL} = 'C';
121 my $wheel = POE::Wheel::Run->new(
122 Program => $cmd,
123 CloseEvent => '_find_prereqs',
124 StdoutEvent => '_stdout',
125 StderrEvent => '_stderr',
126 StdoutFilter => POE::Filter::Line->new,
127 StderrFilter => POE::Filter::Line->new,
130 # need to store the wheel, otherwise the process goes woo!
131 $self->_wheels->{ $wheel->ID } = $wheel;
134 sub is_in_dist {
135 my ($k, $self) = @_[KERNEL, HEAP];
137 # preparing command
138 my $name = $self->name;
139 my $cmd = "urpmq --whatprovides 'perl($name)'";
140 $self->_log_new_step($k, 'Checking if packaged upstream',
141 "Running command: $cmd" );
143 # running command
144 $self->_output('');
145 $ENV{LC_ALL} = 'C';
146 my $wheel = POE::Wheel::Run->new(
147 Program => $cmd,
148 CloseEvent => '_is_in_dist',
149 StdoutEvent => '_stdout',
150 StderrEvent => '_stderr',
151 Conduit => 'pty-pipe', # urpmq wants a pty
152 StdoutFilter => POE::Filter::Line->new,
153 StderrFilter => POE::Filter::Line->new,
156 # need to store the wheel, otherwise the process goes woo!
157 $self->_wheels->{ $wheel->ID } = $wheel;
160 # -- private events
162 sub _find_prereqs {
163 my ($k, $self, $id) = @_[KERNEL, HEAP, ARG0];
165 # terminate wheel
166 my $wheel = delete $self->{_wheels}->{$id};
168 # extract prereqs
169 my @lines =
170 grep { s/^\s+// }
171 split /\n/, $self->_output;
172 shift @lines; # remove the title line
173 my @prereqs =
174 map { (split /\s+/, $_)[0] }
175 @lines;
177 # store prereqs
178 foreach my $prereq ( @prereqs ) {
179 $k->post('ui', 'append', $self, "prereq found: $prereq\n");
180 $self->_prereqs->{$prereq} = 1;
183 $k->post('app', 'prereqs', $self, @prereqs);
186 sub _stderr {
187 my ($k, $self, $line) = @_[KERNEL, HEAP, ARG0];
188 $k->post('ui', 'append', $self, "stderr: $line\n");
191 sub _stdout {
192 my ($k, $self, $line) = @_[KERNEL, HEAP, ARG0];
193 $line .= "\n";
194 $self->_output( $self->_output . $line );
195 $k->post('ui', 'append', $self, "stdout: $line");
199 # -- poe inline states
201 sub _start {
202 my ($k, $self) = @_[KERNEL, HEAP];
204 $k->alias_set($self);
205 $k->post('ui', 'new_module', $self);
206 $k->post('app', 'new_module', $self);
207 $k->yield('is_in_dist');
212 # METHODS
214 # -- private methods
216 sub _log_new_step {
217 my ($self, $k, $step, $comment) = @_;
219 my $out = "\n\n" . '*' x 10 . "\n$step\n\n$comment\n\n";
220 $k->post('ui', 'append', $self, $out);
224 __END__
227 =head1 NAME
229 App::CPAN2Pkg::Module - poe session to drive a module packaging
233 =head1 DESCRIPTION
235 C<App::CPAN2Pkg::Module> implements a POE session driving the whole
236 packaging process of a given module.
238 It is spawned by C<App::CPAN2Pkg> and implements the logic related to
239 the module availability in the distribution.
243 =head1 PUBLIC PACKAGE METHODS
245 =head2 my $id = App::CPAN2Pkg::Module->spawn( $module )
247 This method will create a POE session responsible for packaging &
248 installing the wanted C<$module>.
250 It will return the POE id of the session newly created.
254 =head1 PUBLIC EVENTS ACCEPTED
256 =head2 find_prereqs()
258 Start looking for any other module needed by current module.
261 =head2 is_in_dist()
263 Check whether the package is provided by an existing upstream package.
266 =head1 METHODS
268 This package is also a class, used B<internally> to store private data
269 needed for the packaging stuff. The following accessors are therefore
270 available, but should not be used directly:
272 =over 4
274 =item name() - the module name
276 =item shortname() - the module shortname (only capital letters)
278 =back
282 =head1 SEE ALSO
284 For all related information (bug reporting, source code repository,
285 etc.), refer to C<App::CPAN2Pkg>'s pod, section C<SEE ALSO>.
289 =head1 AUTHOR
291 Jerome Quelin, C<< <jquelin@cpan.org> >>
295 =head1 COPYRIGHT & LICENSE
297 Copyright (c) 2009 Jerome Quelin, all rights reserved.
299 This program is free software; you can redistribute it and/or modify
300 it under the same terms as Perl itself.
302 =cut