Django+uwsgi+Nginx部署到云服务器

前言

当我们在本地运行Django项目

python3 manager runserver

但是,这只适用于Django的开发模式,只支持单用户访问,要想部署到服务器上供大量用户访问,综合网上的许多资料,在此罗列几种不同的Django部署方案

工具:CentOS7腾讯云服务器、 Python3.6、 MySQL5.7、 Django2.2.7、 uwsgi2.0.18、 Nginx1.16.1

Django(Git)+uwsgi+Nginx

概念

python web开发中,我们经常使用uwsgi配合nginx部署一个web框架,如Django或flask。同时我们又会说,框架和web服务器之间要符合WSGI协议,想更深入理解WSGI和uwsgi的可以看这里

Nginx就是一个web服务器,Django或flask就是web框架,只要web服务器和web框架满足WSGI协议,它们就能相互搭配。所以WSGI只是一个协议,一个约定。而不是python的模块、框架等具体的功能

uWSGI,则是实现了WSGI协议的一个web服务器。即用来接受客户端请求,转发响应的程序。实际上,一个uWSGI的web服务器,再加上Django这样的web框架,就已经可以实现网站的功能了。那为什么还需要Nginx呢?

一个普通的个人网站,访问量不大的话,当然可以由uWSGI和Django构成。但是一旦访问量过大,客户端请求连接就要进行长时间的等待。这个时候就出来了分布式服务器,我们可以多来几台web服务器,都能处理请求。但是谁来分配客户端的请求连接和web服务器呢?Nginx就是这样一个管家的存在,由它来分配。这也就是由Nginx实现反向代理,即代理服务器

概述

首先客户端请求服务资源,
nginx作为直接对外的服务接口,接收到客户端发送过来的http请求,会解包、分析,
如果是静态文件请求就根据nginx配置的静态文件目录,返回请求的资源,
如果是动态的请求,nginx就通过配置文件,将请求传递给uWSGI;uWSGI 将接收到的包进行处理,并转发给wsgi,
wsgi根据请求调用django工程的某个文件或函数,处理完后django将返回值交给wsgi,
wsgi将返回值进行打包,转发给uWSGI,
uWSGI接收后转发给nginx,nginx最终将返回值返回给客户端(如浏览器)
注:不同的组件之间传递信息涉及到数据格式和协议的转换

作用

  1. 第一级的nginx并不是必须的,uwsgi完全可以完成整个的和浏览器交互的流程;
  2. 在nginx上加上安全性或其他的限制,可以达到保护程序的作用;
  3. uWSGI本身是内网接口,开启多个work和processes可能也不够用,而nginx可以代理多台uWSGI完成uWSGI的负载均衡;
  4. django在debug=False下对静态文件的处理能力不是很好,而用nginx来处理更加高效。

步骤

安装Python3

在服务器上安装Python3等依赖

# 进入home目录
cd ~

# 安装相关库
yum -y groupinstall "Development tools"
yum -y install zlib-devel bzip2-devel openssl-devel ncurses-devel sqlite-devel readline-devel tk-devel gdbm-devel db4-devel libpcap-devel xz-devel
yum -y install gcc
yum install -y libffi-devel zlib1g-dev
yum install zlib* -y
 
# 下载python3.6安装包
wget https://www.python.org/ftp/python/3.6.8/Python-3.6.8.tgz
 
# 创建一个文件夹
mkdir /usr/local/python3
 
# 解压安装包
tar -zxvf Python-3.6.8.tgz
 
# 进入解压后的目录
cd Python-3.6.8
 
# 配置,使安装路径为/usr/local/python3.6
# 第一个指定安装的路径,不指定的话,安装过程中可能软件所需要的文件复制到其他不同目录,删除软件很不方便,复制软件也不方便.
# 第二个可以提高python10%-20%代码运行速度. 参考:https://blog.csdn.net/whatday/article/details/98053179
# 第三个是为了安装pip需要用到ssl,后面报错会有提到.
./configure --prefix=/usr/local/python3 --enable-optimizations --with-ssl
 
# 编译,安装 时间较长 
make && make install
 
# 创建软连接
ln -s /usr/local/python3/bin/python3 /usr/local/bin/python3
ln -s /usr/local/python3/bin/pip3 /usr/local/bin/pip3
 
# 更新pip
pip3 install --upgrade pip

安装mysql

参考腾讯云Centos7 安装Mysql5.7

分支deploy

本地创建并切换到分支deploy

git checkout -b deploy

注:撤回commit

git reset --soft HEAD^

提交前注意在Django项目,配置setting.py中设置:

DEBUG = False

ALLOWED_HOSTS = ['你的服务器IP或者域名', 'localhost', '127.0.0.1']

然后提交本地Django代码至github项目的deploy分支

