Flask 备注一(单元测试,Debugger, Logger)
Flask 备注一(单元测试,Debugger, Logger)
Flask是一个使用python开发Web程序的框架。依赖于Werkzeug提供完整的WSGI支持,以及Jinja2提供templates支持。Flask的设计理念是提供Micro以及方便的框架。"Micro"是因为除了提供基本特性功能的实现外,其他的功能(例如数据库访问)都是通过extension来完成。方便的特点是因为提供了简单易用的必要特性和功能,提供的功能包含:
- 内置的开发服务器和调试工具。
- 集成单元测试支持。
- 支持Templates(依赖JinJa2提供)
- 完整的WSGI支持(依赖Werkzeug提供)
- 安全Cookie和Sessions
- Thread-Locals,例如在每个Request里面的对象只在线程里面有意义。
Templates
Flask使用jinja2作为templates引擎。jinjia2的默认的语法设定如下:
- {%. .. %} 包含statements(逻辑处理代码块)
- {{ ... }} 包含expressions (输出到tempalte的语句)
- {# ... #} 包含comments (注释,不会输出到template的语句)
- # ... ## 包含单行statements (单行处理代码块)
单元测试
Flask通过test_client提供了非常简单的方式来创建测试的上下文环境(Local Context)。这样可以通过各种方式来完成单元测试,最基本的方式是通过python自带的unittest来完成集成的测试用例。
unittest
使用unittest的Flask单元测试的框架如下:
import os
import xxxx
import unittest
import tempfile
class XxxxTestCase(unittest.TestCase):
def setUp(self):
self.db_fd, xxxx.app.config['DATABASE'] = tempfile.mkstemp()
xxxx.app.config['TESTING'] = True
self.app = xxxx.app.test_client()
xxxx.init_db()
def tearDown(self):
os.close(self.db_fd)
os.unlink(xxxx.app.config['DATABASE'])
if __name__ == '__main__':
unittest.main()
在上述例子里面:
- xxxx是一个自定义的简单的flask程序,xxxx.app是一个Flask类的对象,对应一个Web应用。
- 在setUp函数里面,将Flask的Web应用的TESTING参数配置为True,并创建了一个对应的测试客户端(test_client),包含测试需要的上下文环境。
- 在setUp函数里面,创建了xxxx的数据库文件并进行了初始化。
在tearDown函数里面,关闭了数据库文件并删除。
在unittest中测试用例的函数必须哟以test单词开始,例如testFunc1
,这样测试框架可以将此函数作为测试用例执行。执行测试用例之前先执行setUp函数进行初始化,之后执行tearDown进行资源释放。基本的测试用例如下:def testemptydb(self):
rv = self.app.get('/')
assert 'No entries here so far' in rv.data
Local Context
unittest框架通过setUp创建了testclient以及需要的上下文环境,并且在tearDown函数里面进行销毁。同样可以使用testclient在任意代码中完成验证测试。因为test_client创建了一个临时的上下文环境,在任何区域中都可以通过with语句是此context在范围内有效,因此在这个范围内可以验证request以及session的信息。
验证request信息的简单例子:
with app.test_client() as c:
rv = c.get('/?number=42')
assert request.args['number'] == '42'
验证session信息的简单例子:
with app.test_client() as c:
rv = c.get('/')
assert flask.session['foo'] == 42
如果需要修改testclient创建的临时context中的session信息,可以通过sessiontranscation来获取session对象进行修改。如下简单实例:
with app.test_client() as c:
with c.session_transcation() as sess:
sess['a_key'] = 'a value'
Debug&Logging
在程序中,错误永远无法避免,错误有可能是代码逻辑问题,服务器问题,网络问题或者是硬件问题,环境问题等等。Flask提供了两种方式定位问题,其一可以打开程序的调试模式,通过调试器(debugger)跟踪程序的执行信息;另外就是Flask提供了完善的日志系统,记录程序的运行信息。
Debug
Flask通过必要的参数设置,来确定是否使用Debug模式以及是否使用自带的调试器。这些参数包含:
- debug。True: 设置Debug模式; False: 非Debug模式
- use_debugger。True: 使用内部调试器; Flase: 不实用内部调试器
- use_reloader。True: 在Excpetion时,是否reload和fork当前进程进行调试; False: Nothing.
如果使用第三方类似于Aptana/Eclipse等调试器,需要设置debug为True,usedebugger和usereloader为False。
通过内置的debugger,当程序出现exception时,在错误界面提供一个交互式的界面,而且在这个界面里面可以执行任意的代码进行程序调试。这样存在着巨大的安全隐患,因此永远不要在产品服务器上开启调试模式。
Logging
Flask附带的Logger依赖于Python内置的日志系统,通过默认Logging库设置日志的处理Handler,日志Format以及处理的Level。程序中通过Flask的Logger所写的Log通过系统自带的日志系统进行过滤和格式化,然后输出到所设置的Handler中。
handler可以是文件也可以是邮件,一般情况下的应用场景是,将大部分的日志信息保存到文件中,将重点需要关注的日志信息发送到邮件中。
文件Handler。文件Handler包含四种:
- FileHandler,对应文件系统的一个文件。
- RotatingFileHandler, 对应文件系统的一个文件,在输出一定数量的信息之后,重头开始。
- NTEventLogHandler,对应Windows操作系统的日志系统。
- SysLogHandler,对应Unix Syslog系统。
邮件Handler。通常使用smtphandler进行邮件发送。
Handler的简单示例如下:
if not app.debug:
import logging
from logging.handler import SMTPHandler, FileHandlerfile_handler = FileHandler(/var/test/flask.log)
file_handler.setLevel(logging.WARNING) email_handler = SMTPHandler('127.0.0.1',
'server-error@example.com'
ADMINS,
'Your Application Failed!!!')
app.logger.addHandler(file_handler)
app.logger.addHandler(email_handler)
Formatter是Python自带的日志系统提供的,可以对Handler进行设置需要保存的信息格式。一般情况下,在邮件Handler中保存详细的多行文本信息;在文件Handler里面保存单行重要信息。Log的格式参数如下:
- %(levelname)s || 日志级别(包含:'DEBUG', 'INFO', 'WARNING', 'ERROR', 'CRITICAL') ||
- %(pathname)s || Log产生的源文件路径 ||
- %(filename)s || Log产生的源文件名称 ||
- %(module)s || Log产生的模块 ||
- %(funcName)s || Log产生的函数名称 ||
- %(lineno)d || Log产生的所在文件行号 ||
- %(asctime)s || Log产生的时间格式是:YYYY-MM-DD hh-mm-ss, mmm(,之后是毫秒数)
- %(message)s || Log的信息 ||
Email日志格式设置的简单示例:
from logging import Formatter
mail_handler.setFormatter(Formatter('''
Message type: %(levelname)s
Location: %(pathname)s:%(lineno)d
Module: %(module)s
Function: %(funcName)s
Time: %(asctime)s
Message: %(message)s
'''))
文件日志格式设置的简单示例:
from logging import Formatter
file_handler.setFormatter(Formatter(
'%(asctime)s %(levelname)s: %(message)s '
'[in %(pathname)s:%(lineno)d]'
))
Flask中使用的第三库都有自己的日志策略,同样可以使用getlogger获取每个依赖库的logger设置统一的Handler。如下示例:
from logging import getLogger
loggers = [app.logger, getLogger('sqlalchemy'),
getLogger('otherlibrary')]
for logger in loggers:
logger.addHandler(mail_handler)
logger.addHandler(file_handler)
Flask 备注一(单元测试,Debugger, Logger)的更多相关文章
- Flask备注二(Configurations, Signals)
Flask备注二(Configuration, Signals) Flask是一个使用python开发Web程序的框架.依赖于Werkzeug提供完整的WSGI支持,以及Jinja2提供templat ...
- Flask备注4(Structure)
Flask备注4(Structure) package 通过Flask可以非常简单的通过一个module(一个py文件)创建一个简单的application.这种简单程序的文件结构如下: /youra ...
- Flask备注三(Context)
Flask备注三(Context) Flask支持不同的应用场景下,对应不同的local context(本地上下文环境),用来提供当前环境下的资源.lcoal context和全局变量以及局部变量最 ...
- Flask学习之七 单元测试
英文博客地址:http://blog.miguelgrinberg.com/post/the-flask-mega-tutorial-part-vii-unit-testing 中文翻译地址:http ...
- flask的日志输出current_app.logger.debug
环境部署方式:nginx+supervisord+gunicorn在/etc/supervisord.conf中配置日志的输出路径stdout_logfile=/home/admin/workspac ...
- 如何在Flask的构架中传递logger给子模块
Logger的传递 作为一个新手,如何将主函数的logger传入子模块是一件棘手的事情.某些情况下可以直接将logger作为参数传入子模块的构造函数中,但倘若子模块与主模块存在相互依赖的关系则容易出现 ...
- flask logger
Flask uses standard Python logging. All Flask-related messages are logged under the 'flask' logger n ...
- Flask 教程 第七章:错误处理
本文翻译自The Flask Mega-Tutorial Part VII: Error Handling 这是Flask Mega-Tutorial系列的第七部分,我将告诉你如何在Flask应用中进 ...
- Flask —— 使用Python和OpenShift进行即时Web开发
最近Packtpub找到了我,让我给他们新出版的关于Flask的书写书评.Flask是一个很流行的Python框架.那本书是Ron DuPlain写的<Flask 即时Web开发>.我决定 ...
随机推荐
- MS - 2 - 设计包含 min 函数的栈
定义栈的数据结构,要求添加一个 min 函数,能够得到栈的最小元素. 要求函数 min.push 以及 pop 的时间复杂度都是 O(1). template<typename T> st ...
- sed命令
sed是一个很好的文件处理工具,本身是一个管道命令,主要是以行为单位进行处理,可以将数据行进行替换.删除.新增.选取等特定工作,下面先了解一下sed的用法sed命令行格式为: sed ...
- Orcle基本语句(三)
COMMIT; --查询表内所有内容 SELECT * FROM stu_info; --查询部分列,并赋予别名 SELECT stu_id 学生标号,stu_name 学生姓名 FROM stu_i ...
- closest应用(向上查找最近的资料)
//新增网址信息function AddSitInfo(a) { var obj = $(a).closest("td[name='POIRestaurant_Reference_UR ...
- [小哥Allegro72讲速成视频]
http://v.qq.com/vplus/df932a993679cf80a0b6c87bb849e22c 第01讲 Allegro常用组件介绍 视频链接:http://v.qq.com/boke/ ...
- table变色
<!DOCTYPE html><html><head> <meta http-equiv="Content-Type" content=& ...
- Java画图程序设计
本文讲述一个画图板应用程序的设计,屏幕抓图如下: 『IShape』 这是所有图形类(此后称作模型类)都应该实现接口,外部的控制类,比如画图板类就通过这个接口跟模型类“交流”.名字开头的I表示它是一个接 ...
- select标签非空验证,第一个option value=""即可
select标签非空验证,第一个option value=""即可,否则不能验证
- Unable to load performance pack. Using Java I/O instead的解决办法?
启动的时候报下面的错误: <BEA-000438> <Unable to load performance pack. Using Java I/O instead. Please ...
- SQL Server 2008R2数据库文件导入到SQL Server 2008数据库中
最近,电脑重装系统之后,安装了SQL Server 2008.附加数据库文件的时候,发现无法附加,提示版本不对.想起来,原来的数据库版本是SQL Server 2008R2.低版本的数据库管理工具无法 ...