我们有几个在Linux上运行的Python 2.6应用程序。其中一些是Pylons Web应用程序,其他一些只是长时间运行的进程,我们使用nohup
从命令行运行。我们还在开发和生产中都使用virtualenv
。 将这些应用程序部署到生产服务器的最佳方法是什么?
在开发中,我们只需将源树放入任何目录中,设置一个virtualenv并运行-足够容易。我们可以在生产中做同样的事情,也许这确实是最实用的解决方案,但是在生产中运行svn update
感觉有点不对。我们还尝试了fab
,但是它从来没有第一次起作用。对于每个应用程序,其他地方都会出错。令我惊讶的是,鉴于我们要实现的目标从根本上讲非常简单,整个过程都是太难了。这是我们从部署过程中想要的。
就是这样!它能有多难?
最佳答案
setuptools结合virtualenv和pip使Python代码的开发和部署变得更加容易。
核心思想
我发现最棘手的部分是运行一个开发环境,该环境将尽可能紧密地反射(reflect)已部署的设置,同时尊重Pythonic工具和习惯用法。但是事实证明,使用pip和setuptools非常容易实现这一点,它们一起使您可以将开发树“安装”到Python环境中,而无需移动文件。 (实际上,setuptools本身可以完成所有操作,但是pip可以更好地充当前端处理依赖项。)
另一个关键问题是使用两个环境中已知的软件包设置一个干净的环境。在这方面,Python的virtualenv是天赐之物,它允许您使用自己选择的软件包来配置完全定制的Python环境,而无需root访问权限或OS软件包(rpm或dpkg),并且不受任何软件包和版本的限制恰好安装在您的发行版上。
最后,一个令人讨厌的错误负担是创建与PYTHON_PATH配合使用的命令行脚本的困难。 setuptools也非常优雅地处理了这一问题。
配置
(为简单起见,这是规定性的。请随时进行适当区分。)
$ python <untarred_directory>/virtualenv.py venv
.
是source
的快捷方式):$ . venv/bin/activate
$ easy_install pip
发展历程
一旦树结构准备就绪,您几乎就可以开始编码了。但是现在,相互依赖的程序包无法像在已部署环境中那样相互看到。这个问题可以通过setuptools提供的一个巧妙的小技巧来解决,该技巧可以利用pip。对于您正在开发的每个程序包,运行以下命令(确保您位于项目的虚拟环境中,按照上面的步骤3):
$ pip install -e pkg1
该命令将
pkg1
安装到您的虚拟环境中,并且无需复制任何文件即可执行此操作。它仅向指向包的开发根目录的site-packages
目录添加一个链接,并在该根目录中创建一个egg-info目录。您也可以不使用点子来执行此操作,如下所示:$ cd pkg1
$ python setup.py develop
它通常可以工作,但是如果您具有第三方依赖项(应该在setup.py中列出,如setuptools文档中here所述),则pip会更聪明地查找它们。
需要注意的一个警告是,setuptools和pip都不擅长在您自己的程序包中查找依赖项。如果目录B中的PkgB依赖于目录A中的PkgA,则
pip install -e B
将失败,因为pip无法知道可以在目录A中找到PkgA。相反,它将尝试并失败从其在线存储库源下载PkgA。解决方法是简单地在每个软件包依赖之后安装它们。在这一点上,您可以启动python,加载您的模块之一并开始玩弄它。您可以编辑代码,下次导入时将立即可用。
最后,如果要使用软件包创建命令行工具。不要用手书写。最后,您将遇到一堆糟糕透顶的PYTHON_PATH骇客,这些骇客永远无法正常运作。只需阅读setuptools文档中的automatic script creation。这将使您免于很多痛苦。
部署方式
准备好要执行的程序包时,可以使用setup.py创建部署程序包。这里有太多选择,但是以下几点可以帮助您入门:
$ cd pkg1
$ python setup.py --help
$ python setup.py --help-commands
末端松动
由于问题的广泛性,此答案不一定是完整的。我没有处理长时间运行的服务器,Web框架或实际的部署过程(特别是使用pip install的
--index-url
管理第三方和内部软件包以及-e vcs+...
的私有(private)存储库,这会将软件包从svn中拉出,git,hg或bzr)。但我希望我给了您足够的绳索来将它们绑在一起(只是不要将自己卡在身上:-)。
关于python - 一个简单的Python部署问题-痛苦的世界,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2741507/