nonebot-plugin-l4d2-server 1.0.2__py3-none-any.whl → 1.0.3__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -105,17 +105,27 @@ async def _(
105
105
  async def _(
106
106
  args: Message = CommandArg(),
107
107
  ):
108
+ # 以后有时间补img格式
108
109
  msg: str = args.extract_plain_text().strip()
109
110
  tag_list: List[str] = msg.split(" ", maxsplit=1)
110
- if len(tag_list) < 2:
111
- return await UniMessage.text("格式错误,正确格式:/l4find 组名 玩家名").finish()
112
- group, name = tag_list
113
- out: List[OutServer] = await get_server_detail(group, is_img=False) # type: ignore
114
- out_msg = "未找到玩家"
115
- for one in out:
116
- for player in one["player"]:
117
- if name in player.name:
118
- out_msg = await get_ip_server(f"{one['host']}:{one['port']}")
111
+ if len(tag_list) == 1:
112
+ await UniMessage.text("未设置组,正在全服查找,时间较长").send()
113
+ name = tag_list[0]
114
+ out: List[OutServer] = await get_server_detail(is_img=False) # type: ignore
115
+ out_msg = "未找到玩家"
116
+ for one in out:
117
+ for player in one["player"]:
118
+ if name in player.name:
119
+ out_msg = await get_ip_server(f"{one['host']}:{one['port']}")
120
+ if len(tag_list) == 2:
121
+ group, name = tag_list
122
+ await UniMessage.text(f"正在查询{group}组").send()
123
+ out: List[OutServer] = await get_server_detail(group, is_img=False) # type: ignore
124
+ out_msg = "未找到玩家"
125
+ for one in out:
126
+ for player in one["player"]:
127
+ if name in player.name:
128
+ out_msg = await get_ip_server(f"{one['host']}:{one['port']}")
119
129
 
120
130
  return await UniMessage.text(out_msg).finish()
121
131
 
@@ -1,10 +1,14 @@
1
1
  from pathlib import Path
2
2
 
3
3
  from nonebot import get_plugin_config
4
+ from nonebot.log import logger
4
5
  from pydantic import BaseModel
5
6
 
6
7
  DATAPATH = Path(__file__).parent.joinpath("data")
7
8
  DATAOUT = Path("data/L4D2")
9
+ if not Path(DATAOUT / "l4d2.json").exists():
10
+ logger.info("文件 l4d2.json 不存在,已创建并初始化为 {}")
11
+ Path(DATAOUT / "l4d2.json").write_text("{}", encoding="utf-8")
8
12
  print(DATAOUT.absolute())
9
13
  server_all_path = DATAOUT / "l4d2"
10
14
  server_all_path.mkdir(parents=True, exist_ok=True)
@@ -10,7 +10,7 @@ from ..l4_image.convert import core_font
10
10
  from ..l4_image.model import PluginHelp
11
11
  from .draw import get_help
12
12
 
13
- __version__ = "1.0.2"
13
+ __version__ = "1.0.3"
14
14
  TEXT_PATH = Path(__file__).parent / "texture2d"
15
15
  HELP_DATA = Path(__file__).parent / "Help.json"
16
16
 
@@ -23,6 +23,16 @@ COMMAND = set()
23
23
 
24
24
 
25
25
  async def get_all_server_detail():
26
+ """
27
+ 获取所有服务器的详细信息。
28
+
29
+ Args:
30
+
31
+
32
+ Returns:
33
+ str: 包含所有服务器详细信息的字符串。
34
+
35
+ """
26
36
  out_list: List[AllServer] = []
27
37
  for group in ALLHOST:
28
38
  msg_list = await get_group_detail(group)
@@ -58,11 +68,30 @@ async def get_all_server_detail():
58
68
 
59
69
 
60
70
  async def get_server_detail(
61
- command: str,
71
+ command: str = "",
62
72
  _id: Optional[str] = None,
63
73
  is_img: bool = True,
64
74
  ):
65
- server_json = ALLHOST.get(command)
75
+ """
76
+ 异步获取服务器详细信息。
77
+
78
+ Args:
79
+ command (str): 服务器组名。
80
+ _id (Optional[str], optional): 服务器ID。默认为None。
81
+ is_img (bool, optional): 是否返回图片格式的信息。默认为True。
82
+
83
+ Returns:
84
+ Union[bytes, List[OutServer], None]: 返回服务器详细信息。如果为图片格式,返回bytes类型;
85
+ 如果不是图片格式,返回List[OutServer]类型;如果未找到服务器组,返回None。
86
+
87
+ """
88
+ if command:
89
+ server_json = ALLHOST.get(command)
90
+ else:
91
+ server_json = []
92
+ for servers in ALLHOST.values():
93
+ server_json.extend(servers)
94
+
66
95
  logger.info(server_json)
67
96
  if server_json is None:
68
97
  logger.warning("未找到这个组")
@@ -116,98 +145,139 @@ def reload_ip():
116
145
  # print("正在读取json文件")
117
146
  group_ip = []
118
147
  for item in server_all_path.iterdir():
119
- if item.is_file():
120
- if item.name.endswith("json"):
121
- json_data = json.loads(item.read_text(encoding="utf-8"))
122
- group_server = cast(Dict[str, List[NserverOut]], json_data)
123
-
124
- for group, group_ip in group_server.items():
125
- # 处理ip,host,port关系
126
- for one_ip in group_ip:
127
- if one_ip.get("ip"):
128
- if one_ip.get("host") and one_ip.get("port"):
129
- pass
130
- if one_ip.get("host") and not one_ip.get("port"):
131
- one_ip["port"] = 20715
132
- if not one_ip.get("host"):
133
- one_ip["host"], one_ip["port"] = split_maohao(
134
- one_ip.get("ip"),
135
- )
136
- else:
137
- if one_ip.get("host") and one_ip.get("port"):
138
- one_ip["ip"] = f'{one_ip["host"]}:{one_ip["port"]}'
139
- if one_ip.get("host") and not one_ip.get("port"):
140
- one_ip["ip"] = f'{one_ip["host"]}:20715'
141
- else:
142
- logger.warning(f"{one_ip} 没有ip")
148
+ if item.is_file() and item.name.endswith("json"):
149
+ json_data = json.loads(item.read_text(encoding="utf-8"))
150
+ group_server = cast(Dict[str, List[NserverOut]], json_data)
143
151
 
144
- ALLHOST.update({group: group_ip})
145
- COMMAND.add(group)
146
- logger.success(f"成功加载 {item.name.split('.')[0]} {len(group_ip)}个")
152
+ for group, group_ip in group_server.items():
153
+ # 处理ip,host,port关系
154
+ for one_ip in group_ip:
155
+ if one_ip.get("ip"):
156
+ if one_ip.get("host") and one_ip.get("port"):
157
+ pass
158
+ if one_ip.get("host") and not one_ip.get("port"):
159
+ one_ip["port"] = 20715
160
+ if not one_ip.get("host"):
161
+ one_ip["host"], one_ip["port"] = split_maohao(
162
+ one_ip.get("ip"),
163
+ )
164
+ else:
165
+ if one_ip.get("host") and one_ip.get("port"):
166
+ one_ip["ip"] = f'{one_ip["host"]}:{one_ip["port"]}'
167
+ if one_ip.get("host") and not one_ip.get("port"):
168
+ one_ip["ip"] = f'{one_ip["host"]}:20715'
169
+ else:
170
+ logger.warning(f"{one_ip} 没有ip")
147
171
 
148
- if item.name.endswith("txt"):
149
- """to do"""
172
+ ALLHOST.update({group: group_ip})
173
+ COMMAND.add(group)
174
+ logger.success(f"成功加载 {item.name.split('.')[0]} {len(group_ip)}个")
150
175
 
151
176
 
152
177
  async def tj_request(command: str = "云", tj="tj"):
178
+ map_type = "普通药役"
153
179
  server_json = ALLHOST.get(command)
154
180
  logger.info(server_json)
155
181
  if server_json is None:
156
182
  logger.warning("未找到这个组")
157
183
  return None
158
- # 返回单个
159
- logger.info("正在anne电信服务器信息")
184
+
185
+ logger.info("正在获取电信服务器信息")
160
186
  player_msg = ""
161
187
  right_ip = []
162
- for i in server_json:
163
- ser_list = await L4API.a2s_info([(i["host"], i["port"])], is_player=True)
188
+
189
+ async def _filter_servers(servers: list, tj_mode: str) -> list:
190
+ """筛选符合条件的服务器
191
+ Args:
192
+ servers: 服务器列表
193
+ tj_mode: 筛选模式('tj'或'zl')
194
+ Returns:
195
+ 符合条件的服务器列表
196
+ """
197
+ filtered = []
198
+ for i in servers:
199
+ ser_list = await L4API.a2s_info([(i["host"], i["port"])], is_player=True)
200
+ if not ser_list:
201
+ continue
202
+
203
+ srv = ser_list[0][0]
204
+ players = ser_list[0][1]
205
+
206
+ if tj_mode == "tj" and map_type in srv.map_name:
207
+ score = sum(p.score for p in players[:4])
208
+ t = srv.map_name.split("[")[-1].split("特")[0]
209
+ if t.isdigit() and int(t) * 50 < score:
210
+ logger.info(
211
+ f"符合TJ条件的服务器: {i['host']}:{i['port']}, 地图: {srv.map_name}, 分数: {score}",
212
+ )
213
+ filtered.append(i)
214
+ elif tj_mode == "zl" and map_type in srv.map_name and len(players) <= 4:
215
+ logger.info(
216
+ f"符合ZL条件的服务器: {i['host']}:{i['port']}, 地图: {srv.map_name}, 玩家数: {len(players)}",
217
+ )
218
+ filtered.append(i)
219
+ return filtered
220
+
221
+ async def _format_players(player_list: list) -> str:
222
+ """格式化玩家信息
223
+ Args:
224
+ player_list: 玩家对象列表
225
+ Returns:
226
+ 格式化后的玩家信息字符串
227
+ """
228
+ durations = [await convert_duration(p.duration) for p in player_list]
229
+ max_duration_len = max(len(str(d)) for d in durations)
230
+ max_score_len = max(len(str(p.score)) for p in player_list)
231
+ return "\n".join(
232
+ f"[{p.score:>{max_score_len}}] | {durations[i]:^{max_duration_len}} | {p.name[0]}***{p.name[-1]}"
233
+ for i, p in enumerate(player_list)
234
+ )
235
+
236
+ def _build_message(srv_info, players_msg: str, selected_srv: dict) -> str:
237
+ """构建服务器信息消息
238
+ Args:
239
+ srv_info: 服务器信息对象
240
+ players_msg: 格式化后的玩家信息
241
+ selected_srv: 选中的服务器信息
242
+ Returns:
243
+ 完整的消息字符串
244
+ """
245
+ msg = f"""*{srv_info.server_name}*
246
+ 游戏: {srv_info.folder}
247
+ 地图: {srv_info.map_name}
248
+ 人数: {srv_info.player_count}/{srv_info.max_players}"""
249
+ if srv_info.ping is not None:
250
+ msg += f"\nping: {srv_info.ping * 1000:.0f}ms\n{players_msg}"
251
+ if config.l4_show_ip:
252
+ msg += f"\nconnect {selected_srv['host']}:{selected_srv['port']}"
253
+ return msg
254
+
255
+ try:
256
+ right_ip = await _filter_servers(server_json, tj)
257
+
258
+ if not right_ip:
259
+ logger.warning("没有找到符合条件的服务器")
260
+ return "没有符合条件的服务器"
261
+
262
+ logger.info(
263
+ f"符合条件的服务器列表: {[f'{ip['host']}:{ip['port']}' for ip in right_ip]}",
264
+ )
265
+ s = random.choice(right_ip)
266
+ logger.info(f"最终选择的服务器: {s['host']}:{s['port']}")
267
+ ser_list = await L4API.a2s_info([(s["host"], s["port"])], is_player=True)
268
+ if not ser_list:
269
+ return "获取服务器信息失败"
270
+
164
271
  one_server = ser_list[0][0]
165
272
  one_player = ser_list[0][1]
166
273
 
167
- # 判断坐牢条件
168
- if tj == "tj" and "普通药役" in one_server.map_name:
169
- score: int = 0
170
- for index, player in enumerate(one_player, 1):
171
- if index > 4:
172
- break
173
- score += player.score
174
-
175
- t = one_server.map_name.split("[")[-1].split("特")[0]
176
- if int(t) * 50 < score:
177
- right_ip.append(i)
178
- if tj == "zl" and "普通药役" in one_server.map_name and len(one_player) <= 4:
179
- right_ip.append(i)
180
-
181
- if not right_ip:
182
- return "没有符合条件的服务器"
183
-
184
- s = random.choice(right_ip)
185
- ser_list = await L4API.a2s_info([(s["host"], s["port"])], is_player=True)
186
- one_server = ser_list[0][0]
187
- one_player = ser_list[0][1]
188
- if len(one_player):
189
- max_duration_len = max(
190
- [len(str(await convert_duration(i.duration))) for i in one_player],
191
- )
192
- max_score_len = max(len(str(i.score)) for i in one_player)
193
-
194
- for player in one_player:
195
- soc = "[{:>{}}]".format(player.score, max_score_len)
196
- chines_dur = await convert_duration(player.duration)
197
- dur = "{:^{}}".format(chines_dur, max_duration_len)
198
- name = f"{player.name[0]}***{player.name[-1]}"
199
- player_msg += f"{soc} | {dur} | {name} \n"
200
- else:
201
- player_msg = "服务器感觉很安静啊"
202
- msg = f"""*{one_server.server_name}*
203
- 游戏: {one_server.folder}
204
- 地图: {one_server.map_name}
205
- 人数: {one_server.player_count}/{one_server.max_players}"""
206
- if one_server.ping is not None:
207
- msg += f"""
208
- ping: {one_server.ping * 1000:.0f}ms
209
- {player_msg}"""
210
- if config.l4_show_ip:
211
- msg += f"""
212
- connect {s["host"]}:{s["port"]}"""
213
- return msg
274
+ if one_player:
275
+ player_msg = await _format_players(one_player)
276
+ else:
277
+ player_msg = "服务器感觉很安静啊"
278
+
279
+ return _build_message(one_server, player_msg, s)
280
+
281
+ except Exception as e:
282
+ logger.error(f"tj_request error: {e}")
283
+ return "获取服务器信息时出错"
@@ -31,8 +31,8 @@ async def draw_one_ip(host: str, port: int):
31
31
  dur = "{:^{}}".format(chines_dur, max_duration_len)
32
32
  name_leg = len(player.name)
33
33
  if name_leg > 2:
34
- # xing = "*" * (name_leg - 2)
35
- name = f"{player.name[0]}xing{player.name[-1]}"
34
+ xing = "*" * (name_leg - 2)
35
+ name = f"{player.name[0]}{xing}{player.name[-1]}"
36
36
  else:
37
37
  name = player.name
38
38
  player_msg += f"{soc} | {dur} | {name} \n"
@@ -77,6 +77,12 @@ async def get_much_server(server_json: List[NserverOut], command: str):
77
77
 
78
78
 
79
79
  async def convert_duration(duration: float) -> str:
80
+ """Convert duration in seconds to human-readable string format (e.g. '1h 30m 15s')
81
+ Args:
82
+ duration: Duration in seconds
83
+ Returns:
84
+ Formatted time string
85
+ """
80
86
  minutes, seconds = divmod(duration, 60)
81
87
  hours, minutes = divmod(minutes, 60)
82
88
  time_str = ""
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: nonebot-plugin-l4d2-server
3
- Version: 1.0.2
3
+ Version: 1.0.3
4
4
  Summary: L4D2 server related operations plugin for NoneBot2
5
5
  Keywords: steam,game,l4d2,nonebot2,plugin
6
6
  Author-Email: Agnes_Digital <Z735803792@163.com>
@@ -11,10 +11,12 @@ Classifier: Programming Language :: Python :: 3
11
11
  Classifier: Programming Language :: Python :: 3.9
12
12
  Classifier: Programming Language :: Python :: 3.10
13
13
  Classifier: Programming Language :: Python :: 3.11
14
+ Classifier: Programming Language :: Python :: 3.12
14
15
  Classifier: Operating System :: OS Independent
15
16
  Project-URL: homepage, https://github.com/Agnes4m/nonebot_plugin_l4d2_server
16
17
  Requires-Python: <4.0,>=3.9
17
18
  Requires-Dist: nonebot2>=2.0.0
19
+ Requires-Dist: nonebot2[fastapi]>=2.3.3
18
20
  Requires-Dist: nonebot-plugin-htmlrender>=0.3.0
19
21
  Requires-Dist: nonebot-adapter-onebot>=2.4.4
20
22
  Requires-Dist: nonebot-plugin-alconna>=0.50.0
@@ -29,7 +31,7 @@ Requires-Dist: python-a2s>=1.4.1
29
31
  Requires-Dist: ujson>=5.10.0
30
32
  Requires-Dist: lxml>=5.2.2
31
33
  Requires-Dist: rcon>=2.1.0
32
- Requires-Dist: pillow>9
34
+ Requires-Dist: pillow>10
33
35
  Requires-Dist: beautifulsoup4>=4.12.3
34
36
  Requires-Dist: rarfile>=4.2
35
37
  Requires-Dist: pyunpack>=0.3
@@ -1,15 +1,15 @@
1
- nonebot_plugin_l4d2_server-1.0.2.dist-info/METADATA,sha256=XqVzrC4S7KU6SCMgUx_ztxZobhrTefaqwnhG6vFzUmU,6596
2
- nonebot_plugin_l4d2_server-1.0.2.dist-info/WHEEL,sha256=tSfRZzRHthuv7vxpI4aehrdN9scLjk-dCJkPLzkHxGg,90
3
- nonebot_plugin_l4d2_server-1.0.2.dist-info/entry_points.txt,sha256=6OYgBcLyFCUgeqLgnvMyOJxPCWzgy7se4rLPKtNonMs,34
4
- nonebot_plugin_l4d2_server-1.0.2.dist-info/licenses/LICENSE,sha256=OXLcl0T2SZ8Pmy2_dmlvKuetivmyPd5m1q-Gyd-zaYY,35149
1
+ nonebot_plugin_l4d2_server-1.0.3.dist-info/METADATA,sha256=po5LeZo-Nk7oQNuaxa5Ve_3AANnFc4RxG8fnBbcqeK0,6688
2
+ nonebot_plugin_l4d2_server-1.0.3.dist-info/WHEEL,sha256=tSfRZzRHthuv7vxpI4aehrdN9scLjk-dCJkPLzkHxGg,90
3
+ nonebot_plugin_l4d2_server-1.0.3.dist-info/entry_points.txt,sha256=6OYgBcLyFCUgeqLgnvMyOJxPCWzgy7se4rLPKtNonMs,34
4
+ nonebot_plugin_l4d2_server-1.0.3.dist-info/licenses/LICENSE,sha256=OXLcl0T2SZ8Pmy2_dmlvKuetivmyPd5m1q-Gyd-zaYY,35149
5
5
  nonebot_plugin_l4d2_server/__init__.py,sha256=PIufOk1bxOx-nYqdoinoB7BcntIETNFwfzqKnmu-teE,1624
6
- nonebot_plugin_l4d2_server/__main__.py,sha256=E33d-UIdYmBZfT0ASW8WRkqoTXcF_NWRJIcQD_NS3FA,7981
7
- nonebot_plugin_l4d2_server/config.py,sha256=CavZ4EGdLvooVrUeP8zGsbZ0TJNsxGl2WT9rpyJtvhw,1175
6
+ nonebot_plugin_l4d2_server/__main__.py,sha256=AGqPjt0O6dW_E3D93hcqehtsPsvzSRC1qEtx6HqEAGo,8455
7
+ nonebot_plugin_l4d2_server/config.py,sha256=ZLYJr9lVs1Q5en1m4EA0wF7H366EVm8-6AnN8dcyy7c,1394
8
8
  nonebot_plugin_l4d2_server/data/font/loli.ttf,sha256=Yrh-RPoCrn1-NG94DR0x20ASXYUt8g3Ep6BCt3CdOFk,11125812
9
9
  nonebot_plugin_l4d2_server/l4_anne/__init__.py,sha256=zX4kTu4LAsqm7_li0-GcihnbYI65SBPkfUCPTs79U8k,4124
10
10
  nonebot_plugin_l4d2_server/l4_anne/ranne.py,sha256=vtNQJ-74rJiwVI3IYS3ZgFk_LaCviKU1yxrGk3DFvQs,532
11
11
  nonebot_plugin_l4d2_server/l4_help/Help.json,sha256=hz_k-qmGmGJKwCa8wmkHTLzXu6C-G-E_g7xdK-t6fNk,2673
12
- nonebot_plugin_l4d2_server/l4_help/__init__.py,sha256=cabiFBsUVFWs6EUC2XpXhp6PBWRmdBLWEVRNJAvjN6Y,1535
12
+ nonebot_plugin_l4d2_server/l4_help/__init__.py,sha256=YNR3eob0XvRb_-4wIP9ZcSMey0RuFS4RlHMZ217Bn0E,1535
13
13
  nonebot_plugin_l4d2_server/l4_help/draw.py,sha256=y6yDPUnoZFvwly8cf7g9HRpT1JXTxyA9DC1TuvUinTM,6448
14
14
  nonebot_plugin_l4d2_server/l4_help/icon/介绍.png,sha256=3QC6A38QC-7gDBxmtQSQdbsz7hsefU5LL-oZmJ41zTk,3806
15
15
  nonebot_plugin_l4d2_server/l4_help/icon/任务.png,sha256=nyZ4_kM21ZO95nwanCFnUfCGX-PkmJXYQf9OrJVKomY,3782
@@ -88,12 +88,12 @@ nonebot_plugin_l4d2_server/l4_image/model.py,sha256=FGsCvf_BKbRNJUVy6I5BKnArMY-3
88
88
  nonebot_plugin_l4d2_server/l4_image/vtfs.py,sha256=He_7zzEIOip8MXP55TS7aWPbzo6ac0wPf602nN3GWZM,1461
89
89
  nonebot_plugin_l4d2_server/l4_local/__init__.py,sha256=nz9KUxmuvRrryHakQQ4B30oC9jeGWZ3sON1tCKb-lS0,3603
90
90
  nonebot_plugin_l4d2_server/l4_local/file.py,sha256=hew1Y8kV3uSZvUGplmi09EGKC89-sUJWsWV7SCEstI8,3067
91
- nonebot_plugin_l4d2_server/l4_request/__init__.py,sha256=FRYBGRM30VoImQqAu3k90r82qkeo8wPjR93t6WLx7nk,7144
92
- nonebot_plugin_l4d2_server/l4_request/draw_msg.py,sha256=WjV2eEVVxsd4akNKDzVl6aRIegJ5pPbIzqziC0vBRmM,2697
91
+ nonebot_plugin_l4d2_server/l4_request/__init__.py,sha256=ukb90pROfHtLbIRSya5pRAGSpzB9Bb5gPi3kAH2-41Y,9588
92
+ nonebot_plugin_l4d2_server/l4_request/draw_msg.py,sha256=efmlEBqbbOozf50InPNqONGIZ0D2hI4xAGPDROIcHBY,2883
93
93
  nonebot_plugin_l4d2_server/l4_request/utils.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
94
94
  nonebot_plugin_l4d2_server/utils/api/api.py,sha256=auvDicCEKwvnm6EJYeCxCtugFby61K-zAmmaRWWEwtM,296
95
95
  nonebot_plugin_l4d2_server/utils/api/models.py,sha256=EKAAM5RsFK-hV-a80tc4r35ToNRhnCWVuQ-7imwBjwE,2386
96
96
  nonebot_plugin_l4d2_server/utils/api/request.py,sha256=hugBaPBK-gUr5Xemyk7dSHP1JlzYrEZcXPinkNX0AuU,14987
97
97
  nonebot_plugin_l4d2_server/utils/database/models.py,sha256=SLdcgwsn39r_ZkcBoqf4MLX1EfpCOjGBwWcR16u9Bqo,454
98
98
  nonebot_plugin_l4d2_server/utils/utils.py,sha256=C-QzwtUCbq7QXmHplpmo3OTKbRztE1W7V-IywvufDl0,5971
99
- nonebot_plugin_l4d2_server-1.0.2.dist-info/RECORD,,
99
+ nonebot_plugin_l4d2_server-1.0.3.dist-info/RECORD,,