]>
diplodocus.org Git - flac-archive/blob - fa-flacd
8 B<fa-flacd> - archive CDs to single FLAC files
12 B<fa-flacd> [B<-j> I<jobs>] [B<-v>]
20 use Getopt
::Long
qw(:config gnu_getopt no_ignore_case);
21 use POSIX
':sys_wait_h';
30 $verbose and print(STDERR
$_) for @_;
33 # Return the ARTIST, ALBUM, and DATE followed by a list of all the
34 # lines in the file FN.
44 verbose
("Opening tags file $fn\n");
45 open(TAGS
, $fn) or die("open($fn): $!");
50 ($tag, $value) = split(/=/, $_, 2);
54 verbose
("ARTIST $artist from $fn\n");
55 } elsif (/^ALBUM=/i) {
57 verbose
("ALBUM $album from $fn\n"); # cperl-mode sucks "
58 } elsif (/^DISCNUMBER=/i) {
59 $discnum = int($value);
60 verbose
("DISCNUMBER $discnum from $fn\n");
63 close(TAGS
) or die("close($fn): $!");
65 return ($artist, $album, $discnum, @tags);
68 # Hash of Hashes of Lists Push
75 return push(@{$hash->{$key1}->{$key2}}, $val);
82 while (my ($key, $vall) = each(%$h)) {
83 for my $val (@$vall) {
84 push(@result, "$key=$val")
91 # Process the fa-rip output in the directory DIR.
101 verbose
("Renaming $dir/tags\n");
102 rename("$dir/tags", "$dir/using-tags")
103 or die("rename($dir/tags, $dir/using-tags): $!");
105 ($artist, $album, $discnum, @tags) = get_tags
("$dir/using-tags");
106 for ($artist, $album) {
110 verbose
("mkdir($artist)\n");
111 -d
$artist or mkdir($artist) or die("mkdir($artist): $!");
113 my $outdir = "$artist/$album";
114 verbose
("mkdir($outdir)\n");
115 -d
"$outdir" or mkdir("$outdir") or die("mkdir($outdir): $!");
117 verbose
("chdir($dir)\n");
118 chdir($dir) or die("chdir($dir): $!");
123 for my $tag (@tags) {
124 if ($tag =~ /^([^[]+)\[(\d+)]=(.*)/) {
125 hhlp
(\
%things, $2, $1, $3);
126 } elsif ($tag =~ /^ARTIST=/) {
135 for my $tracknum (sort(map(int, keys(%things)))) {
136 my $title = join(' ', map(split, @{$things{$tracknum}->{'TITLE'}}));
140 ? sprintf('%02d', $discnum)
142 sprintf('%02d', $tracknum),
144 push(@files, "$outdir/$outfile.flac");
145 $outfile = "../$outdir/$outfile";
148 if (exists($things{$tracknum}->{'ARTIST'})) {
154 verbose
("Running flac\n");
155 $status = system('flac', '-o', "$outfile.flac-tmp",
156 '--delete-input-file', '-V', '--no-padding', '--best',
159 grep({ $_ !~ /^ARTIST=/ } @tags),
160 bork_tags
($things{$tracknum})),
161 sprintf('track%02d.cdda.wav', $tracknum));
162 if (WIFEXITED
($status)) {
163 if (($status = WEXITSTATUS
($status)) != 0) {
164 die("flac exited with status $status");
166 } elsif (WIFSIGNALED
($status)) {
167 die("flac killed with signal ", WTERMSIG
($status));
168 } elsif (WIFSTOPPED
($status)) {
169 die("flac stopped with signal ", WSTOPSIG
($status));
171 die("Major horkage on system(flac): \$? = $? \$! = $!");
174 rename("$outfile.flac-tmp", "$outfile.flac")
175 or die("rename($outfile.flac-tmp, $outfile.flac): $!");
178 verbose
("Cleaning up $dir\n");
179 unlink('using-tags') or die("unlink(using-tags): $!");
180 unlink('cue') or die("unlink(cue): $!");
181 rename('log', "../$outdir/log")
182 or die("rename(log, ../$outdir/log): $!");
183 chdir('..') or die("chdir(..): $!");
185 if (-x
"$dir/post-processor") {
186 verbose
("Running './$dir/post-processor'\n");
187 system("./$dir/post-processor", @files);
188 unlink("$dir/post-processor") or die("unlink($dir/post-processor): $!");
191 rmdir($dir) or die("rmdir($dir): $!");
199 while (($pid = waitpid(-1, WNOHANG
)) > 0) {
200 push(@finished, [$pid, $?]);
203 $SIG{CHLD
} = \
&reaper
;
212 if (not defined($pid)) {
217 if ($debug or $pid == 0) {
218 $SIG{CHLD
} = 'DEFAULT';
219 open(STDERR
, ">$dir/log") or die("open(STDERR, >$dir/log): $!");
223 verbose
("new job $pid for $dir\n");
233 $pid = $finished[$i][0];
234 $status = $finished[$i][1];
236 verbose
("$pid finished (");
237 if (WIFEXITED
($status)) {
238 verbose
('exited with status ', WEXITSTATUS
($status));
239 } elsif (WIFSIGNALED
($status)) {
240 verbose
('killed with signal ', WTERMSIG
($status));
241 } elsif (WIFSTOPPED
($status)) {
242 verbose
('stopped with signal ', WSTOPSIG
($status));
246 for ($j = 0; $j <= $#jobs; $j++) {
247 $pid == $jobs[$j] and splice(@jobs, $j, 1) and last;
250 splice(@finished, $i, 1);
259 $SIG{CHLD
} = \
&reaper
;
261 if (scalar(@jobs) <= $MAXJOBS) {
262 foreach $i (glob('*/tags')) {
263 push(@jobs, newjob
(dirname
($i))) <= $MAXJOBS or last;
267 for ($i = 0; $i <= $#finished; $i++) {
271 verbose
(scalar(@jobs), " jobs\n");
282 'debug|X' => \
$debug,
283 'jobs|j=i' => \
$jobs,
284 'verbose|v' => \
$verbose,
285 'help|h|?' => \
$help,
287 $help and pod2usage
(-exitstatus
=>0, -verbose
=>1);
297 B<fa-flacd> and B<fa-rip> together comprise B<flac-archive>, a system
298 for archiving audio CDs to single FLAC files. B<fa-flacd> is the guts
299 of the system. It runs in the directory where the audio archives are
300 stored, scanning for new ripped CDs to encode and rename; it never
301 exits. B<fa-rip> generates the inputs for B<fa-flacd>: the ripped WAV
302 file, Vorbis tags, and a cuesheet.
304 Both programs expect to be run from the same directory. They use that
305 directory to manage directories named by artist. Intermediate files
306 are written to temporary directories here. B<fa-flacd> processes the
307 temporary directories into per-album files in the artist directories.
309 Every 5 seconds, B<fa-flacd> scans its current directory for
310 directories with a file called "tags" and creates a processing job for
311 each one. The number of jobs B<fa-flacd> attempts to run is
312 controlled by the B<-j> option and defaults to 4. B<fa-flacd> will
313 print diagnostic output when the B<-v> option is given.
315 A processing job first renames the directory's "tags" file to
316 "using-tags" so that B<ra-flacd> will not try to start another job for
317 this directory. This file is left as is when an error is encountered,
318 so a new job will not be started until the user corrects the error
319 condition and renames "using-tags" back to "tags". Next, it encodes
320 the "wav" file to a FLAC file, using the "cue" file for the cuesheet
321 and "using-tags" for Vorbis tags. Any diagnostic output is saved in
322 the "log" file. Finally, B<fa-flacd> moves the "cue" and "log" files
323 to the artist directory (named by album) and removes the temporary
326 If the temporary directory contains an executable file named
327 "post-processor", B<fa-flacd> executes that file with the relative
328 path to the output FLAC file as an argument. The output files are in
329 their final location when "post-processor" starts. Possible uses are
330 running B<flac2mp3>, moving the output files to a different location,
331 removing the lock file, or adding to a database. The standard input,
332 output, and error streams are inherited from B<fa-flacd>, so they may
333 be connected to anything from a tty to /dev/null. This means that you
334 may want to redirect these streams, if you want to save them or do any
341 =item B<-j> [B<--jobs>] I<jobs>
343 Run up to I<jobs> jobs instead of the default 4.
345 =item B<-v> [B<--verbose>]
347 Print diagnostic information.
353 Written by Eric Gillespie <epg@pretzelnet.org>.
355 flac-archive is free software; you may redistribute it and/or modify
356 it under the same terms as Perl itself.
361 # cperl-indent-level: 4
362 # perl-indent-level: 4
363 # indent-tabs-mode: nil
366 # vi: set tabstop=4 expandtab: