跨過Nginx上基於uWSGI部署Django項目的坑

来源:http://www.cnblogs.com/qingspace/archive/2017/05/10/6838747.html
-Advertisement-
Play Games

先說說他們的關係,Nginx和uWSGI都是Web伺服器,Nginx負責靜態內容,uWSGI負責Python這樣的動態內容,二者配合共同提供Web服務以實現提高效率和負載均衡等目的。uWSGI實現了多個協議,如WSGI,HTTP協議,還有它自己的uwsgi協議,想瞭解更多關於uWSGI和uwsgi協... ...


先說說他們的關係,Nginx和uWSGI都是Web伺服器,Nginx負責靜態內容,uWSGI負責Python這樣的動態內容,二者配合共同提供Web服務以實現提高效率和負載均衡等目的。uWSGI實現了多個協議,如WSGI,HTTP協議,還有它自己的uwsgi協議,想瞭解更多關於uWSGI和uwsgi協議內容可以查閱這裡。這樣和fastcgi類似,請求和響應的流程如下:

Request > Nginx > uWSGI > Django > uWSGI > Nginx > Response

請求先交由Nginx,如果是靜態內容就自己處理了,如果是動態內容就交給uWSGI伺服器,uWSGI伺服器處理整個Django項目的Python代碼,響應請求,原路返回,但是與fastcgi不同,Nginx、uWSGI和Django可以獨立部署,然後整合。那麼我們從Django開始,這裡的伺服器環境是Ubuntu 16.10。

1. 部署Django的項目

安裝Python和Django,Ubuntu自帶2.7和3.5版本的Python,安裝相應的Django版本,註意在Ubuntu中不同版本Python都有相應的命令

www@cloud-vm-ub01:~$ python --version
Python 2.7.12+
www@cloud-vm-ub01:~$ python3 --version
Python 3.5.2+
www@cloud-vm-ub01:~$ pip -V
pip 9.0.1 from /home/wisesoe/.local/lib/python2.7/site-packages (python 2.7)
www@cloud-vm-ub01:~$ pip3 -V
pip 9.0.1 from /home/wisesoe/.local/lib/python3.5/site-packages (python 3.5)

pip3 install django

將已經完成開發的Django項目pro(pro是Django項目名)拷貝到伺服器,這裡拷貝到了www用戶(www是伺服器可登錄用戶名)路徑下,最後相對路徑是~/work/project/pro,絕對路徑是/home/www/project/pro

進入以上目錄,使用Django的內置伺服器測試看看pro項目是否運行正常。

python3 ./manage.py runserver 127.0.0.1:8080

2. 部署uWSGI伺服器

通過pip安裝uWSGI。

pip3 install uwsgi

測試uWSGI是否正常,在~/work/project/pro目錄中創建一個測試用的Python文件uwsgi_test.py

def application(env, start_response):
        start_response('200 OK',[('Content-Type', 'text/html')])
        #return ['Hello world'] # Python2
        return [b'Hello world'] # Python3

在pro項目路徑下,基於HTTP協議運行uWSGI,如果uWSGI安裝正常的話,可以在瀏覽器中訪問9090埠,看到Hello world字樣

uwsgi --http 127.0.0.1:9090 --wsgi-file uwsgi_test.py

接下來啟動uWSGI載入Django項目,這裡依然使用HTTP協議,將指向具體Python文件--wsgi-file參數替換為指向Django項目的--module參數,參數的值pro.wsgi指向~/work/project/pro/pro/wsgi.py模塊,如果正常可以在瀏覽器http://127.0.0.1:9090埠打開了項目,但是靜態文件路徑有問題,不過沒關係後面再處理。

www@cloud-vm-ub01:~/work/project/pro$ uwsgi --http 127.0.0.1:9090 --module pro.wsgi

對於uWSGI伺服器的配置,如上命令加上很多參數非常麻煩,可以寫成配置文件的方式,在~/work/project/pro中創建一個配置文件uwsgi.ini,註釋掉參數暫時忽略,Django 1.4以前的版本需要配置如env,pythonpath等參數,這裡不再深究了。

其中http參數用於以上測試,而與Nginx交互需要使用socket參數,即使用TCP協議,WSGI和uwsgi協議都在TCP協議之上。socket參數也可以配置為網路地址,如socket=127.0.0.1:7070,但如果Nginx和uWSGI同在一個伺服器上,可以使用socket文件的形式。chmod-socket是為了動態配置socket文件的許可權,因為socket文件會在每次uWSGI啟動時被重新創建。

[uwsgi]
http=127.0.0.1:8000
#socket=/home/www/work/project/pro/nginx_uwsgi.socket
chdir=/home/www/work/project/pro/
#chmod-socket=664
master=true
processes=4
threads=2
module=pro.wsgi
#wsgi-file=uwsgi_test.py
#stats=127.0.0.1:9000

通過下麵命令同樣可以啟動uWSGI載入Djiango項目

uwsgi --ini uwsgi.ini

3. 部署Nginx伺服器

通過apt安裝Nginx

sudo apt install nginx

Nginx可以通過以下命令控制。正常安裝和啟動Nginx後,通過http://127.0.0.1:80可以看到Nginx的歡迎頁

sudo service nginx start
sudo service nginx stop
sudo service nginx restart

接下來修改配置Nginx配置與uWSGI伺服器交互。Nginx的主要配置文件在/etc/nginx/nginx.conf和sites-enabled文件夾里,nginx.conf是全局設置,sites-enabled文件夾里的可以針對不同站點進行配置,其中有個預設的default配置文件,該文件其實是sites-available文件夾里的default文件的軟鏈接,sites-avaliable像個倉庫,但只有sites-enabled里的才有效。我們可以將sites-enabled的default刪除,再cp一份sites-available的default到sites-enabled里重名為nginx-pro,同時cp /etc/nginx/uwsgi_params ~/work/project/pro里以備nginx-pro配置文件調用。

