首先说明下这里的从0到1指的是从没有听说过pylons到开发出一个看上去还不错的网站。一个月前,我没有听说过也不知道什么是pylons,HTML只知道一些标签,JavaScript也不怎么懂,由于只倾向于做后端,对前端没有什么兴趣,一个偶然的机会做了一个系统,感觉前端也挺好玩的。我仅以我的经历介绍下经验。

 ------------------------------

HTML + javaScript

Pylons架构

MySQL

 ------------------------------

那就开始吧^_^

1、 首先介绍下pylons的安装:

这里不包括错误的处理过程,错误的处理过程详见这里的几篇文章:


参考pylons官方文档,更详细的介绍也请看这里:

http://docs.pylonsproject.org/projects/pylons-webframework/en/latest/gettingstarted.html

1>  下载go-pylons.py

$ curl http://pylonshq.com/download/1.0/go-pylons.py

2>  运行脚本,并指定虚拟化环境要被安装的目录:(这里是安装在mydevenv目录下)

$ python go-pylons.py mydevenv

或者为了进一步隔离系统的python库,添加--no-site-packages选项:

$ python go-pylons.py--no-site-packages mydevenv

这一步就会生成虚拟化环境并安装pylons。

3>  激活虚拟化环境:这一步很重要!

$ sourcemydevenv/bin/activate

4>  下载pylons源码:

$ hg clone http://bitbucket.org/bbangert/pylons/

官方文档上的这个链接失效了,建议在下面的链接手动下载,(可以使用wget)

pylons官网http://www.pylonsproject.org/projects/pylons-framework/download

或者github: https://github.com/Pylons/pylons

5>  告诉setuptools使用pylons目录的新版本:

$ cd pylons

$ python setup.py develop

2、 创建一个pylons工程helloworld:

$ paster create -t pylons helloworld

在运行这个命令的时候会让你选择:

1>  使用哪种模板类型?

我选择的默认类型mako(默认mako)

2>  是否包括SQLAlchemy支持?

因为我所要做的网站用到数据库,所以True.(默认False).

填完True.如果没有错误就恭喜你已经创建了一个helloworld工程。

但是你很有可能会遇到错误,因为我就遇到了。错误的原因是系统的SQLAlchemy的问题,需要系统已经安装了SQLAlchemy,安装方法:

We’ll assume you’ve already installed Pylons and have the easy_install command. At the command line, run:

easy_install SQLAlchemy

然后安装数据库引擎:

easy_install pysqlite # If you use SQLite and Python 2.4 (notneeded for Python 2.5)

easy_install MySQL-python # If you use MySQL

easy_install psycopg2 # If you use PostgreSQL

See the Python Package Index(formerly the Cheeseshop)for other database drivers.

Tip: Checking Your Version

To see which version of SQLAlchemy you have, go to aPython shell and look at

sqlalchemy.__version__ :

>>>import sqlalchemy

>>>sqlalchemy.__version__

0.5.8

说明你的SQLAlchemy已经安装成功,版本是0.5.8

3>运行helloworld

如果没有选择SQLAlchemy的话,就不用在development.ini里配置db的信息了,但是如果你选择了,请一定要配置DB信息。

完成了上面的配置,然后启动服务:

$ cd helloworld
$ paster serve --reload development.ini

在浏览器打开http://127.0.0.1:5000/就可以看到welcome信息了。

下面创建一个基本的helloworld应用:

$ paster controller hello

打开helloworld/controllers/hello.py,就能看到默认的文件内容:

 

import logging
from pylons import request, response, session, tmpl_context as c, url
from pylons.controllers.util import abort, redirect
from helloworld.lib.base import BaseController, render
log = logging.getLogger(__name__)
class HelloController(BaseController):
    def index(self):
        # Return a rendered template
        #return render(’/hello.mako’)
        # or, Return a response
        return ’Hello World’

用浏览器打开http://127.0.0.1:5000/hello/index

 

就能看到一个显示helloworld的网页,至此我们的helloworld已经完成。

3、 Helloworld的结构:

[root@host /data/helloworld]$tree

|-- MANIFEST.in

|-- README.txt

|-- development.ini

|-- docs

|-- ez_setup.py

|-- helloworld

|-- helloworld.egg-info

|-- setup.cfg

|-- setup.py

`-- test.ini

这里比较重要的就是development.ini文件和helloworld目录。setup.py我用的系统生成的,如果真要修改这个文件等到进阶篇吧。

关于development.ini文件:

这里面比较常用的设置:

网站的地址和端口:

 

[server:main]
use = egg:Paste#http
 host =127.0.0.1
port = 5000

 

数据库的信息:

 

# SQLAlchemy database URL
sqlalchemy.url =mysql://username:password@host:port/database
sqlalchemy.pool_recycle = 3600

 

还有log信息

关于helloworld子目录下面会详细讲。

4、 helloworld子目录的结构:

[root@host /data/helloworld/helloworld]$tree

|-- __init__.py

|-- __init__.pyc

|-- config

|   |--__init__.py

|   |--__init__.pyc

|   |--deployment.ini_tmpl

|   |--environment.py

|   |--environment.pyc

|   |--middleware.py

|   |--middleware.pyc

|   |--routing.py

|   `--routing.pyc

