bottle.py中的SimpleTemplate
import re
class SimpleTemplate(object):
re_block = re.compile(r'^\s*%\s*((if|elif|else|try|except|finally|for|while|with).*:)\s*$')
re_end = re.compile(r'^\s*%\s*end(.*?)\s*$')
re_code = re.compile(r'^\s*%\s*(.*?)\s*$')
re_inc = re.compile(r'\{\{(.*?)\}\}')
def __init__(self, template):
self.code = "\n".join(self.compile(template))
self.co = compile(self.code, '<string>', 'exec')
def render(self, **args):
''' Returns the rendered template using keyword arguments as local variables. '''
args['stdout'] = []
args['__builtins__'] = __builtins__
eval(self.co, {}, args) # 执行Python代码
return ''.join(args['stdout'])
def compile(self, template):
'''
将模板转换为可执行的Python代码
'''
def code_str(level, line, value): # 可直接输出的字符串
value = "".join(value)
value = value.replace("'","\'").replace('\\','\\\\')
return ' '*level + "stdout.append(r'''%s''')" % value
def code_print(level, line, value): # {{...}}中的内容,需要执行代码后取得字符串再输出
return ' '*level + "stdout.append(str(%s)) # Line: %d" % (value.strip(), line)
def code_raw(level, line, value): # 以%开头,需要作为代码执行
return ' '*level + value.strip() + ' # Line: %d' % line
level = 0 # 缩进深度
ln = 0 # 行号(无具体作用,方便定位问题)
sbuffer = [] # 模板中的普通字符串
for line in template.splitlines(True):
ln += 1
# Line with block starting code
m = self.re_block.match(line)
if m:
if sbuffer:
yield code_str(level, ln, sbuffer)
sbuffer = []
if m.group(2).strip().lower() in ('elif','else','except','finally'):
if level == 0:
raise TemplateError('Unexpected end of block in line %d' % ln)
level -= 1
yield code_raw(level, ln, m.group(1).strip())
level += 1
continue
# Line with % end marker
m = self.re_end.match(line)
if m:
if sbuffer:
yield code_str(level, ln, sbuffer)
sbuffer = []
if level == 0:
raise TemplateError('Unexpected end of block in line %d' % ln)
level -= 1
continue
# Line with % marker
m = self.re_code.match(line)
if m:
yield code_raw(level, ln, m.group(1).strip())
continue
# Line with inline code
lasts = 0
for m in self.re_inc.finditer(line):
sbuffer.append(line[lasts:m.start(0)])
yield code_str(level, ln, sbuffer)
sbuffer = []
lasts = m.end(0)
yield code_print(level, ln, m.group(1))
if lasts:
sbuffer.append(line[lasts:])
continue
# Stupid line
sbuffer.append(line)
if sbuffer:
yield code_str(level, ln, sbuffer)
if __name__ == "__main__":
# 简单替换
t1 = SimpleTemplate('Hello {{name}}!')
print t1.render(name='<b>World</b>')
# 嵌入python语句(必须返回str)
t2 = SimpleTemplate('Hello {{name.title() if name else "stranger"}}!')
print t2.render(name=None)
print t2.render(name="tuzkee")
# %执行Python语句
tmp3="""
%if name:
Hi <b>{{name}}</b>
%else:
<i>Hello stranger</i>
%end"""
t3 = SimpleTemplate(tmp3)
print t3.render(name=None)
tmp4="""
%for i in range(3):
%for j in range(3):
{{"%s:%s"%(i,j)}}
%end
%end"""
t4 = SimpleTemplate(tmp4)
print t4.render()
bottle.py中的SimpleTemplate的更多相关文章
- bottle.py中的路由搜索优化
# Now search regexp routes # ROUTES_REGEXP是一个字典,键是请求方法,值是[路由, 处理函数]的列表 # 例如:{"GET", [[路由1, ...
- bottle.py中的路由解析代码
# Routing def compile_route(route): """ Compiles a route string and returns a precomp ...
- Python的Bottle框架中实现最基本的get和post的方法的教程
这篇文章主要介绍了Python的Bottle框架中实现最基本的get和post的方法的教程,Bottle框架在Python开发者中的人气很高,需要的朋友可以参考下 1.GET方式: # -*- cod ...
- python调用py中rar的路径问题。
1.python调用py,在py中的os.getcwd()获取的不是py的路径,可以通过os.path.split(os.path.realpath(__file__))[0]来获取py的路径. 2. ...
- Django 中 如何使用 settings.py 中的常量
在用django 框架开发 python web 程序的时候 , 在模板页面经常会用到 settings.py 中设置的常量,比如MEDIA_URL, 我尝试过在模板页面用类似如下的方式 程序代码 { ...
- 配置python+mod_wsgi+apache 时 在浏览器中访问服务器时报错:Invalid HTTP_HOST header: 'XXXXX'. You may need to add u'XXXXX' to ALLOWED_HOSTS,在setting.py中添加‘*”无效的原因
配置python+mod_wsgi+apache 时 在浏览器中访问服务器时报错:Invalid HTTP_HOST header: 'XXXXX'. You may need to add u'XX ...
- [django]urls.py 中重定向
Django 1.5 有时候需要对一个链接直接重定向,比如首页啥的重定向到一个内容页等等,在views.py 中可以设定,如果没有参数啥的在urls.py 中设定更加方面 from django.vi ...
- 不在models.py中的models
概述 如何让你定义的model不在models.py中 在app的models目录中的models 你新建一个app后这个models.py就会自动建立,里面只有几行代码.那么如果是一个中大型项目,每 ...
- javascript中的值如何传递到django下的views.py中或者数据库中?
用Ajax,Ajax有很多种写法,包括JQuery和JS,这里贴一个用JQuery写的最通用的Ajax,POST方法传递JSON格式数据: $.ajax({ url: "your url&q ...
随机推荐
- java程序可以跨平台运行的原因
java有虚拟机(JVM),JAVA程序不是直接在电脑上运行的,是在虚拟机上进行的,每个系统平台都是有自己的虚拟机(JVM),所以JAVA语言能跨平台. 1, java代码不是直接运行在CPU上,而是 ...
- win10无法搜索本地文件,修复方法?
win10无法搜索本地文件,实在太不方便了,网上查了一圈没几个方法有效的,筛选出来2个成功解决的问题,具体是哪个起到作用,不太清楚,都放上来,大家自行选择! 方法1:按“Windows+ X”后选择“ ...
- Aspose.Words的Merge Field
今天应客户要求,修改导出word模板.使用的是Aspose.Words插件.这个程序原是同事所写,且自己对Aspose不是很了解.在替换模板上花费了一些时间. 先来一张图:下图是原来的模板.现在要求删 ...
- 站在Web3.0 理解IPFS是什么
尽管网络上,已经有不少文章讨论IPFS,不过真正讲明白IPFS想做什么的很少,文本尝试站在未来Web3.0的高度来看看IPFS究竟用来解决什么问题. DApp 的缺陷 对区块链有所了解的同学,知道区块 ...
- HotSpot虚拟机对象探秘-笔记
学习目的:探讨HotSpot虚拟机在Java堆中对象分配.布局和访问的全过程. 1.对象的创建 虚拟机在执行到一条new指令时,先要检查指令的参数(将要实例化的类)是否已经被加载.解析.初始化过,如果 ...
- java 形式参数和实际参数的区别
1.形参不能离开方法.形参只有在方法内才会发生作用,也只有在方法中使用,不会在方法外可见.而实参可以再程序的任何地方都使用.
- extern "C"的用法
文章转自开源电子论坛:http://www.openedv.com/forum.php?mod=viewthread&tid=7986 看一些程序的时候老是有 “#ifdef __cplusp ...
- 牛客网第9场多校E(思维求期望)
链接:https://www.nowcoder.com/acm/contest/147/E 来源:牛客网 题目描述 Niuniu likes to play OSU! We simplify the ...
- CloudStack 云计算平台框架
前言 CloudStack 和OpenStack 一样都是IaaS层 开源框架,可以管理XenServer.ESXI.KVM.OVM等主流虚拟机,相对OpenStack比较简单.稳定: 二.Cloud ...
- 梳理:python—同一个类中的方法调用
为什么突然在此提到这个梳理问题呢? 因为在自己实践综合练习学过的知识时,突然觉得有些知识点的运用总是不成功,于是翻过课本进行回顾,总是觉得是对的,可是当再进一步思考“既然是对的,为什么在程序中总是不成 ...