1500字范文,内容丰富有趣,写作好帮手!
1500字范文 > 【Web UI自动化测试】Web UI自动化测试之日志收集篇(全网最全)

【Web UI自动化测试】Web UI自动化测试之日志收集篇(全网最全)

时间:2019-01-21 08:22:38

相关推荐

【Web UI自动化测试】Web UI自动化测试之日志收集篇(全网最全)

本文大纲截图:

1、日志介绍【文末免费分享自动化测试学习资源】

日志:用于记录系统运行时的信息,对一个事件的记录,也称为Log

日志作用:

1)调试程序

2)了解系统程序运行的情况是否正常

3)系统程序运行故障分析与问题定位

4)用来做用户行为分析和数据统计

日志级别:

日志级别:指日志信息的优先级、重要性或者严重程度。

常见的日志级别:DEBUGINFOWARNINGERRORCRITICAL

DEBUG:调试级别,打印非常详细的日志信息,通常用于对代码的调试

INFO:信息级别,打印一般的日志信息,突出强调程序的运行过程

WARNING:警告级别,打印警告日志信息,表明会出现潜在错误的情形,一般不影响软件的正常使用

ERROR:错误级别,打印错误异常信息,该级别的错误可能会导致系统的一些功能无法正常使用

CRITICAL:严重错误级别,一个严重的错误,这表明系统可能无法继续运行

代码示例:

importlogginglogging.debug("这是一条调试信息")logging.info("这是一条普通信息")logging.warning("这是一条警告信息")logging.error("这是一条错误信息")logging.critical("这是一条严重错误信息")

说明:

1)上面列表中的日志级别是从上到下依次升高的,即:DEBUG < INFO < WARNING < ERROR < CRITICAL

2)当为程序指定一个日志级别后,程序会记录所有日志级别大于或等于指定日志级别的日志信息,而不是仅仅记 录指定级别的日志信息;

3)一般般建议只使用DEBUG、INFO、WARNING、ERROR这四个级别。

2、日志用法

2.1 基本用法

设置日志级别:

默认:logging中默认的日志级别为WARNING,程序中大于等于该级别的日志才能输出,小于该级别的日志不会被打印出来。

格式:logging.basicConfig(level=logging.DEBUG)

如何选择日志级别?

开发环境和测试环境:为了尽可能详细的查看程序的运行状态来保证上线后的稳定性,可以使用DEBUGINFO级别的日志获取详细的日志信息,这是非常耗费机器性能的。

生产环境:通常只记录程序的异常信息、错误信息等(设置成WARNINGERROR级别),这样既可以 减小服务器的I/O压力,也可以提高获取错误日志信息的效率和方便问题的排查。

设置日志格式:

默认的日志的格式为:format=日志级别:Logger名称:日志内容

format参数中的格式化信息

自定义日志格式:logging.basicConfig(format="%(levelname)s:%(name)s:%(message)s")

代码示例:

importloggingfmt='%(asctime)s%(levelname)s[%(name)s][%(filename)s(%(funcName)s:%(lineno)d)]-%(message)s'logging.basicConfig(level=logging.INFO,format=fmt)logging.debug("调试")logging.info("信息")logging.warning("警告")logging.error("错误")

将日志信息输出到文件中

默认情况下Python的logging模块将日志打印到了标准输出中(控制台)

格式:logging.basicConfig(filename="a.log")

代码:

importloggingfmt='%(asctime)s%(levelname)s[%(name)s][%(filename)s(%(funcName)s:%(lineno)d)]-%(message)s'logging.basicConfig(filename="a.log",level=logging.INFO,format=fmt)logging.debug("调试")logging.info("信息")logging.warning("警告")logging.error("错误")

2.2 高级用法

logging日志模块四大组件:LoggerHandlerFormatterFilter

Logger:日志器,提供了程序使用日志的入口

Handler:处理器,将logger创建的日志记录发送到合适的目的输出

Formatter:格式器,决定日志记录的最终输出格式

Filter:过滤器,提供了更细粒度的控制工具来决定输出哪条日志记录,丢弃哪条日志记录

