Board logo

标题: [原创] bat、vbs、js 原生混编 [打印本页]

作者: CrLf    时间: 2015-1-30 06:05     标题: bat、vbs、js 原生混编

本帖最后由 CrLf 于 2015-8-23 02:22 编辑

发现 mshta 会把 file:// 协议指向的文件当作 html 来解析(注:IUnknown 与 happyxxdhaha 提醒此处必需使用绝对路径,否则不会执行),心里顿时有一万只草泥马奔过,原来如此简单的答案就在身边,却错过了四年
基本框架:
  1. <!-- : www.bathome.net
  2. @echo off
  3. echo I'm Batch!
  4. mshta "file://%~f0"
  5. pause&exit
  6. 使用注释标签囊括批处理部分,条件是批处理部分不能出现注释标签的结束符
  7. -->
  8. <script language=vbs>
  9. Msgbox "I'm VBScript!"
  10. </script>
  11. <script>
  12. alert("I'm JavaScript!")
  13. </script>
  14. <script>close()</script>
复制代码
事实上,file:// 协议名可以省略,而且如果不需要理会界面的话,完全可以不用注释标签:
  1. @echo off
  2. echo I'm Batch!
  3. mshta "%~f0" <nul
  4. pause&exit
  5. 批处理部分之后要加上一串 >,数量要比前文出现的 < 更多,mshta 才能区分哪些是标签
  6. 而且前文出现从文件获取重定向输入的时候,建议加上双引号,例如 <"script"
  7. >>>>>>>>>>>>>>
  8. <script language=vbs>
  9. Msgbox "I'm VBScript!"
  10. </script>
  11. <script>
  12. alert("I'm JavaScript!")
  13. </script>
  14. <script>close()</script>
复制代码
另一用 goto 的种写法可能更直观一点:
  1. @goto :bat
  2. <script language=vbs>
  3. Msgbox "I'm VBScript!"
  4. </script>
  5. <script>
  6. alert("I'm JavaScript!")
  7. </script>
  8. <script>close()</script>
  9. :bat
  10. @echo off
  11. echo I'm Batch!
  12. mshta "%~f0" <nul
  13. pause&exit
复制代码
要注意的是,这里的宿主是 mshta,所以不支持 WSH 宿主的方法和属性(部分属性或方法的替代方案详见后文)
但是!mshta 有嘛不好!
原生支持 setTimeout
原生支持 iframe
原生支持 dom
原生支持 javascript、vbscript 无障碍交互
原生支持 Ajax
原生支持加载外部脚本
原生支持在窗口中选择文件
原生支持复杂的页面交互
...
有这么多便利,那还计较什么呢?
首发于批处理之家
------------------------------------------------------------------------------------------------------------
关于 mshta 宿主的一些知识,参考: https://msdn.microsoft.com/en-us/library/ms536495(VS.85).aspx
感谢 xiaopo 扫盲,才知道联盟早已出现过 mshta 方案的雏形:http://cndos.fam.cx/forum/viewthread.php?tid=39655,回头上镜像站搜下,看看有没有进一步的发展
作者: CrLf    时间: 2015-1-30 06:13

本帖最后由 CrLf 于 2015-8-26 20:52 编辑

好了,我们还要消灭两个障碍~

我们知道批处理可以用 %1~%9 获取切分好的命令行参数,mshta 则没有直接的办法来获取
另一方面,mshta 是个 GUI 宿主,如果不能较好地和控制台交互,调用 mshta 将束手束脚

好在,这些都是可以解决的
-----------------------------------------------------
1、命令行参数的实现
获取 mshta 的启动参数先要创建一个 HTA:APPLICATION 标签,并设置一个 id,例如:
<HTA:APPLICATION id=mshta></HTA>

