spoof HTTP Referer for USO directly (#608)
This commit is contained in:
parent
68ce3a653e
commit
8b8a3d60ab
|
@ -16,6 +16,8 @@ window.addEventListener('showStyles:done', function _() {
|
||||||
const RESULT_ID_PREFIX = 'search-result-';
|
const RESULT_ID_PREFIX = 'search-result-';
|
||||||
|
|
||||||
const BASE_URL = 'https://userstyles.org';
|
const BASE_URL = 'https://userstyles.org';
|
||||||
|
const JSON_URL = BASE_URL + '/styles/chrome/';
|
||||||
|
const API_URL = BASE_URL + '/api/v1/styles/';
|
||||||
const UPDATE_URL = 'https://update.userstyles.org/%.md5';
|
const UPDATE_URL = 'https://update.userstyles.org/%.md5';
|
||||||
|
|
||||||
// normal category is just one word like 'github' or 'google'
|
// normal category is just one word like 'github' or 'google'
|
||||||
|
@ -54,8 +56,10 @@ window.addEventListener('showStyles:done', function _() {
|
||||||
let searchCurrentPage = 1;
|
let searchCurrentPage = 1;
|
||||||
let searchExhausted = false;
|
let searchExhausted = false;
|
||||||
|
|
||||||
let usoFrame;
|
// currently active USO requests
|
||||||
let usoFrameQueue;
|
const xhrSpoofIds = new Set();
|
||||||
|
// used as an HTTP header name to identify spoofed requests
|
||||||
|
const xhrSpoofTelltale = getRandomId();
|
||||||
|
|
||||||
const processedResults = [];
|
const processedResults = [];
|
||||||
const unprocessedResults = [];
|
const unprocessedResults = [];
|
||||||
|
@ -653,7 +657,7 @@ window.addEventListener('showStyles:done', function _() {
|
||||||
function fetchStyleJson(result) {
|
function fetchStyleJson(result) {
|
||||||
return Promise.resolve(
|
return Promise.resolve(
|
||||||
result.json ||
|
result.json ||
|
||||||
downloadInFrame(BASE_URL + '/styles/chrome/' + result.id + '.json').then(json => {
|
downloadFromUSO(JSON_URL + result.id + '.json').then(json => {
|
||||||
result.json = json;
|
result.json = json;
|
||||||
return json;
|
return json;
|
||||||
}));
|
}));
|
||||||
|
@ -667,7 +671,7 @@ window.addEventListener('showStyles:done', function _() {
|
||||||
function fetchStyle(userstylesId) {
|
function fetchStyle(userstylesId) {
|
||||||
return readCache(userstylesId).then(json =>
|
return readCache(userstylesId).then(json =>
|
||||||
json ||
|
json ||
|
||||||
downloadInFrame(BASE_URL + '/api/v1/styles/' + userstylesId).then(writeCache));
|
downloadFromUSO(API_URL + userstylesId).then(writeCache));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -685,8 +689,7 @@ window.addEventListener('showStyles:done', function _() {
|
||||||
return Promise.resolve({'data':[]});
|
return Promise.resolve({'data':[]});
|
||||||
}
|
}
|
||||||
|
|
||||||
const searchURL = BASE_URL +
|
const searchURL = API_URL + 'subcategory' +
|
||||||
'/api/v1/styles/subcategory' +
|
|
||||||
'?search=' + encodeURIComponent(category) +
|
'?search=' + encodeURIComponent(category) +
|
||||||
'&page=' + searchCurrentPage +
|
'&page=' + searchCurrentPage +
|
||||||
'&per_page=10' +
|
'&per_page=10' +
|
||||||
|
@ -697,7 +700,7 @@ window.addEventListener('showStyles:done', function _() {
|
||||||
return readCache(cacheKey)
|
return readCache(cacheKey)
|
||||||
.then(json =>
|
.then(json =>
|
||||||
json ||
|
json ||
|
||||||
downloadInFrame(searchURL).then(writeCache))
|
downloadFromUSO(searchURL).then(writeCache))
|
||||||
.then(json => {
|
.then(json => {
|
||||||
searchCurrentPage = json.current_page + 1;
|
searchCurrentPage = json.current_page + 1;
|
||||||
searchTotalPages = json.total_pages;
|
searchTotalPages = json.total_pages;
|
||||||
|
@ -778,89 +781,74 @@ window.addEventListener('showStyles:done', function _() {
|
||||||
}
|
}
|
||||||
|
|
||||||
//endregion
|
//endregion
|
||||||
//region USO referrer spoofing via iframe
|
//region USO referrer spoofing
|
||||||
|
|
||||||
function downloadInFrame(url) {
|
function downloadFromUSO(url) {
|
||||||
return usoFrame ? new Promise((resolve, reject) => {
|
const requestId = getRandomId();
|
||||||
const id = performance.now();
|
xhrSpoofIds.add(requestId);
|
||||||
const timeout = setTimeout(() => {
|
xhrSpoofStart();
|
||||||
const {reject} = usoFrameQueue.get(id) || {};
|
return download(url, {
|
||||||
usoFrameQueue.delete(id);
|
body: null,
|
||||||
if (reject) reject();
|
responseType: 'json',
|
||||||
}, 10e3);
|
headers: {
|
||||||
const data = {url, resolve, reject, timeout};
|
'Referrer-Policy': 'origin-when-cross-origin',
|
||||||
usoFrameQueue.set(id, data);
|
[xhrSpoofTelltale]: requestId,
|
||||||
usoFrame.contentWindow.postMessage({xhr: {id, url}}, '*');
|
|
||||||
}) : setupFrame().then(() => downloadInFrame(url));
|
|
||||||
}
|
}
|
||||||
|
}).then(data => {
|
||||||
function setupFrame() {
|
xhrSpoofDone(requestId);
|
||||||
usoFrame = $create('iframe', {src: BASE_URL});
|
return data;
|
||||||
usoFrameQueue = new Map();
|
}).catch(data => {
|
||||||
|
xhrSpoofDone(requestId);
|
||||||
const stripHeaders = info => ({
|
return Promise.reject(data);
|
||||||
responseHeaders: info.responseHeaders.filter(({name}) => !/^X-Frame-Options$/i.test(name)),
|
|
||||||
});
|
});
|
||||||
chrome.webRequest.onHeadersReceived.addListener(stripHeaders, {
|
|
||||||
urls: [BASE_URL + '/'],
|
|
||||||
types: ['sub_frame'],
|
|
||||||
}, [
|
|
||||||
'blocking',
|
|
||||||
'responseHeaders',
|
|
||||||
]);
|
|
||||||
|
|
||||||
let frameId;
|
|
||||||
const stripResources = info => {
|
|
||||||
if (!frameId &&
|
|
||||||
info.frameId &&
|
|
||||||
info.type === 'sub_frame' &&
|
|
||||||
(info.initiator === location.origin || !info.initiator) && // Chrome 63+
|
|
||||||
(info.originUrl === location.href || !info.originUrl) && // FF 48+
|
|
||||||
info.url === BASE_URL + '/') {
|
|
||||||
frameId = info.frameId;
|
|
||||||
} else if (frameId === info.frameId && info.type !== 'xmlhttprequest') {
|
|
||||||
return {redirectUrl: 'data:,'};
|
|
||||||
}
|
}
|
||||||
};
|
|
||||||
chrome.webRequest.onBeforeRequest.addListener(stripResources, {
|
|
||||||
urls: ['<all_urls>'],
|
|
||||||
}, [
|
|
||||||
'blocking',
|
|
||||||
]);
|
|
||||||
setTimeout(() => {
|
|
||||||
chrome.webRequest.onBeforeRequest.removeListener(stripResources);
|
|
||||||
}, 10e3);
|
|
||||||
|
|
||||||
window.addEventListener('message', ({data, origin}) => {
|
function xhrSpoofStart() {
|
||||||
if (!data || origin !== BASE_URL) return;
|
if (chrome.webRequest.onBeforeSendHeaders.hasListener(xhrSpoof)) {
|
||||||
const {resolve, reject, timeout} = usoFrameQueue.get(data.id) || {};
|
return;
|
||||||
if (!resolve) return;
|
}
|
||||||
chrome.webRequest.onBeforeRequest.removeListener(stripResources);
|
const urls = [API_URL + '*', JSON_URL + '*'];
|
||||||
usoFrameQueue.delete(data.id);
|
const types = ['xmlhttprequest'];
|
||||||
clearTimeout(timeout);
|
const options = ['blocking', 'requestHeaders'];
|
||||||
// [being overcautious] a string response is used instead of relying on responseType=json
|
// spoofing Referer requires extraHeaders in Chrome 72+
|
||||||
// because it was invoked in a web page context so another extension may have incorrectly spoofed it
|
if (chrome.webRequest.OnBeforeSendHeadersOptions.EXTRA_HEADERS) {
|
||||||
const json = tryJSONparse(data.response);
|
options.push(chrome.webRequest.OnBeforeSendHeadersOptions.EXTRA_HEADERS);
|
||||||
if (json && data.status < 400) {
|
}
|
||||||
resolve(json);
|
chrome.webRequest.onBeforeSendHeaders.addListener(xhrSpoof, {urls, types}, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
function xhrSpoofDone(requestId) {
|
||||||
|
xhrSpoofIds.delete(requestId);
|
||||||
|
if (!xhrSpoofIds.size) {
|
||||||
|
chrome.webRequest.onBeforeSendHeaders.removeListener(xhrSpoof);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function xhrSpoof({requestHeaders}) {
|
||||||
|
let referer, hasTelltale;
|
||||||
|
for (let i = requestHeaders.length; --i >= 0;) {
|
||||||
|
const header = requestHeaders[i];
|
||||||
|
if (header.name.toLowerCase() === 'referer') {
|
||||||
|
referer = header;
|
||||||
|
} else if (header.name === xhrSpoofTelltale) {
|
||||||
|
hasTelltale = xhrSpoofIds.has(header.value);
|
||||||
|
requestHeaders.splice(i, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!hasTelltale) {
|
||||||
|
// not our request (unlikely but just in case)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (referer) {
|
||||||
|
referer.value = BASE_URL;
|
||||||
} else {
|
} else {
|
||||||
reject(data.status);
|
requestHeaders.push({name: 'Referer', value: BASE_URL});
|
||||||
|
}
|
||||||
|
return {requestHeaders};
|
||||||
}
|
}
|
||||||
});
|
|
||||||
|
|
||||||
return new Promise((resolve, reject) => {
|
function getRandomId() {
|
||||||
const done = event => {
|
return btoa(Math.random()).replace(/[^a-z]/gi, '');
|
||||||
chrome.webRequest.onHeadersReceived.removeListener(stripHeaders);
|
|
||||||
(event.type === 'load' ? resolve : reject)();
|
|
||||||
usoFrameQueue.forEach(({url}, id) => {
|
|
||||||
usoFrame.contentWindow.postMessage({xhr: {id, url}}, '*');
|
|
||||||
});
|
|
||||||
};
|
|
||||||
usoFrame.addEventListener('load', done, {once: true});
|
|
||||||
usoFrame.addEventListener('error', done, {once: true});
|
|
||||||
usoFrame.style.setProperty('display', 'none', 'important');
|
|
||||||
document.body.appendChild(usoFrame);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//endregion
|
//endregion
|
||||||
|
|
Loading…
Reference in New Issue
Block a user