组件关系:

日志器(logger)需要通过处理器(handler)将日志信息输出到目标位置,如:文件、sys.stdout、网络等;

不同的处理器(handler)可以将日志输出到不同的位置;

日志器(logger)可以设置多个处理器(handler)将同一条日志记录输出到不同的位置;

每个处理器(handler)都可以设置自己的格式器(formatter)实现同一条日志以不同的格式输出到不同的地 方;

每个处理器(handler)都可以设置自己的过滤器(filter)实现日志过滤,从而只保留感兴趣的日志。

Logger类:

Logger对象的任务:向程序暴露记录日志的方法;基于日志级别或Filter对象来决定要对哪些日志进行后续处理;将日志消息传送给所有感兴趣的日志handlers

创建Logger对象:

格式一:logger = logging.getLogger(),返回root日志器对象

格式二:logger = logging.getLogger("myLogger"),返回myLogger日志器对象

说明:logging.getLogger()方法有一个可选参数name,该参数表示将要返回的日志器的名称标识,如果不提供该参数,则 返回root日志器对象。若以相同的name参数值多次调用getLogger()方法,将会返回指向同一个logger对象的引用。

Logger常用的方法:

打印日志:

logger.debug()

logger.info()

logger.warning()

logger.error()

logger.critical()

设置日志级别:logger.setLevel(),设置日志器将会处理的日志消息的最低严重级别

添加handler对象:logger.addHandler(),为该logger对象添加一个handler对象

添加filter对象:logger.addFilter(),为该logger对象添加一个filter对象

Handler类:

Handler对象的作用:将消息分发到handler指定的位置,比如:控制台、文件、网络、邮件等。Logger对象可以 通过addHandler()方法为自己添加多个handler对象。

创建Handler对象:在程序中不应该直接实例化和使用Handler实例,因为Handler是一个基类,它只定义了Handler应该有的接口。应该使用Handler实现类来创建对象

常用的Handler:

logging.StreamHandler:将日志消息发送到输出到Stream,如std.out, std.err或任何 file-like对象。

logging.FileHandler:将日志消息发送到磁盘文件,默认情况下文件大小会无限增长

logging.handlers.RotatingFileHandler:将日志消息发送到磁盘文件,并支持日志文件按大小切割

logging.hanlders.TimedRotatingFileHandler:将日志消息发送到磁盘文件,并支持日志文件按时间切割

logging.handlers.HTTPHandler:将日志消息以GET或POST的方式发送给一个HTTP服务器

logging.handlers.SMTPHandler:将日志消息发送给一个指定的email地址

Handler常用的方法:

handler.setLevel():设置handler将会处理的日志消息的最低严重级别

handler.setFormatter():为handler设置一个格式器对象

handler.addFilter():为handler添加一个过滤器对象

Formatter类:

Formatter对象作用:Formatter对象用于配置日志信息的格式。

创建Formatter对象:formatter = logging.Formatter(fmt=None, datefmt=None, style='%')

说明:

fmt:指定消息格式化字符串,如果不指定该参数则默认使用message的原始值

datefmt:指定日期格式字符串,如果不指定该参数则默认使用"%Y-%m-%d %H:%M:%S"

style:Python 3.2新增的参数,可取值为 '%', '{'和 '$',如果不指定该参数则默认使用'%'

将日志信息同时输出到控制台和文件中:

步骤:

1)创建日志器对象

2)创建控制台处理器对象

3)创建文件处理器对象

4)创建格式化器对象

5)把格式化器添加到处理器中

6)把处理器添加到日志器中

定义日志格式:

#设置日志格式fmt='%(asctime)s%(levelname)s[%(name)s][%(filename)s(%(funcName)s:%(lineno)d)]-%(message)s'#创建格式化器对象formatter=logging.Formatter(fmt)

把日志输出到控制台:

logger=logging.getLogger()sh=logging.StreamHandler()sh.setFormatter(formatter)logger.addHandler(sh)

把日志输出到文件中:

fh=logging.FileHandler("./b.log")fh.setFormatter(formatter)logger.addHandler(fh)

