qxVm/qxVm_sanbox/z_sanbox/qxVm.sanbox.js
2023-04-20 21:14:25 +08:00

158 lines
7.0 KiB
JavaScript

const fs = require("fs");
// const { VM, VMScript } = require('vm2/index');
// const { NodeVM } = require('vm2/index');
const { VM, VMScript } = require('../lib/vm2/index');
const { NodeVM } = require('../lib/vm2/index');
const QXVM_NODE = require('./qxVm.env.js');
const TOOLS = require('../tools/qxVm.sanbox.tools');
/*
导包顺序就是继承关系的顺序
*/
// 只要这个文件运行了, 那object对象的原型链上的pop方法一定存在
Object.prototype.pop = function (key) {
let value = this[key];
delete this[key];
return value
};
/**
* @return {string}
*/
function ReadCode(name, dir) {
let file_path = dir === undefined ? `${__dirname}/${name}` : `${__dirname}/${dir}/${name}`;
return fs.readFileSync(file_path) + "\r\n"
}
function configObj_to_configStr(user_config) {
let proxy_config = user_config.pop('proxy_config') || { proxy: false, proxy_proto: true, print_log: true };
let env = user_config.pop('env') || {};
let canvas = env.pop('canvas') || "";
let plugin = env.pop('plugin') || [
{
description: "Portable Document Format",
filename: "internal-pdf-viewer",
name: "Chrome PDF Plugin",
MimeTypes: [{
description: "Portable Document Format",
suffixes: "pdf",
type: "application/x-google-chrome-pdf"
}]
},
{
description: "",
filename: "mhjfbmdgcfjbbpaeojofohoefgiehjai",
name: "Chrome PDF Viewer",
MimeTypes: [{ description: "", suffixes: "pdf", type: "application/pdf" }]
},
{
description: "",
filename: "internal-nacl-plugin",
name: "Native Client",
MimeTypes: [
{
description: "Native Client Executable",
suffixes: "",
type: "application/x-nacl"
},
{
description: "Portable Native Client Executable",
suffixes: "",
type: "application/x-pnacl"
}
]
}
];
let window_attribute = user_config.pop('window_attribute') || {};
let proxy_config_str = "",
canvas_str = "",
env_str = "",
plugin_str = "",
window_attribute_str = "";
for (let key in proxy_config) {
proxy_config_str += proxy_config.hasOwnProperty(key) ? `qxVm.config.${key}=${proxy_config[key]};\r\n` : '';
}
canvas_str = `qxVm.default_envs.canvas="${canvas}"\r\n`;
env_str = `qxVm.config_envs(${JSON.stringify(env)})\r\n`;
for (let i = 0; i < plugin.length; i++) {
plugin_str += `qxVm.config_plugin(${JSON.stringify(plugin[i])})\r\n`
}
for (let key in window_attribute) {
window_attribute_str += window_attribute.hasOwnProperty(key) ? `window.${key}=${JSON.stringify(window_attribute[key])};\r\n` : "";
}
return {
proxy_config_str: proxy_config_str,
canvas_str: canvas_str,
env_str: env_str,
plugin_str: plugin_str,
window_attribute_str: window_attribute_str
}
}
function qxVm_sanbox(js_code, func_name, user_config) {
console.log(`
:params js_code 用户自己的代码(可以是一个文件得绝对路径)
:params func_name 需要导出的方法
:params user_config 个性浏览器环境
user_config:{
isTest: Boolean // 如果测试则固定随机数种子, 时间戳
compress: Boolean, // 是否压缩js, 针对 ob,sojson 检测格式化
proxy_config: Object, // 是否挂代理
canvas: String, // 自定义canvas指纹
window_attribute: Object, // 自定义window属性(暴露window出来)
plugin: Array, // 自定义浏览器插件
env: Object { // 自定义浏览器环境
navigator: Object, // 针对 加密UA和请求UA必须一致的情况
location: Object, // 自定义属性
document: Object // 自定义属性
}
}
`);
if (!TOOLS.str_is_funcName(func_name)) {
throw TypeError(`参数 "func_name->${func_name}" 不符合规范`)
}
let compress = user_config.pop('compress') || false; // 是否压缩文件, 防止 sojson , ob, 检测格式化
let isTest = user_config.pop('isTest') || false; // 如果测试则固定随机数种子, 时间戳
let user_config_str = configObj_to_configStr(user_config);
let qxVm_code = QXVM_NODE.qxVm_env(user_config_str);
let export_func = `\r\nqxVm_module.exports = { ${func_name} }`;
js_code = TOOLS.isFilePath(js_code) ? ReadCode(js_code) : js_code; // 判断是不是文件路径, 如果是则去读取文件
js_code = compress ? TOOLS.compress_jsCode(js_code) : js_code;
js_code = isTest ? TOOLS.set_Time_Random(js_code) : js_code;
js_code = `${qxVm_code}${js_code}${export_func}`;
debugger;
let qxVm_module = {
exports: {},
node_func: {
setTimeout: setTimeout,
setInterval: setInterval,
clearTimeout: clearTimeout,
clearInterval: clearInterval
}
};
const vm = new VM({ sandbox: { qxVm_module } });
const script = new VMScript(js_code, `${__dirname}/vmcode.js`);
try {
return vm.run(script);
} catch (err) {
console.error('Failed to execute script.', err);
}
}
module.exports = {
QXVm_sanbox: qxVm_sanbox
};