使用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 24, 2018 - 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
出現下面的網頁,說明成功了。
安裝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/
如圖所示,表示項目啓動成功
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
這裏,先不要執行關閉命令。
由於uwsgi定義的端口是8001。如圖所示,表示項目啓動成功
若是出現(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的基礎上,針對大訪問量網站的需求,添加了不少高級功能和特性。主要特性,請查看官網:
從官網下載最新版本。目前最新穩定版本是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訪問便可。
出現如下頁面,說明成功了!
進入/www/mysite1/static/目錄,建立3個目錄
cd /www/mysite1/static/
mkdir css
mkdir js
mkdir images
上傳一個圖片到images目錄
網頁訪問圖片
http://192.168.11.103/static/images/zly.jpg
測試一下,表單提交
由於這裏尚未數據庫,直接使用寫入文件方式來存儲數據。
準備靜態文件
下載Bootstrap,官方網址爲:
下載最新穩定版本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/
默認有一條數據
添加2條數據
最終效果以下: