Flask 项目目录结构

翻译原文:https://lepture.com/en/2018/structure-of-a-flask-project
原文作者:lepture, pallets 小组成员,flask 系列核心开发者

flask 非常灵活,它没有一个固定的项目目录组织结构。这里写的只是我的一些建议

Flask 非常灵活,它可以让有经验的开发人员按照他们自己喜欢来组织项目的目录结构。但是对于新手来说会感到困惑,他们在组织项目目录结构时需要一些指导,并且通常情况下他们会去找一些项目结构示例,但这些示例总不是那么好(甚至很糟糕)。

我并不知道还有这样的问题,直到有人在 Authlib 中提 issue。刚开始我还不能理解问题,后来有人以项目结构的方式向我解释,我终于明白了。

我吓坏了,因为很多文章,样板文件是从项目根 __init__.py 逆向导入模块的:

# project/__init__.py
from flask import Flask
from flask_sqlalchemy import SQLAlchemy

db = SQLAlchemy()

def create_app():
    app = Flask(__name__)
    db.init_app(app)

# project/auth/models.py
from .. import db
class User(db.Model):
    # define columns

代码本身是可以工作的,但当你项目变大时,你将面临一个循环依赖的问题。例如,另一个扩展需要与用户模型一起初始化:

# project/__init__.py
from flask_sqlalchemy import SQLAlchemy
from another_extension import AnotherExtension
from project.auth.models import User

db = SQLAlchemy()
ext = AnotherExtension(User)

糟糕,发生​依赖关系包循环导入问题。因为 auth.models 正在从项目根目录下导入 db,而根目录下不能导入 User module。这是一个常见的循环导入问题,不只限于 flask 。这个问题很容易解决,但是初级开发人员来说可能会很难。所以为什么不在开始的时候就避免它呢。事实上,如果你阅读了官方文档,你会在应用工厂中找到这段代码:

def create_app(config_filename):
    app = Flask(__name__)
    app.config.from_pyfile(config_filename)
    from yourapplication.model import db
    db.init_app(app)
    from yourapplication.views.admin import admin
    from yourapplication.views.frontend import frontend
    app.register_blueprint(admin)
    app.register_blueprint(frontend)
    return app

这里,我们把 db 放在 yourapplication.model 中。

我在编写模块和包时,总是保持这个特定的规则:

不要从根目录下的 __init__.py 中逆向导入

这就是在我发现这个问题之后尽我可能快的提交 ticket 到 Flask 的原因。人们在项目目录结构组织时需要一个指导。这里我分享下我的建议,但是还是要自己思考,不要把我当作金科玉律。

基于功能的结构


设置项目目录结构有很多种方式。一种是按它的功能来组织,例如:

project/
  __init__.py
  models/
    __init__.py
    base.py
    users.py
    posts.py
    ...
  routes/
    __init__.py
    home.py
    account.py
    dashboard.py
    ...
  templates/
    base.html
    post.html
    ...
  services/
    __init__.py
    google.py
    mail.py
    ...

一切都是按功能分组的。如果它的行为像模型,则将它放在 models 目录;如果它的行为像路由,则将它放入 routes 目录。在 project/__init__.py 中创建一个 create_app 工厂函数,并且初始化所有应用 init_app:

# project/__init__.py
from flask import Flask

def create_app()
    from . import models, routes, services
    app = Flask(__name__)
    models.init_app(app)
    routes.init_app(app)
    services.init_app(app)
    return app

这里是我的一个小技巧。在官方文档中,Flask-SQlAlchemy 的 db 是通过这个方式注册的:

from project.models import db
db.init_app(app)

所是我的技巧是在每个目录下的 __init__.py 中定义一个 init_app 函数,并且统一初始化进程:

# project/models/__init__.py
from .base import db

def init_app(app):
    db.init_app(app)

# project/routes/__init__.py
from .users import user_bp
from .posts import posts_bp
# ...

def init_app(app):
    app.register_blueprint(user_bp)
    app.register_blueprint(posts_bp)    

# ...

基于应用的结构


另一个著名的目录结构是基于应用的结构,这意味着按照业务项目的应用程序来分组。例如:

project/
  __init__.py
  db.py
  auth/
    __init__.py
    route.py
    models.py
    templates/
  blog/
    __init__.py
    route.py
    models.py
    templates/
...

每个目录都对应一个应用。Django 默认是使用这种方式来组织目录。当然这并不意味该方式是很好的,你需要按照项目来选择目录结构。某些时候,你将不得不使用一个混合模式。

类似于上面的,你可以像这样来 init_app :

# project/__init__.py
from flask import Flask

def create_app()
    from . import db, auth, blog
    app = Flask(__name__)
    db.init_app(app)
    auth.init_app(app)
    blog.init_app(app)
    return app

配置


加载配置将是另一个问题,我不知道其他人是怎么做的,我只是分享我的解决方案。

  1. 在项目目录下放一个 settings.py 文件,把它当作静态配置。
  2. 从环境变量中加载配置。
  3. create_app 中更新配置。

这是一个配置的基础目录结构:

conf/
  dev_config.py
  test_config.py
project/
  __init__.py
  settings.py
app.py

定义一个 create_app 来加载配置和环境变量:

# project/__init__.py
import os
from flask import Flask

def create_app(config=None)
    app = Flask(__name__)
    # load default configuration
    app.config.from_object('project.settings')
    # load environment configuration
    if 'FLASK_CONF' in os.environ:
        app.config.from_envvar('FLASK_CONF')
    # load app sepcified configuration
    if config is not None:
        if isinstance(config, dict):
            app.config.update(config)
        elif config.endswith('.py'):
            app.config.from_pyfile(config)
    return app

FLASK_CONF 是一个包含配置的 python 文件(包含路径)。这可以是任何你想要的名称, 如项目叫做 Expanse,你可以叫它 Expanse_CONF

我使用 FLASK_CONF 来加载生产环境的配置。

再一次说明,Flask 是非常灵活的,没有固定的模式。在 Flask 中你总是可以找到你所喜欢的。以上只是我的建议方案,不要被任何人蒙住眼睛。

我不喜欢写这样的文章。但有很多错误的指导,我希望这篇文章能得到更好的搜索引擎优化,这样不好的帖子就不会再误导人了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,905评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,140评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,791评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,483评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,476评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,516评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,905评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,560评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,778评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,557评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,635评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,338评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,925评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,898评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,142评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,818评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,347评论 2 342

推荐阅读更多精彩内容