摘要

很久之前就想着要写个脚本,要么去刷12306的票,要么就登QQ空间。为什么呢?你想啊,别人刚发一个说说,然后你就能检测到并秒赞回去,这得多让人惊讶。(不小心暴露了异想天开的本质啦,⊙﹏⊙b汗)。

一开始学习Python的时候就模拟着试了试,除非借助于cookie那块,不然也是没法成功的。然而这次歪打正着。本学期有个《软件工程导论》课程,刚好讲到了测试这块,然后就提到了自动化测试,以及在此行业中的翘楚Selenium,结果就是发现了新大陆一般,觉得拿来测试模拟登陆应该会比较不错。

于是,真的成功啦。在此记录一下,走过的历程,填过的坑。

环境搭建

Selenium是一个支持多语言的自动化测试框架,不管是Java, Ruby,还是Python,都能使用其支持的库来进行自动化测试。我本人最喜欢Python语言,所以这里将会以Python语言进行测试。

驱动

Selenium是典型的CS框架,浏览器作为Server执行Client(selenium代码)的请求,通过“代理”这么个理念实现自动化测试。 这么说可能不太通俗,换句话就是selenium执行的时候会调用浏览器,根据设置好的代码运行,最终实现自动化测试。

selenium2之后,以webdriver代替Proxy功能,处理所有请求。

所以不管怎么说,浏览器都是必不可少的啦。因此我们需要安装一下浏览器驱动。

关于浏览器驱动的问题,可能就是阻挡了大部分人使用selenium的拦路虎吧。

下载

下载驱动的话,可以到 http://www.seleniumhq.org/projects/webdriver/

这个网址进行下载。按需下载即可,待会会讲解怎么使用,我这里下载的是firefoxdriver.exe

下载webdriver驱动

selenium

安装selenium也是很方便的。

如果您已安装比较旧的版本:

pip install -U selenium

如果您还未安装selenium:

pip install selenium

这样就完事了。

驱动安装问题集

下面聊聊我在这个过程中遇到的一些奇怪的问题,可能不具备普适性。

未发现驱动

如果驱动没有正确放置,就会报出下面的错误。
驱动未正确配置

解决办法就是将刚才下载的driver放置到Python的Path或者放到任意一个系统能找得到的Path中。我个人建议放置到Python的根目录中,这样便于管理。

firefox驱动错误

主要症状就是无法开启,闪退。这个时候我们需要下载一个geckodriver.exe。

可以在下面的链接中找到适合自己系统版本的来使用。

https://github.com/mozilla/geckodriver/releases

下载完之后放置到Python路径中即可,处理方法和刚才的那个一样。然后问题就解决了(反正我是这么解决的啦)。

模拟登陆

下面开始步入正题了,使用selenium进行模拟登陆其实就可以想象成有一双无形的手在进行用户名,密码填写,然后点击按钮等等。这样就会变得很容易理解了。

这里关于selenium选择器等等的基础性的知识点就不再讲解了,网上资料很多,也很详细。相信大家一看就会明白的。

我自己最常用也最喜欢的就是:

driver.get_element_by_id()
driver.get_element_by_xpath()

然后要登录QQ空间,就得先看看人家长什么样吧,如图:

QQ空间首页

首败

按下F12就可以查看网页的源代码了,所以什么用户名啊,密码啊,登陆按钮啊都不是事了。不多说,直接上代码。

#coding: utf8

from selenium import webdriver

driver = webdriver.Firefox()

driver.get('http://i.qq.com/')

driver.find_element_by_id('switcher_plogin').click()

driver.find_element_by_name('u').clear()
driver.find_element_by_name('u').send_keys('你的QQ号')
driver.find_element_by_name('p').clear()
driver.find_element_by_name('p').send_keys('你的密码')



driver.find_element_by_xpath('//*[@id="loginform"]/div[4]/a').click()
driver.find_element_by_id('login_button').click()

print driver.current_url

然而我发现我想多了,真正的考验才刚刚开始。这样根本就找不到网页源代码中看到的那些个id啊class什么的。

再败

既然找不到,可能就是代码的问题了。然后我又仔仔细细的查看了一下代码,发现也没啥错误啊。然后不甘心,又去看了看网页的源代码。结果还真的被我发现了。

尼玛,这叫什么事嘛。

如下图,不难发现了吧。
多了个iframe

有一个iframe,怪不得Selenium找不到,既然如此,那咱们就乘胜追击。修改一下代码呗。

#coding: utf8

from selenium import webdriver

driver = webdriver.Firefox()

driver.get('http://i.qq.com/')
driver.switch_to.frame('login_frame')
driver.find_element_by_id('switcher_plogin').click()

driver.find_element_by_name('u').clear()
driver.find_element_by_name('u').send_keys('你的QQ号')
driver.find_element_by_name('p').clear()
driver.find_element_by_name('p').send_keys('你的密码')



driver.find_element_by_xpath('//*[@id="loginform"]/div[4]/a').click()
driver.find_element_by_id('login_button').click()

print driver.current_url

加上了这么一行:

driver.switch_to.frame(‘login_frame’)

作用就是根据iframe的id或name来跳转到这个iframe上。

本以为完事大吉咯,终于可以秒赞了,耶。

然而事实给了我一个残酷的打击。到最后一个“登录”按钮点击的时候停了下来。结果还是失败了。

成功

它越是这样,我就越想写出来。没办法,硬着头皮上呗,直觉上还是QQ空间源代码的问题,于是这次就直接继续看源代码去了。
果不其然,发现了下图这么个代码。
hidefocus问题

然后我就想使用JS来把这个故意隐藏不让focus的给显现出来。

恩,我就是这么做的,然后真的就成功了。核心就添加了下面的这行代码。

driver.execute_script("document.getElementById('login_button').parentNode.hidefocus=false;")

下面贴出来完整的代码吧,方便参考。

#coding: utf8

from selenium import webdriver

driver = webdriver.Firefox()

driver.get('http://i.qq.com/')
driver.switch_to.frame('login_frame')
driver.find_element_by_id('switcher_plogin').click()

driver.find_element_by_name('u').clear()
driver.find_element_by_name('u').send_keys('你的QQ号')
driver.find_element_by_name('p').clear()
driver.find_element_by_name('p').send_keys('你的密码')

driver.execute_script("document.getElementById('login_button').parentNode.hidefocus=false;")


driver.find_element_by_xpath('//*[@id="loginform"]/div[4]/a').click()
driver.find_element_by_id('login_button').click()

print driver.current_url

演示

下面使用一个gif图来演示一下模拟登陆的效果,方便流畅性的观看,也更具说服力。

QQ空间模拟登陆

总结

到此基本上已经可以拿来使用了。无非在登陆成功的界面下使用Selenium模拟进行一些点击操作。如果想发点文字的话也是很方便的,send_keys可以很好的解决这个问题。什么秒赞,秒答都不是什么事了。

但是需要注意的是,QQ空间对登陆频率是做了限制的,就算是手动登陆,连续几次之后就会让我们输入验证码。这也是一种安全机制罢了。

如果有需要的话,Python处理验证码的模块也是非常好用的。这里就不再介绍了。

最后,Selenium给我的感悟就是:我自己对Selenium有点大材小用了,这样一款优秀的测试框架,却拿来做这种小玩意,确实是有点说不过去。

另外我也认识到了,测试的目的是为了提供更好的服务。而不是拿来当做攻击的手段。恶意的发水机,这种行为极不负责任,希望我们都能谨言慎行,共同营造一个和谐的网络家园。

心正了,人才会正直。感谢Selenium给我的这个启迪。


本文转载:CSDN博客