为啥要弄这个?
我是天津大学的博士生,天大校园网络使用PPPOE方式登录,在OS X下需要首先在系统网络设置中连接PPPOE,再登录g.tju.edu.cn,最后再去点击“登录”按钮。就算在浏览器端事先保存好了用户名和密码,同时设置浏览器打开首页即为登录页,也需要至少4下鼠标点击操作,虽然操作不多,但人眼、手、脑协同控制光标在屏幕上跑来跑去是个纯粹的“闭环控制”,为这么一个登录操作而花这么多动作其实是件让人久而生厌的事情。
如果系统能帮我一键登录该有多好,即不再需要我控制光标“瞄准”了任务栏上的PPPOE连接标,也不需要“瞄准”登录按钮再点击,而是单纯地执行某一个命令就“开环”地实现了所有这些操作。
这件事能实现么?
Mac OS X为用户准备了一套系统脚本语言,即AppleScript(简称AS),用户可以利用AS控制整个系统及运行在其上的应用程序。AS允许用户在不具备专业编程知识的情况下依然能写出自己想要的高质量脚本程序。不过话虽这么说,个人感觉,一方面绝大多数非程序员用户是不会碰触AS的,另一方面对于一个从没接触过编程的人来说除非他特别聪明否则不太可能搞得定AS。
话再说回来,既然OS X系统为我提供了AS,这种登录网络的小case就不成问题。
这件事做起来耗时耗力么?
没想象中那么耗时。更新补充:妈蛋其实还是很耗时的!
虽然AS对我来说是一门新语言,但我有一些编程基础,语言和语言是相通的,无非是语法格式不同而已,代码逻辑依然要靠人来组织。而且,Yosemite原生支持了JavaScript(简称JS),并且所有已存在的AS命令都可以以JS的形式实现,这意味着我不需要去特意掌握一个小众的缺少参考资料的AS,直接看看漫山遍野的JS相关资料和例程就好。
说到具体功能的实现,这个就有点坑了,OS X 的man pages里的命令真是浩如烟海,想找一个shell命令只能靠搜索和猜测。OS X下ScriptEditor里的Dictionary实在是太精炼,一句例程都没有,必须得去看一下官网的Language Guide才能明白词典到底怎么用。这里最耗时间了,当然,这一步通过了,后面的事情就简单了。更新补充:其实后面的事情还是不那么简单……
这件事需要哪些命令实现?
1. 连接PPPOE
登录PPPOE需要用到系统的shell script,在官网os x man pages中section 8中networksetup页可以找到。
查询当前系统的PPPOE服务都有什么,可利用:
networksetup -listpppoeservices
得到返回值是tjuPPPOE,其实这个值就是当初建立PPPOE网络连接时自己起的名字,直接在UI界面下的网络设置就可以看到。
查询当前系统的连接状态,可利用:
networksetup -showpppoestatus tjuPPPOE
得到返回值为"connected"或"disconnected"。
连接PPPOE,可利用:
networksetup -connectpppoeservice tjuPPPOE
以上的所有shell脚本的实现需要借助doShellScript命令来实现。
2. 打开Safari进入登录页
打开Safari进入g.tju.edu.cn需要创建Safari的object,对指定的标签页的url做设定即可。需要注意的是,打开Safari之前需要首先判断一下PPPOE是否已经连接好了,这里不再详细说明。
使Safari打开g.tju.edu.cn有两种方式,一种是在某个固定的标签页打开,比如首个标签页,另一种是在原窗口中新建立一个标签页,在这个新建的标签页中打开。
对于第一种方法
在Safari的首个标签页中打开g.tju.edu.cn,可利用如下代码来实现:
app.doShellScript('open /Applications/safari.app');
Safari = Application('Safari');
for(;Safari.windows[0].visible()==false;) {
delay(0.1);
}
delay(0.1);
Safari.windows[0].tabs[0].url = "http://g.tju.edu.cn";
简单解释一下,若Safari应用已打开但窗口未打开,即Dock栏上Safari下有小光点但系统并未打开任何Safari窗口,则windows[0]
是存在的,但tabs[0]
不存在,此时脚本无法对tabs作操作。故首句app.doShellScript()
是为了保证无论Safari的应用是否打开,或者Safari的窗口是否visible,都能激活一个Safari窗口出来,以使tabs存在。第三句的for循环以及循环后的delay(0.1)
是为了保证在窗口完全打开之后再去执行tabs的操作,若在窗口未完全打开时就写入地址,会使Safari崩溃。
对于第二种方法
在新建立的标签页中打开g.tju.edu.cn,可利用如下代码来实现:
Safari = Application('Safari');
Safari.activate();
SystemEvents = Application('System Events');
safariUI = SystemEvents.processes['Safari'];
safariUI.menuBars[0].menus[2].menuItems[2].click();
for(;Safari.windows[0].visible()==false;) {
delay(0.1);
}
delay(0.1);
Safari.windows[0].currentTab.url = "http://g.tju.edu.cn";
这里利用了系统事件,虚拟点击了Safari菜单栏中的新建标签按钮,此操作是由safariUI.menuBars[0].menus[2].menuItems[2].click();
实现的。另外,对当前标签页的操作可由原来的tabs[0]
更改为currentTab
。
有关系统事件(SystemEvents)的内容可以参考苹果官网的JavaScript for Automation Release Notes以及WWDC2014的相关session视频(Javascript for Automation)。
3. “虚拟”点击登录按钮
这一步需要提前对网页的source做分析,找到按钮对应的函数后执行。与步骤2类似,需要首先判断一下网页是否被加载完成,可使用JavaScript语言中的indexOf()
来匹配网页源文件里的内容,即对于首标签打开网页的情况:
for(;Safari.windows[0].tabs[0].source().indexOf("do_login()")==-1;){delay(0.1);}
如果是新建标签,则将tabs[0]
更改为currentTab
。
分析了网页的源文件之后,发现登录按钮对应的函数叫do_login()
,是用JavaScript写的,所以只要网页事先勾选好了保存密码,直接执行do_login()
就能实现登录,这里需要借助doJavaScript
命令来实现,即:
Safari.doJavaScript('do_login()',{in: Safari.windows[0].tabs[0]});
如果是新建标签,则将tabs[0]
更改为currentTab
。
为了保证在网页加载完全之后再执行do_login()
,在Safari.doJavaScript()
前再加0.2秒的延迟让浏览器上的各Object加载完成,如果Safari加载速度不够快的话,可以适当增加一点延时时间。
放整体代码:
首个标签页打开登录页的情况:
app = Application.currentApplication();
app.includeStandardAdditions = true;
var isOutOfTime = 0;
var myPPPOEstatus = app.doShellScript("networksetup -showpppoestatus tjuPPPOE");
if (myPPPOEstatus != "connected") {
app.doShellScript("networksetup -connectpppoeservice tjuPPPOE");
}
for (var countTime=0;myPPPOEstatus!="connected";countTime++) {
myPPPOEstatus = app.doShellScript("networksetup -showpppoestatus tjuPPPOE");
delay(0.1);
if (countTime==30) {
isOutOfTime = 1;
app.displayDialog("Out of Time.",{buttons:"OK"});
break;
}
}
if (isOutOfTime == 0) {
Safari = Application('Safari');
Safari.activate();
SystemEvents = Application('System Events');
safariUI = SystemEvents.processes['Safari'];
safariUI.menuBars[0].menus[2].menuItems[2].click();
for(;Safari.windows[0].visible()==false;) {
delay(0.1);
}
delay(0.1);
Safari.windows[0].tabs[0].url = "http://g.tju.edu.cn";
for(;Safari.windows[0].tabs[0].source().indexOf("do_login()")==-1;) {
delay(0.1);
}
delay(0.2);
Safari.doJavaScript('do_login()',{in: Safari.windows[0].tabs[0]});
}
新建标签页打开登录页的情况:
app = Application.currentApplication();
app.includeStandardAdditions = true;
var isOutOfTime = 0;
var myPPPOEstatus = app.doShellScript("networksetup -showpppoestatus tjuPPPOE");
if (myPPPOEstatus != "connected") {
app.doShellScript("networksetup -connectpppoeservice tjuPPPOE");
}
for (var countTime=0;myPPPOEstatus!="connected";countTime++) {
myPPPOEstatus = app.doShellScript("networksetup -showpppoestatus tjuPPPOE");
delay(0.1);
if (countTime==30) {
isOutOfTime = 1;
app.displayDialog("Out of Time.",{buttons:"OK"});
break;
}
}
if (isOutOfTime == 0) {
Safari = Application('Safari');
Safari.activate();
SystemEvents = Application('System Events');
safariUI = SystemEvents.processes['Safari'];
safariUI.menuBars[0].menus[2].menuItems[2].click();
for(;Safari.windows[0].visible()==false;) {
delay(0.1);
}
delay(0.1);
Safari.windows[0].currentTab.url = "http://g.tju.edu.cn";
for(;Safari.windows[0].currentTab.source().indexOf("do_login()")==-1;) {
delay(0.1);
}
delay(0.2);
Safari.doJavaScript('do_login()',{in: Safari.windows[0].currentTab});
}
最后的效果就是,无论我是刚开机,还是因休眠而断开了网络,都只需要执行一下脚本,系统会自动连接PPPOE,并自动打开Safari,进入g.tju.edu.cn,然后自动帮我虚拟点击登录按钮,正所谓一劳永逸。
欢迎反馈使用情况与交流意见,谢谢!