web.py 项目架构分析之 blog

Published: 18 Nov 2013 Category: 源代码阅读

web.py 项目之 blog

目录树

src/
├── blog.py
├── model.py
├── schema.sql
└── templates
    ├── base.html
    ├── edit.html
    ├── index.html
    ├── new.html
    └── view.html

1 directory, 8 files

项目说明

项目来自 webpy.org, 主要实现一个基于web的博客系统,实现了基本的增删查改。

结构分析

控制模块

控制模块包括 blog.py

""" Basic blog using webpy 0.3 """
import web
import model

### Url mappings

urls = (
    '/', 'Index',
    '/view/(\d+)', 'View',
    '/new', 'New',
    '/delete/(\d+)', 'Delete',
    '/edit/(\d+)', 'Edit',
)


### Templates
t_globals = {
    'datestr': web.datestr
}
render = web.template.render('templates', base='base', globals=t_globals)


class Index:

    def GET(self):
        """ Show page """
        posts = model.get_posts()
        return render.index(posts)


class View:

    def GET(self, id):
        """ View single post """
        post = model.get_post(int(id))
        return render.view(post)


class New:

    form = web.form.Form(
        web.form.Textbox('title', web.form.notnull, 
            size=30,
            description="Post title:"),
        web.form.Textarea('content', web.form.notnull, 
            rows=30, cols=80,
            description="Post content:"),
        web.form.Button('Post entry'),
    )

    def GET(self):
        form = self.form()
        return render.new(form)

    def POST(self):
        form = self.form()
        if not form.validates():
            return render.new(form)
        model.new_post(form.d.title, form.d.content)
        raise web.seeother('/')


class Delete:

    def POST(self, id):
        model.del_post(int(id))
        raise web.seeother('/')


class Edit:

    def GET(self, id):
        post = model.get_post(int(id))
        form = New.form()
        form.fill(post)
        return render.edit(post, form)


    def POST(self, id):
        form = New.form()
        post = model.get_post(int(id))
        if not form.validates():
            return render.edit(post, form)
        model.update_post(int(id), form.d.title, form.d.content)
        raise web.seeother('/')


app = web.application(urls, globals())

if __name__ == '__main__':
    app.run()

可以看出,主要的逻辑,即对各个请求的处理,都在这个模块中, 包括表单的生成,数据的获取,以及对模板引擎的调用。这个 模块主要是在顶层调用,具体的细节不在这里实现。

显示模块

显示模块没有具体实现逻辑的代码,只有模板引擎用到的一些HTML 文件,这些文件的大体内容已经定下来,只是具体需要填充的内容 需要在控制模块中进行设定。

templates/base.html

$def with (page)

<html>
<head>
    <title>My Blog</title>
    <style>
        #menu {
            width: 200px;
            float: right;
        }
    </style>
</head>
<body>

<ul id="menu">
    <li><a href="/">Home</a></li>
    <li><a href="/new">New Post</a></li>
</ul>

$:page

</body>
</html>

base.html 为每个文件提供都基本的模板,其它文件只负责自己的那 一块内容。

templates/index.html


$def with (posts)

<h1>Blog posts</h1>

<ul>
$for post in posts:
    <li>
        <a href="/view/$post.id">$post.title</a> 
        from $datestr(post.posted_on) 
        <a href="/edit/$post.id">Edit</a>
    </li>
</ul>

index.html 负责文章列表的显示。

templates/new.html


$def with (form)


<h1>New Blog Post</h1>
<form action="" method="post">
$:form.render()
</form>

new.html 负责新建文件页面。

templates/edit.html


$def with (post, form)

<h1>Edit $form.d.title</h1>

<form action="" method="post">
$:form.render()
</form>


<h2>Delete post</h2>
<form action="/delete/$post.id" method="post">
    <input type="submit" value="Delete post"/>
</form>

edit.html 负责文章的编辑页面。

templates/view.html

$def with (post)

<h1>$post.title</h1>
$datestr(post.posted_on)<br/>

$post.content

view.html 负责显示文章页面。

数据模块

数据模块包含两个文件 schema.sqlmodel.py

schema.sql

CREATE TABLE entries (
    id INT AUTO_INCREMENT,
    title TEXT,
    content TEXT,
    posted_on DATETIME,
    primary key (id)
);

schema.sql 包含了数据库中表的格式

model.py


import web, datetime

db = web.database(dbn='mysql', db='miniblog', user='root', pw="7102155")

def get_posts():
    return db.select('entries', order='id DESC')

def get_post(id):
    try:
        return db.select('entries', where='id=$id', vars=locals())[0]
    except IndexError:
        return None

def new_post(title, text):
    db.insert('entries', title=title, content=text, posted_on=datetime.datetime.utcnow())

def del_post(id):
    db.delete('entries', where="id=$id", vars=locals())

def update_post(id, title, text):
    db.update('entries', where="id=$id", vars=locals(),
        title=title, content=text)

model.py 包含了数据库连接的创建与数据库操作的具体细节。 此模块中的函数被控制模块调用,从数据库中读取数据。

总的来说,控制模块在最顶层,负责对请求进行处理,调用数据模块, 从数据库中读取,写入数据。再将从数据库中取出的数据送给显示模块 进行显示。

感觉项目再大一点就应该把控制模块再分割了。

comments powered by Disqus