Allocine releases finder

Vérifie si des releases (warez) sont disponible pour une film donné.

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name Allocine releases finder
// @namespace Allocine scripts
// @match http://www.allocine.fr/film/*
// @match https://predb.me/*#to-close
// @grant GM_xmlhttpRequest
// @grant GM_openInTab
// @grant window.close
// @connect https://predb.me
// @require https://cdn.jsdelivr.net/gh/v-garcia/oleoo@b7d9fe652ba8dec5bc3afbcbf9ffcf0e7db810d1/src/index.js
// @require https://cdn.jsdelivr.net/gh/Nycto/PicoModal/src/picoModal.js
// @version 0.0.1.20190525002937
// @description Vérifie si des releases (warez) sont disponible pour une film donné.
// ==/UserScript==

const base64Images = {
  available:
    '',
  availableLowQ:
    '',
  notAvailable:
    '',
  frenchLang:
    '',
  frenchLangSt:
    '',
  torrentz2:
    '',
  yggTorrent:
    ''
};

const carriageReturn = '\n';

function limitPromiseDuration(prom, duration = 15000) {
  return Promise.race([
    prom,
    new Promise((_, rej) => setTimeout(() => rej(`Promise has timed out (${duration} ms)`), duration))
  ]);
}

function isOnPreDb() {
  return window.location.href.startsWith('https://predb.me/');
}

function appendMultipleChildren(element, childrensToAppend, prepend = false) {
  const fnName = prepend ? 'prepend' : 'append';
  for (let toInsert of childrensToAppend) {
    if (prepend) {
      element.insertBefore(toInsert, element.firstChild);
    } else {
      element.appendChild(toInsert);
    }
  }
}

function closePreDbWhenDdosChallengeIsOk() {
  const closeWindowIfOk = () => {
    const title = document.querySelector('title').textContent;
    if (title.includes('PreDB.me')) {
      window.close();
    }
  };

  closeWindowIfOk();
  // Just by security if window if DOM is updated by JS
  setInterval(closeWindowIfOk, 250);
}

function arrayToString(arr) {
  return arr.reduce((prev, currentLine) => prev + carriageReturn + currentLine, '');
}

function createImage(imgName, altName, title) {
  const img = new Image(32, 32);
  img.src = base64Images[imgName];
  img.alt = altName;
  img.title = title;
  img.style = 'margin:5px 5px 0px 5px;';
  return img;
}

function normalizeStr(str) {
  return str.normalize('NFD').replace(/[\u0300-\u036f]/g, '');
}

function createImageLink(imgName, linkName, link) {
  const linkElem = document.createElement('a');
  const img = createImage(imgName, imgName, linkName);
  linkElem.title = linkName;
  linkElem.href = link;
  linkElem.target = '_blank';
  linkElem.appendChild(img);
  return linkElem;
}

function getYggLinkElem(searchTerm) {
  return createImageLink(
    'yggTorrent',
    `Search '${searchTerm}' on yggTorrent`,
    `https://www2.yggtorrent.ch/engine/search?category=2145&sub_category=all&name=${encodeURIComponent(
      searchTerm
    )}&do=search`
  );
}

function getTorrentz2LinkElem(searchTerm) {
  return createImageLink(
    'torrentz2',
    `Search '${searchTerm}' on torrentz2`,
    `https://torrentz2.eu/search?f=${encodeURIComponent(normalizeStr(searchTerm))}`
  );
}

function getMovieTitleFromDetail() {
  return getOriginalTitleFromDetail() || getFrenchTitleFromDetail();
}

function getFrenchTitleFromDetail() {
  const frenchTitleElem = document.querySelector('.titlebar-title-lg');
  const frenchTitle = frenchTitleElem.innerHTML.trim();
  return frenchTitle;
}

function getOriginalTitleFromDetail() {
  const objOriginalTitle = getMovieDetails().find(({ title }) => title === 'Titre original');

  return objOriginalTitle ? objOriginalTitle.value : null;
}

