Jarvis-Brain 0.1.7.1__py3-none-any.whl → 0.1.8.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.
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: Jarvis_Brain
3
- Version: 0.1.7.1
3
+ Version: 0.1.8.3
4
4
  Summary: Jarvis brain mcp
5
5
  Requires-Python: >=3.10
6
6
  Requires-Dist: beautifulsoup4
@@ -0,0 +1,11 @@
1
+ mcp_tools/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
2
+ mcp_tools/dp_tools.py,sha256=B39pf4GedGnO-7eh1b9hc-qoXrIZEQ7AFsiCLuvVheM,13366
3
+ mcp_tools/main.py,sha256=1oM11IvEfwBvJ4bigIQzr2AigTISX-LIJ63oFtYnwFY,1049
4
+ tools/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
5
+ tools/browser_manager.py,sha256=zeYcWuzxoohMdnYUoZbRH7axFC_VtV8MsncfN8y0yw0,2023
6
+ tools/browser_proxy.py,sha256=sybZoxjp_its4fzEGO3E7aeQhmTVv1kIFm-nuOjBHiw,6797
7
+ tools/tools.py,sha256=TaWs-CNXy-py9BFmCnJrQ09ke938xXpImf-N2Qo_Rvc,4708
8
+ jarvis_brain-0.1.8.3.dist-info/METADATA,sha256=vfEWVhj6kYKaqJQ4SyT4JLurD_Pu_Ox_vXbPwBJQEdg,241
9
+ jarvis_brain-0.1.8.3.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
10
+ jarvis_brain-0.1.8.3.dist-info/entry_points.txt,sha256=YFQT4xpkUqt5dM5wlKPQQOqcjMuFrT9iuRAzIpAyH7U,51
11
+ jarvis_brain-0.1.8.3.dist-info/RECORD,,
mcp_tools/dp_tools.py CHANGED
@@ -15,7 +15,7 @@ waf_status_code_dict = {
15
15
  521: "加速乐"
16
16
  }
17
17
  # 一轮最大输入,以免单个html最大长度超过ai最大输入
18
- ont_turn_max_token = 25000
18
+ one_turn_max_token = 20000
19
19
 
20
20
 
21
21
  def register_visit_url(mcp: FastMCP, browser_manager: BrowserManager, client_manager: DPProxyClientManager):
@@ -37,16 +37,47 @@ def register_visit_url(mcp: FastMCP, browser_manager: BrowserManager, client_man
37
37
  )
38
38
 
39
39
 
40
- def register_get_html(mcp: FastMCP, browser_manager, client_manager: DPProxyClientManager):
40
+ def register_get_new_tab(mcp: FastMCP, browser_manager, client_manager: DPProxyClientManager):
41
+ @mcp.tool(name="get_new_tab",
42
+ description="使用Drissionpage创建一个新的tab页,在新的tab页中打开url,并开始监听新的tab页的所有XHR请求")
43
+ async def get_new_tab(browser_port: int, url: str) -> dict[str, Any]:
44
+ _browser = browser_manager.get_browser(browser_port)
45
+ tab = _browser.new_tab()
46
+ client_manager.create_client(tab)
47
+ tab.get(url)
48
+ _browser.activate_tab(tab)
49
+ tab_id = tab.tab_id
50
+ return dp_mcp_message_pack(f"已创建新的tab页,并打开链接:{url}", tab_id=tab_id)
51
+
52
+
53
+ def register_pop_first_packet(mcp: FastMCP, browser_manager, client_manager: DPProxyClientManager):
54
+ @mcp.tool(name="pop_first_packet",
55
+ description="每调用一次就会弹出传入的tab页所监听到的数据包中的第一个packet_message,当一个packet_message的response body过长时会被切分成多个包,具体一个请求是否还有下一个包,可以参考body_completed字段")
56
+ async def pop_first_packet(browser_port: int, tab_id: str) -> dict[str, Any]:
57
+ _browser = browser_manager.get_browser(browser_port)
58
+ client = client_manager.get_client(tab_id)
59
+ packet_message = client.pop_first_packet()
60
+ message = f"tab页:【{tab_id}】,暂时没有监听到XHR数据包"
61
+ if packet_message:
62
+ message = f"tab页:【{tab_id}】,监听到XHR数据包",
63
+ return dp_mcp_message_pack(
64
+ message,
65
+ browser_port=browser_port,
66
+ tab_id=tab_id,
67
+ packet_message=packet_message
68
+ )
69
+
70
+
71
+ def register_get_html(mcp: FastMCP, browser_manager):
41
72
  @mcp.tool(name="get_html", description="使用Drissionpage获取某一个tab页的html")