拉取Django项目
git clone https://github.com/xxx/xxx.git

关于在github上 下载源码 clone 非 master 分支的代码

拉取后默认是master分支,需切换到deploy分支

git checkout deploy
安装Django+uwsgi+nginx

这样在你的服务器上就有了Django项目的代码,但是我们还没安装Django等第三方库

接着服务器上安装:

pip3 install django==2.2.7
pip3 install uwsgi

# 通过yum安装nginx
yum install nginx -y

# 创建软连接
ln -s /usr/local/python3/bin/uwsgi /usr/bin/uwsgi
# ln -s /usr/local/python3/bin/virtualenv /usr/bin/virtualenv
# ln -s /usr/local/python3/bin/gunicorn /usr/bin/gunicorn
ln -s /usr/local/python3/bin/django-admin.py /usr/bin/django-admin
测试Django

进入项目根目录

python3 manage.py runserver

我的出现报错:

django.core.exceptions.ImproperlyConfigured: mysqlclient 1.3.13 or newer is required; you have 0.7.11.None.

根据报错安装mysqlclient

pip3 install mysqlclient

又报错:

ERROR: Command errored out with exit status 1:
     command: /usr/local/python3/bin/python3.6 -c 'import sys, setuptools, tokenize; sys.argv[0] = '"'"'/tmp/pip-install-p6v1q25n/mysqlclient/setup.py'"'"'; __file__='"'"'/tmp/pip-install-p6v1q25n/mysqlclient/setup.py'"'"';f=getattr(tokenize, '"'"'open'"'"', open)(__file__);code=f.read().replace('"'"'\r\n'"'"', '"'"'\n'"'"');f.close();exec(compile(code, __file__, '"'"'exec'"'"'))' egg_info --egg-base /tmp/pip-install-p6v1q25n/mysqlclient/pip-egg-info
         cwd: /tmp/pip-install-p6v1q25n/mysqlclient/
    Complete output (10 lines):
    /bin/sh: mysql_config: command not found
    Traceback (most recent call last):
      File "<string>", line 1, in <module>
      File "/tmp/pip-install-p6v1q25n/mysqlclient/setup.py", line 18, in <module>
        metadata, options = get_config()
      File "/tmp/pip-install-p6v1q25n/mysqlclient/setup_posix.py", line 53, in get_config
        libs = mysql_config("libs_r")
      File "/tmp/pip-install-p6v1q25n/mysqlclient/setup_posix.py", line 28, in mysql_config
        raise EnvironmentError("%s not found" % (mysql_config.path,))
    OSError: mysql_config not found
    ----------------------------------------
ERROR: Command errored out with exit status 1: python setup.py egg_info Check the logs for full command output.

找了找资料,原来需要安装mysqlclient所需依赖,略耗时

yum install mysql-devel gcc gcc-devel python-devel

再次安装mysqlclient

pip3 install mysqlclient

成功安装,但是再次runserver运行Django 还是报错:

File "/usr/local/python3/lib/python3.6/site-packages/django/db/backends/mysql/base.py", line 36, in <module>
    raise ImproperlyConfigured('mysqlclient 1.3.13 or newer is required; you have %s.' % Database.__version__)
django.core.exceptions.ImproperlyConfigured: mysqlclient 1.3.13 or newer is required; you have 0.7.11.None.

找了找资料,django2.2和pymysql版本不匹配。mysqldb不支持python3 参考django.core.exceptions.ImproperlyConfigured: mysqlclient 1.3.13 or newer is required; you have 0.9.2

解决办法:

找到Python安装路劲下的Python36-32\Lib\site-packages\django\db\backends\mysql\base.py文件

vi /usr/local/python3/lib/python3.6/site-packages/django/db/backends/mysql/base.py

将文件中的如下代码注释即可

if version < (1, 3, 3):
    raise ImproperlyConfigured("mysqlclient 1.3.3 or newer is required; you have %s" % Database.__version__)

接着

python3 manage.py makemigrations

python3 manage.py migrate

python3 manage.py createsuperuser

python3 manage.py runserver

期间登录超级管理员却出现CSRF验证失败. 请求被中断.

关闭防火墙试试

systemctl stop firewalld.service
附:
# 一.查看防火墙的状态
systemctl status  firewalld

# 或者
firewall-cmd --state

# 二.关闭防火墙
systemctl stop firewalld.service

# 禁止防火墙在开机时启动
systemctl disable firewalld.service

# 三,开启防火墙
systemctl start firewalld.service

# 防火墙在开机时启动
systemctl enable firewalld.service

# 重启防火墙
systemctl restart firewalld.service

然后就可以登录

测试uwsgi

查看安装的uwsgi版本

uwsgi --version

编写测试脚本,可以与Django项目放在同一级路径下