function getMovieDetails() {
  const detailElems = Array.from(document.querySelectorAll('#synopsis-details .ovw-synopsis-info .item'));

  return detailElems.length
    ? detailElems.map(el => ({
        title: el.querySelector('.what').innerText.trim(),
        value: el.querySelector('.that').innerText.trim()
      }))
    : [];
}

function quoteString(string) {
  return `"${string}"`;
}

function log(string, type = 'log') {
  console[type](`allocine_release_finder: ${string}`);
}

function getTitleFromMovieItem(movieItemElem) {
  return movieItemElem.querySelector('.meta-title a').innerText.trim();
}

function getMovieIdFromMovieItem(movieItemElem) {
  const linkElem = movieItemElem.querySelector('.meta-title a');
  const allocineLink = linkElem.getAttribute('href');
  const [fst, snd, allocineId] = allocineLink.match(/(_cfilm=)(\d+)/);
  return Number(allocineId);
}

async function executeScriptOnMovieDetail() {
  log('Start script for movie detail');

  const currentMovieTitle = getMovieTitleFromDetail();

  const infosIconsElem = getScriptButtonsElem(currentMovieTitle);

  document.querySelector('.meta-body').appendChild(infosIconsElem);
}

function executeScriptOnMoviesList() {
  log('Start script for movie detail');

  const movieElems = Array.from(document.querySelectorAll('ol li.mdl, ul li.mdl')).filter(x =>
    x.querySelector('[data-entity-id]')
  );

  if (!movieElems.length) {
    log('No movies found in this movie list');
    return;
  }

  movieElems.forEach(executeScriptOnMovieItem);
}

async function executeScriptOnMovieItem(targetElement) {
  const movieTitleFr = getTitleFromMovieItem(targetElement);
  const movieId = getMovieIdFromMovieItem(targetElement);

  const movieOriginalTitle = await getMovieOriginalTitle(movieTitleFr, movieId);

  const scriptButtonsElem = getScriptButtonsElem(movieOriginalTitle);

  targetElement.querySelector('.meta').appendChild(scriptButtonsElem);

  if (movieTitleFr !== movieOriginalTitle) {
    log(`'${movieTitleFr}' original title is '${movieOriginalTitle}'`);
  }
}

async function getAutoCompleteResults(term) {
  // No need to use GM_xmlhttpRequest for query here as it's a same origin query
  const response = await fetch(`http://essearch.allocine.net/fr/autocomplete?q=${encodeURIComponent(term)}`);

  if (response.status !== 200) {
    throw `Bad response status why searching '${term}' in autocomplete`;
  }

  const jsonResponse = await response.json();
  return jsonResponse;
}

async function getMovieAutoCompleteInfo(term, movieId) {
  const autoCompleRes = await getAutoCompleteResults(term);

  const movieInfo = autoCompleRes.find(({ id }) => id === movieId);

  return movieInfo;
}

async function getMovieOriginalTitle(term, movieId) {
  const { title2: originalTitle } = (await getMovieAutoCompleteInfo(term, movieId)) || {};
  return originalTitle;
}

function getScriptButtonsElem(title) {
  const iconsCtnElem = document.createElement('span');

  appendMultipleChildren(iconsCtnElem, getDownloadButtonElems(title));

  getDownloadInfoElems(title)
    .then(els => {
      appendMultipleChildren(iconsCtnElem, els, true);
    })
    .catch(err => {
      console.error(err);
    });

  return iconsCtnElem;
}

async function searchForReleases(title) {
  const releasesResponse = await limitPromiseDuration(preDbSearch(quoteString(title)));
  log(`${releasesResponse.length} releases found on preDb for '${title}'`);

  const parsedReleases = orderReleaseByInterest(
    addCustomPropertiesToReleases(parseReleasesWithOleoo(releasesResponse))
  );

  return parsedReleases;
}