#nginx-pro
 
upstream django{
        server unix:///home/wisesoe/Work/Project/Python/duty/nginx_uwsgi.sock; # file socket
        #server 127.0.0.1:7070; # TCP socket
}


server {
        listen 80 default_server;
        listen [::]:80 default_server;
        root /var/www/html;
        index index.html index.htm index.nginx-debian.html;
        server_name 127.0.0.1; # IP or FQDN

        location /static {
                alias /home/www/work/project/pro/static;
        }

        location / {
                uwsgi_pass django;
                include /home/www/work/project/pro/uwsgi_params;
                #try_files $uri $uri/ =404;
        }
}

uwsgi_params文件是Nginx向uWSGI傳遞的參數,uwsgi_pass的意思動態內容請求都通過名為django的upstream傳遞給uWSGI,這使用文件socket的方式,那麼與之前uwsgi.ini里的socket參數配置一致。

4. Nginx許可權問題

以上全部配置完成了,但是還有一個重要的許可權問題,如果啟動uWSGI和Nginx(以下需要兩個終端視窗,因為uwsgi命令會占據一個),會報錯

uwsgi --ini uwsgi.ini
sudo service nginx restart

在/var/log/nginx/error.log中會看到Permission denied字樣,是對home/www/work/project/pro/nginx_uwsgi.socket文件沒有讀寫許可權,即運行Nginx工作進程的用戶需要socket文件的讀寫許可權。

運行Nginx的工作進程的用戶在/etc/nginx/nginx.conf中有配置,是user的值www-data,但查看/etc/group發現www-data是個用戶組

user www-data;
worker_processes auto;
pid /run/nginx.pid;

events {
        worker_connections 768;
        # multi_accept on;
}

我們可以將www用戶加入該用戶組

usermod -G www-data www

也可以將socket文件及其上級目錄pro的用戶組改為www-data,併為該用戶組授予讀寫許可權

chown :www-data ~/home/work/project/pro
chown :www-data ~/home/work/project/pro/nginx_uwsgi.socket
chmod g+rw ~/home/work/project/pro/nginx_uwsgi.socket

5.Nginx和Django靜態文件處理

Django項目可以正常打開,但是靜態文件引用路徑還有問題,在Django開發時Django自己可以正確處理靜態文件的路徑,但是部署後Nginx去無法找到靜態文件路徑。

檢查Nginx配置文件夾sites-enabled里的nginx-pro文件,確保裡面預設的try_files要刪掉或者註釋掉,否則Nginx會因此檢查靜態文件是否存在。

將Django的靜態文件集中起來,Django為此有專門的工具

現在Django的Settings文件中加上StATIC_ROOT,把靜態文件都集中到這個路徑下

STATIC_ROOT = os.path.join(BASE_DIR, "static/")

執行命令

python3 ./manage.py collectstatic

這樣所有Django前後臺的靜態文件都會集中到項目文件夾pro下static中,另外nginx-pro其中一個配置location /static即可讓Nginx來處理靜態內容。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、 官網下載mysql5.5 下載地址: http://dev.mysql.com/downloads/mysql/5.5.html#downloads 2、 安裝mysql5.5 註意,安裝之前,請關閉殺毒軟體。 (1) 打開下載的mysql-5.5.53-winx64.msi (2) 點擊下一 ...
  • " 1、靜態數據字典 " "1.1、實用靜態數據字典" "1.2、運用靜態數據字典" " 2、動態數據字典 " "2.1、實用動態性能視圖" "2.2、運用動態性能視圖" " 3、死鎖 " "3.1、定位死鎖" "3.2、解鎖方法" "3.3、強制刪除已連接用戶" " 4、總結 " 數據字典是 Or ...
  • PXC三節點安裝: node1:10.157.26.132 node2:10.157.26.133 node3:10.157.26.134 配置伺服器ssh登錄無密碼驗證 ssh-keygen實現三台主機之間相互免密鑰登錄,保證三台主機之間能ping通 1)在所有的主機上執行: # ssh-keyg ...
  • 背景簡介: 本文為針對一次windows平臺RAC資料庫遷移至Linux平臺RAC的筆記,基本步驟為: 1.搭建windows RAC到Linux 單實例資料庫的DataGuard 2.做switchover,將備庫IP修改為原RAC資料庫的scanip 3.搭建單實例到Linux RAC的Data ...
  • ZooKeeper是一個分散式開源框架,提供了協調分散式應用的基本服務,它向外部應用暴露一組通用服務——分散式同步(Distributed Synchronization).命名服務(Naming Service).集群維護(Group Maintenance)等,簡化分散式應用協調及其管理的難度, ...
  • ElasticSearch 2.4版本支持Java正則表達式查詢,但是,在對大段的文本(Text Block)進行挖掘之前,必須瞭解正則表達式查詢的特殊之處。由於分析器會對文本欄位進行分詞,移除停用詞,小寫轉換等操作,最終存儲在倒轉索引中的是小寫的標記流(Token Stream),預設情況下,每一... ...
  • 以下摘自官方文檔: 語法: Or: If you declare an alias for a table, you must use the alias when referring to the table: Correct: ...
  • 一、最小化安裝 1、進入系統之後,要配置network網路。 首先ping www.baidu.com (Ctrl+z 推出正在執行的命令) 如果ping不通,則修改: vi /etc/sysconfig/network-scripts/ifcfg-ens33 ONBOOT=yes 修改之後重啟ne ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...