mirror of https://github.com/jumpserver/jumpserver
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
286 lines
9.4 KiB
286 lines
9.4 KiB
import os
|
|
import tempfile
|
|
import time
|
|
from enum import Enum
|
|
from subprocess import CREATE_NO_WINDOW
|
|
|
|
from selenium import webdriver
|
|
from selenium.webdriver.chrome.service import Service
|
|
from selenium.webdriver.common.by import By
|
|
from selenium.webdriver.remote.webelement import WebElement
|
|
|
|
from code_dialog import CodeDialog
|
|
from common import (Asset, User, Account, Platform, Step)
|
|
from common import (BaseApplication)
|
|
from common import (notify_err_message, block_input, unblock_input)
|
|
|
|
|
|
class Command(Enum):
|
|
TYPE = 'type'
|
|
CLICK = 'click'
|
|
OPEN = 'open'
|
|
CODE = 'code'
|
|
SELECT_FRAME = 'select_frame'
|
|
SLEEP = 'sleep'
|
|
|
|
|
|
def _execute_type(ele: WebElement, value: str):
|
|
ele.send_keys(value)
|
|
|
|
|
|
def _execute_click(ele: WebElement, value: str):
|
|
ele.click()
|
|
|
|
|
|
commands_func_maps = {
|
|
Command.CLICK: _execute_click,
|
|
Command.TYPE: _execute_type,
|
|
Command.OPEN: _execute_type,
|
|
}
|
|
|
|
|
|
class StepAction:
|
|
methods_map = {
|
|
"NAME": By.NAME,
|
|
"ID": By.ID,
|
|
"CLASS_NAME": By.CLASS_NAME,
|
|
"CSS_SELECTOR": By.CSS_SELECTOR,
|
|
"CSS": By.CSS_SELECTOR,
|
|
"XPATH": By.XPATH
|
|
}
|
|
|
|
def __init__(self, target='', value='', command=Command.TYPE, **kwargs):
|
|
self.target = target
|
|
self.value = value
|
|
self.command = command
|
|
|
|
def execute(self, driver: webdriver.Chrome) -> bool:
|
|
if not self.target:
|
|
return True
|
|
if self.command == 'select_frame':
|
|
self._switch_iframe(driver, self.target)
|
|
return True
|
|
elif self.command == 'sleep':
|
|
self._sleep(driver, self.target)
|
|
return True
|
|
target_name, target_value = self.target.split("=", 1)
|
|
by_name = self.methods_map.get(target_name.upper(), By.NAME)
|
|
ele = driver.find_element(by=by_name, value=target_value)
|
|
if not ele:
|
|
return False
|
|
if self.command == 'type':
|
|
ele.send_keys(self.value)
|
|
elif self.command in ['click', 'button']:
|
|
ele.click()
|
|
elif self.command in ['open']:
|
|
driver.get(self.value)
|
|
elif self.command == 'code':
|
|
unblock_input()
|
|
code_string = CodeDialog(title="Code Dialog", label="Code").wait_string()
|
|
block_input()
|
|
ele.send_keys(code_string)
|
|
return True
|
|
|
|
def _execute_command_type(self, ele, value):
|
|
ele.send_keys(value)
|
|
|
|
def _switch_iframe(self, driver: webdriver.Chrome, target: str):
|
|
"""
|
|
driver: webdriver.Chrome
|
|
target: str
|
|
target support three format str below:
|
|
index=1: switch to frame by index, if index < 0, switch to default frame
|
|
id=xxx: switch to frame by id
|
|
name=xxx: switch to frame by name
|
|
"""
|
|
target_name, target_value = target.split("=", 1)
|
|
if target_name == 'id':
|
|
driver.switch_to.frame(target_value)
|
|
elif target_name == 'index':
|
|
index = int(target_value)
|
|
if index < 0:
|
|
driver.switch_to.default_content()
|
|
else:
|
|
driver.switch_to.frame(index)
|
|
elif target_name == 'name':
|
|
driver.switch_to.frame(target_value)
|
|
else:
|
|
driver.switch_to.frame(target)
|
|
|
|
def _sleep(self, driver: webdriver.Chrome, target: str):
|
|
try:
|
|
sleep_time = int(target)
|
|
except Exception as e:
|
|
# at least sleep 1 second
|
|
sleep_time = 1
|
|
time.sleep(sleep_time)
|
|
|
|
|
|
def execute_action(driver: webdriver.Chrome, step: StepAction) -> bool:
|
|
try:
|
|
return step.execute(driver)
|
|
except Exception as e:
|
|
print(e)
|
|
notify_err_message(str(e))
|
|
return False
|
|
|
|
|
|
class WebAPP(object):
|
|
|
|
def __init__(self, app_name: str = '', user: User = None, asset: Asset = None,
|
|
account: Account = None, platform: Platform = None, **kwargs):
|
|
self.app_name = app_name
|
|
self.user = user
|
|
self.asset = asset
|
|
self.account = account
|
|
self.platform = platform
|
|
self._steps = list()
|
|
|
|
extra_data = self.asset.spec_info
|
|
autofill_type = extra_data.autofill
|
|
if not autofill_type:
|
|
protocol_setting = self.platform.get_protocol_setting("http")
|
|
if not protocol_setting:
|
|
print("No protocol setting found")
|
|
return
|
|
extra_data = protocol_setting
|
|
autofill_type = extra_data.autofill
|
|
if autofill_type == "basic":
|
|
self._steps = self._default_custom_steps(extra_data)
|
|
elif autofill_type == "script":
|
|
script_list = extra_data.script
|
|
steps = sorted(script_list, key=lambda step_item: step_item.step)
|
|
for item in steps:
|
|
val = item.value
|
|
if val:
|
|
val = val.replace("{USERNAME}", self.account.username)
|
|
val = val.replace("{SECRET}", self.account.secret)
|
|
item.value = val
|
|
self._steps.append(item)
|
|
|
|
def _default_custom_steps(self, spec_info) -> list:
|
|
account = self.account
|
|
default_steps = [
|
|
Step({
|
|
"step": 1,
|
|
"value": account.username,
|
|
"target": spec_info.username_selector,
|
|
"command": "type"
|
|
}),
|
|
Step({
|
|
"step": 2,
|
|
"value": account.secret,
|
|
"target": spec_info.password_selector,
|
|
"command": "type"
|
|
}),
|
|
Step({
|
|
"step": 3,
|
|
"value": "",
|
|
"target": spec_info.submit_selector,
|
|
"command": "click"
|
|
})
|
|
]
|
|
return default_steps
|
|
|
|
def execute(self, driver: webdriver.Chrome) -> bool:
|
|
if not self.asset.address:
|
|
return True
|
|
|
|
for step in self._steps:
|
|
action = StepAction(target=step.target, value=step.value,
|
|
command=step.command)
|
|
ret = execute_action(driver, action)
|
|
if not ret:
|
|
unblock_input()
|
|
notify_err_message(f"执行失败: target: {action.target} command: {action.command}")
|
|
block_input()
|
|
return False
|
|
return True
|
|
|
|
|
|
def load_extensions():
|
|
extensions_root = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'extensions')
|
|
extension_names = os.listdir(extensions_root)
|
|
extension_paths = [os.path.join(extensions_root, name) for name in extension_names]
|
|
return extension_paths
|
|
|
|
|
|
def default_chrome_driver_options():
|
|
options = webdriver.ChromeOptions()
|
|
options.add_argument("--start-maximized")
|
|
|
|
# 忽略证书错误相关
|
|
options.add_argument('--ignore-ssl-errors')
|
|
options.add_argument('--ignore-certificate-errors')
|
|
options.add_argument('--ignore-certificate-errors-spki-list')
|
|
options.add_argument('--allow-running-insecure-content')
|
|
|
|
# 加载 extensions
|
|
extension_paths = load_extensions()
|
|
for extension_path in extension_paths:
|
|
options.add_argument('--load-extension={}'.format(extension_path))
|
|
# 禁用开发者工具
|
|
options.add_argument("--disable-dev-tools")
|
|
# 禁用 密码管理器弹窗
|
|
prefs = {
|
|
"credentials_enable_service": False,
|
|
"profile.password_manager_enabled": False
|
|
}
|
|
options.add_experimental_option("prefs", prefs)
|
|
options.add_experimental_option("excludeSwitches", ['enable-automation'])
|
|
return options
|
|
|
|
|
|
class AppletApplication(BaseApplication):
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
self.driver = None
|
|
self.app = WebAPP(app_name=self.app_name, user=self.user,
|
|
account=self.account, asset=self.asset, platform=self.platform)
|
|
self._tmp_user_dir = tempfile.TemporaryDirectory()
|
|
self._chrome_options = default_chrome_driver_options()
|
|
self._chrome_options.add_argument("--app={}".format(self.asset.address))
|
|
self._chrome_options.add_argument("--user-data-dir={}".format(self._tmp_user_dir.name))
|
|
|
|
def run(self):
|
|
service = Service()
|
|
# driver 的 console 终端框不显示
|
|
service.creationflags = CREATE_NO_WINDOW
|
|
self.driver = webdriver.Chrome(options=self._chrome_options, service=service)
|
|
self.driver.implicitly_wait(10)
|
|
if self.app.asset.address != "":
|
|
ok = self.app.execute(self.driver)
|
|
if not ok:
|
|
print("执行失败")
|
|
self.driver.maximize_window()
|
|
|
|
def wait(self):
|
|
disconnected_msg = "Unable to evaluate script: disconnected: not connected to DevTools\n"
|
|
closed_msg = "Unable to evaluate script: no such window: target window already closed"
|
|
|
|
while True:
|
|
time.sleep(5)
|
|
logs = self.driver.get_log('driver')
|
|
if len(logs) == 0:
|
|
continue
|
|
ret = logs[-1]
|
|
if isinstance(ret, dict):
|
|
message = ret.get('message', '')
|
|
if disconnected_msg in message or closed_msg in message:
|
|
break
|
|
print("ret: ", ret)
|
|
self.close()
|
|
|
|
def close(self):
|
|
if self.driver:
|
|
try:
|
|
# quit 退出全部打开的窗口
|
|
self.driver.quit()
|
|
except Exception as e:
|
|
print(e)
|
|
try:
|
|
self._tmp_user_dir.cleanup()
|
|
except Exception as e:
|
|
print(e)
|