function preDbSearch(term, triesLeft = 3) {
  const baseUrl = `https://predb.me/?cats=movies&search=${encodeURIComponent(normalizeStr(term))}`;

  if (triesLeft < 1) {
    const err = `Max preDb tries exceeded for '${term}'`;
    log(err);
    return Promise.reject(err);
  }

  return new Promise((resolve, reject) => {
    const redoSearch = () => preDbSearch(term, --triesLeft).then(resolve, reject);
    log(`Looking for movie '${term}' on predb.me (${triesLeft} tries left)`);

    GM_xmlhttpRequest({
      method: 'GET',
      url: `${baseUrl}&rss=1`,
      headers: { Accept: 'text/html' },
      onerror: () => {
        const redoIn = 1500;
        log(`OnError callback thrown for '${term}', redoing in ${redoIn}ms`, 'warn');
        window.setTimeout(redoSearch, redoIn);
      },
      onload: response => {
        const { status, responseText, responseXML } = response;
        if (status === 503 && responseText.indexOf('DDoS protection by Cloudflare') > -1) {
          log('Trying to bypass Ddos protect by cloud fare');

          const openedTab = GM_openInTab(`${baseUrl}#to-close`, { active: false, insert: true });
          // One the tab is closed, the Cloud Fare challenge has been done
          openedTab.onclose = redoSearch;
          return;
        }
        if (status === 503 && responseText.indexOf('Service Temporarily Unavailable') > -1) {
          const redoIn = 1500;
          log(`Too much preDb query, redoing in ${redoIn}ms`, 'warn');
          window.setTimeout(redoSearch, redoIn);
          return;
        }

        // Check status code
        if (status !== 200) {
          reject(`Unexpected status ${status} for an preDb.me search`, 'error');
          return;
        }

        log(`Success for movie '${term}' on predb.me (${triesLeft} tries left)`);

        // Try to parse response
        try {
          var xmlResponse = new DOMParser().parseFromString(responseText, 'text/xml');
        } catch (ex) {
          reject('Unable to parse result');
        }

        // Convert to object
        const releaseItemsElems = Array.from(xmlResponse.querySelectorAll('item') || []);
        const releaseItems = releaseItemsElems.map(el => el.querySelector('title').innerHTML);

        resolve(releaseItems);
      }
    });
  });
}

function orderReleaseByInterest(releases) {
  const getReleaseScore = ({ isSourceOk, hasFrenchVersion }) => isSourceOk + hasFrenchVersion;
  return releases.slice(0).sort((r1, r2) => getReleaseScore(r2) - getReleaseScore(r1));
}

function addCustomPropertiesToReleases(releases) {
  return releases.map(rel => ({
    ...rel,
    hasFrenchVersion: hasFrenchVersion(rel),
    isSourceOk: isSourceOk(rel)
  }));
}

function parseReleasesWithOleoo(releases) {
  return releases.map(x => window.oleoo.parse(x));
}

function isSourceOk({ source }) {
  // We consider that screener is not good enough, but it depends
  return ['DVDRip', 'BDRip', 'HDRip', 'WEB-DL', 'DVD-R', 'BLURAY', 'PDTV', 'SDTV', 'HDTV'].includes(source);
}

function getBestFrenchTranslation(releases) {
  return releases.reduce((acc, { language }) => {
    const isFrench = ['FRENCH', 'MULTI', 'TRUEFRENCH'].includes(language);
    const isSubFr = language === 'VOSTFR';

    if (isFrench) {
      return 'VFR';
    }
    if (isSubFr) {
      return 'VOSTFR';
    }

    return acc;
  }, 'OTHER');
}

function hasFrenchVersion({ language }) {
  return ['FRENCH', 'MULTI', 'VOSTFR', 'TRUEFRENCH'].includes(language);
}

function getNotAvailableImgElem() {
  const text = 'No releases found for this movie';
  return createImage('notAvailable', text, text);
}

function getFrenchLangElem() {
  return createImage('frenchLang', 'French lang available', 'French version (or MULTI) available for this movie');
}

function getFrenchStLangElem() {
  return createImage(
    'frenchLangSt',
    'Fr subtitles available',
    'Release with french subtitles available for this movies'
  );
}