42
73
  async def get_html(browser_port: int, tab_id: str) -> dict[str, Any]:
43
74
  _browser = browser_manager.get_browser(browser_port)
44
75
  tab = _browser.get_tab(tab_id)
45
- file_name_prefix = str(tab.title).replace("/", "_").replace(":", "_")
76
+ file_name_prefix = hashlib.md5(str(tab.title).encode('utf-8')).hexdigest()
46
77
  if not os.path.exists(html_source_code_local_save_path):
47
78
  os.makedirs(html_source_code_local_save_path)
48
79
  min_html, compress_rate = compress_html(tab.html)
49
- html_str_list = [min_html[i:i + ont_turn_max_token] for i in range(0, len(min_html), ont_turn_max_token)]
80
+ html_str_list = [min_html[i:i + one_turn_max_token] for i in range(0, len(min_html), one_turn_max_token)]
50
81
  html_file_list = []
51
82
  for index, html_str in enumerate(html_str_list):
52
83
  file_name = file_name_prefix + f"_{tab_id}_segment{index}.html"
@@ -58,20 +89,7 @@ def register_get_html(mcp: FastMCP, browser_manager, client_manager: DPProxyClie
58
89
  return dp_mcp_message_pack(message, tab_id=tab_id, htmls_local_path=html_file_list)
59
90
 
60
91
 
61
- def register_get_new_tab(mcp: FastMCP, browser_manager, client_manager: DPProxyClientManager):
62
- @mcp.tool(name="get_new_tab",
63
- description="使用Drissionpage创建一个新的tab页,在新的tab页中打开url,并开始监听新的tab页的所有XHR请求")
64
- async def get_new_tab(browser_port: int, url: str) -> dict[str, Any]:
65
- _browser = browser_manager.get_browser(browser_port)
66
- tab = _browser.new_tab()
67
- client_manager.create_client(tab)
68
- tab.get(url)
69
- _browser.activate_tab(tab)
70
- tab_id = tab.tab_id
71
- return dp_mcp_message_pack(f"已创建新的tab页,并打开链接:{url}", tab_id=tab_id)
72
-
73
-
74
- def register_switch_tab(mcp: FastMCP, browser_manager, client_manager: DPProxyClientManager):
92
+ def register_switch_tab(mcp: FastMCP, browser_manager):
75
93
  @mcp.tool(name="switch_tab", description="根据传入的tab_id切换到对应的tab页", )
76
94
  async def switch_tab(browser_port: int, tab_id: str) -> dict[str, Any]:
77
95
  _browser = browser_manager.get_browser(browser_port)
@@ -79,7 +97,7 @@ def register_switch_tab(mcp: FastMCP, browser_manager, client_manager: DPProxyCl
79
97
  return dp_mcp_message_pack(f"已将tab页:【{tab_id}】切换至最前端")
80
98
 
81
99
 
82
- def register_close_tab(mcp: FastMCP, browser_manager, client_manager: DPProxyClientManager):
100
+ def register_close_tab(mcp: FastMCP, browser_manager):
83
101
  @mcp.tool(name="close_tab", description="根据传入的tab_id关闭tab页", )
84
102
  async def close_tab(browser_port, tab_id) -> dict[str, Any]:
85
103
  _browser = browser_manager.get_browser(browser_port)
@@ -87,9 +105,13 @@ def register_close_tab(mcp: FastMCP, browser_manager, client_manager: DPProxyCli
87
105
  return dp_mcp_message_pack(f"已将tab页:【{tab_id}】关闭")
88
106
 
89
107
 
90
- def register_check_selector(mcp: FastMCP, browser_manager, client_manager: DPProxyClientManager):
91
- @mcp.tool(name="check_selector", description="查找tab页中是否包含元素,并返回元素attr_name所对应的值")
92
- async def check_selector(browser_port: int, tab_id: str, css_selector: str, attr_name: str = "text") -> dict[
108
+ def register_check_selector(mcp: FastMCP, browser_manager):
109
+ @mcp.tool(name="check_selector",
110
+ description="查找tab页中是否包含元素,并返回元素attr_name所对应的值。"
111
+ "当要选择的元素包含过多元素时,需要传入offset和page_size来分批查看元素,一般不建议调整page_size,更推荐你调整offset"
112
+ "同时如果单个元素属性值太长,函数会进行截断。一般的单个元素的属性值超过300个字符的就会触发截断,截断后会在最后拼接'...'")
113
+ async def check_selector(browser_port: int, tab_id: str, css_selector: str, attr_name: str = "text",
114
+ offset: int = 0, page_size: int = 10) -> dict[
93
115
  str, Any]:
94
116
  _browser = browser_manager.get_browser(browser_port)
95
117
  target_tab = _browser.get_tab(tab_id)
@@ -100,21 +122,33 @@ def register_check_selector(mcp: FastMCP, browser_manager, client_manager: DPPro
100
122
  exist_flag = False
101
123
  if len(target_eles) != 0:
102
124
  exist_flag = True
125
+ if len(target_eles) > page_size:
126
+ target_eles = target_eles[offset:offset + page_size]
127
+ slice_seg = max(300, one_turn_max_token // (page_size + 6))
103
128
  if attr_name == "text":
104
- ele_text_list = [i.text.replace("\n", "") for i in target_eles]
105
- attr_output = "\n".join(ele_text_list)
129
+ ele_attr_list = [i.text.replace("\n", "") for i in target_eles]
130
+ ele_attr_list = [attr_str[:slice_seg] for attr_str in ele_attr_list]
131
+ # 如果经过截断遍历后的字符串长度与截断长度相等,则默认截断了
132
+ ele_attr_list = [attr_str + "..." if len(attr_str) == slice_seg else attr_str for attr_str in ele_attr_list]
133
+ attr_output = "\n".join(ele_attr_list)
106
134
  else:
107
- attr_output = json.dumps([i.attr(attr_name) for i in target_eles])
135
+ ele_attr_list = [i.attr(attr_name) for i in target_eles]
136
+ ele_attr_list = [attr_str[:slice_seg] for attr_str in ele_attr_list]
137
+ ele_attr_list = [attr_str + "..." if len(attr_str) == slice_seg else attr_str for attr_str in ele_attr_list]
138
+ attr_output = json.dumps(ele_attr_list)
139
+ # 对attr_output逐个截断,截断的长度为:一轮最大token除以元素个数+3个点+两个引号和逗号
108
140
  return dp_mcp_message_pack(
109
141
  f"已完成tab页:【{tab_id}】对:【{css_selector}】的检查",
110
142
  tab_id=tab_id,
111
143
  selector=css_selector,
112
144
  selector_ele_exist=exist_flag,
145
+ page_size=page_size,
146
+ offset=offset,
113
147
  attr_output=attr_output
114
148
  )
115
149
 
116
150
 
117
- def register_quit_browser(mcp: FastMCP, browser_manager, client_manager: DPProxyClientManager):
151
+ def register_quit_browser(mcp: FastMCP, browser_manager):
118
152
  @mcp.tool(name="quit_browser", description="退出浏览器会话,关闭浏览器")
119
153
  async def quit_browser(browser_port: int) -> dict[str, Any]:
120
154
  flag, _browser = browser_manager.remove_page(browser_port)
@@ -127,25 +161,7 @@ def register_quit_browser(mcp: FastMCP, browser_manager, client_manager: DPProxy
127
161
  )
128
162
 
129
163
 
130
- def register_pop_first_packet(mcp: FastMCP, browser_manager, client_manager: DPProxyClientManager):
131
- @mcp.tool(name="pop_first_packet",
132
- description="每调用一次就会弹出传入的tab页所监听到的数据包中的第一个packet_message")
133
- async def pop_first_packet(browser_port: int, tab_id: str) -> dict[str, Any]:
134
- _browser = browser_manager.get_browser(browser_port)
135
- client = client_manager.get_client(tab_id)
136
- packet_message = client.pop_first_packet()
137
- message = f"tab页:【{tab_id}】,暂时没有监听到XHR数据包"
138
- if packet_message:
139
- message = f"tab页:【{tab_id}】,监听到XHR数据包",
140
- return dp_mcp_message_pack(
141
- message,
142
- browser_port=browser_port,
143
- tab_id=tab_id,
144
- packet_message=packet_message
145
- )
146
-
147
-
148
- def register_assert_waf(mcp: FastMCP, browser_manager, client_manager: DPProxyClientManager):
164
+ def register_assert_waf(mcp: FastMCP, browser_manager):
149
165
  @mcp.tool(name="assert_waf",
150
166
  description="通过对比requests、有头浏览器、无头浏览器获取到的html,判断网页是否使用了waf以及是否为动态渲染的网页")
151
167
  async def assert_waf(browser_port: int, tab_id: str) -> dict[str, Any]:
@@ -193,3 +209,36 @@ def register_assert_waf(mcp: FastMCP, browser_manager, client_manager: DPProxyCl
193
209
  raw_headless_rate_difference=h_hless_rate_diff,
194
210
  head_headless_rate_difference=h_hless_rate_diff
195
211
  )
212
+
213
+
214
+ def register_click_action(mcp: FastMCP, browser_manager):
215
+ @mcp.tool(name="click_action", description="尝试点击tab页中的元素,返回元素是否可以被点击,以及是否点击成功。")
216
+ async def click_action(browser_port: int, tab_id: str, css_selector: str) -> dict[str, Any]:
217
+ _browser = browser_manager.get_browser(browser_port)
218
+ target_tab = _browser.get_tab(tab_id)
219
+ css_selector = css_selector
220
+ if "css:" not in css_selector:
221
+ css_selector = "css:" + css_selector
222
+ target_eles = target_tab.eles(css_selector)
223
+ click_success = False
224
+ element_clickable = False
225
+ if len(target_eles) == 1:
226
+ target_element = target_eles[0]
227
+ element_clickable = target_element.states.is_clickable
228
+ try:
229
+ target_element.click()
230
+ click_success = True
231
+ except Exception as e:
232
+ click_success = False
233
+ message = f"点击【{css_selector}】 {'成功' if click_success else '失败'}"
234
+ else:
235
+ message = f"传入的css_selector找到了{len(target_eles)}个元素,请确保传入的css_selector可以找到唯一的一个元素"
236
+ return dp_mcp_message_pack(
237
+ message=message,
238
+ browser_port=browser_port,
239
+ tab_id=tab_id,
240
+ css_selector=css_selector,
241
+ element_clickable=element_clickable,
242
+ click_success=click_success,
243
+ extra_message="点击成功,页面可能有更新,请重新获取页面html,并重新分析页面Selector" if click_success else ""
244
+ )
mcp_tools/main.py CHANGED
@@ -12,17 +12,19 @@ base_cwd = os.getenv("BASE_CWD", os.path.expanduser('~'))
12
12
 
13
13
  if "TeamNode-Dp" in enabled_modules:
14
14
  # 页面管理
15
- register_close_tab(mcp, browser_manager, client_manager)
16
- register_switch_tab(mcp, browser_manager, client_manager)
15
+ register_close_tab(mcp, browser_manager)
16
+ register_switch_tab(mcp, browser_manager)
17
17
  register_get_new_tab(mcp, browser_manager, client_manager)
18
- # 功能
18
+ # 基础功能
19
19
  register_visit_url(mcp, browser_manager, client_manager)
20
- register_get_html(mcp, browser_manager, client_manager)
21
- register_check_selector(mcp, browser_manager, client_manager)
20
+ register_get_html(mcp, browser_manager)
21
+ register_check_selector(mcp, browser_manager)
22
22
  register_pop_first_packet(mcp, browser_manager, client_manager)
23
+ # 页面交互
24
+ register_click_action(mcp, browser_manager)
23
25
 
24
26
  if "JarvisNode" in enabled_modules:
25
- register_assert_waf(mcp, browser_manager, client_manager)
27
+ register_assert_waf(mcp, browser_manager)
26
28
 
27
29
 
28
30
  def main():
tools/browser_manager.py CHANGED
@@ -4,6 +4,9 @@ from typing import Optional, Tuple
4
4
  import os
5
5
  from DrissionPage import ChromiumPage, ChromiumOptions
6
6
  import platform
7
+ from DrissionPage.common import Settings
8
+
9
+ Settings.set_raise_when_click_failed(True)
7
10
 
8
11
 
9
12
  class BrowserManager:
tools/browser_proxy.py CHANGED
@@ -1,12 +1,14 @@
1
1
  import threading
2
2
  from collections import deque
3
-
3
+ import time
4
4
  from DrissionPage import ChromiumPage, ChromiumOptions
5
5
  from DrissionPage._pages.chromium_tab import ChromiumTab
6
6
  from DrissionPage._units.listener import DataPacket
7
7
  from typing import Tuple, Optional
8
8
  import json
9
9
 
10
+ one_turn_max_token = 20000
11
+
10
12
 
11
13
  class DPProxyClient:
12
14
  def __init__(self, driver: ChromiumTab, self_kill=False):
@@ -58,7 +60,7 @@ class DPProxyClientManager:
58
60
  def create_client(self, tab: ChromiumTab, self_kill=False) -> Tuple[str, DPProxyClient, ChromiumTab]:
59
61
  """创建新的tab页面代理实例"""
60
62
  client = DPProxyClient(tab, self_kill=self_kill)
61
- tab = client.get_driver(True)
63
+ tab = client.get_driver(True, timeout=60 * 10)
62
64
  tab_id = tab.tab_id
63
65
  self.tab_pool[tab_id] = {"client": client, "driver": tab}
64
66
  return tab_id, client, tab
@@ -133,21 +135,27 @@ def check_data_packet(packet: DataPacket, client: DPProxyClient):
133
135
  data = None
134
136
  if packet.request.hasPostData:
135
137
  data = packet.request.postData
136
- temp_dict = {
137
- "url": url,
138
- "method": method,
139
- "request_data": data,
140
- "request_headers": packet.request.headers,
141
- "response_headers": packet.response.headers,
142
- "response_body": packet.response.body,
143
- }
144
- print(temp_dict)
145
- client.packet_queue.append(temp_dict)
138
+ body = packet.response.body
139
+ body_str = json.dumps(body, ensure_ascii=False)
140
+ body_str_list = [body_str[i:i + one_turn_max_token] for i in range(0, len(body_str), one_turn_max_token)]
141
+ body_completed = True
142
+ for index, body_str in enumerate(body_str_list):
143
+ if (index + 1) != len(body_str_list):
144
+ body_completed = False
145
+ temp_dict = {
146
+ "url": url,
147
+ "body_completed": body_completed,
148
+ "method": method,
149
+ "request_data": data,
150
+ "request_headers": dict(packet.request.headers),
151
+ "response_headers": dict(packet.response.headers),
152
+ "response_body_segment": body_str,
153
+ }
154
+ client.packet_queue.append(temp_dict)
146
155
 
147
156
 
148
157
  client_manager = DPProxyClientManager()
149
158
 
150
- #
151
159
  # if __name__ == '__main__':
152
160
  # co = ChromiumOptions().set_user_agent(
153
161
  # "Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/143.0.0.0 Mobile Safari/537.36")
@@ -1,11 +0,0 @@
1
- mcp_tools/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
2
- mcp_tools/dp_tools.py,sha256=ZOmHTIofSeuqiY3YZHNnK72-_MRo22tBAB9NJ4KyEbo,10416
3
- mcp_tools/main.py,sha256=Fdt2N3oKGwvruuno_ywnuWSlm1BexE9ZY669H2LTo9w,1056
4
- tools/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
5
- tools/browser_manager.py,sha256=EXM7n-sDOtdQGpWkVTAZHWhepVU-7PAoUTDNgGF9_fQ,1938
6
- tools/browser_proxy.py,sha256=YbqXT6h0Zz_H1TMGBd8wwPABIGpE0nGLwYzDlA4t674,6320
7
- tools/tools.py,sha256=TaWs-CNXy-py9BFmCnJrQ09ke938xXpImf-N2Qo_Rvc,4708
8
- jarvis_brain-0.1.7.1.dist-info/METADATA,sha256=nLX_3KOUhtCzLP_qScoyY6sdH0CtAWekIlqtfM2RZR0,241
9
- jarvis_brain-0.1.7.1.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
10
- jarvis_brain-0.1.7.1.dist-info/entry_points.txt,sha256=YFQT4xpkUqt5dM5wlKPQQOqcjMuFrT9iuRAzIpAyH7U,51
11
- jarvis_brain-0.1.7.1.dist-info/RECORD,,