这里以 id=mshta 为例,可以用 mshta.commandLine 获取未切分的命令行参数
然后就可以按 windows 的规则来解析,例如:
  1. <script>var argv = getopts(mshta.commandLine)
  2. for(var i=0;i<argv.length;i++)alert(argv[i])
  3. function getopts(strArg){
  4.         var re = /[^"\s,;=]*"([^"]*("[^"\s,;=]*")*)*("[^"\s,;=]*|$)|[^"\s,;=]+/g
  5.         var argv = []
  6.         strArg.replace(re,function($0){argv.push($0.replace(/^"(.*)"$/g,'$1'))})
  7.         return argv
  8. }
  9. </script>
复制代码
原方案有误,现已修正了 getopts 的实现方式,正则自己写,爽爽的
示例脚本:[attach]9005[/attach]
-----------------------------------------------------
2、StdIn、StdOut、StdErr 的实现
使用 fso 控件的 GetStandardStream 方法可实现(由 terse 传授,详见原帖
得到的对象与 WScript 中的 StdIn、StdOut、StdErr 有完全相同的方法和属性,例如:
  1. <script>var fso = new ActiveXObject('Scripting.Filesystemobject')
  2. var StdIn = fso.GetStandardStream(0),
  3.      StdOut = fso.GetStandardStream(1),
  4.      StdErr = fso.GetStandardStream(2)
  5. alert(StdIn.ReadAll())
  6. StdOut.Write('标准输出测试')
  7. StdErr.Write('错误输出测试')</script>
复制代码
示例脚本:[attach]8295[/attach]
作者: CrLf    时间: 2015-1-30 06:13

本帖最后由 CrLf 于 2015-8-26 18:51 编辑

与 bat+js 混编的异同
bat+hta 混编 bat+js 混编
宿主mshta.exewscript.exe 或 cscript.exe
宿主对象与 <HTA:APPLICATION> 的 id 同名WSH 或 WScript
子系统GUI GUI 或 CLI
界面原生 html
实际上,如果抛掉 DOM、BOM 之类由宿主强加上去的部分,代码的基本语法是一致的(hta 加载 WSH.js 之后,就和 wscript 里的环境很像了)
mshta 比 wscript 更方便的是,它有界面(事实上它本来就是用来写桌面程序),而且可以动态加载文件(内核是 ie)
例如我用批处理之家在线 js 脚本库的 loadFirebug.js 加载一个 FireBug:
  1. <p>按 F12 调出控制台</p><script src=http://www.bathome.net/lib/diy/loadFirebug.js></script>
复制代码
这是 WScript 所做不到的
作者: CrLf    时间: 2015-1-30 06:20

本帖最后由 CrLf 于 2015-3-6 05:57 编辑

配合批处理之家 js 在线脚本库,可扩展更多功能,详见:
http://www.bathome.net/thread-34544-1-1.html

例如模拟 WSH 环境的完整实现和第三方工具的下载:
  1. @echo off
  2. mshta "%~f0" psexec
  3. .\psexec /?
  4. pause
  5. >>>>>>>>>>
  6. <script src=http://bbs.bathome.net/lib/diy/hide.js></script>
  7. <script src=http://bbs.bathome.net/lib/diy/Tools.js></script>
  8. <script src=http://bbs.bathome.net/lib/diy/WSH.js></script>
  9. <script>Tools.get(WScript.Arguments(0))</script>
复制代码

作者: xxpinqz    时间: 2015-1-30 07:59

起的这么早
期待大作。。。。。。
作者: Demon    时间: 2015-2-1 18:01

火钳
作者: CrLf    时间: 2015-2-2 03:56

忙啊忙,放一阵,下次更新动作很大,得酝酿一下
作者: apang    时间: 2015-2-5 00:00

这个要顶,期待中。
作者: Spring    时间: 2015-2-5 11:14

与直接使用hta文件相比,这有什么好处吗?
作者: CrLf    时间: 2015-2-5 16:05

本帖最后由 CrLf 于 2015-2-5 17:03 编辑

回复 9# Spring


    分成两个独立文件互相调用当然没问题,不过放在一起省一个步骤,写的时候不用切换窗口,用的时候也比较绿色简便
    作为批处理论坛,求助者可能默认把代码当成批处理保存,混编以后确实就是个批处理,不需要再向楼主说明这个是 .vbs 那个是 .bat,也不要求一定要保存成 外部调用.vbs 才能被认到
    你看现在论坛上几位经常发话的坛友就经常用 Bat+JS 混编,多敲两行语句,大概可以省许多口水
    纯 js 和 vbs 虽然很强大,但某些操作比较麻烦,而且这里毕竟是批处理之家,要是放眼望去全是“请保存为 .vbs”,那让人情何以堪,要是 powershell 倒是可以接受,毕竟一脉相承
----------------------------------------------------------------------------------------------------------------
    往远了说,各种脚本与批处理混编也许会成为论坛的特色,用 batcher 的话说,“批处理的圈子就屁股大”,无论是圈子里的人还是圈子里的发展空间都适用,纯批已经几乎到尽头了,那么再走下去是什么呢?
    曾和 tmplinshi 讨论过批处理的未来将是外部命令还是 vbs,我当时认为是 vbs,理由无他,便携、安全、系统自带这三大优势足以和外部命令抗衡
    但受语法所制,vbs 很难和批处理相容,即使学会了 mshta vbscript:xxxx 的用法,仍不是最好的选择
    后来,powerbat 有一阵子一直在推广 bat+js 混编,起初觉得很神奇,却没有引起注意,直到学会了 js 才开始跟着推广,当时觉得在新一代命令行 powershell 面前,这就是批处理的未来
    可惜 js 也有缺点,有些方面终究是不如 vbs 的,于是如何实现 bat+vbs 混编就成为一根心头刺,尝试过汇编修改 WScript.exe 和 CScript.exe 以兼容 bat/cmd 后缀的 WSH 文件,虽然做出来了,但终究治标不治本
    那天无意中发现(其实我是想试试看能不能用这种方式下载文件),mshta 会将 file://xxxx 里的文本内容当作 html 解析,顿时豁然开朗,bat+vbs 目前最优的混编方案原来藏在这里
    于是就发了这个帖子
----------------------------------------------------------------------------------------------------------------
    所以混编至少对批处理论坛来说还是有实用价值的
    过阵子(也许要等到春节前后),我会更新这个帖子,到时候,这个混编方案会比现在强大得多,不知道自己的想法能不能被接受,先做出来再说
作者: 凡凡之呗    时间: 2015-2-6 14:16

回复 1# CrLf 不错,综合这么多,能有如此发现
作者: 凡凡之呗    时间: 2015-2-6 14:20

回复 1# CrLf


    英文网页看不大懂
作者: CrLf    时间: 2015-3-5 17:51

帖子已更新,简化了实现方法...
作者: IUnknown    时间: 2015-3-5 22:38

回复 1# CrLf


    mshta后面的第一个参数如果是文件,会当作html来解析,这不是很明显吗,为什么现在才发现呢?而且DOS联盟早就有不少类似的多语言混编的帖子,包括hta。
作者: IUnknown    时间: 2015-3-5 22:47

"%~0" 最好用 "%~f0",因为运行bat脚本时可能是用相对路径。
而mshta的文件路径必须是绝经路径,否则进程不退出,也不执行文件内容。
%0的确切含义,好像以前某个批处理论坛有帖子分析过,可惜那个论坛貌似也关闭很久了。
批处理之家,够坚挺!
作者: happyxxdhaha    时间: 2015-3-6 01:41

没错,是应该把 "%~0" 改成 "%~f0",一般我们在资源管理器中打开一个bat/cmd的文件,%0就是完全合格的路径,等价于把一个bat/cmd文件拖到CMD窗口中一样,路径中如果有空格,就会自动加双引号。如果我要从CMD窗口中运行一个bat/cmd的文件,我一般只会输入一个文件名,那么%0就等于我输入的文件名,我如果输入一个文件名加一个扩展名,那么%0就等于文件名加一个扩展名,只有我输入绝对路径,%0才等于绝对路径。一句话,我在命令提示符中输入什么样的路径来执行bat,那么%0就等价于什么样的路径。所以,如果我要在命令提示符中运行顶楼的bat,只输入文件名,执行到mshta所在的行就会出现问题。
作者: CrLf    时间: 2015-3-6 06:10

本帖最后由 CrLf 于 2015-3-6 14:36 编辑

回复 14# IUnknown


    同为 windows 自带的脚本宿主,wscript、cmd、powershell、regedit 都会检查后缀名,个人感觉 mshta 不看文件类型直接执行这才叫少数派呢
    mshta 也许是特殊在于第一个参数被视为 url,才没有设下这种限制
    兄台应该是 dos 联盟的前辈了,拱手一个。我介入得晚,没碰上那个黄金时代,也许无意间把前人的技术又“发明”了一次。
    不过奇怪的是,如果早就有这技术,为什么在此之前我看到 bat+vbs 混编最像样的也只有 est 的 On Error Resume Next 方案和 mshta 的 vbscript: 方案呢?前者太苛刻,后者受命令行参数长度所制,特别是如果要兼容空格还得写更长
-----------------------------------------------------------------------------------------------------------------------------------------
    非常感谢指出 %~0 的错误!之前没发现还有这问题,毕竟url地址是允许相对路径的,script 的 src 也支持相对路径,没往深处想。
    想想也是,url 的相对路径都是基于 location.href 的,那么一开始必须存在一个绝对路径作为参照
    确实是个隐患,已修改,谢谢指正
    另外,测试中发现个有趣的现象,把下面的内容分别保存为 txt、bat、hta,mshta 竟然很聪明地不把 txt 里的内容当脚本看,看起来执行前还是有检查后缀名的,不过想不明白用意何在
  1. <script>
  2. alert('haha')
  3. </script>
复制代码

作者: CrLf    时间: 2015-3-6 06:24

继续测试:
  1. mshta http://batch-cn.qiniudn.com/test/测试.rar
  2. mshta http://batch-cn.qiniudn.com/test/测试.txt
  3. mshta http://batch-cn.qiniudn.com/test/测试.html
  4. mshta http://batch-cn.qiniudn.com/test/测试.js
  5. mshta http://batch-cn.qiniudn.com/test/测试.bat
  6. mshta file://D:/测试/测试.rar
  7. mshta file://D:/测试/测试.txt
  8. mshta file://D:/测试/测试.html
  9. mshta file://D:/测试/测试.js
  10. mshta file://D:/测试/测试.bat
复制代码
对比结果发现 http:// 下几种后缀名效果一致,而在 file:// 协议下 txt 不会被解析为 dom...
这是为什么呢
作者: 慕夜蓝化    时间: 2015-3-8 18:20

顶一下,我也要学学其它的脚本语言,还有一些工具,不能老是小打小闹。
如果能看明白你写的什么就好了。
赞赞赞。。。
作者: ptsdy    时间: 2015-3-14 15:02

好贴,绝对是一种创新!省去临时文件,多种语言混写语法支持的麻烦!
代码不难,关键是给大家提供的一种范例!至少论坛中少有这种写法。
可以看出楼主有代码洁僻,精益求精!赞!
作者: 林小七    时间: 2015-7-29 13:44

火        钳
作者: xiaopo    时间: 2015-8-22 22:54

本帖最后由 xiaopo 于 2015-8-22 22:59 编辑

学习一下,在dos联盟找到一个之前的
  1. :<!--
  2. ::::::::::::::::::::::::::::::::BAT::::::::::::::::::::::::::::::::
  3. ::::::::1.执行HTML代码之前的BAT代码::::::::
  4. @echo off
  5. call :e Starting mshta...
  6. pause
  7. ::::::::1.执行HTML代码之前的BAT代码::::::::
  8. ::执行HTML代码:
  9. start mshta %0
  10. ::::::::2.执行HTML代码之后的BAT代码::::::::
  11. call :e Mshta is executing HTML codes...
  12. pause
  13. ::::::::2.执行HTML代码之后的BAT代码::::::::
  14. ::退出BAT:
  15. exit /b
  16. :::::::BAT函数定义部分:::::::
  17. :e
  18. echo %*
  19. goto :eof
  20. :::::::BAT函数定义部分:::::::
  21. ::::::::::::::::::::::::::::::::BAT::::::::::::::::::::::::::::::::
  22. -->
  23. <!--此句用来清除第一行的:-->
  24. <script>document.body.innerText=""</script>
  25. <script language=vbs>
  26. Msgbox "I'm VBScript!"
  27. </script>
  28. <script>
  29. alert("I'm JavaScript!")
  30. </script>
  31. <script>close()</script>
  32. <!--------------------------HTML-------------------------->
  33. <body onkeypress=window.close()>
  34. <hr color=red>
  35. <marquee><font color=green>HTML Codes</font></marquee>
  36. <hr color=red>
  37. <!--------------------------HTML-------------------------->
  38. <!--        BAT & HTML        {s11ss@www.cn-dos.net/forum 2008-4-22}
  39. 思路:当此文件被当作BAT文件执行时,未执行到HTML代码部分时就已退出;
  40.       当此文件被当作HTML文件执行时,BAT代码部分被注释,不会被执行。
  41. -->
复制代码
原文地址:http://cndos.fam.cx/forum/viewthread.php?tid=39655
作者: CrLf    时间: 2015-8-23 01:52

回复 22# xiaopo


    原来联盟已经发明过了,膜拜,这么好的办法竟然没普及,真费解…或者说,我所熟知的 mshta vbscript:xxxxx 的写法是其最终的演化结果?
    真可惜错过了那个黄金时代,迟到七年,“原创”就成了“改良”,不过若没有来自联盟潜移默化的传承,我大概也想不到用 mshta
    事实上 mshta 输出 stdout 的方法我是从 terse 的代码中学到的,而他是联盟版主,所以这技巧有可能也是来自联盟,不知道联盟健在的时候 mshta 混编是发展到什么地步
    幸好两贴的方法和延伸略有差别,轮子造得更圆了一点
作者: givengenius    时间: 2022-11-12 10:12

最近也在研究混排,在stackoverflow找了個批處理混排vbs及js的方案,就是通過wsh,可能我的需求略簡單,通過這種方式實現了目的;很早也看到過mshta的方案,但是沒仔細研究,特來研究下
作者: givengenius    时间: 2022-11-12 10:14

還有個小問題,文件要保存為bat還是html?




欢迎光临 批处理之家 (http://www.bathome.net/) Powered by Discuz! 7.2