Django + Uwsgi + Nginx 的生產環境部署

    使用runserver可使咱們的django項目很便捷的在本地運行起來,但這隻能在局域網內訪問,若是在生產環境部署django,就要多考慮一些問題了。好比靜態文件處理,安全,效率等等,本篇文章總結概括了一下基於uwsgi+Nginx下django項目生產環境的部署javascript


    準備條件:php

    linux上已部署好python環境,且已安裝好項目所需的模塊css

    安裝python環境,請參考如下連接html

    http://www.py3study.com/Article/details/id/320.html java


    建立django項目node

    [root@localhost ~]# cd /www/
    [root@localhost www]# django-admin startproject mysite1
    [root@localhost www]# cd mysite1
    [root@localhost mysite1]# python manage.py startapp blog
    [root@localhost mysite1]# mkdir static
    #編輯配置文件
    [root@localhost mysite1]# vim mysite1/settings.py
    #容許全部IP,注意:'*'必須用引號包起來
    ALLOWED_HOSTS = ['*']

    若是提示-bash: django-admin: 未找到命令python

    請使用命令pip3 install django 安裝jquery


    啓動項目,監聽本機全部IP的8001端口linux

    [root@localhost mysite1]# python manage.py runserver 0.0.0.0:8001
    Performing system checks...
    System check identified no issues (0 silenced).
    You have 14 unapplied migration(s). Your project may not work properly until you apply the migrations for app(s): admin, auth, contenttypes, sessions.
    Run 'python manage.py migrate' to apply them.
    June 242018 - 03:48:20
    Django version 2.0.6, using settings 'mysite1.settings'
    Starting development server at http://0.0.0.0:8001/
    Quit the server with CONTROL-C.

    訪問頁面:
    nginx

    http://192.168.11.103:8001/

    出現下面的網頁,說明成功了。

    blob.png


    安裝uwsgi

    uwsgi是python的一個模塊,安裝uwsgi只需簡單的pip命令就能夠了

    pip3 install uwsgi

    若是提示:

    You should consider upgrading via the 'pip install --upgrade pip' command.

    使用命令:pip3 install --upgrade pip 進行升級


    基於uwsgi+django的實現

    1.使用命令啓動uwsgi

    先關閉上面啓動的Django項目,使用Ctrl+c,就能夠取消。

    第一步:進入django項目

    cd /www/mysite1/

    第二步:命令測試啓動

    uwsgi --http 0.0.0.0:8080 --file mysite1/wsgi.py --static-map=/static=static

    參數說明:

    --http 這個就和runserver同樣指定IP 端口

    --file 這個文件就裏有一個反射,若是你在調用他的時候沒有指定Web Server就使用默認的

        注意:mysite1是一個相對路徑。--file它的絕對路徑是/www/mysite1/mysite1/wsgi.py

    --static 作一個映射,指定靜態文件。


    此時,訪問http://192.168.11.103:8080/

    如圖所示,表示項目啓動成功

    blob.png



    2.使用uwsgi配置文件啓動django項目

    使用Ctrl+c,取消uwsgi啓動。

    第一步:在django項目同級目錄建立script目錄,用於存放配置腳本等等

    mkdir script

    項目結構以下:

    mysite1/
    ├── blog
    │   ├── admin.py
    │   ├── apps.py
    │   ├── __init__.py
    │   ├── migrations
    │   │   └── __init__.py
    │   ├── models.py
    │   ├── tests.py
    │   └── views.py
    ├── db.sqlite3
    ├── manage.py
    ├── mysite1
    │   ├── __init__.py
    │   ├── settings.py
    │   ├── urls.py
    │   └── wsgi.py
    ├── script
    └── static

    第二步:進入script目錄,建立一個uwsgi.ini文件

    cd script

    vim uwsgi.ini

    編輯uwsgi.ini文件內容以下:

    # uwsig使用配置文件啓動
    [uwsgi]
    # 項目目錄
    chdir=/www/mysite1/
    # 指定項目的application
    module=mysite1.mysite1.wsgi:application
    # 指定sock的文件路徑
    socket=/www/mysite1/mysite1.sock
    # 進程個數
    workers=5
    pidfile=/www/mysite1/script/uwsgi.pid
    # 指定IP端口
    http=0.0.0.0:8001
    # 指定靜態文件
    static-map=/static=/www/mysite1/static
    # 啓動uwsgi的用戶名和用戶組
    uid=root
    gid=root
    # 啓用主進程
    master=true
    # 自動移除unix Socket和pid文件當服務中止的時候
    vacuum=true
    # 序列化接受的內容,若是可能的話
    thunder-lock=true
    # 啓用線程
    enable-threads=true
    # 設置自中斷時間
    harakiri=30
    # 設置緩衝
    post-buffering=8192
    # 設置日誌目錄
    daemonize=/www/mysite1/script/uwsgi.log
    wsgi-file = /www/mysite1/mysite1/wsgi.py


    注意:

    chdir的目錄要正確,目錄後面要加斜槓

    module的配置,要特別當心

    mysite1.mysite1.wsgi這一句表示mysite1項目下的mysite1目錄下的wsgi.py文件。

    不少教程都是這樣寫的mysite1.wsgi:application

    那是由於他們直接將uwsgi.ini放到和manage.py在同一級目錄。

    可是我建立了script目錄,須要將uwsgi.ini放到script目錄。因此路徑必須多加一層才行!


    啓動項目:

    注意:必須切換到script目錄

    cd /www/mysite1/script

    uwsgi --ini uwsgi.ini

    它會輸出以下信息:

    [uWSGI] getting INI configuration from uwsgi.ini
    [uwsgi-static] added mapping for /static => /www/mysite1/static

    若是須要關閉項目,使用命令:

    cd /www/mysite1/script

    uwsgi --stop uwsgi.pid


    這裏,先不要執行關閉命令。

    訪問http://192.168.11.103:8001/

    由於uwsgi定義的端口是8001。如圖所示,表示項目啓動成功


    blob.png

    若是出現(HTTP/1.1 500) 錯誤,請仔細檢查

    chdir,module,socket,wsgi-file 這幾項配置是否正確!



    安裝tengine

    說到tengine,首先仍是得說下nginx了,你們對於nginx並不陌生,對於基本的需求都能知足,若是是涉及高級性能,那麼就必須使用商用版nginx plus了,一談到商用,你們就特別敏感,有沒有開源免費的呢,有的,因此tengine誕生了。

        Tengine(http://tengine.taobao.org/index_cn.html)是由淘寶網發起的Web服務器項目。它在Nginx的基礎上,針對大訪問量網站的需求,添加了不少高級功能和特性。主要特性,請查看官網:

    http://tengine.taobao.org/


    從官網下載最新版本。目前最新穩定版本是2.2.2,下載連接爲:

    http://tengine.taobao.org/download/tengine-2.2.2.tar.gz


    安裝依賴包

    yum install -y gcc gcc-c++ autoconf automake pcre pcre-devel openssl openssl-devel

    解壓安裝

    tar zxvf tengine-2.2.2.tar.gz -C /usr/src/

    cd /usr/src/tengine-2.2.2/

    ./configure --prefix=/usr/local/tengine --with-http_sub_module --with-http_stub_status_module --with-http_gzip_static_module

    make && make install


    新建用戶和組

    groupadd www

    useradd -g www -s /sbin/nologin www

    進入tengine目錄,備份配置文件,編輯配置文件

    cd /usr/local/tengine/conf

    mv nginx.conf nginx.conf.bak

    vim nginx.conf

    完整內容以下:

    user  www www;
    worker_processes  4;  # cpu核心數
    
    error_log  logs/error.log;
    #error_log  logs/error.log  notice;
    #error_log  logs/error.log  info;
    
    pid        logs/nginx.pid;
    
    
    events {
        worker_connections  60240;
    #接受盡量多的鏈接
        multi_accept on;
    #網絡I/O模型
        use epoll;
    }
    
    #進程打開的最多文件描述符數目
    worker_rlimit_nofile 65535;
    
    http {
        include       mime.types;
        default_type  application/octet-stream;
    
        #隱藏版本號
        server_tokens off;
        keepalive_timeout 30;
        sendfile on;
        tcp_nopush on;
        tcp_nodelay on;
        gzip on;
        gzip_min_length 1000;
        gzip_comp_level 9;
        gzip_proxied any;
        gzip_types text/plain text/css text/xml
                   application/x-javascript application/xml
                   application/atom+xml text/javascript
               application/x-httpd-php image/jpeg
               image/gif image/png;
    
        #log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
        #                  '$status $body_bytes_sent "$http_referer" '
        #                  '"$http_user_agent" "$http_x_forwarded_for"';
        log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
                          '$status $body_bytes_sent "$http_referer" '
                          '"$http_user_agent" "$http_x_forwarded_for"'
                           '$upstream_response_time $request_time ';
    
        access_log  logs/access.log  main;
        #緩存打開的文件描述符
        open_file_cache max=100000 inactive=20s;
        #多長時間檢查一次緩存的有效信息
        open_file_cache_valid 30s;
        #open_file_cache指令中的inactive參數時間內文件的最少使用次數
        open_file_cache_min_uses 2;
        #指定是否在搜索一個文件是記錄cache錯誤
        open_file_cache_errors on;
    
        #容許客戶端請求的最大單文件字節數
        client_max_body_size 64M;
        #緩衝區代理緩衝用戶端請求的最大字節數
        client_body_buffer_size  432K;
        #設定請求緩衝
        client_header_buffer_size 16k;
        #指定客戶端的響應超時時間
        send_timeout 60;
        #可經過keep-alive鏈接的客戶端請求數
        keepalive_requests 100000;
        large_client_header_buffers 4 64k;
        proxy_connect_timeout 300s;
        proxy_read_timeout 300s;
        proxy_send_timeout 300s;
        proxy_buffer_size 128k;
        proxy_buffers 32 32k;
        proxy_busy_buffers_size 128k;
        proxy_temp_file_write_size 128k;
        proxy_ignore_client_abort on;
    
        server_names_hash_bucket_size 512;
    
        #虛擬主機配置文件目錄
        include vhosts/*;
    
    }

    建立虛擬目錄

    mkdir vhosts

    cd vhosts/

    編輯配置文件mysite.conf

    vim mysite.conf

    內容以下:

    server {
        listen 80;
        server_name localhost;
    
        # 指定項目路徑uwsgi
        location / {
            include uwsgi_params; # 導入一個Nginx模塊他是用來和uWSGI進行通信的
            uwsgi_connect_timeout 30# 設置鏈接uWSGI超時時間
            uwsgi_pass unix:/www/mysite1/mysite1.sock; # 指定uwsgi的sock文件全部動態請求就會直接丟給他
        }
    
        # 指定靜態文件路徑
        location /static/ {
            alias /www/mysite1/static/;
        }
    
    }

    判斷配置文件是否有錯誤,並啓動nginx

    [root@localhost vhosts]# ../../sbin/nginx -t
    nginx: the configuration file /usr/local/tengine/conf/nginx.conf syntax is ok
    nginx: configuration file /usr/local/tengine/conf/nginx.conf test is successful
    [root@localhost vhosts]# ../../sbin/nginx

    訪問首頁,直接IP訪問便可。

    http://192.168.11.103/


    出現如下頁面,說明成功了!

    blob.png

    進入/www/mysite1/static/目錄,建立3個目錄

    cd /www/mysite1/static/

    mkdir css

    mkdir js

    mkdir images


    上傳一個圖片到images目錄

    網頁訪問圖片

    http://192.168.11.103/static/images/zly.jpg


    blob.png


    測試一下,表單提交

    由於這裏尚未數據庫,直接使用寫入文件方式來存儲數據。

    準備靜態文件


    下載Bootstrap,官方網址爲:

    http://www.bootcss.com/

    下載最新穩定版本3.3.7,選擇用於生產環境的

    https://v3.bootcss.com/getting-started/#download


    將壓縮包裏面的bootstrap.min.css放到css目錄

    bootstrap.min.js放到js目錄

    下載jquery:

    https://code.jquery.com/jquery-3.3.1.min.js


    將jquery-3.3.1.min.js放到js目錄


    修改django相關文件


    修改urls.py,增長路徑userInfo 

    vim /www/mysite1/mysite1/urls.py

    from django.contrib import admin
    from django.urls import path
    from blog import views
    
    urlpatterns = [
        path('admin/', admin.site.urls),
        path('userInfo/', views.userInfo),
    ]

    修改views.py,增長視圖函數userInfo

    vim /www/mysite1/blog/views.py

    from django.shortcuts import render,HttpResponse
    import os
    import json
    
    # Create your views here.
    
    def userInfo(req):
        filename = 'userInfo.txt'
        #判斷請求類型
        if req.method == "POST":
            #獲取表單數據,若是獲取不到,則爲None
            username = req.POST.get("username",None)
            password = req.POST.get("password"None)
            email = req.POST.get("email"None)
            # print(username,password,email)
            #定義字典
            user = {'username':username,'password':password,'email':email}
            #追加到列表中
            f = open(filename, 'a', encoding='utf-8')
            f.write(json.dumps(user) + '\n')
    
    
        # 判斷認證文件是否存在,不然自動建立
        if os.path.exists(filename) == False:
            with open(filename, encoding='utf-8', mode='w'as mk:
                #寫入默認數據
                # default =
                mk.write(json.dumps({'username':'xiao','password':'123','email':'123@qq.com'})+'\n')
    
        #讀取文件數據
        f = open(filename, 'r', encoding='utf-8')
        user_list = []  # 定義空列表
        for i in f:
            # print(json.loads(i.strip()))
            user_list.append(json.loads(i.strip()))
        f.close()
        # 將列表傳給模板index.html
        return render(req, "index.html", {"user_list": user_list})

    建立目錄templates

    cd /www/mysite1

    mkdir templates

    修改index.html文件

    vim /www/mysite1/templates/index.html

    內容以下:

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <!--告訴IE使用最新的引擎渲染網頁,chrome=1則能夠激活Chrome Frame-->
     <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"/>
        <!--適用於移動設備,禁止頁面縮放-->
     <meta name="viewport" content="width=device-width, initial-scale=1">
        <!-- Bootstrap -->
     <script src='/static/js/jquery-3.3.1.min.js'></script>
        <link href="/static/css/bootstrap.min.css" rel="stylesheet">
        <!-- 最新的 Bootstrap 核心 JavaScript 文件 -->
     <script src="/static/js/bootstrap.min.js"></script>
    
        <title>Title</title>
        <style>
            .col-center-block {
                float: none;
     display: block;
     margin-left: auto;
     margin-right: auto;
     }
    
        </style>
    </head>
    <body>
    
    <!--正文-->
    <div class="container">
        <div class="row">
            <div class="col-md-6 col-center-block">
                <!--面板-->
                <!--panel-success顯示綠色-->
     <div class="panel panel-success">
                    <div class="panel-heading">
                        <h3 class="panel-title text-center">註冊</h3>
                    </div>
                    <div class="panel-body">
                        <form class="form-horizontal" action="/userInfo/" method="post">
                            <div class="form-group">
                                <label for="inputUser1" class="col-sm-2 control-label">用戶名</label>
                                <div class="col-sm-10">
                                    <input type="text" name="username" class="form-control" id="inputUser1"
     placeholder="請輸入用戶名">
                                </div>
                            </div>
                            <div class="form-group">
                                <label for="inputPassword1" class="col-sm-2 control-label">密碼</label>
                                <div class="col-sm-10">
                                    <input type="password" name="password" class="form-control" id="inputPassword1"
     placeholder="請輸入密碼">
                                </div>
                            </div>
                            <div class="form-group">
                                <label for="inputEmail3" class="col-sm-2 control-label">郵箱</label>
                                <div class="col-sm-10">
                                    <input type="email" name="email" class="form-control" id="inputEmail3"
     placeholder="請輸入郵箱">
                                </div>
                            </div>
                            <div class="form-group">
                                <div class="col-sm-offset-2 col-sm-10">
                                    <button type="submit" class="btn btn-success text-left">註冊</button>
    
                                </div>
                            </div>
                        </form>
    
    
                        {#判斷列表有數據的狀況下#}
                        {% if user_list %}
     <hr/>
                        <h2>數據展現</h2>
                        <!--面板-->
     <div class="panel panel-default">
                            <div class="panel-heading">
                                <h3 class="panel-title">標籤</h3>
                            </div>
                            <div class="panel-body">
                                <table class="table table-striped table-bordered table-hover table-condensed">
                                    <thead>
                                    <tr>
                                        <th>姓名</th>
                                        <th>密碼</th>
                                        <th>郵箱</th>
                                    </tr>
                                    </thead>
                                    <tbody>
                                    {#使用for循環遍歷列表#}
                                    {% for i in user_list %}
     <tr>
                                        {#展現數據#}
     <td>{{i.username}}</td>
                                        <td>{{i.password}}</td>
                                        <td>{{i.email}}</td>
                                    </tr>
                                    {#結束for循環#}
                                    {% endfor %}
    
     </tbody>
                                </table>
                            </div>
                        </div>
                        {#必定要寫結束符#}
                        {% endif %}
    
     </div>
                </div>
    
            </div>
        </div>
    </div>
    
    </body>
    </html>

    修改settings.py

    vim /www/mysite1/mysite1/settings.py

    關閉CSRF

    MIDDLEWARE = [
        'django.middleware.security.SecurityMiddleware',
        'django.contrib.sessions.middleware.SessionMiddleware',
        'django.middleware.common.CommonMiddleware',
        #'django.middleware.csrf.CsrfViewMiddleware',
        'django.contrib.auth.middleware.AuthenticationMiddleware',
        'django.contrib.messages.middleware.MessageMiddleware',
        'django.middleware.clickjacking.XFrameOptionsMiddleware',
    ]

    定義templates目錄

    TEMPLATES = [
        {
            'BACKEND''django.template.backends.django.DjangoTemplates',
            'DIRS': [os.path.join(BASE_DIR, 'templates')],
            'APP_DIRS'True,
            'OPTIONS': {
                'context_processors': [
                    'django.template.context_processors.debug',
                    'django.template.context_processors.request',
                    'django.contrib.auth.context_processors.auth',
                    'django.contrib.messages.context_processors.messages',
                ],
            },
        },
    ]

    定義static目錄

    STATIC_URL = '/static/'
    STATICFILES_DIRS = (
        os.path.join(BASE_DIR,"static"),
    )


    項目解構以下:

    mysite1/
    ├── blog
    │   ├── admin.py
    │   ├── apps.py
    │   ├── __init__.py
    │   ├── migrations
    │   │   └── __init__.py
    │   ├── models.py
    │   ├── tests.py
    │   └── views.py
    ├── db.sqlite3
    ├── manage.py
    ├── mysite1
    │   ├── __init__.py
    │   ├── settings.py
    │   ├── urls.py
    │   └── wsgi.py
    ├── mysite1.sock
    ├── script
    │   ├── uwsgi.ini
    │   ├── uwsgi.log
    │   └── uwsgi.pid
    ├── static
    │   ├── css
    │   │   └── bootstrap.min.css
    │   ├── images
    │   │   └── zly.jpg
    │   └── js
    │       ├── bootstrap.min.js
    │       └── jquery-3.3.1.min.js
    └── templates
        └── index.html

    關閉uwsgi

    cd /www/mysite1/script

    uwsgi --stop uwsgi.pid

    啓動uwsgi

    uwsgi --ini uwsgi.ini


    訪問網頁:

    http://192.168.11.103/userInfo/


    blob.png

    默認有一條數據

    添加2條數據


    最終效果以下:

    blob.png

相關文章
相關標籤/搜索