将机场ss节点批量转换成ss字符串链接批量添加到passwall

最近我使用的机场节点都更新了,我在一个新的 immortalwrt 环境上使用原有的订阅链接,始终无法订阅到节点,我还一直以为是我找的订阅转换服务有问题。试了好几个服务,都无法订阅到节点。

测试了这些订阅转换服务(subscription converter)

订阅全部都失败了。passwall 给出的错误要么是“订阅失败,可能是订阅地址无效,或是网络问题,请诊断”,要么是 “成功解析【xxx_airport】节点数量: 0”。

我开始了研究。

旧 immortalwrt 环境上的 passwall 版本

# opkg list-installed | grep passwall
luci-app-passwall - 25.9.23-r1
luci-i18n-passwall-zh-cn - 25.270.37028~65fa739

新 immortalwrt 环境上的 passwall 版本

# opkg list-installed | grep passwall
luci-app-passwall - 25.12.16-r1
luci-i18n-passwall-zh-cn - 25.350.06350~ba7f272

版本确实不一样,源里的版本升级了。但因为 op 源里一般只保存最新的软件包,我放弃了降级,也放弃了卸载和重新安装同样的版本(要去找同版本 ipk 文件),后面的经历证明此举明智。

我开始对订阅链接产生兴趣,看看能发现什么规律。

果然!发现规律了,我使用的机场提供的订阅链接,我一般只使用 ?clash=1 的那个,在 passwall 中订阅时需要先转换,把节点目标设成 shadowsocks(SIP002)

注:SIP002 是 Shadowsocks 官方定义的标准 URI 格式规范(即标准的“ss://”链接格式)

我把 clash=1 的订阅拉到本地发现就是 clash 的 yaml 文本,最终发现所有节点都是 type: ssr,原来是这样!!!就是因为 clash 的订阅链接只返回了 ssr 节点,但我本地的 passwall 使用 ss/trojan/vmess/vless/hysteria2 节点,因为没有安装 ssr。

clash订阅的全部是ssr节点.jpg

我又从机场的web后台复制了一个 ss 节点的 ss:// 链接手动导入,可以成功导入!遗憾的是,机场提供了批量复制 ssr 链接功能,但是没有提供批量复制 ss 链接功能。于是我开始想办法手动转换。

我本地访问了 quanx 的订阅链接 ?list=quantumultx,得到一个94行的文本,乍一看就能看出规律,47个节点,每个节点各有一个 ss 链接和一个 ssr 链接。长得像这样

shadowsocks=xxx.com:3044, method=chacha20-ietf, password=xxx, ssr-protocol=auth_aes128_sha1, ssr-protocol-param=xxx, obfs=plain, obfs-host="xxx, tag=Standard|台湾|IEPL|01
shadowsocks=xxx.com:30333, method=rc4-md5, password=xxx, obfs=http, obfs-host=xxx, obfs-uri=/, tag=Standard|台湾|IEPL|01

于是我将他们转到 linux 下,

cat > sub.txt<<'EOF'
94 行文本
EOF

执行命令 grep -v 'ssr-protocol=' sub.txt > sub_new.txt,将 ssr 节点对应的行全都删掉,重定向到新文件,新文件里全部都是 ss 节点信息。

最后我就是找规律,看机场给的 ss 链接是根据什么规律生成的,然后将剩下的 47 个节点数据全部转换成 ss:// 文本链接。附上我使用的 java 代码(java17)

import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.Base64;
import java.util.HashMap;
import java.util.Map;

public class SSUrlParser {

    public static String parseAndGenerate(String line) throws Exception {
        Map<String, String> map = new HashMap<>();
        for (String part : line.split(",")) {
            String[] kv = part.trim().split("=", 2);
            map.put(kv[0], kv[1]);
        }

        String[] hostPort = map.get("shadowsocks").split(":");
        String host = hostPort[0];
        int port = Integer.parseInt(hostPort[1]);

        String userInfo = map.get("method") + ":" + map.get("password");
        String userInfoBase64 = Base64.getEncoder().encodeToString(userInfo.getBytes(StandardCharsets.UTF_8));

        // 插件参数
        String pluginRaw = "obfs-local;obfs=" + map.get("obfs") + ";obfs-host=" + map.get("obfs-host");
        String pluginEncoded = URLEncoder.encode(pluginRaw, StandardCharsets.UTF_8.toString());

        // group 参数:Base64 编码
        String group = map.getOrDefault("tag", ""); // 没有单独 group 时用 tag
//        String groupBase64 = Base64.getEncoder().encodeToString(group.getBytes(StandardCharsets.UTF_8));

        String groupName = "机场名称特有字符串"; // 这里的规律,可能不同机场不一样
        String groupBase64 = Base64.getEncoder()
                .withoutPadding()
                .encodeToString(groupName.getBytes(StandardCharsets.UTF_8));

        // tag 参数 URL encode
        String tagEncoded = URLEncoder.encode(group, StandardCharsets.UTF_8.toString());

        return "ss://" + userInfoBase64
                + "@" + host + ":" + port
                + "/?plugin=" + pluginEncoded
                + "&group=" + groupBase64
                + "#" + tagEncoded;
    }

    public static void main(String[] args) throws Exception {
        String input = """
                shadowsocks=xxx.com:30333, method=rc4-md5, password=xxx, obfs=http, obfs-host=xxx, obfs-uri=/, tag=Standard|台湾|IEPL|01
                shadowsocks=xxx.com:30334, method=rc4-md5, password=xxx, obfs=http, obfs-host=xxx, obfs-uri=/, tag=Standard|台湾|IEPL|02
                shadowsocks=xxx.com:30335, method=rc4-md5, password=xxx, obfs=http, obfs-host=xxx, obfs-uri=/, tag=Standard|台湾|IEPL|03
                """;  // 使用 Text Blocks 特性
        // 遍历每一行
        input.lines()
                .map(String::trim)        // 去掉首尾空白
                .filter(line -> !line.isEmpty()) // 判空
                .forEach(line -> {
                    try {
                        System.out.println(parseAndGenerate(line));
                    }
                    catch (Exception e) {
                        throw new RuntimeException("当前行计算出了问题...");
                    }
                }); // 调用自定义方法
    }
}

上述代码执行后即可将机场ss节点信息全都转换成 ss链接,但是注意,这可能不适用于你的机场。最终我将47行ss链接成功一次性导入 passwall:节点列表标签页下 -> 通过链接添加节点功能 -> 一行一个,全部复制到这里,可以添加一个单独的分组。

passwall 节点列表 - 通过链接添加节点- 一行一个.jpg

没有使用订阅的方式(机场没有提供符合我需求的订阅链接),成功批量导入节点!完美。

没有使用订阅的方式-成功批量导入节点.jpg

不过后来我发现,将 ?list=quantumultx 这个订阅链接作为源,然后转换为 ShadowsocksSIP002,最终生成临时订阅链接放进 passwall,成功订阅 😂️,解析到了93个,应该是把所有的节点 ss/ssr 都识别出来了,passwall 这边每个节点显示了两次,至于为什么还少一个(应该94个),就不再探究了。

最终学到了:
ss:// URL 中,只有 method / password / host / port / plugin 是“协议级可推导”的
grouptag 都是“订阅生成器的业务字段”

还有

String groupBase64 = Base64.getEncoder()
                .withoutPadding()
                .encodeToString(groupName.getBytes(StandardCharsets.UTF_8));

String groupBase64 = Base64.getEncoder()
                .encodeToString(groupName.getBytes(StandardCharsets.UTF_8));

的区别。

添加新评论