每日生成一个日志文件:

定义Handler对象

fh=logging.handlers.TimedRotatingFileHandler(filename,when='h',interval=1,backupCount=0)#将日志信息记录到文件中,以特定的时间间隔切换日志文件。#filename:日志文件名#when:时间单位,可选参数#S-Seconds#M-Minutes#H-Hours#D-Days#midnight-rolloveratmidnight#W{0-6}-rolloveronacertainday;0-Monday#interval:时间间隔#backupCount:日志文件备份数量。#如果backupCount大于0,那么当生成新的日志文件时,将只保留backupCount个文件,删除最老的文件。

示例代码:

importlogging.handlerslogger=logging.getLogger()logger.setLevel(logging.DEBUG)#日志格式fmt="%(asctime)s%(levelname)s[%(filename)s(%(funcName)s:%(lineno)d)]-%(message)s"formatter=logging.Formatter(fmt)#输出到文件,每日一个文件fh=logging.handlers.TimedRotatingFileHandler("./a.log",when='MIDNIGHT',interval=1,backupCount=3)fh.setFormatter(formatter)fh.setLevel(logging.INFO)logger.addHandler(fh)

日志封装成工具(tools):

定义日志器方法,封装日志

#导包importlogging.handlers#定义日志器方法,封装日志defget_logger():#获取日志器logger,并设置名称adminlogger=logging.getLogger("admin")#设置日志级别为infologger.setLevel(logging.INFO)#获取控制台处理器sh=logging.StreamHandler()#获取文件处理器,根据时间分割th=logging.handlers.TimedRotatingFileHandler(filename="../log/logger.log",when="S",interval=1,backupCount=2,encoding="utf-8")#设置文件处理器日志级别th.setLevel(logging.ERROR)#获取格式器fmt="%(asctime)s%(levelname)s[%(name)s][%(filename)s(%(funcName)s%(lineno)d)]-%(message)s"fm=logging.Formatter(fmt)#将格式器添加到处理器中sh.setFormatter(fm)th.setFormatter(fm)#将处理器添加到日志器中logger.addHandler(sh)logger.addHandler(th)#返回loggerreturnloggerif__name__=='__main__':logger=get_logger()#日志器应用logger.info("这是info信息")logger.warning("这是warning信息")logger.error("这是error信息")

3、日志应用

日志应用:PO模式中,在base操作层、page对象层、scripts业务层都可添加日志,以及将日志用单例模式封装成工具类放在tools文件夹中。

base(在操作层文件中添加日志),以公共方法封装文件 base.py 为例

代码示例:

