mirror of
https://github.com/imputnet/cobalt.git
synced 2025-06-28 17:38:31 +00:00
155 lines
4.6 KiB
JavaScript
155 lines
4.6 KiB
JavaScript
import Cookie from './cookie.js';
|
|
|
|
import { readFile, writeFile } from 'fs/promises';
|
|
import { Red, Green, Yellow } from '../../misc/console-text.js';
|
|
import { parse as parseSetCookie, splitCookiesString } from 'set-cookie-parser';
|
|
import * as cluster from '../../misc/cluster.js';
|
|
import { isCluster } from '../../config.js';
|
|
|
|
const WRITE_INTERVAL = 60000;
|
|
const VALID_SERVICES = new Set([
|
|
'instagram',
|
|
'instagram_bearer',
|
|
'reddit',
|
|
'twitter',
|
|
'youtube_oauth'
|
|
]);
|
|
|
|
const invalidCookies = {};
|
|
let cookies = {}, dirty = false, intervalId;
|
|
|
|
function writeChanges(cookiePath) {
|
|
if (!dirty) return;
|
|
dirty = false;
|
|
|
|
const cookieData = JSON.stringify({ ...cookies, ...invalidCookies }, null, 4);
|
|
writeFile(cookiePath, cookieData).catch((e) => {
|
|
console.warn(`${Yellow('[!]')} failed writing updated cookies to storage`);
|
|
console.warn(e);
|
|
clearInterval(intervalId);
|
|
intervalId = null;
|
|
})
|
|
}
|
|
|
|
const setupMain = async (cookiePath) => {
|
|
try {
|
|
cookies = await readFile(cookiePath, 'utf8');
|
|
cookies = JSON.parse(cookies);
|
|
for (const serviceName in cookies) {
|
|
if (!VALID_SERVICES.has(serviceName)) {
|
|
console.warn(`${Yellow('[!]')} ignoring unknown service in cookie file: ${serviceName}`);
|
|
} else if (!Array.isArray(cookies[serviceName])) {
|
|
console.warn(`${Yellow('[!]')} ${serviceName} in cookies file is not an array, ignoring it`);
|
|
} else if (cookies[serviceName].some(c => typeof c !== 'string')) {
|
|
console.warn(`${Yellow('[!]')} some cookie for ${serviceName} contains non-string value in cookies file`);
|
|
} else continue;
|
|
|
|
invalidCookies[serviceName] = cookies[serviceName];
|
|
delete cookies[serviceName];
|
|
}
|
|
|
|
if (!intervalId) {
|
|
intervalId = setInterval(() => writeChanges(cookiePath), WRITE_INTERVAL);
|
|
}
|
|
|
|
cluster.broadcast({ cookies });
|
|
|
|
console.log(`${Green('[✓]')} cookies loaded successfully!`);
|
|
} catch (e) {
|
|
console.error(`${Yellow('[!]')} failed to load cookies.`);
|
|
console.error('error:', e);
|
|
}
|
|
}
|
|
|
|
const setupWorker = async () => {
|
|
cookies = (await cluster.waitFor('cookies')).cookies;
|
|
}
|
|
|
|
export const loadFromFile = async (path) => {
|
|
if (cluster.isPrimary) {
|
|
await setupMain(path);
|
|
} else if (cluster.isWorker) {
|
|
await setupWorker();
|
|
}
|
|
|
|
dirty = false;
|
|
}
|
|
|
|
export const setup = async (path) => {
|
|
await loadFromFile(path);
|
|
|
|
if (isCluster) {
|
|
const messageHandler = (message) => {
|
|
if ('cookieUpdate' in message) {
|
|
const { cookieUpdate } = message;
|
|
|
|
if (cluster.isPrimary) {
|
|
dirty = true;
|
|
cluster.broadcast({ cookieUpdate });
|
|
}
|
|
|
|
const { service, idx, cookie } = cookieUpdate;
|
|
cookies[service][idx] = cookie;
|
|
}
|
|
}
|
|
|
|
if (cluster.isPrimary) {
|
|
cluster.mainOnMessage(messageHandler);
|
|
} else {
|
|
process.on('message', messageHandler);
|
|
}
|
|
}
|
|
}
|
|
|
|
export function getCookie(service) {
|
|
if (!VALID_SERVICES.has(service)) {
|
|
console.error(`${Red('[!]')} ${service} not in allowed services list for cookies.`
|
|
+ ' if adding a new cookie type, include it there.');
|
|
return;
|
|
}
|
|
|
|
if (!cookies[service] || !cookies[service].length) return;
|
|
|
|
const idx = Math.floor(Math.random() * cookies[service].length);
|
|
|
|
const cookie = cookies[service][idx];
|
|
if (typeof cookie === 'string') {
|
|
cookies[service][idx] = Cookie.fromString(cookie);
|
|
}
|
|
|
|
cookies[service][idx].meta = { service, idx };
|
|
return cookies[service][idx];
|
|
}
|
|
|
|
export function updateCookieValues(cookie, values) {
|
|
let changed = false;
|
|
|
|
for (const [ key, value ] of Object.entries(values)) {
|
|
changed = cookie.set(key, value) || changed;
|
|
}
|
|
|
|
if (changed && cookie.meta) {
|
|
dirty = true;
|
|
if (isCluster) {
|
|
const message = { cookieUpdate: { ...cookie.meta, cookie } };
|
|
cluster.send(message);
|
|
}
|
|
}
|
|
|
|
return changed;
|
|
}
|
|
|
|
export function updateCookie(cookie, headers) {
|
|
if (!cookie) return;
|
|
|
|
const parsed = parseSetCookie(
|
|
splitCookiesString(headers.get('set-cookie')),
|
|
{ decodeValues: false }
|
|
), values = {}
|
|
|
|
cookie.unset(parsed.filter(c => c.expires < new Date()).map(c => c.name));
|
|
parsed.filter(c => !c.expires || c.expires > new Date()).forEach(c => values[c.name] = c.value);
|
|
|
|
updateCookieValues(cookie, values);
|
|
}
|