feat: 支持按顺序合并本地和远程订阅(前端版本 > 2.14.14 可输入)

This commit is contained in:
xream 2023-11-29 03:42:25 +08:00
parent 3054d5cd5d
commit c541b83037
No known key found for this signature in database
GPG Key ID: 1D2C5225471789F9
5 changed files with 68 additions and 14 deletions

View File

@ -44,7 +44,7 @@ jobs:
run: | run: |
cd backend cd backend
SUBSTORE_RELEASE=`node --eval="process.stdout.write(require('./package.json').version)"` SUBSTORE_RELEASE=`node --eval="process.stdout.write(require('./package.json').version)"`
echo "::set-output name=release_tag::$SUBSTORE_RELEASE" echo "release_tag=$SUBSTORE_RELEASE" >> $GITHUB_OUTPUT
- name: Prepare release - name: Prepare release
run: | run: |
cd backend cd backend

View File

@ -1,6 +1,6 @@
{ {
"name": "sub-store", "name": "sub-store",
"version": "2.14.104", "version": "2.14.106",
"description": "Advanced Subscription Manager for QX, Loon, Surge, Stash and ShadowRocket.", "description": "Advanced Subscription Manager for QX, Loon, Surge, Stash and ShadowRocket.",
"main": "src/main.js", "main": "src/main.js",
"scripts": { "scripts": {

View File

@ -20,18 +20,22 @@ async function downloadSubscription(req, res) {
req.query.target || getPlatformFromHeaders(req.headers) || 'JSON'; req.query.target || getPlatformFromHeaders(req.headers) || 'JSON';
$.info(`正在下载订阅:${name}`); $.info(`正在下载订阅:${name}`);
let { url, ua, content } = req.query; let { url, ua, content, mergeSources } = req.query;
if (url) { if (url) {
url = decodeURIComponent(url); url = decodeURIComponent(url);
$.info(`指定 url: ${url}`); $.info(`指定远程订阅 URL: ${url}`);
} }
if (ua) { if (ua) {
ua = decodeURIComponent(ua); ua = decodeURIComponent(ua);
$.info(`指定 ua: ${ua}`); $.info(`指定远程订阅 User-Agent: ${ua}`);
} }
if (content) { if (content) {
content = decodeURIComponent(content); content = decodeURIComponent(content);
$.info(`指定 content: ${content}`); $.info(`指定本地订阅: ${content}`);
}
if (mergeSources) {
mergeSources = decodeURIComponent(mergeSources);
$.info(`指定合并来源: ${mergeSources}`);
} }
const allSubs = $.read(SUBS_KEY); const allSubs = $.read(SUBS_KEY);
@ -45,6 +49,7 @@ async function downloadSubscription(req, res) {
url, url,
ua, ua,
content, content,
mergeSources,
}); });
if (sub.source !== 'local' || url) { if (sub.source !== 'local' || url) {

View File

@ -16,7 +16,10 @@ async function compareSub(req, res) {
const sub = req.body; const sub = req.body;
const target = req.query.target || 'JSON'; const target = req.query.target || 'JSON';
let content; let content;
if (sub.source === 'local') { if (
sub.source === 'local' &&
!['localFirst', 'remoteFirst'].includes(sub.mergeSources)
) {
content = sub.content; content = sub.content;
} else { } else {
try { try {
@ -38,6 +41,11 @@ async function compareSub(req, res) {
); );
return; return;
} }
if (sub.mergeSources === 'localFirst') {
content.unshift(sub.content);
} else if (sub.mergeSources === 'remoteFirst') {
content.push(sub.content);
}
} }
// parse proxies // parse proxies
const original = (Array.isArray(content) ? content : [content]) const original = (Array.isArray(content) ? content : [content])
@ -85,7 +93,12 @@ async function compareCollection(req, res) {
const sub = findByName(allSubs, name); const sub = findByName(allSubs, name);
try { try {
let raw; let raw;
if (sub.source === 'local') { if (
sub.source === 'local' &&
!['localFirst', 'remoteFirst'].includes(
sub.mergeSources,
)
) {
raw = sub.content; raw = sub.content;
} else { } else {
raw = await Promise.all( raw = await Promise.all(
@ -95,6 +108,11 @@ async function compareCollection(req, res) {
.filter((i) => i.length) .filter((i) => i.length)
.map((url) => download(url, sub.ua)), .map((url) => download(url, sub.ua)),
); );
if (sub.mergeSources === 'localFirst') {
raw.unshift(sub.content);
} else if (sub.mergeSources === 'remoteFirst') {
raw.push(sub.content);
}
} }
// parse proxies // parse proxies
let currentProxies = (Array.isArray(raw) ? raw : [raw]) let currentProxies = (Array.isArray(raw) ? raw : [raw])

View File

@ -22,14 +22,24 @@ export default function register($app) {
$app.get('/api/sync/artifact/:name', syncArtifact); $app.get('/api/sync/artifact/:name', syncArtifact);
} }
async function produceArtifact({ type, name, platform, url, ua, content }) { async function produceArtifact({
type,
name,
platform,
url,
ua,
content,
mergeSources,
}) {
platform = platform || 'JSON'; platform = platform || 'JSON';
if (type === 'subscription') { if (type === 'subscription') {
const allSubs = $.read(SUBS_KEY); const allSubs = $.read(SUBS_KEY);
const sub = findByName(allSubs, name); const sub = findByName(allSubs, name);
let raw; let raw;
if (url) { if (content && !['localFirst', 'remoteFirst'].includes(mergeSources)) {
raw = content;
} else if (url) {
raw = await Promise.all( raw = await Promise.all(
url url
.split(/[\r\n]+/) .split(/[\r\n]+/)
@ -37,9 +47,15 @@ async function produceArtifact({ type, name, platform, url, ua, content }) {
.filter((i) => i.length) .filter((i) => i.length)
.map((url) => download(url, ua)), .map((url) => download(url, ua)),
); );
} else if (content) { if (mergeSources === 'localFirst') {
raw = content; raw.unshift(content);
} else if (sub.source === 'local') { } else if (mergeSources === 'remoteFirst') {
raw.push(content);
}
} else if (
sub.source === 'local' &&
!['localFirst', 'remoteFirst'].includes(sub.mergeSources)
) {
raw = sub.content; raw = sub.content;
} else { } else {
raw = await Promise.all( raw = await Promise.all(
@ -49,6 +65,11 @@ async function produceArtifact({ type, name, platform, url, ua, content }) {
.filter((i) => i.length) .filter((i) => i.length)
.map((url) => download(url, sub.ua)), .map((url) => download(url, sub.ua)),
); );
if (sub.mergeSources === 'localFirst') {
raw.unshift(sub.content);
} else if (sub.mergeSources === 'remoteFirst') {
raw.push(sub.content);
}
} }
// parse proxies // parse proxies
let proxies = (Array.isArray(raw) ? raw : [raw]) let proxies = (Array.isArray(raw) ? raw : [raw])
@ -102,7 +123,12 @@ async function produceArtifact({ type, name, platform, url, ua, content }) {
try { try {
$.info(`正在处理子订阅:${sub.name}...`); $.info(`正在处理子订阅:${sub.name}...`);
let raw; let raw;
if (sub.source === 'local') { if (
sub.source === 'local' &&
!['localFirst', 'remoteFirst'].includes(
sub.mergeSources,
)
) {
raw = sub.content; raw = sub.content;
} else { } else {
raw = await await Promise.all( raw = await await Promise.all(
@ -112,6 +138,11 @@ async function produceArtifact({ type, name, platform, url, ua, content }) {
.filter((i) => i.length) .filter((i) => i.length)
.map((url) => download(url, sub.ua)), .map((url) => download(url, sub.ua)),
); );
if (sub.mergeSources === 'localFirst') {
raw.unshift(sub.content);
} else if (sub.mergeSources === 'remoteFirst') {
raw.push(sub.content);
}
} }
// parse proxies // parse proxies
let currentProxies = (Array.isArray(raw) ? raw : [raw]) let currentProxies = (Array.isArray(raw) ? raw : [raw])