vim test.py
def application(env, start_response):
    start_response('200 OK', [('Content-Type','text/html')])
    return [b"Hello World"]

然后运行

uwsgi --http :8008 --wsgi-file test.py

然后访问:http://服务器IP:8008/

出现Hello World则成功

测试uwsgi+Django

执行如下命令来检验uwsgi是否能与django项目成功结合

uwsgi --http :8008 --chdir /home/LVideo --wsgi-file LVideo.wsgi.py --master --processes 4 --threads 2 --stats 127.0.0.1:9192

常用选项如下所示:

http       :协议类型和端口号
processes  :开启的进程数量
workers    :开启的进程数量,等同于processes(官网的说法是spawn the specified number ofworkers / processes)
chdir      :指定运行目录(chdir to specified directory before apps loading)
wsgi-file  :载入wsgi-file(load .wsgi file)
stats      :在指定的地址上,开启状态服务(enable the stats server on the specified address)
threads    :运行线程。由于GIL的存在,我觉得这个真心没啥用。(run each worker in prethreaded mode with the specified number of threads)
master     :允许主进程存在(enable master process)
daemonize  :使进程在后台运行,并将日志打到指定的日志文件或者udp服务器(daemonize uWSGI)。实际上最常用的,还是把运行记录输出到一个本地文件上。
pidfile    :指定pid文件的位置,记录主进程的pid号。
vacuum     :当服务器退出的时候自动清理环境,删除unix socket文件和pid文件(try to remove all of the generated file/sockets)

注意:--wsgi-file后面跟的是相对目录

访问http://IP:8008成功,出现你的Django项目的页面

参数太多,可以将其写入ini文件中

我是在Django项目的同级目录下创建lvideo_uwsgi.ini文件,写入如下内容(采用字典格式)

我的/home下的文件,LVideo是clone下来的项目:

[root@VM_0_14_centos home]# ls
LVideo  lvideo_uwsgi.ini  test.py  uwsgi.log  uwsgi.pid

新建lvideo_uwsgi.ini

vi lvideo_uwsgi.ini

这是参考:

# lvideo_uwsgi.ini file
[uwsgi]

# Django-related settings
http = :8008
# 真实服务的端口

# Django项目根目录 (绝对路径)
chdir           = /home/LVideo

# wsgi.py文件在项目中的位置
module          = LVideo.wsgi

# process-related settings
# master
master          = true

# 运行的进程数
processes       = 4

# ... with appropriate permissions - may be needed
# chmod-socket    = 664
# clear environment on exit
# 当服务器退出的时候自动清理环境,删除unix socket文件和pid文件
vacuum          = true

# 使进程在后台运行,并将日志打到指定的日志文件或者udp服务器
daemonize = /home/uwsgi.log

# 指定pid文件的位置,记录主进程的pid号
pidfile = /home/uwsgi.pid

# 不记录请求信息的日志,只记录错误以及uWSGI内部消息到日志中
disable-logging = true

你可以复制这里的内容,注意修改chdir、module、daemonize、pidfile:

[uwsgi]

socket          = :8008
chdir           = /home/LVideo
module          = LVideo.wsgi
master          = true
processes       = 4
vacuum          = true
daemonize       = /home/uwsgi.log
pidfile         = /home/uwsgi.pid
disable-logging = true

其中的端口有两种,分为:

http   =:8008  这是在测试通过浏览器访问时,可以成功
socket =:8008  这是为了与nginx配置时需要的

两个颠倒了就会出错,所以浏览器测试完后需要修改回socket

这样保存好后,启动

uwsgi --ini lvideo_uwsgi.ini

若你的ini文件中是http =:8008则可以通过浏览器访问到8008端口,然后注意修改回socket =:8008

因为我们后面要结合nginx

注:这是配置了daemonize后的uwsgi运行成功的样子

[root@VM_0_14_centos home]# uwsgi --ini  lvideo_uwsgi.ini 
[uWSGI] getting INI configuration from lvideo_uwsgi.ini
测试Nginx

前面已经通过yum安装了nginx

查看Nginx版本

nginx -v

然后访问服务器IP就可以看到CentOS的欢迎界面?我TM也不知道为什么不是Nginx的欢迎界面...(好像说是index.html的问题)

看看有没有进程占用80端口

lsof -i:80

这里有

