TwitchAdSolutions/strip-alt/strip-alt.user.js
pixeltris f37af09192 Script improvements #17 #24 #28
- Change notify-strip and notify-reload algorithms to handle midrolls better and hopefully fix audio desync
- Fetch native resolution earlier
- Enforce site player type to avoid issues on embedded sites
- Removed unused / broken scripts
- Improve readme and full list info
- Experimentally testing rolling device id, may remove
2021-06-12 05:35:11 +01:00

205 lines
33 KiB
JavaScript

// ==UserScript==
// @name TwitchAdSolutions (strip-alt)
// @namespace https://github.com/pixeltris/TwitchAdSolutions
// @version 1.1
// @description Multiple solutions for blocking Twitch ads (strip-alt)
// @updateURL https://github.com/pixeltris/TwitchAdSolutions/raw/master/strip-alt/strip-alt.user.js
// @downloadURL https://github.com/pixeltris/TwitchAdSolutions/raw/master/strip-alt/strip-alt.user.js
// @author pixeltris
// @match *://*.twitch.tv/*
// @run-at document-start
// @grant none
// ==/UserScript==
(function() {
'use strict';
var twitchMainWorker = null;
const oldWorker = window.Worker;
window.Worker = class Worker extends oldWorker {
constructor(twitchBlobUrl) {
if (twitchMainWorker) {
super(twitchBlobUrl);
return;
}
var jsURL = getWasmWorkerUrl(twitchBlobUrl);
if (typeof jsURL !== 'string') {
super(twitchBlobUrl);
return;
}
var newBlobStr = `
${processM3U8.toString()}
${hookWorkerFetch.toString()}
${pushSegUrlInfo.toString()}
AD_SIGNIFIER = 'stitched-ad';
LIVE_SIGNIFIER = ',live';
IsMidroll = false;
HasAd = false;
StreamUrlCache = [];
hookWorkerFetch();
importScripts('${jsURL}');
`
super(URL.createObjectURL(new Blob([newBlobStr])));
twitchMainWorker = this;
this.onmessage = function(e) {
if (e.data.key == 'UboShowAdBanner') {
var adDiv = getAdDiv();
if (adDiv != null) {
adDiv.P.textContent = 'Blocking' + (e.data.isMidroll ? ' midroll' : '') + ' ads...';
adDiv.style.display = 'block';
}
} else if (e.data.key == 'UboHideAdBanner') {
var adDiv = getAdDiv();
if (adDiv != null) {
adDiv.style.display = 'none';
}
if (e.data.resetPlayer) {
// There's some audio sync issues from the replaced segments. Resetting the player should hopefully fix this.
resetTwitchPlayer();
console.log('[strip-alt] Reset player');
}
}
}
function getAdDiv() {
var playerRootDiv = document.querySelector('.video-player');
var adDiv = null;
if (playerRootDiv != null) {
adDiv = playerRootDiv.querySelector('.ubo-overlay');
if (adDiv == null) {
adDiv = document.createElement('div');
adDiv.className = 'ubo-overlay';
adDiv.innerHTML = '<div class="player-ad-notice" style="color: white; background-color: rgba(0, 0, 0, 0.8); position: absolute; top: 0px; left: 0px; padding: 10px;"><p></p></div>';
adDiv.style.display = 'none';
adDiv.P = adDiv.querySelector('p');
playerRootDiv.appendChild(adDiv);
}
}
return adDiv;
}
}
}
function getWasmWorkerUrl(twitchBlobUrl) {
var req = new XMLHttpRequest();
req.open('GET', twitchBlobUrl, false);
req.send();
return req.responseText.split("'")[1];
}
function pushSegUrlInfo(segUrl, isLive) {
var segInfo = {
expireDate: new Date(Date.now() + 120000),
isAd: !isLive,
url: segUrl
};
StreamUrlCache[segUrl] = segInfo;
return segInfo;
}
async function processM3U8(url, textStr, realFetch) {
var haveAdTags = textStr.includes(AD_SIGNIFIER);
if (haveAdTags) {
var dateNow = new Date();
for (const [segUrl, segUrlInfo] of Object.entries(StreamUrlCache)) {
if (segUrlInfo.expireDate < dateNow) {
delete StreamUrlCache[segUrl];
}
}
// FIXME: Twitch ad banner issues. Maybe detect and remove from DOM?
// FIXME: Sometimes freezes after midroll?
// NOTE: Midroll might invoke player-by-picture player? Might need to change MIDROLL to PREROLL?
IsMidroll = textStr.includes('"MIDROLL"') || textStr.includes('"midroll"');
var lines = textStr.replace('\r', '').split('\n');
var isLive = false;
for (var i = 0; i < lines.length; i++) {
var line = lines[i];
if (line.includes('stitched-ad')) {
var replaceTags = ['X-TV-TWITCH-AD-URL', 'X-TV-TWITCH-AD-CLICK-TRACKING-URL'];
for (var j = 0; j < replaceTags.length; j++) {
var adTag = replaceTags[j] + '="';
var adTagIndex = line.indexOf(adTag);
var adTagEndIndex = line.indexOf('"', adTagIndex + adTag.length);
line = line.substring(0, adTagIndex) + adTag + 'http://twitch.tv' + line.substring(adTagEndIndex);
}
lines[i] = line;
} else if (line.startsWith('#EXTINF') && lines.length > i + 1) {
isLive = !(pushSegUrlInfo(lines[i + 1], line.includes(LIVE_SIGNIFIER))).isAd;
} else if (line.startsWith('#EXT-X-TWITCH-PREFETCH:')) {
if ((pushSegUrlInfo(line.substring(line.indexOf(':') + 1), isLive || !IsMidroll)).isAd) {
console.log('[strip-alt] Removing prefetch url');// NOTE: This currently strips some legit prefetch urls (might invalidate low latency). Preroll shouldn't have a prefetch ad, assume live segment to avoid 2 second delay on stream starting.
}
} else if (line.startsWith('#EXT-X-DISCONTINUITY')) {
isLive = false;
}
}
textStr = lines.join('\n');
}
return textStr;
}
function hookWorkerFetch() {
var realFetch = fetch;
fetch = async function(url, options) {
if (typeof url === 'string') {
if (url.endsWith('.ts')) {
var segUrlInfo = StreamUrlCache[url];
if (segUrlInfo && segUrlInfo.isAd) {
url = '';
postMessage({key:'UboShowAdBanner',isMidroll:IsMidroll});
HasAd = true;
} else {
postMessage({key:'UboHideAdBanner',resetPlayer:HasAd});
HasAd = false;
}
}
if (url.endsWith('m3u8')) {
return new Promise(function(resolve, reject) {
var processAfter = async function(response) {
var str = await processM3U8(url, await response.text(), realFetch);
resolve(new Response(str));
};
var send = function() {
return realFetch(url, options).then(function(response) {
processAfter(response);
})['catch'](function(err) {
console.log('fetch hook err ' + err);
reject(err);
});
};
send();
});
}
}
return realFetch.apply(this, arguments);
}
}
function resetTwitchPlayer(isPausePlay) {
// Taken from ttv-tools / ffz
// https://github.com/Nerixyz/ttv-tools/blob/master/src/context/twitch-player.ts
// https://github.com/FrankerFaceZ/FrankerFaceZ/blob/master/src/sites/twitch-twilight/modules/player.jsx
function findReactNode(root, constraint) {
if (root.stateNode && constraint(root.stateNode)) {
return root.stateNode;
}
let node = root.child;
while (node) {
const result = findReactNode(node, constraint);
if (result) {
return result;
}
node = node.sibling;
}
return null;
}
var reactRootNode = null;
var rootNode = document.querySelector('#root');
if (rootNode && rootNode._reactRootContainer && rootNode._reactRootContainer._internalRoot && rootNode._reactRootContainer._internalRoot.current) {
reactRootNode = rootNode._reactRootContainer._internalRoot.current;
}
if (!reactRootNode) {
console.log('Could not find react root');
return;
}
var player = findReactNode(reactRootNode, node => node.setPlayerActive && node.props && node.props.mediaPlayerInstance);
player = player && player.props && player.props.mediaPlayerInstance ? player.props.mediaPlayerInstance : null;
if (!player) {
console.log('Could not find player');
return;
}
player.seekTo(0);
}
})();