importtimefromtimeimportsleepfromselenium.webdriver.support.waitimportWebDriverWaitfromday11_tpshopimportpagefromday11_tpshop.base.get_loggerimportGetLogger#获取log日志器log=GetLogger().get_logger()classBase:def__init__(self,driver):log.info("[base:]正在获取初始化driver对象:{}".format(driver))self.driver=driver#查找元素方法封装defbase_find(self,loc,timeout=30,poll=0.5):log.info("[base]:正在定位:{}元素,默认定位超时时间为:{}".format(loc,timeout))#使用显式等待查找元素returnWebDriverWait(self.driver,timeout=timeout,poll_frequency=poll).until(lambdax:x.find_element(*loc))#点击元素方法封装defbase_click(self,loc):log.info("[base]:正在对:{}元素执行点击事件".format(loc))self.base_find(loc).click()#输入元素方法封装defbase_input(self,loc,value):log.info("[base]:正在获取:{}元素".format(loc))#获取元素el=self.base_find(loc)log.info("[base]:正在对:{}元素执行清空操作".format(loc))#输入前清空el.clear()log.info("[base]:正在给{}元素输入内容:{}".format(loc,value))#输入el.send_keys(value)#获取文本信息方法封装defbase_get_text(self,loc):log.info("[base]:正在获取{}元素文本值".format(loc))returnself.base_find(loc).text#截图方法封装defbase_get_img(self):log.info("[base]:断言出错,调用截图")self.driver.get_screenshot_as_file("../image/{}.png".format(time.strftime("%Y_%m_%d%H_%M_%S")))#判断元素是否存在方法封装defbase_elememt_is_exist(self,loc):try:self.base_find(loc,timeout=2)log.info("[base]:{}元素查找成功,存在页面".format(loc))returnTrue#代表元素存在except:log.info("[base]:{}元素查找失败,不存在当前页面".format(loc))returnFalse#代表元素不存在#回到首页(购物车、下订单、支付)都需要用到此方法defbase_index(self):#暂停2秒sleep(2)log.info("[base]:正在打开首页")self.driver.get(page.URL)#切到frame表单方法以元素属性切换defbase_switch_frame(self,element):log.info("[base]:正在切换到frame表单")self.driver.switch_to.frame(element)#回到默认目录方法defbase_default_content(self):log.info("[base]:正在返回默认目录")self.driver.switch_to.default_content()#切换窗口方法defbase_switch_to_window(self,title):log.info("正在执行切换title值为:{}窗口".format(title))self.base_get_title_handle(title)#self.driver.switch_to.window(self.base_get_title_handle(title))#获取指定title页面的handle方法defbase_get_title_handle(self,title):#获取当前页面所有的handleshandles=self.driver.window_handles#遍历handleforhandleinhandles:log.info("正在遍历handles:{}-->{}".format(handle,handles))#切换handleself.driver.switch_to.window(handle)log.info("切换:{}窗口".format(handle))#获取当前页面title并判断是否等于指定参数titlelog.info("条件成立!返回当前handle{}".format(handle))ifself.driver.title==title:#返回handlereturnhandle

page(在对象层文件中添加日志):以登录为例,page_login.py

代码示例:

fromday11_tpshopimportpagefromday11_tpshop.base.baseimportBasefromday11_tpshop.base.get_loggerimportGetLogger#获取log日志器log=GetLogger().get_logger()classPageLogin(Base):#点击登录链接defpage_click_login_link(self):log.info("[page_login]:执行{}元素点击链接操作".format(page.login_link))self.base_click(page.login_link)#输入用户名defpage_input_username(self,username):log.info("[page_login]:对{}元素输入用户名{}操作".format(page.login_username,username))self.base_input(page.login_username,username)#输入密码defpage_input_pwd(self,pwd):log.info("[page_login]:对{}元素输入密码{}操作".format(page.login_pwd,pwd))self.base_input(page.login_pwd,pwd)#输入验证码defpage_input_verify_code(self,verify_code):log.info("[page_login]:对{}元素输入验证码{}操作".format(page.login_verify_code,verify_code))self.base_input(page.login_verify_code,verify_code)#点击登录defpage_click_login_btn(self):log.info("[page_login]:执行{}元素点击操作".format(page.login_btn))self.base_click(page.login_btn)#获取错误提示信息defpage_get_err_info(self):returnself.base_get_text(page.login_err_info)#点击错误提示框确定按钮defpage_click_error_alert(self):log.info("[page_login]:执行{}元素点击操作".format(page.login_err_ok_btn))self.base_click(page.login_err_ok_btn)#判断是否登录成功defpage_if_login_success(self):#注意一定要将找元素的结果返回,True:存在returnself.base_elememt_is_exist(page.login_logout_link)#点击安全退出defpage_click_logout_link(self):self.base_click(page.login_logout_link)#判断是否退出成功defpage_if_logout_success(self):returnself.base_elememt_is_exist(page.login_link)#组合业务方法登录业务直接调用defpage_login(self,username,pwd,verify_code):log.info("[page_login]:正在执行登录操作,用户名:{},密码:{},验证码:{}".format(username,pwd,verify_code))self.page_input_username(username)self.page_input_pwd(pwd)self.page_input_verify_code(verify_code)self.page_click_login_btn()#组合登录业务方法给(购物车模块、订单模块、支付模块)依赖登录使用defpage_login_success(self,username="13800001111",pwd="123456",verify_code="8888"):#点击登录链接self.page_click_login_link()log.info("[page_login]:正在执行登录操作,用户名:{},密码:{},验证码:{}".format(username,pwd,verify_code))self.page_input_username(username)self.page_input_pwd(pwd)self.page_input_verify_code(verify_code)self.page_click_login_btn()

