2 # Copyright (c) 2012 Steve McIntyre <93sam@debian.org>
3 # This code is hereby licensed for public consumption under either the
4 # GNU GPL v2 or greater, or Larry Wall's Artistic license - your choice.
6 # You should have received a copy of the GNU General Public License along
7 # with this program; if not, write to the Free Software Foundation, Inc.,
8 # 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
10 # abcde-musicbrainz-tool
12 # Helper script for abcde to work with the MusicBrainz WS API (v2)
18 use MusicBrainz
::DiscID
;
19 use WebService
::MusicBrainz
::Release
;
20 use WebService
::MusicBrainz
::Artist
;
21 use WebService
::MusicBrainz
::Response
::Track
;
22 use WebService
::MusicBrainz
::Response
::TrackList
;
25 my $FRAMES_PER_SEC = 75;
27 my ($device, $command, $discid, @discinfo, $workdir);
28 Getopt
::Long
::Configure
('no_ignore_case');
29 Getopt
::Long
::Configure
('no_auto_abbrev');
30 GetOptions
("device=s" => \
$device,
31 "command=s" => \
$command,
32 "discid=s" => \
$discid,
33 "discinfo=i{5,}" => \
@discinfo,
34 "workdir=s" => \
$workdir);
36 if (!defined($device)) {
37 $device = "/dev/cdrom";
39 if (!defined($command)) {
42 if (!defined($workdir)) {
48 my $s = Digest
::SHA
->new(1);
49 $s->addfile($filename);
53 if ($command =~ m/^id/) {
54 my $disc = new MusicBrainz
::DiscID
($device);
56 # read the disc in the default disc drive */
57 if ( $disc->read() == 0 ) {
58 printf STDERR
"Error: %s\n", $disc->error_msg();
62 printf("%s ", $disc->id());
63 printf("%d ", $disc->last_track_num() + 1 - $disc->first_track_num());
65 for ( my $i = $disc->first_track_num;
66 $i <= $disc->last_track_num; $i++ ) {
67 printf("%d ", $disc->track_offset($i));
69 printf("%d\n", $disc->sectors() / $FRAMES_PER_SEC);
72 } elsif ($command =~ m/data/) {
73 my $ws = WebService
::MusicBrainz
::Release
->new();
74 my $response = $ws->search({ DISCID
=> $discid });
75 my @releases = $response->release_list();
79 foreach my $release (@releases) {
80 my $a_artist = $release->artist()->name();
82 if ($a_artist =~ /Various Artists/) {
86 open (OUT
, "> $workdir/cddbread.$releasenum");
88 print OUT
"# xmcd style database file\n";
90 print OUT
"# Track frame offsets:\n";
91 # Assume standard pregap
93 my @tracks = @
{$release->track_list()->tracks()};
94 for (my $i = 0; $i < scalar(@tracks); $i++) {
95 printf OUT
"# %d\n", ceil
($total_len * $FRAMES_PER_SEC / 1000.0);
96 $total_len += $tracks[$i]->duration();
99 printf OUT
"# Disc length: %d seconds\n", $total_len / 1000.0;
101 print OUT
"# Submitted via: XXXXXX\n";
103 print OUT
"#blues,classical,country,data,folk,jazz,newage,reggae,rock,soundtrack,misc\n";
104 print OUT
"#CATEGORY=none\n";
105 print OUT
"DISCID=" . $discid . "\n";
106 print OUT
"DTITLE=" . $a_artist. " / " . $release->title() . "\n";
107 print OUT
"DYEAR=\n";
108 print OUT
"DGENRE=\n";
110 my @tracks = @
{$release->track_list()->tracks()};
111 for (my $i = 0; $i < scalar(@tracks); $i++) {
112 my $track = $tracks[$i];
113 my $t_name = $track->title;
115 my $t_artist = $track->artist->name;
116 printf OUT
"TTITLE%d=%s / %s\n", $i, $t_artist, $t_name;
118 printf OUT
"TTITLE%d=%s\n", $i, $t_name;
123 for (my $i = 0; $i < scalar(@tracks); $i++) {
124 printf OUT
"EXTT%d=\n", $i;
126 print OUT
"PLAYORDER=\n";
130 # Check to see that this entry is unique; generate a checksum
131 # and compare to any previous checksums
132 my $checksum = calc_sha1
("$workdir/cddbread.$releasenum");
133 foreach my $sum (@sums) {
134 if ($checksum eq $sum) {
135 unlink("$workdir/cddbread.$releasenum");
140 push (@sums, $checksum);
142 } elsif ($command =~ m/calcid/) {
143 # Calculate MusicBrainz ID from disc offsets; see
144 # http://musicbrainz.org/doc/DiscIDCalculation
146 my ($first, $last, $leadin, $leadout, @offsets) = @discinfo;
148 my $s = Digest
::SHA
->new(1);
149 $s->add(sprintf "%02X", int($first));
150 $s->add(sprintf "%02X", int($last));
153 for (my $i = 0; $i < 100; $i++) {
157 foreach my $o ($leadout, @offsets) {
158 $a[$i++] = int($o) + int($leadin);
160 for (my $i = 0; $i < 100; $i++) {
161 $s->add(sprintf "%08X", $a[$i]);
164 my $id = $s->b64digest;
165 # CPAN Digest modules do not pad their Base64 output, so we have to do it.
166 while (length($id) % 4) {