Greasy Fork is available in English.
Vérifie si des releases (warez) sont disponible pour une film donné.
// ==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: '###2eg3+6rN+48fPbq7YePX4Hgw5pF82dM7WyoiIur62mOgGrgFFJUcnB0cnVz93Bx3vX85Zv3QMXfweDTukXzZ0/rbq8uL2/ohWvg4BMRkZYxNjWzsbG1PfbmHdDw7z9+/AaBL+uXLpw9Hayhsa8lCWYDv7CouIyMgZGpBVADWD1YNRB8BWuYgqFBRERSCqjBwsLW7viHDyD1f/78BYGfG5YunDN9Sld1FVBDK5IGcXGgeiMLS1u7Ex8+fv/xG6waCBAaqhqnoWgAWWBuYWlndwLkIIjxQPBzw7KFc2ZN6arF0CAB1mAN1PD5K8KCv782LFsC0lBTVdWOQ8MpZAvgGoA2YNFgYQl00mmgBqhqICCkwdrOfshowB5K1NAAjjisMY2iAREPyGkJiwZYxCGsAGkAJu82kA2oiQ+kAZI0viIljT+Q5A3MDxUVqMkbogEU0+DEB0zdf37/AecHRI5D1QBK3ibg1Prh4zd4/vkByqIzp3U3VpSj5Gl+EVFQ8gZnoBPADATKoT9+gPL0hzUL5s7obWwoRy0EgDkOnCEsrCBZ9DOwCPj29eP71y+eLG9JSUyIioqMjomtTwuHagAWAqLgLArM09agQuDDh3evnj64efTQ/i3Berra2lqaaqrqGkGhYVANzOx8wiKSMgaGJhYWVjuePnl4+/KRA1s3b9q87fCVi0B5MLiwL0snBEiDNCgzMrFxCQgKiYnJy8t5A5Vu3Ljl4DWwMkwA0vD/f4C+AjcLCysrD6+yz/7Dt8BiOABEAwlgVAMxgNYa/v8HABkzBp0quLI/AAAAAElFTkSuQmCC', frenchLangSt: '', torrentz2: '', yggTorrent: '###nGCsmmStsGSzpmO9o2K8qmS/q2iwp2i2q2a5s2W/sGW9uGW+vWq8umW9wWTFrGHAsGbPsmfNtWrFtGzDvWjJtWnMtWjLumvLvmjNuGzLv2fSsmXdtGfbuGXcuGbdvGvQs2nStWrWtW7Vt2nbtWnauGravGjcuGjdvWzdvXHbtWTgu2fFxGvJw23Lw23LxG3MwGvOyGbewWbexW7Uxm3Ry2zTz2jdwWndxW3ZwmzZxG3cwGzdxWvZyWrdyWvdzW/bym3dyWzdzmvf02ve1Wze0m3e1Wvf2m3d2W7e3XHRwXDWzHbcw3PeyXHez37dz3HX0XDW1HDd0nLd1XDe23Hd3Hfc3W3e4G7e5HDe4WbgwGfgx2jgx2jgzGrg1mng2Wrg3mzg2Gzg3X3gy3vi1Wvg4Gzg4W3g5IjexIjf0I/f3YXgwobky43gzo/k0JXjy57lzJDg3ZXl3pjl3J3j25nl4p7o5qPlzqLozqPh16Xk2aXo0Krp2Kno37Do1rjq2qTm4KXn5KTp7K/p4qnp67Dr6bHr7Lns4Lrr6Lnq7QAAAIFIZFcAAAEAdFJOU////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////wBT9wclAAAACXBIWXMAAA7DAAAOwwHHb6hkAAAAFnRFWHRTb2Z0d2FyZQBwYWludC5uZXQgNC4xTO/F/wAAA1RJREFUSEvtkmmUDFcUx82Mqnqea0hw2hqxRM4MscchjFgSa2LfmbEbxL4GiSUhEXurFunWbR3G0sWMoiuqVNkZbexD25dB7MSaxdLuq+o5xwdzjM8z/9On69Z9/9+9dd97uYLvqRwgK8ruwKv//slMz0MWVAZw5+C5c6mZ6NCh86n+2yGjBfzvP2z4DSNgrgcCAcMwzqYaBv4MLIP/gbMn/c9MqwncNpjFzxiDuf0sSDUCJzFvxiwdCPyVAdzYpuvru/SMHaXretrq2Li47s2/aDZ4b/fYHssPDPu6TpNxq7vGddmXpqddsoCXGlMlSvL+dkI3WgGKJ0DKE67AmJrA4UsFAJKAnlNPTSAFC+vaFELhS1VbZyMURYgAkRDxMaECjwUo5RI0NGkM+DeNNTiR8hVQMkntIBA0RLSfOqoMdiEEig/9uU8xAC4hBV36DQQe3Lx1RlX/3L8iP9DPd5Si2IFrqWraT0RAgI7QtK19gXIrVQ2Tu82hg3sURVXVdgSEulgWQBguybLPFsmA7T6fPD0SuFWKqijKIxO4r2xmWlsSsCitzFNo4/V4ZoSjn5DvMYwXKDfPpxxXLlrbeswnm+rNExJR4MdCFAr2/2N8NLNTKPfdokEf4QzzfPKWXda27pJDgByFlgZHW0Me4MM4gQK1IcLnxk2m3BxZTn5hAulSSGuSq+H6DGljDBuEFMbjGF0Bt5fABxHAzZbXXEU/At6QX5JGYoNGRyVpY79aVco3/IEjwpBNnWpEVW08ADvM9Uqn09OvILAz0Wtpw6cECs/EQPo1yetNnEyJEJ88y+tJOtJNgPC5UiIakxB46LD83niO0Kb4nFiP/2bmhgkVCfCdP7N1XLhobFGgJZKY33ONDb3E4/CgFn8igG2Ww7MkGktzYVhUqFGEDxfCOLwctI3pcZu79ER0OByioxdO8C1Gy6aVoXjZIB+J/mUgTo7Xg9L6zLPYcdc6h+uiKLrcLWrXjvndjOa3rV70w9K1OrpcrmktokoULBsz1u3CFf###Q7B4DW70ym67fYFLCuKdtHudrncopOFGLGEaLfbLzOvCQQfO/H9HXLi96AsAD9raSifiZzXQ8YMAPXk3t17b9ffj0MW1BtA1pQDZEXZEAgGXwOihQJyBvbupgAAAABJRU5ErkJggg==' }; 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 getAutoCompleteR###lts(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 getAutoCompleteR###lts(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 r###lt'); } // 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'); } }