|-- controllers

|   |--__init__.py

|   `--error.py

|-- lib

|   |--__init__.py

|   |--__init__.pyc

|   |--app_globals.py

|   |--app_globals.pyc

|   |--base.py

|   |--helpers.py

|   `--helpers.pyc

|-- model

|   `--__init__.py

|-- public

|   |--bg.png

|   |--favicon.ico

|   |--index.html

|   `--pylons-logo.gif

|-- templates

|-- tests

|   |--__init__.py

|   |--functional

|  |   `-- __init__.py

|   `--test_models.py

`-- websetup.py

这个目录里面比较重要的文件和目录有:config/routing.py, controllers子目录,model子目录,template子目录,public子目录。

关于routing.py这个文件,主要是一个映射的功能,将URL请求映射到处理请求的方法。(A Python library called Routes handles mapping URLs tocontrollers and their methods, or their action as Routes refers to them)

一般格式为:

map.connect('/{controller}/{action}')

或者

map.connect('/{controller}/{action}/{id}')

例如routing.py中的这一行:

map.connect('/save-template',controller='Notifylist',action='SaveTemplate')

这句话就是前端发来的save-template请求,处理这个请求的就是Notifylist 这个Controller中的SaveTemplate这个函数。

关于controllers这个子目录,包含controllers文件。比如上面的Notifylist.py这个文件。我们来看一下Notifylist.py中的一个代表性的函数def index(self):

 

def index(self):
       #return a rendered template
       return render('/helloworld.mako')
       #or, return a string
       #return "hello world"

在routing.py中有这个一行:

 

 

map.connect('/index',controller='Notifylist',action='index')

你就应该明白了当浏览器请求index的时候,通过routing这个路由,就转到index这个函数来处理,返回helloworld.mako这个文件。

 

关于model这个子目录,包含各个数据库表文件,比如有个person表,创建一个person.py模块。person.py文件的内容大致如下,Person类中一般有两个函数:__init__和__repr__。如果有其他的表类内容也相差不大。

 

"""Person model"""
from sqlalchemy import Column
from sqlalchemy.types import Integer, String
from myapp.model.meta import Base
 
class Person(Base):
    __tablename__= "person"
    id = Column(Integer, primary_key=True)
    name = Column(String(100))
    email = Column(String(100))
    def __init__(self, name=’’, email=’’):
        self.name = name
        self.email = email
    def __repr__(self):
        return "<Person(’%s’)" % self.name

关于 public 这个子目录 ,主要是一些静态的文件,如html,js,css,png等等。

 

关于templates这个子目录,主要是mako文件,mako文件和平时的html文件一样。

5、 到这里pylons的初级篇已经讲得差不多了。

6、 下面讲讲在mako文件中进行数据库请求的过程吧。以保存到数据库为例。

1.  button请求:

保存请求一般会来自html中的save button,例如这个id为btn_save_schemaid_info的button:

 

<button id= "btn_save_schemaid_info"type="button">保存</button>

 

2.  在javascript中就会有处理这个button的点击的函数:

$(btn_save_schemaid_info).click(

...
$.ajax('/save-schema-info-by-id?id=' +schema_id_btn
           + '&type=' + schema_type_btn,{
           'success':function(text){
           alert("success:" + text);
           },
           'error':function(text){
           alert("error:" + text);
            }})
...
);

在这个处理btn_save_schemaid_info的函数中,产生了一个ajax请求,save-schema-info-by-id,这个ajax请求有两个参数:id和type,都带有相应的值。

3.  再看看routing中的这个map:

map.connect('/save-schema-info-by-id',controller='Notifylist',action='SaveSchemaInfoById')

在routing中,这个请求又转给了Notifylist这个controller中的SaveSchemaInfoById这个函数。

4.  Notifylist这个controller中的SaveSchemaInfoById函数

 

def SaveSchemaInfoById(self):
       if "id" in request.params:
            id = int(request.params['id'])
       if "type" in request.params:
            type = int(request.params['type'])
       if id > 0:
            a = self.tablename.filter_by(id =id).first()
            if a:
                a.type = type
                Session.commit()
                return "保存成功"
            else:
                return "此id不存在"
       return "参数错误."

这样就保存到数据库里面了。从这个请求的全过程我们就可以看到pylons架构中MVC各个模块的明显了。多么优越啊~~~

 

7、 关于json数据格式特殊字符问题.

1>  处理方法就是在发送ajax请求时,对含有特殊格式的字符序列进行两次编码:

encodeURIComponent(encodeURIComponent(data))

2>  如果里面含有中文,需要再在Controller里面:

urllib2.unquote(str(request.params['data']))

3>  读出来的时候controller再进行逆操作:

urllib2.quote(a.template.encode("utf8"))

4>  最后解码:

decodeURIComponent(data)

 



以上简单介绍了Pylon的各个部分 (setup and configuration, routing, models, controllers, 和 templates) ,希望你已经对 Pylons 的强大有所认识, 你会发现用 Pylons 创建复杂的应用程序是多么的容易 !