kinsing   11406 root    5u  IPv4 3066592      0t0  TCP VM_0_14_centos:47296->ip255.ip-139-99-50.net:http (ESTABLISHED)
kinsing   11406 root    7u  IPv4 3068905      0t0  TCP VM_0_14_centos:48120->ip255.ip-139-99-50.net:http (ESTABLISHED)
kdevtmpfs 11537 root   15u  IPv4 3021919      0t0  TCP VM_0_14_centos:46146->45.89.230.240:http (ESTABLISHED)
kdevtmpfs 11537 root  175u  IPv4 3068497      0t0  TCP VM_0_14_centos:42542->178.170.189.5:http (SYN_SENT)
kdevtmpfs 11537 root  176u  IPv4 3068779      0t0  TCP VM_0_14_centos:42604->178.170.189.5:http (SYN_SENT)
kdevtmpfs 11537 root  177u  IPv4 3068921      0t0  TCP VM_0_14_centos:42656->178.170.189.5:http (SYN_SENT)

挖矿病毒(我买的腾讯的服务器),等会再处理,说明80端口被占,那么我们就配置为8088端口

输入nginx -t 看看nginx.conf在哪

[root@VM_0_14_centos ~]# nginx -t
nginx: the configuration file /etc/nginx/nginx.conf syntax is ok
nginx: configuration file /etc/nginx/nginx.conf test is successful

然后修改

vi /etc/nginx/nginx.conf

把server部分用下面的替代

server {
    listen       8088;
    server_name  localhost;

    # Load configuration files for the default server block.
    include /etc/nginx/default.d/*.conf;

    charset UTF-8;
    # 这里存放日志文件
    access_log  /var/log/nginx/LVideo_access.log;
    error_log   /var/log/nginx/LVideo_error.log;

    client_max_body_size 75M;
    location / {
        # 你的uwsgi_params文件的路径
        include /etc/nginx/uwsgi_params;
        # 你的uwsgi端口
        uwsgi_pass 127.0.0.1:8008;
        # 链接超时时间
        uwsgi_read_timeout 30;
    }
}

保存

看看访问8088端口有没有页面

若出现502 Bad Gateway nginx/1.16.1

先输入以下命令看看是否有已存在的uwsgi或者nginx进程

ps -ef | grep uwsgi

ps -ef | grep nginx

有的话可以杀死

killall -9 uwsgi

killall -9 nginx

然后重新运行uwsgi和nginx

uwsgi --ini lvideo_uwsgi.ini

systemctl start nginx.service

然后一定一定可以看到8088端口的页面,我发四(前前后后调整了10多次)!!!

如果你的项目界面变得非常low,那么是因为你还没配置nginx解析静态文件,在nginx.conf的 location / {} 前面加上Django项目静态文件夹的路径

location /static/ {
        # 你的static的绝对路径
        alias /home/LVideo/LVideo/static/;
    }
uwsgi+nginx进程

在启动uwsgi时,如果指定了pid,可以通过pid停止uwsgi,如果没指定,直接kill uwsgi的进程id,会导致uwsgi重启,无法关闭成功

附:

# nginx开机自启动
systemctl enable nginx.service

# 终止uwsgi方法
# 1、未指定 daemonize:
Ctrl+c(快捷键可能有所不同)

# 2、指定daemonize和pidfile:
uwsgi --stop uwsgi.pid

# 3、指定daemonize,但未指定pidfile 通过ps,查看uwsgi相关进程
ps aux|grep uwsgi
# kill pid会发送SIGTERM,只会导致重启,而不是结束掉。需要发送SIGINT或SIGQUIT,对应着是INT才可以
killall -s INT /usr/local/bin/uwsgi

# 若出现
-bash: killall: command not found

# 则
# debian、ubuntu系统下:
  apt-get install psmisc

# centos 下:
  yum install psmisc

# 进程查看及杀死:
lsof -i:80

ps -ef | grep nginx

killall -9 nginx
提交代码

更新服务器Django代码到github

find / -name id_rsa

没有ssh钥匙

在 cd ~ 下 :ssh-keygen -t rsa -C "xxx@xx.com"

一路回车

可以看到:Enter file in which to save the key (/root/.ssh/id_rsa)

cat /root/.ssh/id_rsa.pub

复制内容到github上即可

Django(Git)+uwsgi+Nginx+venv

参考

django项目部署到服务器+虚拟环境
解决nginx+uwsgi部署Django的所有问题
如何把本地的Django项目部署到服务器(亲测)
centos6.5下配置django+uwsgi+nginx
centos下配置django、uwsgi和nginx(亲测成功)
Python+Django+Nginx+Uwsgi(史上最全步骤)
uWSGI+django+nginx的工作原理流程与部署历程
Linux vim命令
uwsgi、wsgi和nginx的区别和关系!!

linux如何彻底杀掉uwsgi进程
uWSGI的安装与配置(官网摘录)
centos7 nginx安装/启动/进程状态/杀掉进程
nginx报502错误,日志connect() failed (111: Connection refused) while connecting to upstream的解决
Linux 下建立 Git 与 GitHub 的连接并克隆到本地

使用Django + Vue.js快速而优雅地构建前后端分离项目

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

推荐阅读更多精彩内容