以ubuntu爲例:css
#!/bin/bash #初始化用戶 sudo useradd -rm -s /bin/bash demo sudo adduser demo sudo sudo passwd demo sudo apt-get install build-essential python-software-properties software-properties-common -y sudo apt-get install vim nano -y sudo apt-get install supervisor -y sudo add-apt-repository ppa:nginx/stable sudo add-apt-repository -y ppa:rwky/redis sudo apt-get update sudo apt-get install nginx aria2 axel wget curl -y sudo apt-get install redis-server -y sudo apt-get install build-essential libssl-dev libffi-dev python-dev python3-dev -y sudo apt-get install python3-pip -y sudo apt-get install convmv libevent-dev libssl-dev libffi-dev libsasl2-dev libpq-dev libxml2-dev libxslt1-dev libldap2-dev -y vim ~/.pip/pip.conf #[global] #index-url = https://pypi.douban.com/simple #豆瓣源,能夠換成其餘的源 #disable-pip-version-check = true #取消pip版本檢查,排除每次都報最新的pip #timeout = 120 sudo pip3 install virtualenv mkdir venv && cd venv virtualenv mdwiki source mdwiki/bin/activate pip3 install gunicorn #具體配置文件後續說明 sudo vim mdwiki/gunicorn.conf.py sudo vim /etc/supervisor/conf.d/default.conf sudo vim /etc/nginx/conf.d/default.conf #添加開機自啓 sudo vim /etc/rc.local #/usr/bin/supervisord -c /etc/supervisor/supervisord.conf sudo update-rc.d nginx disable
celery+virtualenv+supervisor的情形,其實只要指定celery程序爲virtaulenv下面的那個便可,例如/home/xby/venv/mdwiki/bin/celerynode
若是在gunicorn下出現'ascii' codec can't encode...報錯,那麼請在supervisor加入environment以下python
environment=LANG="en_US.utf8", LC_ALL="en_US.UTF-8", LC_LANG="en_US.UTF-8"
[program:nginx] command=/usr/sbin/nginx -g "daemon off;" stopsignal=QUIT priority=1 ;user=www-data [program:celeryworker] directory=/opt/www/mdwiki command=/path/to/celery worker -A app.util.tasks.celery_app -f celery.worker.log -l info priority=5 autostart=true autorestart=true startsecs=10 user=www-data [program:celerybeat] directory=/opt/www/mdwiki command=/path/to/celery beat -A app.util.tasks.celery_app -f celery.beat.log -l info priority=6 autostart=true autorestart=true startsecs=10 user=www-data [program:mdwiki] ;environment=SECRET_KEY=value,aliyun_api_key=value,aliyun_secret_key=value,MAIL_PASSWORD=value ;command=/usr/bin/gunicorn -n mdwiki -w 4 -b 127.0.0.1:4000 -k gevent app:app environment=LANG="en_US.utf8", LC_ALL="en_US.UTF-8", LC_LANG="en_US.UTF-8" command=/path/to/gunicorn app:app -c /path/to/gunicorn.conf.py directory=/opt/www/mdwiki ;user=www-data autostart=true autorestart=true priority=10 redirect_stderr = true stdout_logfile_maxbytes = 20MB stdout_logfile_backups = 20 stdout_logfile = /var/log/mdwiki/mdwiki.log ; environment=PYTHONPATH=$PYTHONPATH:/path/to/somewhere
server { listen 80; listen 443 ssl; server_name demo.com; ssl_certificate /etc/nginx/ssl/nginx.crt; ssl_certificate_key /etc/nginx/ssl/nginx.key; server_tokens off; charset utf-8; client_max_body_size 20M; set $projdir "/opt/www/mdwiki"; location / { proxy_set_header Host $host; proxy_set_header X-Real-IP $remote_addr; proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; proxy_redirect off; proxy_pass http://127.0.0.1:4000; } location ~ ^/[^static].*\.(jpg|png|gif|bmp|zip|docx)$ { expires 30d; root $projdir; } location ~ ^/static.*\.(js|css|png|jpg|gif|bmp|map|ico|eot|svg|ttf|woff)$ { expires 30d; root $projdir/app; } }
#!/bin/bash import multiprocessing bind = "127.0.0.1:4000" workers = multiprocessing.cpu_count() * 2 + 1 worker_class='gevent' proc_name = "mdwiki" user = "www-data" chdir='/opt/www/mdwiki' #daemon=False #group = "nginx" loglevel = "info" errorlog = chdir+"/log/gunicorn/error.log" accesslog= chdir+"/log/gunicorn/access.log" raw_env = [ r'MAIL_PASSWORD=pass', r'SECRET_KEY=\xe6' ] #ssl #keyfile= #certfile= #ca_certs=
fab -f fabfile.py deploy
fabfile.py文件nginx
from fabric.api import * import os,sys import tarfile from contextlib import contextmanager from fabric.contrib.files import exists #$ fab -f fabfile.py -H localhost,remote host_type def host_type(): run('uname -s') env.user= os.environ.get('USER','') env.hosts= os.environ.get('HOST','').split(',') env.password= os.environ.get('PASSWORD','') env.sudo_password= os.environ.get('PASSWORD','') active='source /home/xby/venv/mdwiki/bin/activate' srcPath=r'C:\Users\taojw\Desktop\pywork\mdwiki' distPath=r'C:\Users\taojw\Desktop\pywork\mdwiki\dist' distFile=distPath+os.sep+'mdwiki.tar.gz' #用於處理virtualenv環境,將其包裝成with上下文 @contextmanager def virtualenv(): with prefix(active): yield if not os.path.exists(distPath): os.mkdir(distPath) #本地打包分發文件 def pack(): def ecludefiles(path): for name in ['venv','node_modules','websrc','__pycache__','.git','.idea','dist']: if path.find(os.sep+name)>0: return True return False if os.path.exists(distFile): os.remove(distFile) #壓縮成tar.gz格式 with tarfile.open(distFile,'w:gz') as f: f.add(srcPath,arcname='mdwiki',exclude=ecludefiles) #部署 def deploy(): #local pack dist file pack() remote_tmp='/tmp/mdwiki.tar.gz' localsize=os.path.getsize(distFile) remotesize=0 #check if should upload again if there is a same file if exists(remote_tmp): remotesize=int(run("stat -c '%s' {0}".format(remote_tmp))) print(str(localsize)+":"+str(remotesize)) if localsize!=remotesize: sudo('rm -f %s' % remote_tmp) # upload dist file put(distFile,remote_tmp) if not exists('/opt/www'): sudo('mkdir /opt/www') sudo('chown www-data:www-data /opt/www') #stop app and bak now with settings(warn_only=True): #delete previous bak sudo('rm -rf /opt/www/mdwiki_bak') sudo('supervisorctl stop all') if exists('/opt/www/mdwiki'): sudo('mv /opt/www/mdwiki /opt/www/mdwiki_bak') sudo('tar -zxvf /tmp/mdwiki.tar.gz -C /opt/www/') with cd('/opt/www/'): #replace data dir if exists('mdwiki_bak/data'): sudo('rm -rf mdwiki/data') sudo('cp -R mdwiki_bak/data mdwiki/') if exists('mdwiki_bak/app.db'): sudo('cp mdwiki_bak/app.db mdwiki/') sudo('chown -R www-data:www-data mdwiki') with virtualenv(): run('pip3 install -r mdwiki/requirements.txt') sudo('rm -f %s' % remote_tmp) sudo('supervisorctl start all') #in your local shell run 'fab deploy' command