-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathmerge_clash.py
More file actions
261 lines (214 loc) · 7.92 KB
/
merge_clash.py
File metadata and controls
261 lines (214 loc) · 7.92 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
import hashlib
import socket
from collections import defaultdict
from ipaddress import ip_address
import re
from typing import Dict # 添加此导入以修复 Dict 类型提示
import requests
import yaml
try:
import geoip2.database
except ImportError:
geoip2 = None
# ================= 配置 =================
CONFIG_URLS = [
"https://raw.githubusercontent.com/free18/v2ray/refs/heads/main/c.yaml",
# 添加其他订阅链接...
]
MMDB_PATH = "GeoLite2-Country.mmdb"
DOMAIN_RULES = {
"hk": "HK", "tw": "TW", "us": "US", "sg": "SG", "jp": "JP", "kr": "KR",
"868863.xyz": "HK", "bestcdn": "HK", "netvora.space": "HK",
"206352.xyz": "HK", "206353.xyz": "HK", "206355.xyz": "HK",
"cloudflare": "US", "akamai": "US", "fastly": "US",
"cdn": "HK", "v2ray": "CN", "proxy": "CN", "speed": "HK",
"xyz": "HK", "space": "HK",
"visa.com": "US", "dpdns.org": "US", "db-link.in": "US",
}
CLOUDFLARE_PREFIXES = [
"104.16.", "104.17.", "104.18.", "104.19.", "104.20.", "104.21.",
"162.158.", "162.159.", "172.64.", "172.65.", "172.66.", "172.67.",
"173.245.48.", "173.245.49.", "173.245.50.", "173.245.51.", "173.245.52.",
"173.245.53.", "173.245.54.", "173.245.55.", "173.245.56.", "173.245.57.",
"173.245.58.", "173.245.59.",
]
# 排除的国家代码
EXCLUDED_COUNTRIES = ["CN", "RU"]
# 国旗生成
def country_code_to_flag(code: str) -> str:
if not code or len(code) != 2:
return "🇽🇽"
code = code.upper()
offset = 0x1F1E6 - ord('A')
try:
return chr(ord(code[0]) + offset) + chr(ord(code[1]) + offset)
except:
return "🇽🇽"
# 中文名称(常见节点地区)
country_names = {
'US': '美国', 'CA': '加拿大', 'GB': '英国', 'AU': '澳大利亚',
'DE': '德国', 'FR': '法国', 'IT': '意大利', 'ES': '西班牙',
'NL': '荷兰', 'SE': '瑞典', 'NO': '挪威', 'DK': '丹麦',
'FI': '芬兰', 'CH': '瑞士', 'BE': '比利时', 'AT': '奥地利',
'IE': '爱尔兰', 'NZ': '新西兰', 'ZA': '南非', 'IN': '印度',
'CN': '中国', 'JP': '日本', 'KR': '韩国', 'SG': '新加坡',
'MY': '马来西亚', 'TH': '泰国', 'VN': '越南', 'ID': '印尼',
'PH': '菲律宾', 'BR': '巴西', 'AR': '阿根廷', 'MX': '墨西哥',
'RU': '俄罗斯', 'TR': '土耳其', 'SA': '沙特', 'AE': '阿联酋',
'EG': '埃及', 'IL': '以色列', 'GR': '希腊', 'PT': '葡萄牙',
'CZ': '捷克', 'HU': '匈牙利', 'SK': '斯洛伐克', 'HK': '香港',
'TW': '台湾', 'MO': '澳门', 'PL': '波兰', 'UA': '乌克兰',
'RO': '罗马尼亚', 'RS': '塞尔维亚',
'XX': '未知',
}
# GeoLite2 初始化
reader = None
if geoip2:
try:
reader = geoip2.database.Reader(MMDB_PATH)
print("GeoLite2 加载成功")
except Exception as e:
print(f"GeoLite2 加载失败: {e}")
# ================= 提取原有名称中的国家 =================
def extract_country_from_name(name: str) -> str | None:
if not name:
return None
name_upper = name.upper()
# 1. 国旗 emoji
flag_pattern = r'[\U0001F1E6-\U0001F1FF]{2}'
flags = re.findall(flag_pattern, name)
for flag in flags:
code = (
chr(ord(flag[0]) - 0x1F1E6 + ord('A')) +
chr(ord(flag[1]) - 0x1F1E6 + ord('A'))
)
if code in country_names:
return code
# 2. 两字母代码
common_codes = list(country_names.keys()) + ['UK', 'KOR', 'JPN']
for code in common_codes:
if re.search(r'\b' + re.escape(code) + r'\b', name_upper):
return code
# 3. 中文名称
for code, cn in country_names.items():
if cn in name:
return code
# 4. 括号/前缀
match = re.search(r'\[([A-Z]{2})\]', name_upper)
if match and match.group(1) in country_names:
return match.group(1)
return None
def get_country(ip: str, server: str = "", original_name: str = "") -> str:
# 优先:原名称提取
if original_name:
code = extract_country_from_name(original_name)
if code:
print(f"原名称提取: {original_name} → {code}")
return code
server_lower = server.lower()
# 域名规则
for key, cc in DOMAIN_RULES.items():
if key in server_lower:
print(f"域名匹配: {server} → {cc}")
return cc.upper()
# Cloudflare IP
for prefix in CLOUDFLARE_PREFIXES:
if ip.startswith(prefix):
print(f"Cloudflare IP: {ip} → US")
return "US"
# GeoLite2
if reader:
try:
resp = reader.country(ip)
cc = resp.country.iso_code
if cc:
print(f"GeoLite2: {ip} → {cc}")
return cc.upper()
except Exception as e:
print(f"GeoLite2 失败 {ip}: {e}")
# 兜底
print(f"未知: {server} / {ip}")
return "XX"
def normalize_proxy_key(proxy: Dict) -> str:
key_fields = ["type", "server", "port"]
p_type = proxy.get("type", "")
if p_type in ["vmess", "vless"]:
key_fields.append("uuid")
elif p_type == "trojan":
key_fields.append("password")
elif p_type == "ss":
key_fields.extend(["cipher", "password"])
values = [str(proxy.get(k, "")) for k in key_fields]
return hashlib.md5("".join(values).encode()).hexdigest()
def fetch_config(source: str) -> Dict:
if source.startswith(("http://", "https://")):
r = requests.get(source, timeout=10)
r.raise_for_status()
return yaml.safe_load(r.text)
else:
with open(source, encoding="utf-8") as f:
return yaml.safe_load(f)
# ================= 主逻辑 =================
all_proxies = []
seen_keys = set()
for src in CONFIG_URLS:
print(f"处理: {src}")
try:
config = fetch_config(src)
for p in config.get("proxies", []):
key = normalize_proxy_key(p)
if key not in seen_keys:
seen_keys.add(key)
all_proxies.append(p)
except Exception as e:
print(f"跳过 {src} : {e}")
print(f"去重后节点数:{len(all_proxies)}")
# 查询国家并分组(排除 CN/RU)
country_groups = defaultdict(list)
for proxy in all_proxies:
server = proxy.get("server", "")
original_name = proxy.get("name", "")
if not server:
continue # 跳过无效节点
try:
ip = str(ip_address(server))
except ValueError:
try:
ip = socket.gethostbyname(server)
except Exception:
print(f"解析失败: {server}")
continue # 跳过解析失败节点
else:
country = get_country(ip, server, original_name)
else:
country = get_country(ip, server, original_name)
if country in EXCLUDED_COUNTRIES:
print(f"排除节点: {original_name} ({country})")
continue
country_groups[country].append(proxy)
# 重命名
sorted_countries = sorted(country_groups)
new_proxies = []
name_mapping = {}
counters = {c: 1 for c in sorted_countries}
for country in sorted_countries:
flag = country_code_to_flag(country)
display_name = country_names.get(country, country)
prefix = f"{flag}{display_name}" if flag else country
for proxy in country_groups[country]:
old = proxy["name"]
new = f"{prefix}-{counters[country]:02d}"
proxy["name"] = new
name_mapping[old] = new
new_proxies.append(proxy)
counters[country] += 1
# 构建输出配置
base_config = fetch_config(CONFIG_URLS[0]) if CONFIG_URLS else {}
base_config["proxies"] = new_proxies
if "proxy-groups" in base_config:
for group in base_config["proxy-groups"]:
if "proxies" in group and isinstance(group["proxies"], list):
group["proxies"] = [name_mapping.get(n, n) for n in group["proxies"] if n in name_mapping]
with open("merged-clash.yaml", "w", encoding="utf-8") as f:
yaml.safe_dump(base_config, f, allow_unicode=True, sort_keys=False)
print("完成 → merged-clash.yaml")