Flask子域名
一般用于數量比較少的子域名,一個模塊對應一個子域名。先看下面一個例子:
modules.py:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
from flask import Blueprint public = Blueprint( 'public' , __name__) @public .route( '/' ) def home(): return 'hello flask' app.py: app = Flask(__name__) app.config[ 'SERVER_NAME' ] = 'example.com' from modules import public app.register_blueprint(public, subdomain = 'public' ) |
現在可以通過public.example.com/來訪問public模塊了。
通配符子域
通配符子域,即通過一個模塊來匹配很多個子域名。比如某些網站提供的個性化域名功能,就是這種形式。
先來看段示例代碼:
modules.py:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
from flask import Blueprint member = Blueprint( 'member' , __name__) @member .route( '/' ) def home(): return g.subdomain app.py: app = Flask(__name__) app.config[ 'SERVER_NAME' ] = 'example.com' from modules import member app.register_blueprint(member, subdomain = '<subdomain>' ) |
這段代碼和上一節的第像,不同之處是這里的subdomain使用了動態參數<subdomain>(路由中的URL變量也是這種方式)。我們可以用這個參數在請求回調函數之前利用的組合的url處理器來獲取相關的用戶。這樣我們就可以通過*.example.com的形式來訪問member模塊了。
下面是為任何Flask或Blueprint對象增加子域名支持的便捷函數:
1
2
3
4
5
6
7
8
9
10
|
def add_subdomain_to_global(endpoint, values): g.subdomain = values.pop( 'subdomain' , None ) def add_subdomain_to_url_params(endpoint, values): if not 'subdomain' in values: values[ 'subdomain' ] = g.subdomain def add_subdomain_support(app): app.url_value_preprocessor(add_subdomain_to_global) app.url_defaults(add_subdomain_to_url_params) |
然后你可以使用before_request回調函數來處理子域名:
1
2
3
4
5
6
7
|
add_subdomain_support(blueprint) @blueprint .before_request def add_user_to_global(): g.user = None if g.subdomain: g.user = User.query.filter_by(username = g.subdomain).first_or_404() |
注:這里的blueprint請改為實際對象。
特別說明:通配符子域調試不是不太方便,需要做泛域名解析才可以。修改hosts文件來指定域名的方法是不可行的(子域名較少時可以逐個添加,子域名多了就不太現實了)。本機調試時,可以安裝DNS服務器(比如LINUX BIND服務等),并做好泛域名解析,然后再進行調試。當然使用公網域名和服務器來調試也未嘗不可。
PS:
1.如果某個blueprint默認就需要實用a.domain.com,那么在定義blueprint時候:
1
|
a = Blueprint(‘a ' ,__name__,subdomain=' a') |
這樣,該bp下面的所有url routing走的都是a.domain.com/xxx
2.在某個具體的url routing定義時,如果需要實用a.domain.com,那么這么寫:
1
2
3
|
@www .route(‘ / hello ',methods=[' GET ',' POST '],subdomain=' a') def xxx(): ….. |
3.我在具體實踐中,默認的routing都是走的www,這是在__init__.py中:
1
|
app.url_map.default_subdomain = 'www' |
其實就是設置默認子域名,這樣默認不做設置的話,路由走的就是www。
那這個時候如果訪問domain.com,即不帶www的話,就會報404了,怎么辦呢,我是在nginx層面解決這個問題的,在nginx.conf增加一個server:
1
2
3
4
|
server { server_name domain.com; rewrite ^(.*) http://www.domain.com$1 permanent; } |