测试web前端、前端SDK、web渗透流程等时,难免会需要调用前端js里面自定义的方法。
那么该如何用python编写脚本去调用js文件里面的函数呢?
这里将以SDK的调用为例,提供一些解决方法,供大家参考!
【方法一】
自主研发:直接用前端开发语言来研发SDK的测试demo。(缺点:对技术要求较高,对js不熟悉的我就不多言语了。)
【方法二】
Python+Selenium: 这个一般被用来做UI自动化。selenium可以通过定位元素的方式来模拟用户在界面上的点击操作,从而实现自动化测试。除此之外,它还可以直接调用js文件的函数,通过浏览器驱动的execute_script方法来完成。
【方法三】
**Python+Pyexecjs: PyExecJS库相当于js的引擎,在Windows上默认有个JScript库,可以直接运行execjs。**不同的操作系统依赖不同的执行环境,PyExecJS库提供了JScript、Node、PhantomJS环境,可以根据情况进行配置。Python就是通过这个引擎来执行js函数的。
下面我将重点跟大家讲解下第三种方法!
1.Python开发环境:部署安装省略。
2.安装PyExecJS模块
(1)推荐:pip install pyexecjs
(2)到官网下载安装包,进行安装。(https://pypi.org/project/PyExecJS/)
(3)推荐:直接通过开发工具安装第三方依赖库。
1.引入模块的库名:import execjs
2.引入需要被调用的js文件:建议放置于python文件的同级目录。
3.Python调用过程:
(1)打开并读取js文件(注意编码格式)
rFile = open("./static/js/test.js", 'r', encoding='UTF-8')
(2)通过execjs库加载并编译js文件
testResult = execjs.compile(rFile.read()).call('test')
(3)方法解说
eval()
输入参数:source(JS语句)、cwd(路径)
返回值:result(语句执行结果)
compile()
输入参数:source(JS语句)、cwd(路径)
返回值:编译后的对象
call()
输入参数:name(要调用的JS方法名称)、*args(方法所需要的参数,可多个)
返回值:result(运行方法的返回结果)
# -*- coding:utf-8 -*-
from selenium import webdriver
import os,sys,io,time
import execjs
# #执行本地的js
# def get_js():
# # D:\Pycharm\PycharmProjects\SDKTestDemo\demo\static\js
# f = open("./static/js/test.js", 'r', encoding='utf-8')
# jsstr = f.read()
# line = f.readline()
# htmlstr = ''
# while line:
# htmlstr = htmlstr + line
# line = f.readline()
# print(htmlstr)
# return htmlstr
# # return jsstr
def login(driver,url):
# 通过驱动不同的浏览器:需要安装对应的驱动版本
driver = webdriver.Chrome()
driver.maximize_window()
time.sleep(1)
url = "https://liveweb.haoshitong.com:8080/demo/index.html?uri=85AcOKEaS"
driver.get(url)
time.sleep(3)
def testPy():
print('当前目录是:', os.getcwd())
rFile = open("./static/js/test.js", 'r', encoding='UTF-8')
testResult = execjs.compile(rFile.read()).call('test')
print(testResult)
if __name__ == '__main__':
#运行环境:JScript,Node,PhantomJS
os.environ["EXECJS_RUNTIME"] = "Node"
testPy()
1.无Windows目录?
2.没有Windows对象?找不到某个属性?
解决方法:
在js文件的头部添加如下代码:
const jsdom = require("jsdom");
const { JSDOM } = jsdom;
const dom = new JSDOM(`<!DOCTYPE html><p>Hello world</p>`);
window = dom.window;
document = window.document;
location = window.location;
XMLHttpRequest = window.XMLHttpRequest;
3.找不到路径?
4.中文出现乱码?
sys.stdout = io.TextIOWrapper(sys.stdout.buffer, encoding=‘gb18030’)
5.在windows环境运行存在gbk编码报错问题,错误信息:
“UnicodeEncodeError: ‘gbk’ codec can’t encode character ‘\xa9’ in position 75879: illegal multibyte sequence”
解决方法:
将python的Lib库subprocess.py文件中的586初始化函数的参数:encoding=None改成:encoding="utf-8"
6.未定义的错误
【错误1】execjs._exceptions.ProgramError: ReferenceError: navigator is not defined
【错误2】execjs._exceptions.ProgramError: TypeError: Cannot read property ‘userAgent’ of undefined
解决方法:
在js文档头部添加如下代码:
global.navigator={
userAgent: 'node.js',
};
8.函数名未定义错误,报错信息:execjs._exceptions.ProgramError: ReferenceError: getUsingSignup is not defined
1.如何调用第三方封装好的函数?
2.如何调用SDK?
3.存在多个依赖文件时,如何引用调用?
4.Js中的变量被加密后,python如何解析?
官方说明: https://pypi.org/project/PyExecJS/