详解如何在Apache中运行Python WSGI应用


Posted in Python onJanuary 02, 2019

在生产环境上,一般会使用比较健壮的Web服务器,如Apache来运行我们的应用。如果我们的Web应用是采用Python开发,而且符合WSGI规范,比如基于Django,Flask等框架,那如何将其部署在Apache中呢?本文中,我们就会介绍如何使用Apache模块mod_wsgi来运行Python WSGI应用。

安装mod_wsgi

我们假设你已经有了Apache和Python环境,在Linux或者Mac上,那第一步自然是安装。在Ubuntu或Debian环境中,你可以使用apt-get命令来安装:

$ sudo apt-get install libapache2-mod-wsgi
$ sudo apt-get install libapache2-mod-wsgi-py3  # For Python 3

不过我们建议采用编译安装,这样在任何系统中都可以安装成功,具体步骤如下:

1、下载源码包

mod_wsgi的源码托管在Github上,你可以从https://github.com/GrahamDumpleton/mod_wsgi/releases下载它各个版本的源码包。

2、解压后,配置编译选项

一般采用默认配置即可,即执行:

$ ./configure

如果要指定Apache和Python环境,那你需要加上”?with-apxs”和”?with-python”选项:

$ ./configure --with-apxs=/usr/local/apache/bin/apxs --with-python=/usr/local/bin/python

3、编译并安装

$ sudo make && make install

4、在Apache配置文件中载入mod_wsgi

让我们打开Apache的配置文件httpd.conf,默认是在:

$ sudo vi /etc/httpd/conf/httpd.conf  # For Linux
$ sudo vi /etc/apache2/httpd.conf   # For Mac

在所有”Load Module”配置项的最后,加上载入mod_wsgi的配置,注意Linux和Mac的模块加载路径不同:

LoadModule wsgi_module modules/mod_wsgi.so  # For Linux
# LoadModule wsgi_module libexec/apache2/mod_wsgi.so  # For Mac

5、重启Apache来启用配置

$ sudo service httpd restart    # For Linux
$ sudo service apachectl restart  # For Mac

测试mod_wsgi

最简单的测试方法自然是Hello World,让我们在Apache的DocumentRoot根目录下创建一个文件”test.wsgi”。在文件中,我们写入这样的内容:

def application(environ, start_response):
  status = '200 OK'
  output = 'Hello World!'
 
  response_headers = [('Content-type', 'text/plain'),
            ('Content-Length', str(len(output)))]
  start_response(status, response_headers)
 
  return [output]

这里的函数application即为WSGI应用对象,它返回的值就是该应用收到请求后的响应。然后,再打开Apache的配置文件httpd.conf,在其最后加上URL路径映射:

WSGIScriptAlias /test /var/www/test.wsgi

这里我们假设Apache的文档根目录是”/var/www”。

现在你可以打开浏览器,访问一下”http://localhost/test”,如果看到”Hello World!”了,就说明mod_wsgi已经安装成功。

我们可以试试运行Flask应用,当然首先是你本地Python环境已经安装了Flask,我们将”test.wsgi”改为:

from flask import Flask
application = Flask(__name__)
 
@application.route('/')
def index():
  return '<h1>Hello World!</h1>'

注意,这里必须要将Flask应用对象命名为”application”,这样才能被mod_wsgi识别。再用浏览器访问下,是不是能看到大标题”Hello World!”?

使用Python虚拟环境

一般我们会将应用安装在虚拟环境中,这样应用的更新只需改变虚拟环境即可,不会影响到其他应用环境。要使用虚拟环境来运行当前WSGI应用的话,你必须在”.wsgi”文件中先执行虚拟环境的启用脚本,基于上面的代码,我们来做如下改动:

activate_this = '/home/bjhee/virtualenv/bin/activate_this.py'
execfile(activate_this, dict(__file__=activate_this))
 
from flask import Flask
application = Flask(__name__)
 
@application.route('/')
def index():
  return '<h1>Hello World!</h1>'

上例中,我们的虚拟环境在目录”/home/bjhee/virtualenv”下,你可以在其”/bin”子目录中找到启用脚本”activate_this.py”。在WSGI应用的一开始执行它即可。

补充内容

当我们的Python环境中有模块是以.egg压缩包安装的话,WSGI应用运行时需要将.egg压缩包解开。默认的解压路径很有可能没有访问权限,比如Mac下是”/Library/WebServer/.python-eggs”,因此你需要指定临时解压目录。方法有两种,一是在Apache的httpd.conf文件中,使用”WSGIPythonEggs”配置项,配置参数就是我们的临时目录路径;二是设置系统环境变量”PYTHON_EGG_CACHE”。我们建议采用第二种,并将其写在”.wsgi”文件中,这样就不会影响其他的应用:

