自定义过滤器
在之前的版本里,我们允许用户使用内置的几个过滤器进行节点过滤。现在 Surgio 已经支持在 Provider 和 Surgio 配置中自定义 Filter。需要提醒一下大家,原来内置的国别过滤器已经拓展了不少,可以在 这里 查看。
如何自定义
在 Provider 配置或 surgio.conf.js
中加入 customFilters
,然后参照下面例子来自定义。
看不懂的同学也不用怕,依葫芦画瓢很简单的哈 🙋♂️。
module.exports = {
url: 'http://example.com/ss-sub.txt',
type: 'shadowsocks_subscribe',
customFilters: {
wo_yao_netflix_jie_dian: nodeConfig => nodeConfig.nodeName.includes('Netflix'),
},
};
提示
这里只是举个例子,并不推荐你重复实现 Surgio 已经内置的过滤器。
入参 nodeConfig
包含节点的所有属性,其中有一些你可能会用到的属性写在下面:
nodeName
- 节点名type
- 节点类型(例如shadowsocks
)
如何使用
在模板(.tpl
文件)中你可以使用 customFilters.wo_yao_netflix_jie_dian
。
<!-- .tpl 文件 -->
{{ getSurgeNodes(nodeList, customFilters.wo_yao_netflix_jie_dian) }}
工具方法
很多同学在自定义过滤器时绝大部分时候都希望定义关键词或正则式过滤器,所以 Surgio 提供了几个工具方法方便大家不用写太多代码。
在开干前要介绍一个知识点。这几个工具方法是通过模块引入的,你需要在你的 Provider 文件头部加上一行代码,才能使用它们。
mergeFilters
把多个过滤器合并。第二个入参是开启严格模式。
默认情况下不同的过滤器间是「或」的关系,开启严格模式后是「和」的关系。
const { utils } = require('surgio');
module.exports = {
url: 'http://example.com/ss-sub.txt',
type: 'shadowsocks_subscribe',
customFilters: {
usNetflixFilter: utils.mergeFilters([utils.usFilter, utils.netflixFilter], true), // 美国的 Netflix 节点
},
};
useKeywords
生成一个关键词过滤器。第二个入参是开启严格模式。
const { utils } = require('surgio');
module.exports = {
url: 'http://example.com/ss-sub.txt',
type: 'shadowsocks_subscribe',
customFilters: {
gameFilter: utils.useKeywords(['游戏']),
},
};
注意
该过滤器不保证顺序。
discardKeywords
生成一个反向关键词过滤器。第二个入参是开启严格模式。
注意
discardKeywords
的目的是 过滤掉,useKeywords
的目的是 过滤出。
useRegexp
生成一个正则表达式过滤器。
const { utils } = require('surgio');
module.exports = {
url: 'http://example.com/ss-sub.txt',
type: 'shadowsocks_subscribe',
customFilters: {
iplcFilter: utils.useRegexp(/iplc/i),
},
};
useSortedKeywords
生成一个排序型关键词过滤器。
假设某机场在输出节点列表时按照编码正序排列,你可以采用该过滤器将某几个节点过滤并排序输出。
const { utils } = require('surgio');
module.exports = {
url: 'http://example.com/ss-sub.txt',
type: 'shadowsocks_subscribe',
customFilters: {
preferNodes: utils.useSortedKeywords(['V001', 'V009', 'V003']),
},
};
mergeSortedFilters
合并多个过滤器,生成一个新的排序型过滤器。
假设某机场在输出节点列表时按照国家或地区进行归集,你可以采用该过滤器将某地区的节点过滤并排序输出。
const { utils } = require('surgio');
module.exports = {
url: 'http://example.com/ss-sub.txt',
type: 'shadowsocks_subscribe',
customFilters: {
preferNodes: utils.mergeSortedFilters([utils.hkFilter, utils.usFilter]), // 也支持 useRegexp
},
};
注意
- 不能合并排序型过滤器;
- 若某个节点同时匹配多个规则,只会出现在第一次匹配的位置;
- Provider 的配置项
nodeFilter
也支持排序类型的过滤器,但我们并不建议您这么用,因为 Provider 的nodeFilter
配置项仅针对当前 Provider 而非合并进来的所有的节点,再者如果你同时在nodeFilter
和getNodeNames
(包括但不限于)中使用排序过滤器,会进行多次排序,很难避免不出错。所以请尽可能在nodeFilter
中使用普通的过滤器;
useProviders
合并 Provider 之后若是想快速地过滤出某几个 Provider 中的节点作为一个策略组,可以使用该过滤器。
第二个入参是开启严格模式。严格模式下会严格匹配 Provider 名称(注意和 useKeywords
的严格模式不同)。
// surgio.conf.js
const { utils } = require('surgio');
module.exports = {
customFilters: {
providerFilter: utils.useProviders(['dlercloud', 'maying']),
looseProviderFilter: utils.useProviders(['dlercloud', 'maying'], false), // 松散模式
},
};
注意
- 该过滤器不保证顺序;
- 默认开启严格模式;
discardProviders
合并 Provider 之后若是想快速地舍弃某几个 Provider 中的节点作为一个策略组,可以使用该过滤器。
第二个入参是开启严格模式。严格模式下会严格匹配 Provider 名称(注意和 useKeywords
的严格模式不同)。
// surgio.conf.js
const { utils } = require('surgio');
module.exports = {
customFilters: {
providerFilter: utils.discardProviders(['dlercloud', 'maying']),
looseProviderFilter: utils.discardProviders(['dlercloud', 'maying'], false), // 松散模式
},
};
注意
- 该过滤器不保证顺序;
- 默认开启严格模式;
useGlob
能够使用 glob 语法筛选出节点。
支持的语法见 这里。
// surgio.conf.js
const { utils } = require('surgio');
module.exports = {
customFilters: {
premiumNodes: utils.useGlob('*Premium*'),
},
}
discardGlob
能够使用 glob 语法移出节点。
支持的语法见 这里。
// surgio.conf.js
const { utils } = require('surgio');
module.exports = {
customFilters: {
hongKongNodes: utils.discardGlob('*(US|JP)*'), // 不匹配 US 和 JP
},
}
reverseFilter
你可以使用 reverseFilter
来反转过滤器的结果。
const notUSFilter = reverseFilter(usFilter)
mergeReversedFilters
你可以使用 mergeReversedFilters
来合并多个反转过滤器,discardKeywords
, discardProviders
, discardGlob
过滤器。
// 丢弃 US 和包含 BGP 关键字的节点
const notUSAndNotBGP = mergeReversedFilters(
[notUSFilter, discardKeywords(['BGP'])],
true, // 严格模式
)
// 香港 BGP ✅
// 香港 IPLC ✅
// 洛杉矶 BGP 🚫
// 洛杉矶 IPLC ✅
如何在自定义过滤器时引用内置的过滤器
你可能需要在自定义过滤器时引用内置的过滤器(你可以在 这里 找到所有内置的过滤器)。
假设我需要增强内置的 netflixFilter
,使得所有满足内置 netflixFilter
的节点或者名称包含「流媒体」的节点,作为新的 netflixFilter
。
const { utils } = require('surgio');
const myNetflixFilter = utils.useKeywords(['流媒体']);
module.exports = {
type: 'clash',
url: '',
netflixFilter: utils.mergeFilters([utils.netflixFilter, myNetflixFilter]), // 记得这里应该是松散模式
};