scripts(在业务层文件中添加日志):以登录业务为例,test01_login.py

代码示例:

importunittestfromday11_TPshop项目.base.get_driverimportGetDriverfromday11_TPshop项目.base.get_loggerimportGetLoggerfromday11_TPshop项目.page.page_loginimportPageLoginfromday11_TPshop项目.tools.read_txtimportread_txtfromparameterizedimportparameterized#获取log日志器log=GetLogger().get_logger()defget_data():arrs=[]fordatainread_txt("login.txt"):arrs.append(tuple(data.strip().split(",")))returnarrs[1:]#新建登录测试类并继承unittest.TestCaseclassTestLogin(unittest.TestCase):#新建setUpClass@classmethoddefsetUpClass(cls)->None:try:#实例化并获取drivercls.driver=GetDriver.get_driver()#实例化PageLogin()cls.login=PageLogin(cls.driver)#点击登录链接cls.login.page_click_login_link()exceptExceptionase:#截图cls.login.base_get_img()#日志log.error("错误:{}".format(e))#新建tearDownClass@classmethoddeftearDownClass(cls)->None:#关闭driver驱动对象GetDriver.quit_driver()#新建登录测试方法@parameterized.expand(get_data())deftest_login(self,username,pwd,verify_code,expect_result,status):try:#调用登录业务方法self.login.page_login(username,pwd,verify_code)#判断是否为正向ifstatus=="true":#断言是否登录成功try:self.assertTrue(self.login.page_if_login_success())exceptExceptionase:#截图self.login.base_get_img()#日志log.error("出错了:{}".format(e))#点击安全退出self.login.page_click_logout_link()#点击登录链接self.login.page_click_login_link()#逆向用例else:#获取错误提示信息msg=self.login.page_get_err_info()print("msg:",msg)try:self.assertEqual(msg,expect_result)exceptExceptionase:#截图self.login.base_get_img()#日志log.error("出错了:{}".format(e))#点击错误提示框确定按钮self.login.page_click_error_alert()exceptExceptionase:#截图self.login.base_get_img()#日志log.error("出错了:{}".format(e))

tools(存放封装的日志工具 get_logger.py):使用单例封装logger日志对象

代码示例:

importlogging.handlersimporttimeclassGetLogger:logger=None#获取logger@classmethoddefget_logger(cls):ifcls.loggerisNone:#获取logger日志器并设置名称为“admin”cls.logger=logging.getLogger("admin")#设置日志级别cls.logger.setLevel(logging.INFO)#获取控制台处理器sh=logging.StreamHandler()#获取文件处理器根据时间分割th=logging.handlers.TimedRotatingFileHandler(filename="../log/{}.log".format(time.strftime("%Y_%m_%d%H_%M_%S")),when="S",interval=1,backupCount=3,encoding="utf-8")#设置文件处理器日志级别th.setLevel(logging.ERROR)#获取格式器fmt="%(asctime)s%(levelname)s[%(name)s][%(filename)s%(funcName)s%(lineno)d]-%(message)s"fm=logging.Formatter(fmt)#将格式器添加到处理器sh.setFormatter(fm)th.setFormatter(fm)#将处理器添加到日志器cls.logger.addHandler(sh)cls.logger.addHandler(th)#返回日志器returncls.loggerif__name__=='__main__':logger=GetLogger.get_logger()#日志器应用logger.info("这是info日志信息")logger.debug("这是debug日志信息")logger.warning("这是warning日志信息")logger.error("这是error日志信息")

log(保存日志文件)(内容略)

自动化测试学习视频

面试资料

我们学习软件测试必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有阿里大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。

这些资料,对于做【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!凡事要趁早,特别是技术行业,一定要提升技术功底。希望对大家有所帮助…….

加油吧,测试人!如果你需要提升规划,那就行动吧,在路上总比在起点观望的要好。

未来的你肯定会感谢现在拼命的自己!

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。