function getReleaseImgElem(releases) {
  const firstRelease = releases[0];
  const pictureToChoose = firstRelease.isSourceOk ? 'available' : 'availableLowQ';
  const imgAlt = firstRelease.isSourceOk ? 'Releases found' : 'Low quality releases found';
  const concatNames = arrayToString(releases.slice(0, 20).map(x => x.original));
  const title = firstRelease.isSourceOk
    ? `${releases.length} releases has been found ${carriageReturn}`
    : `${releases.length} releases has been found ${carriageReturn}/!\\ But source qualities are poor${carriageReturn}`;

  const onReleaseImageClick = evt => {
    evt.preventDefault();
    evt.stopPropagation();
    showReleaseInfoModal(releases);
  };

  const btnLink = document.createElement('a');
  btnLink.setAttribute('href', 'release-modal');
  btnLink.addEventListener('click', onReleaseImageClick);

  const releasesInfoImg = createImage(pictureToChoose, imgAlt, title + concatNames);
  releasesInfoImg.addEventListener('click', onReleaseImageClick);
  btnLink.appendChild(releasesInfoImg);
  return btnLink;
}

function getReleaseListItemModal(release) {
  const { original } = release;
  const liElem = document.createElement('li');

  const iconsLang = getLanguageImageElem([release]);

  if (iconsLang) {
    //iconsLang.setAttribute('style', 'height:1em;width:1em;margin:0;');
    liElem.appendChild(iconsLang);
  } else {
    liElem.style.paddingLeft = '42px';
  }

  const downloadButtons = getDownloadButtonElems(original);

  appendMultipleChildren(liElem, downloadButtons);

  const releaseName = document.createElement('span');
  releaseName.setAttribute('style', 'display:inline-block;height:2em;padding-left:.25em;max-width:calc(100% - 85px);');
  releaseName.innerText = original;
  liElem.appendChild(releaseName);

  return liElem;
}

function showReleaseInfoModal(releases, btn) {
  const [{ title: movieTitle }] = releases;
  const modalContentElem = document.createElement('div');

  const modalTitleElem = document.createElement('h1');
  modalTitleElem.innerText = `${releases.length} releases found for '${movieTitle}'`;
  modalTitleElem.setAttribute('style', 'margin-bottom:1em;');
  modalContentElem.appendChild(modalTitleElem);

  const modalListElem = document.createElement('ul');

  appendMultipleChildren(modalListElem, releases.map(getReleaseListItemModal));

  modalContentElem.appendChild(modalListElem);

  picoModal({
    content: modalContentElem
  })
    .afterClose(function(modal) {
      modal.destroy();
    })
    .show();
}

function isShowingMovieList() {
  return Boolean(document.querySelector('ol li.mdl, ul li.mdl'));
}

function isShowingMovieDetail() {
  return /^http(s)?:\/\/www.allocine.fr\/film\/fichefilm*/.test(window.location.href);
}

function getLanguageImageElem(releases) {
  const bestFrTranslation = getBestFrenchTranslation(releases);

  if (bestFrTranslation === 'VFR') {
    return getFrenchLangElem();
  }

  if (bestFrTranslation === 'VOSTFR') {
    return getFrenchStLangElem();
  }

  return null;
}

function getDownloadButtonElems(title) {
  return [getYggLinkElem(title), getTorrentz2LinkElem(title)];
}

async function getDownloadInfoElems(title) {
  try {
    var releases = await searchForReleases(title);
  } catch (ex) {
    throw ex;
    log(`PreDbSearch failled for '${title}', only torrents links will be displayed`);
    return [];
  }

  if (!releases.length) {
    return [getNotAvailableImgElem()];
  }

  const iconReleases = getReleaseImgElem(releases);
  const iconsLang = getLanguageImageElem(releases);

  return iconsLang ? [iconsLang, iconReleases] : [iconReleases];
}

if (isOnPreDb()) {
  closePreDbWhenDdosChallengeIsOk();
} else {
  if (isShowingMovieDetail()) {
    executeScriptOnMovieDetail();
  } else if (isShowingMovieList()) {
    executeScriptOnMoviesList();
  } else {
    log('Script cannot be applied on this page');
  }
}