activate_this = '/home/bjhee/virtualenv/bin/activate_this.py'
execfile(activate_this, dict(__file__=activate_this))
 
import os
os.environ['PYTHON_EGG_CACHE'] = '/home/bjhee/.python-eggs'
 
from flask import Flask
application = Flask(__name__)
 
@application.route('/')
def index():
  return '<h1>Hello World!</h1>'

运行前,请确保临时目录(上例中的”/home/bjhee/.python-eggs”)有访问及写权限。

更多内容请参阅mod_wsgi的官方文档。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
探究Python中isalnum()方法的使用
May 18 Python
Python考拉兹猜想输出序列代码实践
Jul 05 Python
python连接打印机实现打印文档、图片、pdf文件等功能
Feb 07 Python
python实现梯度下降和逻辑回归
Mar 24 Python
python实现将列表中各个值快速赋值给多个变量
Apr 02 Python
解决pyqt5异常退出无提示信息的问题
Apr 08 Python
详解Python中namedtuple的使用
Apr 27 Python
python基于socket函数实现端口扫描
May 28 Python
python 偷懒技巧——使用 keyboard 录制键盘事件
Sep 21 Python
如何设置PyCharm中的Python代码模版(推荐)
Nov 20 Python
python周期任务调度工具Schedule使用详解
Nov 23 Python
使用Django框架创建项目
Jun 10 Python
漂亮的Django Markdown富文本app插件的实现
Jan 02 #Python
对Python发送带header的http请求方法详解
Jan 02 #Python
Django渲染Markdown文章目录的方法示例
Jan 02 #Python
使用python 打开文件并做匹配处理的实例
Jan 02 #Python
对Xpath 获取子标签下所有文本的方法详解
Jan 02 #Python
python之验证码生成(gvcode与captcha)
Jan 02 #Python
Python lxml解析HTML并用xpath获取元素的方法
Jan 02 #Python
You might like
星际初学者游戏中永远要做的事
2020/03/04 星际争霸
基于ThinkPHP5.0实现图片上传插件
2017/09/25 PHP
为你的 Laravel 验证器加上多验证场景的实现
2020/04/07 PHP
jQuery库与其他JS库冲突的解决办法
2010/02/07 Javascript
javascript 模式设计之工厂模式详细说明
2010/05/10 Javascript
避免回车键导致的页面无意义刷新的解决方法
2011/04/12 Javascript
简单漂亮的js弹窗可自由拖拽且兼容大部分浏览器
2013/10/22 Javascript
javascript获取url上某个参数的方法
2013/11/08 Javascript
利用js实现前台动态添加文本框,后台获取文本框内容(示例代码)
2013/11/25 Javascript
javascript面向对象之访问对象属性的两种方式分析
2015/01/13 Javascript
JavaScript实现cookie的写入、读取、删除功能
2015/11/05 Javascript
jquery判断密码强度的验证代码
2020/04/22 Javascript
分享一个原生的JavaScript拖动方法
2016/09/25 Javascript
微信小程序 image组件binderror使用例子与js中的onerror区别
2017/02/15 Javascript
JS实现下拉菜单列表与登录注册弹窗效果
2017/08/10 Javascript
Angular移动端页面input无法输入的解决方法
2017/11/14 Javascript
详解基于Vue-cli搭建的项目如何和后台交互
2018/06/29 Javascript
Vue中Table组件行内右键菜单实现方法(基于 vue + AntDesign)
2019/11/21 Javascript
JS实现简单的表格增删
2020/01/16 Javascript
JS实现小米轮播图
2020/09/21 Javascript
微信小程序实现星星评分效果
2020/11/01 Javascript
Python实现数通设备端口使用情况监控实例
2015/07/15 Python
Python八大常见排序算法定义、实现及时间消耗效率分析
2018/04/27 Python
python实现音乐下载的统计
2018/06/20 Python
Pycharm中安装Pygal并使用Pygal模拟掷骰子(推荐)
2020/04/08 Python
使用python编写一个语音朗读闹钟功能的示例代码
2020/07/14 Python
python抢购软件/插件/脚本附完整源码
2021/03/04 Python
英国最好的温室之家:Greenhouses Direct
2019/07/13 全球购物
为什么如下的代码int a=100,b=100;long int c=a * b;不能工作
2013/11/29 面试题
中专毕业生自我鉴定范文
2013/11/09 职场文书
投资合作协议书
2014/04/17 职场文书
慈善晚会策划方案
2014/05/14 职场文书
公司年底活动方案
2014/08/17 职场文书
学校机关党总支领导班子整改工作方案
2014/10/26 职场文书
革命电影观后感
2015/06/18 职场文书
Java存储没有重复元素的数组
2022/04/29 Java/Android