source: extensions/gsdl-video/trunk/perllib/plugins/MultimediaPlugin.pm@ 18556

Last change on this file since 18556 was 18556, checked in by davidb, 15 years ago

Restructing of VideoPlugin to be like ImagePlugin (with its supporting ImageConverter plugin). The pattern was then repeated for Audio, so we now have an AudioPlugin and AudioConverter. Where possible code is shared in a Multimedia base class

File size: 8.4 KB
Line 
1######################################################################
2#
3# MultimediaPlugin.pm -- internal plugin to support processing audio and video
4# A component of the Greenstone digital library software
5# from the New Zealand Digital Library Project at the
6# University of Waikato, New Zealand.
7#
8# Copyright (C) 1999 New Zealand Digital Library Project
9#
10# This program is free software; you can redistribute it and/or modify
11# it under the terms of the GNU General Public License as published by
12# the Free Software Foundation; either version 2 of the License, or
13# (at your option) any later version.
14#
15# This program is distributed in the hope that it will be useful,
16# but WITHOUT ANY WARRANTY; without even the implied warranty of
17# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18# GNU General Public License for more details.
19#
20# You should have received a copy of the GNU General Public License
21# along with this program; if not, write to the Free Software
22# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
23#
24###########################################################################
25
26# -- Largely modeled on how ImagePlugin works
27
28package MultimediaPlugin;
29
30use strict;
31no strict 'refs'; # allow filehandles to be variables and viceversa
32no strict 'subs';
33
34use gsprintf;
35
36use BasePlugin;
37
38sub BEGIN {
39 @MultimediaPlugin::ISA = ('BasePlugin');
40
41 if (!defined $ENV{'GEXTVIDEO'}) {
42 print STDERR "Warning: Greenstone Video extension not detected.\n";
43 }
44
45}
46
47
48
49
50my $arguments =
51 [ { 'name' => "use_ascii_only_filenames",
52 'desc' => "{VideoPlug.use_ascii_only_filenames}",
53 'type' => "flag",
54 'reqd' => "no" },
55 { 'name' => "excerpt_duration",
56 'desc' => "{MultimediaPlug.excerpt_duration}",
57 'type' => "string",
58 'deft' => "",
59 'reqd' => "no" },
60 { 'name' => "converttotype",
61 'desc' => "{MultimediaPlug.converttotype}",
62 'type' => "string",
63 'deft' => "",
64 'reqd' => "no" },
65 { 'name' => "enable_streaming",
66 'desc' => "{MultimediaPlug.enable_streaming}",
67 'type' => "flag",
68 'deft' => "1",
69 'reqd' => "no" } ];
70
71my $options = { 'name' => "MultimediaPlugin",
72 'desc' => "{MultimediaPlug.desc}",
73 'abstract' => "yes",
74 'inherits' => "yes",
75 'args' => $arguments };
76
77
78sub new {
79 my ($class) = shift (@_);
80 my ($pluginlist,$inputargs,$hashArgOptLists) = @_;
81 push(@$pluginlist, $class);
82
83 push(@{$hashArgOptLists->{"ArgList"}},@{$arguments});
84 push(@{$hashArgOptLists->{"OptList"}},$options);
85
86 my $self = new BasePlugin($pluginlist, $inputargs, $hashArgOptLists);
87
88 # Check that ffmpeg is installed and available on the path (except for Windows 95/98)
89 # This test is "inherited" from ImagePlugin where ImageMagick support
90 # for building cannot be used on these two versions of Windows as they
91 # do not have enough flexablity in the use of stdout and stderr to
92 # support how our code works. It seems reasonable to assume the
93 # same is true for MultimediaPlugin work using ffmpeg.
94
95 if (($ENV{'GSDLOS'} ne "windows" || Win32::IsWinNT())) {
96
97 my $result = `ffmpeg -h 2>&1`;
98
99
100 if (!defined $result || $result !~ m/^FFmpeg version/) {
101 $self->{'ffmpeg_installed'} = 0;
102 print STDERR $result;
103 }
104 else {
105 $self->{'ffmpeg_installed'} = 1;
106 }
107
108 }
109
110 return bless $self, $class;
111}
112
113
114
115sub begin {
116 my $self = shift (@_);
117 my ($pluginfo, $base_dir, $processor, $maxdocs) = @_;
118
119 $self->SUPER::begin(@_);
120
121 $self->{'base_dir'} = $base_dir;
122}
123
124
125sub init {
126 my $self = shift (@_);
127 my ($verbosity, $outhandle, $failhandle) = @_;
128
129 $self->SUPER::init(@_);
130}
131
132# Want to hash on doc.xml rather than original file which in the case of
133# audio and video can be HUGE
134
135sub get_oid_hash_type {
136
137 my $self = shift (@_);
138
139 return "hash_on_ga_xml";
140}
141
142
143
144# Create the keyframes, thumbnail and screenview images, and discover
145# the Video's size, width, and height using the ffmpeg utility.
146
147sub run_convert {
148 my $self = shift (@_);
149
150 die "$self The inheriting class must implement run_convert method";
151}
152
153
154sub read_into_doc_obj {
155 my $self = shift (@_);
156 my ($pluginfo, $base_dir, $file, $block_hash, $metadata, $processor, $maxdocs, $total_count, $gli) = @_;
157
158 my $outhandle = $self->{'outhandle'};
159
160 # should we move this to read? What about secondary plugins?
161 print STDERR "<Processing n='$file' p='$self->{'plugin_type'}'>\n" if ($gli);
162 print $outhandle "$self->{'plugin_type'} processing $file\n"
163 if $self->{'verbosity'} > 1;
164
165 my ($filename_full_path, $filename_no_path) = &util::get_full_filenames($base_dir, $file);
166
167 # create a new document
168 my $doc_obj = new doc ($filename_full_path, "indexed_doc", $self->{'file_rename_method'});
169
170
171 my $top_section = $doc_obj->get_top_section();
172
173
174 $doc_obj->add_utf8_metadata($top_section, "Plugin", "$self->{'plugin_type'}");
175 $doc_obj->add_utf8_metadata($top_section, "FileSize", (-s $filename_full_path));
176
177 # sets the UTF8 filename (Source) for display and sets the url ref to URL encoded version
178 # of the UTF8 filename (SourceFile) for generated files
179 $self->set_Source_metadata($doc_obj, $filename_no_path);
180
181
182 # plugin specific stuff - what args do we need here??
183 unless (defined ($self->process($pluginfo, $base_dir, $file, $metadata, $doc_obj, $gli))) {
184 print STDERR "<ProcessingError n='$file'>\n" if ($gli);
185 return (-1,undef);
186 }
187
188 # include any metadata passed in from previous plugins
189 # note that this metadata is associated with the top level section
190 my $section = $doc_obj->get_top_section();
191 # can we merge these two methods??
192 $self->add_associated_files($doc_obj, $filename_full_path);
193 $self->extra_metadata ($doc_obj, $section, $metadata);
194 $self->auto_extract_metadata($doc_obj);
195
196 # if we haven't found any Title so far, assign one
197 # this was shifted to here from inside read()
198 $self->title_fallback($doc_obj,$section,$filename_no_path);
199
200 $self->add_OID($doc_obj);
201
202 return (1,$doc_obj);
203}
204
205sub add_dummy_text {
206 my $self = shift(@_);
207 my ($doc_obj, $section) = @_;
208
209 # add NoText metadata so we can hide this dummy text in format statements
210 $doc_obj->add_metadata($section, "NoText", "1");
211 $doc_obj->add_text($section, &gsprintf::lookup_string("{BasePlugin.dummy_text}",1));
212
213}
214
215
216
217
218# do plugin specific processing of doc_obj
219sub process {
220 my $self = shift (@_);
221 # options??
222 my ($pluginfo, $base_dir, $file, $metadata, $doc_obj, $gli) = @_;
223
224 my $outhandle = $self->{'outhandle'};
225
226 my ($filename_full_path, $filename_no_path) = &util::get_full_filenames($base_dir, $file);
227
228
229 if ($self->{'ffmpeg_installed'}) {
230 my $utf8_filename_no_path = $self->filepath_to_utf8($filename_no_path);
231 my $url_encoded_filename = &util::rename_file($utf8_filename_no_path, $self->{'file_rename_method'});
232
233
234 #run convert to get the thumbnail and extract size and type info
235 my $result = $self->run_convert($base_dir, $filename_full_path,
236 $url_encoded_filename, $doc_obj);
237
238 if (!defined $result) {
239 if ($gli) {
240 print STDERR "<ProcessingError n='$file'>\n";
241 }
242 print $outhandle "MultimediaPlugin: couldn't process \"$filename_full_path\"\n";
243 return (-1,undef); # error during processing
244 }
245 }
246 else {
247 if ($gli) {
248 &gsprintf(STDERR, "<Warning p='MultimediaPlugin' r='{MultimediaConverter.noconversionavailable}: {MultimediaConverter.".$self->{'no_multimedia_conversion_reason'}."}'>");
249 }
250 # all we do is add the original video file as an associated file, and set up srclink etc
251 my $assoc_file = $doc_obj->get_assocfile_from_sourcefile();
252 my $section = $doc_obj->get_top_section();
253
254 $doc_obj->associate_file($filename_full_path, $assoc_file, "", $section);
255
256 $doc_obj->add_metadata ($section, "srclink", "<a href=\"_httpprefix_/collect/[collection]/index/assoc/[assocfilepath]/[SourceFile]\">");
257 $doc_obj->add_metadata ($section, "/srclink", "</a>");
258 $doc_obj->add_metadata ($section, "srcicon", "<img src=\"_httpprefix_/collect/[collection]/index/assoc/[assocfilepath]/[SourceFile]\" width=\"100\">");
259
260 }
261 #we have no text - adds dummy text and NoText metadata
262 $self->add_dummy_text($doc_obj, $doc_obj->get_top_section());
263
264 return 1;
265
266}
267
268
269
2701;
271
272
273
274
275
276
277
278
279
280
281
Note: See TracBrowser for help on using the repository browser.