-
Notifications
You must be signed in to change notification settings - Fork 120
/
Copy pathgenerate.py
330 lines (291 loc) · 12.3 KB
/
generate.py
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
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
#!/usr/bin/env python3
# 针对部分网站显示IP归属地的分流规则
# anti-ip-attribution generate.py
# https://github.com/SunsetMkt/anti-ip-attribution
# 从rules.yaml生成配置文件,由Actions调用。
# 读取文件:
# rules.yaml 配置文件
# parser-header.yaml parser.yaml的生成模板
# 输出文件:
# parser.yaml 适用于Clash for Windows的配置文件预处理功能,详见https://docs.cfw.lbyczf.com/contents/parser.html
# rule-provider.yaml 适用于Clash的Rule Provider功能,详见https://lancellc.gitbook.io/clash/clash-config-file/rule-provider
# rule-provider-direct.yaml rule-provider-proxy.yaml rule-provider-reject.yaml
# surge.list Surge分流规则
# quantumultx.list QuantumultX分流规则
# quantumultx-domesticsocial.list QuantumultX分流规则,策略组名称为DomesticSocial
import copy
import os
import sys
import git
import yaml
def read_yaml(file):
'''读取YAML文件'''
with open(file, 'r', encoding='utf-8') as f:
return yaml.safe_load(f)
def save_yaml(config, filename):
'''保存YAML文件'''
with open(filename, 'w', encoding='utf-8') as f:
yaml.dump(config, f, default_flow_style=False)
def get_yaml_string(config):
'''获取YAML字符串'''
return yaml.dump(config, default_flow_style=False)
def save_string(string, filename):
'''保存字符串'''
with open(filename, 'w', encoding='utf-8') as f:
f.write(string)
def get_git_hash():
'''获取Git提交'''
repo = git.Repo(search_parent_directories=True)
sha = repo.head.object.hexsha
return str(sha)
def get_head_comment(config, filename='', description=''):
'''获取头部注释'''
comment = ''
comment += '# ' + config['config']['description'] + '\n'
comment += '# ' + config['config']['name'] + ' ' + \
filename + ' ' + config['config']['version'] + \
' ' + get_git_hash() + '\n'
comment += '# ' + config['config']['url'] + '\n'
comment += '# ' + description + '\n'
return comment
def seprate_comma(string):
'''分割字符串'''
return string.split(',')
def yaml_list(rules):
'''yaml写法转list写法'''
ret=[]
for rule in rules:
seprated = seprate_comma(rule)
# if rule.startswith('IP-CIDR'):
# # 'IP-CIDR,203.107.1.0/24,REJECT,no-resolve'
# del seprated[2]
# else:
# # 'DOMAIN-SUFFIX,zijieapi.com,DIRECT'
# seprated.pop()
del seprated[2]
ret.append(','.join(seprated))
return ret
def get_list_string(list):
return '\n'.join(list)
def check_rules(config):
'''检查规则是否有误'''
rules = config['config']['rules']
# 预期中可用的规则方法
expected_methods = ['DOMAIN', 'DOMAIN-SUFFIX',
'DOMAIN-KEYWORD', 'IP-CIDR', 'GEOIP', 'DST-PORT', 'IP-CIDR6']
# 容易出错的规则方法
questioned_methods = ['SRC-IP-CIDR',
'SRC-PORT', 'RULE-SET', 'MATCH', 'IP6-CIDR']
# 预期中可用的规则部分
expected_rules = ['DIRECT', 'REJECT', 'no-resolve']
# rules中不需要出现IP类关键词,生成脚本会按需自动添加
should_not_appear_rules = ['IP', 'IP归属地']
for rule in rules:
rule = rule.strip()
print(rule)
# 检查空行
if rule == '':
print('规则空行')
return False
else:
print('非空行')
# 检查逗号分隔
length = len(seprate_comma(rule))
print('length: ' + str(length))
if length == 1:
print('规则不完整?没有逗号分隔?:' + rule)
return False
elif length == 2:
pass
elif length == 3:
pass
else:
print('规则有误?逗号分隔超过三个部分:' + rule)
return False
# 检查规则方法
method = seprate_comma(rule)[0]
print('method: ' + method)
if method not in expected_methods and method not in questioned_methods:
print('规则方法有误?:' + rule)
return False
if method.upper() in questioned_methods:
print('规则方法有误?' + method + '可能是常见错误:' + rule)
return False
if method not in expected_methods and method.upper() in expected_methods:
print('规则方法未大写?:' + rule)
return False
# 检查规则部分
if length == 3:
print('length == 3')
this_rule = seprate_comma(rule)[2]
print('rule: ' + this_rule)
if this_rule not in expected_rules:
print('规则有误?' + this_rule + '是预期之外的规则:' + rule)
return False
for should_not_appear_rule in should_not_appear_rules:
if should_not_appear_rule in this_rule.upper():
print('规则有误?rules中不需要出现“IP归属地”类规则描述:' + rule)
return False
return True
def generate_parser(config):
'''生成parser.yaml'''
head = read_yaml('parser-header.yaml')
print(head)
comment = get_head_comment(
config, 'parser.yaml', '适用于Clash for Windows的配置文件预处理功能,详见https://docs.cfw.lbyczf.com/contents/parser.html')
rules = []
for rule in config['config']['rules']:
rule = rule.strip()
if len(seprate_comma(rule)) == 2:
rules.append(rule+',IP归属地')
else:
rules.append(rule)
head['parsers'][0]['yaml']['prepend-rules'] = rules
output = get_yaml_string(head)
output = comment + output
save_string(output, os.path.join('generated', 'parser.yaml'))
def generate_rule_provider(config):
'''生成rule-provider.yaml'''
comment = get_head_comment(
config, 'rule-provider.yaml', '适用于Clash的Rule Provider功能,详见https://lancellc.gitbook.io/clash/clash-config-file/rule-provider')
# 为了不影响其他软件规则的生成,使用深拷贝
rules = copy.deepcopy(config['config']['rules'])
# 检查 IP rules 有无 `no-resolve` 字段,没有的话加上, 防止 DNS 泄漏
for i in range(len(rules)):
if rules[i].startswith('IP-CIDR') and 'no-resolve' not in rules[i]:
rules[i] = rules[i] + ',no-resolve'
# https://github.com/SunsetMkt/anti-ip-attribution/issues/23#issuecomment-1223931835
direct = []
proxy = []
reject = []
for rule in rules:
if 'REJECT' in rule:
reject.append(rule)
elif 'DIRECT' in rule:
direct.append(rule)
else:
proxy.append(rule)
# Summary of rules
print('生成rule-provider.yaml')
output = {}
output['payload'] = rules
output = comment + get_yaml_string(output)
save_string(output, os.path.join('generated', 'rule-provider.yaml'))
# Direct rules list
print('生成rule-set-direct.list')
comment = get_head_comment(config, 'rule-set-direct.list',
'适用于Clash RULE-SET')
output = yaml_list(direct)
output = comment + get_list_string(output)
save_string(output, os.path.join('generated', 'rule-set-direct.list'))
# Proxy rules list
print('生成rule-set-proxy.list')
comment = get_head_comment(config, 'rule-set-proxy.list',
'适用于Clash RULE-SET')
output = proxy
output = comment + get_list_string(output)
save_string(output, os.path.join('generated', 'rule-set-proxy.list'))
# Reject rules list
print('生成rule-set-reject.list')
comment = get_head_comment(config, 'rule-set-reject.list',
'适用于Clash RULE-SET')
output = yaml_list(reject)
output = comment + get_list_string(output)
save_string(output, os.path.join('generated', 'rule-set-reject.list'))
# Direct rules
print('生成rule-provider-direct.yaml')
comment = get_head_comment(config, 'rule-provider-direct.yaml',
'适用于Clash的Rule Provider功能,详见https://lancellc.gitbook.io/clash/clash-config-file/rule-provider')
output = {}
output['payload'] = direct
output = comment + get_yaml_string(output)
save_string(output, os.path.join('generated', 'rule-provider-direct.yaml'))
# Proxy rules
print('生成rule-provider-proxy.yaml')
comment = get_head_comment(config, 'rule-provider-proxy.yaml',
'适用于Clash的Rule Provider功能,详见https://lancellc.gitbook.io/clash/clash-config-file/rule-provider')
output = {}
output['payload'] = proxy
output = comment + get_yaml_string(output)
save_string(output, os.path.join('generated', 'rule-provider-proxy.yaml'))
# Reject rules
print('生成rule-provider-reject.yaml')
comment = get_head_comment(config, 'rule-provider-reject.yaml',
'适用于Clash的Rule Provider功能,详见https://lancellc.gitbook.io/clash/clash-config-file/rule-provider')
output = {}
output['payload'] = reject
output = comment + get_yaml_string(output)
save_string(output, os.path.join('generated', 'rule-provider-reject.yaml'))
def generate_surge(config):
'''生成surge.list'''
comment = get_head_comment(
config, 'surge.list', 'Surge分流规则')
# 为了不影响其他软件规则的生成,使用深拷贝
rules = copy.deepcopy(config["config"]["rules"])
# 检查 IP rules 有无 `no-resolve` 字段,没有的话加上, 防止 DNS 泄漏
for i in range(len(rules)):
if rules[i].startswith("IP-CIDR") and "no-resolve" not in rules[i]:
rules[i] = rules[i] + ",no-resolve"
gen_rules = ''
for rule in rules:
gen_rules += rule + '\n'
output = comment + gen_rules
save_string(output, os.path.join('generated', 'surge.list'))
def generate_quantumultx(config):
'''生成quantumultx.list'''
comment = get_head_comment(
config, 'quantumultx.list', 'QuantumultX分流规则')
commentDomesticSocial = get_head_comment(
config, 'quantumultx-domesticsocial.list', 'QuantumultX分流规则,策略组名称为DomesticSocial')
rules = ''
rulesDomesticSocial = ''
for rule in config['config']['rules']:
rule = rule.strip()
# Quantumult X 中, IP-CIDR6 的相关规则名称为 IP6-CIDR, 无法识别前缀为 IP-CIDR6 的规则.
# https://github.com/KooriMoe
if "IP-CIDR6" in seprate_comma(rule)[0]: # 仅处理出现在第一个位置的规则
rule = rule.replace('IP-CIDR6', 'IP6-CIDR',
1) # 替换第一个 IP-CIDR6 字符串
print('针对Quantumult X替换IP-CIDR6为IP6-CIDR:' + rule)
if len(seprate_comma(rule)) == 2:
rules += rule + ',IP\n'
rulesDomesticSocial += rule + ',DomesticSocial\n'
else:
rules += rule + '\n'
rulesDomesticSocial += rule + '\n'
output = comment + rules
outputDomesticSocial = commentDomesticSocial + rulesDomesticSocial
save_string(output, os.path.join('generated', 'quantumultx.list'))
save_string(outputDomesticSocial, os.path.join(
'generated', 'quantumultx-domesticsocial.list'))
if __name__ == '__main__':
config = read_yaml('rules.yaml')
print(get_head_comment(config, 'generate.py', '配置文件生成脚本'))
print('=====================')
print('开始生成配置文件...')
print('=====================')
print(config)
print('=====================')
print('检查规则是否有误...')
if check_rules(config):
print('规则检查通过!')
print('=====================')
print('生成parser.yaml...')
generate_parser(config)
print('=====================')
print('生成rule-provider.yaml...')
generate_rule_provider(config)
print('=====================')
print('生成surge.list...')
generate_surge(config)
print('=====================')
print('生成quantumultx.list...')
generate_quantumultx(config)
print('=====================')
print('生成配置文件完成!')
else:
print('规则检查未通过!')
print('=====================')
print('请检查规则是否有误!')
print('生成配置文件失败!')
sys.exit(1)