2014-12-21 20:55:08 +00:00
|
|
|
|
// ==UserScript==
|
|
|
|
|
// @name Musicbrainz DiscIds Detector
|
|
|
|
|
// @namespace http://userscripts.org/users/22504
|
2023-06-24 17:38:51 +00:00
|
|
|
|
// @version 2023.6.24.1
|
2014-12-21 20:55:08 +00:00
|
|
|
|
// @description Generate MusicBrainz DiscIds from online EAC logs, and check existence in MusicBrainz database.
|
2014-12-21 21:05:56 +00:00
|
|
|
|
// @downloadURL https://raw.githubusercontent.com/murdos/musicbrainz-userscripts/master/mb_discids_detector.user.js
|
|
|
|
|
// @updateURL https://raw.githubusercontent.com/murdos/musicbrainz-userscripts/master/mb_discids_detector.user.js
|
2014-12-21 20:55:08 +00:00
|
|
|
|
// @include http://avaxhome.ws/music/*
|
2019-02-22 14:35:02 +00:00
|
|
|
|
// @include https://orpheus.network/torrents.php?id=*
|
2016-12-11 11:04:38 +00:00
|
|
|
|
// @include https://passtheheadphones.me/torrents.php?id=*
|
2017-02-22 10:23:25 +00:00
|
|
|
|
// @include https://redacted.ch/torrents.php?id=*
|
2014-12-21 20:55:08 +00:00
|
|
|
|
// @include http*://lztr.us/torrents.php?id=*
|
2016-02-20 22:53:28 +00:00
|
|
|
|
// @include http*://lztr.me/torrents.php?id=*
|
2014-12-21 20:55:08 +00:00
|
|
|
|
// @include http*://mutracker.org/torrents.php?id=*
|
2018-07-14 14:57:55 +00:00
|
|
|
|
// @include https://notwhat.cd/torrents.php?id=*
|
2015-06-20 13:43:35 +00:00
|
|
|
|
// @require http://ajax.googleapis.com/ajax/libs/jquery/2.1.4/jquery.js
|
2015-06-20 13:30:47 +00:00
|
|
|
|
// @require http://pajhome.org.uk/crypt/md5/sha1.js
|
2015-06-20 14:13:52 +00:00
|
|
|
|
// @require lib/logger.js
|
2014-12-21 20:55:08 +00:00
|
|
|
|
// ==/UserScript==
|
|
|
|
|
|
2015-06-20 14:42:32 +00:00
|
|
|
|
// prevent JQuery conflicts, see http://wiki.greasespot.net/@grant
|
|
|
|
|
this.$ = this.jQuery = jQuery.noConflict(true);
|
2014-12-21 20:55:08 +00:00
|
|
|
|
|
2016-12-11 11:04:38 +00:00
|
|
|
|
LOGGER.setLevel('info');
|
|
|
|
|
|
2018-11-20 22:18:49 +00:00
|
|
|
|
var CHECK_IMAGE =
|
|
|
|
|
'%3D';
|
2014-12-21 20:55:08 +00:00
|
|
|
|
|
2020-04-05 14:01:21 +00:00
|
|
|
|
$(document).ready(function () {
|
2019-02-22 14:35:02 +00:00
|
|
|
|
if (window.location.host.match(/orpheus\.network|redacted\.ch|passtheheadphones\.me|lztr\.(us|me)|mutracker\.org|notwhat\.cd/)) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
LOGGER.info('Gazelle site detected');
|
2015-06-20 14:42:32 +00:00
|
|
|
|
gazellePageHandler();
|
|
|
|
|
} else if (window.location.host.match(/avaxhome\.ws/)) {
|
|
|
|
|
avaxHomePageHandler();
|
|
|
|
|
}
|
|
|
|
|
});
|
2015-06-20 14:31:50 +00:00
|
|
|
|
|
2015-06-20 14:42:32 +00:00
|
|
|
|
function avaxHomePageHandler() {
|
|
|
|
|
// Find artist and release titles
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let artistName = '';
|
|
|
|
|
let releaseName = '';
|
|
|
|
|
let m = $('div.title h1')
|
|
|
|
|
.text()
|
|
|
|
|
.match(/(.*) (?:-|–) (.*)( \(\d{4}\))?/);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (m) {
|
|
|
|
|
artistName = m[1];
|
|
|
|
|
releaseName = m[2];
|
|
|
|
|
}
|
2018-11-20 22:18:49 +00:00
|
|
|
|
if (artistName == 'VA') artistName = 'Various Artists';
|
2015-06-20 14:42:32 +00:00
|
|
|
|
|
|
|
|
|
// Find and analyze EAC log
|
2018-11-20 22:18:49 +00:00
|
|
|
|
$('div.spoiler')
|
2020-04-05 14:01:21 +00:00
|
|
|
|
.filter(function () {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
return $(this)
|
|
|
|
|
.find('a')
|
|
|
|
|
.text()
|
|
|
|
|
.match(/(EAC|log)/i);
|
|
|
|
|
})
|
2015-06-20 14:42:32 +00:00
|
|
|
|
.find('div')
|
2020-04-05 14:01:21 +00:00
|
|
|
|
.each(function () {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let $eacLog = $(this);
|
|
|
|
|
let discs = analyze_log_files($eacLog);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
|
|
|
|
|
// Check and display
|
2018-11-20 22:18:49 +00:00
|
|
|
|
check_and_display_discs(
|
|
|
|
|
artistName,
|
|
|
|
|
releaseName,
|
|
|
|
|
discs,
|
2020-04-05 14:01:21 +00:00
|
|
|
|
function (mb_toc_numbers, discid, discNumber) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
$eacLog
|
|
|
|
|
.parents('div.spoiler')
|
|
|
|
|
.prevAll('div.center:first')
|
|
|
|
|
.append(
|
|
|
|
|
`<br /><strong>${discs.length > 1 ? `Disc ${discNumber}: ` : ''}MB DiscId </strong><span id="${discid}" />`
|
|
|
|
|
);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
},
|
2020-04-05 14:01:21 +00:00
|
|
|
|
function (mb_toc_numbers, discid, discNumber, found) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let url = computeAttachURL(mb_toc_numbers, artistName, releaseName);
|
|
|
|
|
let html = `<a href="${url}">${discid}</a>`;
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (found) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
html = `${html}<img src="${CHECK_IMAGE}" />`;
|
2015-06-20 14:31:50 +00:00
|
|
|
|
}
|
2018-11-20 22:18:49 +00:00
|
|
|
|
$(`#${discid.replace('.', '\\.')}`).html(html);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
});
|
|
|
|
|
}
|
2014-12-21 20:55:08 +00:00
|
|
|
|
|
2015-06-20 14:42:32 +00:00
|
|
|
|
function gazellePageHandler() {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let serverHost = window.location.host;
|
2014-12-21 20:55:08 +00:00
|
|
|
|
|
2015-06-20 14:42:32 +00:00
|
|
|
|
// Determine Artist name and Release title
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let titleAndArtists = $('#content div.thin h2:eq(0)').text();
|
|
|
|
|
let pattern = /(.*) - (.*) \[.*\] \[.*/;
|
2023-01-20 11:30:50 +00:00
|
|
|
|
if (serverHost.match(/orpheus/)) {
|
2023-06-24 17:28:43 +00:00
|
|
|
|
pattern = /(.*) [-–] (.*) \[.*\]( \[.*)?/;
|
2023-01-20 11:30:50 +00:00
|
|
|
|
}
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let artistName, releaseName;
|
|
|
|
|
if ((m = titleAndArtists.match(pattern))) {
|
2015-06-20 14:42:32 +00:00
|
|
|
|
artistName = m[1];
|
|
|
|
|
releaseName = m[2];
|
|
|
|
|
}
|
2018-11-20 22:18:49 +00:00
|
|
|
|
LOGGER.debug('artist:', artistName, '- releaseName:', releaseName);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
|
|
|
|
|
// Parse each torrent
|
2018-11-20 22:18:49 +00:00
|
|
|
|
$('tr.group_torrent')
|
2020-04-05 14:01:21 +00:00
|
|
|
|
.filter(function () {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
return $(this).attr('id');
|
|
|
|
|
})
|
2020-04-05 14:01:21 +00:00
|
|
|
|
.each(function () {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let torrentInfo = $(this).next();
|
|
|
|
|
|
|
|
|
|
$(torrentInfo)
|
|
|
|
|
.find('a')
|
|
|
|
|
// Only investigate the ones with a log
|
2020-04-05 14:01:21 +00:00
|
|
|
|
.filter(function (index) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
return $(this)
|
|
|
|
|
.text()
|
|
|
|
|
.match(/View\s+Log/i);
|
|
|
|
|
})
|
2020-04-05 14:01:21 +00:00
|
|
|
|
.each(function () {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
LOGGER.debug('Log link', this);
|
|
|
|
|
if (
|
|
|
|
|
$(this)
|
|
|
|
|
.attr('onclick')
|
|
|
|
|
.match(/show_logs/)
|
|
|
|
|
) {
|
2019-02-22 14:35:02 +00:00
|
|
|
|
if (window.location.host.match(/orpheus/)) {
|
|
|
|
|
LOGGER.debug('Orpheus');
|
2018-11-20 22:18:49 +00:00
|
|
|
|
var logAction = 'viewlog';
|
|
|
|
|
} else if (window.location.host.match(/redacted|passtheheadphones/)) {
|
|
|
|
|
LOGGER.debug('RED');
|
|
|
|
|
var logAction = 'loglist';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// LzTR
|
|
|
|
|
else if (
|
|
|
|
|
$(this)
|
|
|
|
|
.attr('onclick')
|
|
|
|
|
.match(/get_log/)
|
|
|
|
|
) {
|
|
|
|
|
LOGGER.debug('LzTR');
|
|
|
|
|
var logAction = 'log_ajax';
|
|
|
|
|
}
|
|
|
|
|
// NotWhat.CD
|
|
|
|
|
else if (
|
|
|
|
|
$(this)
|
|
|
|
|
.attr('onclick')
|
|
|
|
|
.match(/show_log/)
|
|
|
|
|
) {
|
|
|
|
|
LOGGER.debug('NotWhat.CD');
|
2016-12-11 11:04:38 +00:00
|
|
|
|
var logAction = 'viewlog';
|
2018-11-20 22:18:49 +00:00
|
|
|
|
} else {
|
|
|
|
|
return true;
|
2016-12-11 11:04:38 +00:00
|
|
|
|
}
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let targetContainer = $(this).parents('.linkbox');
|
|
|
|
|
let torrentId = /(show_logs|get_log|show_log)\('(\d+)/.exec($(this).attr('onclick'))[2];
|
|
|
|
|
let logUrl = `/torrents.php?action=${logAction}&torrentid=${torrentId}`;
|
|
|
|
|
LOGGER.info('Log URL: ', logUrl);
|
|
|
|
|
LOGGER.debug('targetContainer: ', targetContainer);
|
|
|
|
|
|
|
|
|
|
// Get log content
|
2020-04-05 14:01:21 +00:00
|
|
|
|
$.get(logUrl, function (data) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
LOGGER.debug('Log content', $(data).find('pre'));
|
|
|
|
|
let discs = analyze_log_files($(data).find('pre'));
|
|
|
|
|
LOGGER.debug('Number of disc found', discs.length);
|
|
|
|
|
check_and_display_discs(
|
|
|
|
|
artistName,
|
|
|
|
|
releaseName,
|
|
|
|
|
discs,
|
2020-04-05 14:01:21 +00:00
|
|
|
|
function (mb_toc_numbers, discid, discNumber) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
targetContainer.append(
|
|
|
|
|
`<br /><strong>${
|
|
|
|
|
discs.length > 1 ? `Disc ${discNumber}: ` : ''
|
|
|
|
|
}MB DiscId: </strong><span id="${torrentId}_disc${discNumber}" />`
|
|
|
|
|
);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
},
|
2020-04-05 14:01:21 +00:00
|
|
|
|
function (mb_toc_numbers, discid, discNumber, found) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let url = computeAttachURL(mb_toc_numbers, artistName, releaseName);
|
|
|
|
|
let html = `<a href="${url}">${discid}</a>`;
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (found) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
html = `${html}<img src="${CHECK_IMAGE}" />`;
|
2015-06-20 14:31:50 +00:00
|
|
|
|
}
|
2018-11-20 22:18:49 +00:00
|
|
|
|
LOGGER.debug(`#${torrentId}_disc${discNumber}`);
|
|
|
|
|
$(`#${torrentId}_disc${discNumber}`).html(html);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
}
|
|
|
|
|
);
|
2018-11-20 22:18:49 +00:00
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
});
|
2015-06-20 14:42:32 +00:00
|
|
|
|
}
|
2014-12-21 20:55:08 +00:00
|
|
|
|
|
|
|
|
|
// Common functions
|
|
|
|
|
|
2015-06-20 14:42:32 +00:00
|
|
|
|
function computeAttachURL(mb_toc_numbers, artistName, releaseName) {
|
2018-11-20 22:45:38 +00:00
|
|
|
|
let url = `${'http://musicbrainz.org/cdtoc/attach?toc='}${mb_toc_numbers.join('%20')}&artist-name=${encodeURIComponent(
|
2018-11-20 22:18:49 +00:00
|
|
|
|
artistName
|
|
|
|
|
)}&release-name=${encodeURIComponent(releaseName)}`;
|
2015-06-20 14:42:32 +00:00
|
|
|
|
return url;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function analyze_log_files(log_files) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let discs = [];
|
2020-04-05 14:01:21 +00:00
|
|
|
|
$.each(log_files, function (i, log_file) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let discsInLog = MBDiscid.log_input_to_entries($(log_file).text());
|
2015-06-20 14:42:32 +00:00
|
|
|
|
for (var i = 0; i < discsInLog.length; i++) {
|
|
|
|
|
discs.push(discsInLog[i]);
|
|
|
|
|
}
|
|
|
|
|
});
|
2014-12-21 20:55:08 +00:00
|
|
|
|
|
2015-06-20 14:42:32 +00:00
|
|
|
|
// Remove dupes discs
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let keys = new Object();
|
|
|
|
|
let uniqueDiscs = new Array();
|
|
|
|
|
for (let i = 0; i < discs.length; i++) {
|
|
|
|
|
let discid = MBDiscid.calculate_mb_discid(discs[i]);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (discid in keys) {
|
|
|
|
|
continue;
|
|
|
|
|
} else {
|
|
|
|
|
keys[discid] = 1;
|
|
|
|
|
uniqueDiscs.push(discs[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
discs = uniqueDiscs;
|
|
|
|
|
return discs;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function check_and_display_discs(artistName, releaseName, discs, displayDiscHandler, displayResultHandler) {
|
|
|
|
|
// For each disc, check if it's in MusicBrainz database
|
2018-11-20 22:18:49 +00:00
|
|
|
|
for (let i = 0; i < discs.length; i++) {
|
|
|
|
|
let entries = discs[i];
|
|
|
|
|
let discNumber = i + 1;
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (entries.length > 0) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let mb_toc_numbers = MBDiscid.calculate_mb_toc_numbers(entries);
|
|
|
|
|
let discid = MBDiscid.calculate_mb_discid(entries);
|
|
|
|
|
LOGGER.info(`Computed discid :${discid}`);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
displayDiscHandler(mb_toc_numbers, discid, discNumber);
|
|
|
|
|
|
|
|
|
|
// Now check if this discid is known by MusicBrainz
|
2020-04-05 14:01:21 +00:00
|
|
|
|
(function (discid, discNumber, mb_toc_numbers) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let query = $.getJSON(`//musicbrainz.org/ws/2/discid/${discid}?cdstubs=no`);
|
2020-04-05 14:01:21 +00:00
|
|
|
|
query.done(function (data) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let existsInMusicbrainz = !('error' in data) && data.error != 'Not found';
|
2015-06-20 14:42:32 +00:00
|
|
|
|
displayResultHandler(mb_toc_numbers, discid, discNumber, existsInMusicbrainz);
|
|
|
|
|
});
|
2020-04-05 14:01:21 +00:00
|
|
|
|
query.fail(function () {
|
2015-06-20 14:42:32 +00:00
|
|
|
|
// If discid is not found, the webservice returns a 404 http code
|
|
|
|
|
displayResultHandler(mb_toc_numbers, discid, discNumber, false);
|
|
|
|
|
});
|
|
|
|
|
})(discid, discNumber, mb_toc_numbers);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* -------------------------------------------- */
|
|
|
|
|
|
2015-06-20 14:45:29 +00:00
|
|
|
|
// MBDiscid code comes from https://gist.github.com/kolen/766668
|
|
|
|
|
// Copyright 2010, kolen
|
|
|
|
|
// Released under the MIT License
|
|
|
|
|
|
2020-04-05 14:01:21 +00:00
|
|
|
|
var MBDiscid = (function () {
|
2015-06-20 14:42:32 +00:00
|
|
|
|
this.SECTORS_PER_SECOND = 75;
|
|
|
|
|
this.PREGAP = 150;
|
|
|
|
|
this.DATA_TRACK_GAP = 11400;
|
|
|
|
|
|
|
|
|
|
this.toc_entry_matcher = new RegExp(
|
2018-11-20 22:18:49 +00:00
|
|
|
|
'^\\s*' +
|
2020-09-13 16:05:28 +00:00
|
|
|
|
'(\\d+)' + // 1 - track number
|
|
|
|
|
'\\s*\\|\\s*' +
|
|
|
|
|
'([0-9:.]+)' + // 2 - time start
|
|
|
|
|
'\\s*\\|\\s*' +
|
|
|
|
|
'([0-9:.]+)' + // 3 - time length
|
|
|
|
|
'\\s*\\|\\s*' +
|
|
|
|
|
'(\\d+)' + // 4 - start sector
|
|
|
|
|
'\\s*\\|\\s*' +
|
|
|
|
|
'(\\d+)' + // 5 - end sector
|
2018-11-20 22:18:49 +00:00
|
|
|
|
'\\s*$'
|
2015-06-20 14:42:32 +00:00
|
|
|
|
);
|
2020-04-05 14:01:21 +00:00
|
|
|
|
this.log_input_to_entries = function (text) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let discs = [];
|
2015-06-20 14:42:32 +00:00
|
|
|
|
var entries = [];
|
2020-04-05 14:01:21 +00:00
|
|
|
|
$.each(text.split('\n'), function (index, value) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let m = toc_entry_matcher.exec(value);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (m) {
|
|
|
|
|
// New disc
|
|
|
|
|
if (parseInt(m[1], 10) == 1) {
|
|
|
|
|
if (entries.length > 0) {
|
|
|
|
|
discs.push(entries);
|
|
|
|
|
}
|
|
|
|
|
entries = [];
|
|
|
|
|
}
|
|
|
|
|
entries.push(m);
|
2015-06-20 14:31:50 +00:00
|
|
|
|
}
|
|
|
|
|
});
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (entries.length > 0) {
|
|
|
|
|
discs.push(entries);
|
|
|
|
|
}
|
2015-06-20 14:31:50 +00:00
|
|
|
|
|
2018-11-20 22:18:49 +00:00
|
|
|
|
for (let i = 0; i < discs.length; i++) {
|
2015-06-20 14:42:32 +00:00
|
|
|
|
var entries = discs[i];
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let layout_type = get_layout_type(entries);
|
2015-06-20 14:42:32 +00:00
|
|
|
|
var entries_audio;
|
2018-11-20 22:18:49 +00:00
|
|
|
|
if (layout_type == 'with_data') {
|
2015-06-20 14:42:32 +00:00
|
|
|
|
entries_audio = entries.slice(0, entries.length - 1);
|
2015-06-20 14:31:50 +00:00
|
|
|
|
} else {
|
2015-06-20 14:42:32 +00:00
|
|
|
|
entries_audio = entries;
|
2015-06-20 14:31:50 +00:00
|
|
|
|
}
|
2015-06-20 14:42:32 +00:00
|
|
|
|
discs[i] = entries_audio;
|
2015-06-10 22:22:38 +00:00
|
|
|
|
}
|
2015-06-20 14:31:50 +00:00
|
|
|
|
return discs;
|
2015-06-20 14:42:32 +00:00
|
|
|
|
};
|
|
|
|
|
|
2020-04-05 14:01:21 +00:00
|
|
|
|
this.get_layout_type = function (entries) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let type = 'standard';
|
|
|
|
|
for (let i = 0; i < entries.length - 1; i++) {
|
|
|
|
|
let gap = parseInt(entries[i + 1][4], 10) - parseInt(entries[i][5], 10) - 1;
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (gap != 0) {
|
|
|
|
|
if (i == entries.length - 2 && gap == DATA_TRACK_GAP) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
type = 'with_data';
|
2015-06-20 14:42:32 +00:00
|
|
|
|
} else {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
type = 'unknown';
|
2015-06-20 14:42:32 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
2014-12-21 20:55:08 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-06-20 14:42:32 +00:00
|
|
|
|
return type;
|
|
|
|
|
};
|
2015-06-20 14:31:50 +00:00
|
|
|
|
|
2020-04-05 14:01:21 +00:00
|
|
|
|
this.calculate_mb_toc_numbers = function (entries) {
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (entries.length == 0) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2015-06-20 14:31:50 +00:00
|
|
|
|
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let leadout_offset = parseInt(entries[entries.length - 1][5], 10) + PREGAP + 1;
|
2015-06-20 14:31:50 +00:00
|
|
|
|
|
2020-04-05 14:01:21 +00:00
|
|
|
|
let offsets = $.map(entries, function (entry) {
|
2015-06-20 14:42:32 +00:00
|
|
|
|
return parseInt(entry[4], 10) + PREGAP;
|
2018-11-20 22:18:49 +00:00
|
|
|
|
});
|
2015-06-20 14:42:32 +00:00
|
|
|
|
return [1, entries.length, leadout_offset].concat(offsets);
|
|
|
|
|
};
|
|
|
|
|
|
2020-04-05 14:01:21 +00:00
|
|
|
|
this.calculate_cddb_id = function (entries) {
|
|
|
|
|
let sum_of_digits = function (n) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let sum = 0;
|
2015-06-20 14:42:32 +00:00
|
|
|
|
while (n > 0) {
|
|
|
|
|
sum = sum + (n % 10);
|
|
|
|
|
n = Math.floor(n / 10);
|
2015-06-20 14:31:50 +00:00
|
|
|
|
}
|
2015-06-20 14:42:32 +00:00
|
|
|
|
return sum;
|
2018-11-20 22:18:49 +00:00
|
|
|
|
};
|
2015-06-20 14:35:22 +00:00
|
|
|
|
|
2020-04-05 14:01:21 +00:00
|
|
|
|
let decimalToHexString = function (number) {
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (number < 0) {
|
2018-11-20 22:18:49 +00:00
|
|
|
|
number = 0xffffffff + number + 1;
|
2015-06-20 14:31:50 +00:00
|
|
|
|
}
|
2014-12-21 20:55:08 +00:00
|
|
|
|
|
2015-06-20 14:42:32 +00:00
|
|
|
|
return number.toString(16).toUpperCase();
|
2018-11-20 22:18:49 +00:00
|
|
|
|
};
|
2015-06-20 14:31:50 +00:00
|
|
|
|
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let length_seconds = Math.floor(
|
|
|
|
|
(parseInt(entries[entries.length - 1][5], 10) - parseInt(entries[0][4], 10) + 1) / SECTORS_PER_SECOND
|
|
|
|
|
);
|
|
|
|
|
let checksum = 0;
|
2020-04-05 14:01:21 +00:00
|
|
|
|
$.each(entries, function (index, entry) {
|
2015-06-20 14:42:32 +00:00
|
|
|
|
checksum += sum_of_digits(Math.floor((parseInt(entry[4], 10) + PREGAP) / SECTORS_PER_SECOND));
|
2018-11-20 22:18:49 +00:00
|
|
|
|
});
|
2014-12-21 20:55:08 +00:00
|
|
|
|
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let xx = checksum % 255;
|
|
|
|
|
let discid_num = (xx << 24) | (length_seconds << 8) | entries.length;
|
2015-06-20 14:42:32 +00:00
|
|
|
|
//return discid_num
|
|
|
|
|
return decimalToHexString(discid_num);
|
|
|
|
|
};
|
2015-06-20 14:31:50 +00:00
|
|
|
|
|
2020-04-05 14:01:21 +00:00
|
|
|
|
this.calculate_mb_discid = function (entries) {
|
|
|
|
|
let hex_left_pad = function (input, totalChars) {
|
|
|
|
|
input = `${parseInt(input, 10).toString(16).toUpperCase()}`;
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let padWith = '0';
|
2015-06-20 14:42:32 +00:00
|
|
|
|
if (input.length < totalChars) {
|
|
|
|
|
while (input.length < totalChars) {
|
|
|
|
|
input = padWith + input;
|
2015-06-20 14:35:22 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2018-11-20 22:18:49 +00:00
|
|
|
|
if (input.length > totalChars) {
|
|
|
|
|
//if padWith was a multiple character string and num was overpadded
|
|
|
|
|
input = input.substring(input.length - totalChars, totalChars);
|
2015-06-20 14:31:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-06-20 14:42:32 +00:00
|
|
|
|
return input;
|
2015-06-20 14:35:22 +00:00
|
|
|
|
};
|
2015-06-20 14:31:50 +00:00
|
|
|
|
|
2018-11-20 22:18:49 +00:00
|
|
|
|
let mb_toc_numbers = calculate_mb_toc_numbers(entries);
|
|
|
|
|
let message = '';
|
|
|
|
|
let first_track = mb_toc_numbers[0];
|
|
|
|
|
let last_track = mb_toc_numbers[1];
|
|
|
|
|
let leadout_offset = mb_toc_numbers[2];
|
2015-06-20 14:42:32 +00:00
|
|
|
|
message = message + hex_left_pad(first_track, 2);
|
|
|
|
|
message = message + hex_left_pad(last_track, 2);
|
|
|
|
|
message = message + hex_left_pad(leadout_offset, 8);
|
2018-11-20 22:18:49 +00:00
|
|
|
|
for (let i = 0; i < 99; i++) {
|
|
|
|
|
let offset = i + 3 < mb_toc_numbers.length ? mb_toc_numbers[i + 3] : 0;
|
2015-06-20 14:42:32 +00:00
|
|
|
|
message = message + hex_left_pad(offset, 8);
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-20 22:18:49 +00:00
|
|
|
|
b64pad = '=';
|
|
|
|
|
let discid = b64_sha1(message);
|
2020-04-05 14:01:21 +00:00
|
|
|
|
discid = discid.replace(/\+/g, '.').replace(/\//g, '_').replace(/=/g, '-');
|
2015-06-20 14:42:32 +00:00
|
|
|
|
return discid;
|
|
|
|
|
};
|
2014-12-21 20:55:08 +00:00
|
|
|
|
|
2015-06-20 14:42:32 +00:00
|
|
|
|
return this;
|
2015-06-10 22:22:38 +00:00
|
|
|
|
})();
|