在uWSGI和Nginx

时间:2017-11-10 13:12:15

标签: python nginx flask uwsgi flask-admin

我使用 uWSGI Nginx AWS 中举办了 python Flask 项目,最初一切正常,但后来我决定实施 Flask-admin 模块。现在页面显示内部服务器错误消息。

我检查了UWSGI日志并说:

  Traceback (most recent call last):
  File "/var/www/project/run.py", line 2, in <module>
    from app import app
  File "/var/www/project/app/__init__.py", line 22, in <module>
    from app import views
  File "/var/www/project/app/views.py", line 15, in <module>
    from flask_admin.contrib.sqla import ModelView
  File "/var/www/project/venv/local/lib/python2.7/site-packages/flask_admin/contrib/sqla/__init__.py", line 2, in <module>
    from .view import ModelView
  File "/var/www/project/venv/local/lib/python2.7/site-packages/flask_admin/contrib/sqla/view.py", line 17, in <module>
    from flask_admin.contrib.sqla.tools import is_relationship
  File "/var/www/project/venv/local/lib/python2.7/site-packages/flask_admin/contrib/sqla/tools.py", line 4, in <module>
    from sqlalchemy.ext.declarative.clsregistry import _class_resolver
ImportError: cannot import name _class_resolver
Fri Nov 10 07:41:08 2017 - unable to load app 0 (mountpoint='project.domain.com|') (callable not found or import error)
Fri Nov 10 07:41:08 2017 - --- no python application found, check your startup logs for errors ---

我的app / views.py文件:

# -*- coding: utf-8 -*-
import os, time
import flask
from flask import render_template, request, jsonify, send_from_directory, send_file, session, redirect, g, url_for, make_response, session, request
from flask_sqlalchemy import SQLAlchemy
from models import RequestsLog
import requests
import json
from app import app, db, appname #Line edited
import flask_admin as admin
#from flask_admin import Admin, helpers, expose  ------Line Erased
from flask_admin.contrib.sqla import ModelView
#from flask_admin.contrib import sqla   ----------- Line Erased

class MyAdminIndexView(admin.AdminIndexView):
        @admin.expose('/')
        def index(self):
                count = 56
                return self.render('admin/index.html', count=count)

class CustomView(ModelView):
       can_create = False
       can_edit = False
       can_delete = False  # disable model deletion
       can_export = True
       export_columns = ['phone_number', 'doc_number', 'date', 'doc_type']
       can_view_details = True       
       column_export_exclude_list = ['job_id', 'status']
       list_template = 'list.html'
       details_modal = True
       page_size = 50
       can_set_page_size = True

admin = admin.Admin(app, appname, template_mode='bootstrap3', index_view=MyAdminIndexView())
admin.add_view(CustomView(RequestsLog, db.session, menu_icon_type ='fa', menu_icon_value='fa-calendar'))

修改 如果我评论这一行,一切正常:

from flask_admin.contrib.sqla import ModelView

谷歌搜索我注意到 ImportError 可能是由于冗余导入造成的,但我认为所有的flask-admin导入都是必要的。

我认为问题不是由uWSGI引起的,而是配置文件:

[uwsgi]
uid = www-data
gid = www-data

plugins=python
vhost=true
socket=/tmp/project.sock
chmod-socket = 666
chown-socket = www-data:www-data

enable-threads = true
procname-prefix = project_
chdir = /var/www/project

我再次创建了虚拟环境,但所有内容都完美安装。

编辑:应该注意的是,当我直接执行 run.py 文件时,一切都按预期工作,即使使用flask-admin导入,问题就在于uWSGI。

最后,我的项目托管在Ubuntu 14.04 Trusty AWS依赖项中。

修改

添加run.py文件:

#!venv/bin/python
from app import app

if __name__ == '__main__':

    app.run(host='0.0.0.0', threaded=True, debug=True, port=5010)

编辑2:

添加 __ init __。py 文件:

# -*- coding: utf-8 -*-
from flask import Flask
from flask_sqlalchemy import SQLAlchemy
from flask_socketio import SocketIO
#import flask_admin as admin
from flask_babel import Babel

#I've defined the next three lines to avoid encoding conflicts
import sys
reload(sys)
sys.setdefaultencoding("utf-8")

app = Flask(__name__)
app.config.from_pyfile('../config.py')
appname = 'Project'

db = SQLAlchemy(app, session_options={'autoflush': False})
babel = Babel(app)

async_mode = None
socketio = SocketIO(app, async_mode=async_mode, ping_timeout=1800)

from app import views

venv/bin/pip freeze输出:

Babel==2.5.1
Flask==0.12.2
Flask-Admin==1.5.0
Flask-Babel==0.11.2
Flask-SQLAlchemy==2.3.2
Flask-SocketIO==2.9.2
Jinja2==2.9.6
MarkupSafe==1.0
SQLAlchemy==1.1.14
WTForms==2.1
Werkzeug==0.12.2
argparse==1.2.1
asn1crypto==0.22.0
certifi==2017.7.27.1
chardet==3.0.4
click==6.7
click-completion==0.2.1
colorama==0.3.9
crayons==0.1.2
elibom==1.2
gunicorn==19.7.1
idna==2.6
itsdangerous==0.24
psycopg2==2.7.3.1
py==1.4.34
pycycle==0.0.8
pytest==3.2.3
python-engineio==1.7.0
python-socketio==1.8.1
pytz==2017.3
requests==2.18.4
six==1.11.0
uWSGI==2.0.15
urllib3==1.22
wsgiref==0.1.2

Nginx配置文件:

server {
        listen  80 ssl;
    server_name project.domain.com; 
    listen 443 ssl;

        ssl_certificate /etc/nginx/project.domain.com/fullchain.pem;
        ssl_certificate_key /etc/nginx/project.domain.com/privkey.pem;
        ssl_session_timeout 1d;
        ssl_session_cache shared:SSL:50m;

        ssl_protocols TLSv1 TLSv1.1 TLSv1.2;
    ssl_ciphers "EECDH+ECDSA+AESGCM EECDH+aRSA+AESGCM EECDH+ECDSA+SHA384 EECDH+ECDSA+SHA256 EECDH+aRSA+SHA384 EECDH+aRSA+SHA256 !aNULL !eNULL !LOW !3DES !MD5 !EXP !PSK !SRP !DSS !MEDIUM !RC4";
        ssl_prefer_server_ciphers on;

        ssl_stapling on;
        ssl_stapling_verify on;

    location /static/ {
                autoindex               on;
                alias                   /var/www/project/app/static/;
        }

        location / {
                include                 uwsgi_params;
                uwsgi_pass              unix:/tmp/project.sock;
                uwsgi_read_timeout      300;
                uwsgi_param             UWSGI_PYHOME    /var/www/project/venv;
                uwsgi_param             UWSGI_CHDIR     /var/www/project;
                uwsgi_param             UWSGI_MODULE    run;
                uwsgi_param             UWSGI_CALLABLE  app;
        }

        error_page 404 /404.html;

        location /socket.io {
            include proxy_params;
            proxy_http_version 1.1;
            proxy_buffering off;
            proxy_set_header Upgrade $http_upgrade;
            proxy_set_header Connection "Upgrade";
            proxy_pass http://127.0.0.1:5010/socket.io;
    }
}

非常感谢你的帮助。

1 个答案:

答案 0 :(得分:0)

我将从即时错误开始,然后解决它可能与uWSGI不同的原因。

立即错误:

这里有一些快速和宽松的导入,这很常见,因为你正在构建一些东西或者修改功能。一个好的起点是将其削减,以减少冗余并清理它。即使您没有在途中解决问题,每次删除都会使剩余的导入更容易思考。

除了Luis在评论中建议的更改之外,您可以将以下两行使用不同的约定组合在同一个地方导入:

以管理员身份导入flask_admin 来自flask_admin导入管理员,帮助者,公开

您可以直接导入flask_admin:

import flask_admin
...
class MyAdminIndexView(flask_admin.AdminIndexView):
    @flask_admin.expose('/')
...
admin = flask_admin.Admin...

或者您可以将AdminIndexView添加到显式导入列表中:

from flask_admin import Admin, AdminIndexView, helpers, expose

我遇到的大多数Flask导入错误来自于尝试从您的应用中导入另一个尚未正确初始化的对象(通常是因为循环导入) ,所以我猜它隐藏在这3行中的一条之后:

from app import app
...
from models import RequestsLog
...
from app import db, appname

原因可能会对你产生影响,但有时候这个周期是几个模块。如果它没有,那么下一步就是通过Pylint这样的linter来运行你的项目。如果您已经在运行Pylint,则可能需要仔细检查您的设置并确保启用与导入相关的规则!如果您还没有使用它,可以从命令行手动运行它,或者通过编辑器插件使用它,它可以在您工作时运行代码(它可能会抱怨很多问题)如果你还没有linting - 但我只关注导入/模块相关的代码来开始。)

你可能也可以使用像pycycle这样更专注的工具更快地发现它,虽然我还没有尝试过。

有时候,您可以通过精心设计或稍微重构来避免Flask中的循环导入问题,但随着应用程序复杂性的增加,新功能更有可能引导您进入循环导入。如果您发现问题但无法找到明显的方法,则下一步通常是采用Flask application factory pattern。这种模式并不神奇 - 你仍然可以将自己描绘成导入问题的角落 - 但它确实会让你更容易重构自己。

从uWSGI运行:

目前,我认为这两个最可能的原因是,当您从命令行通过python运行时,以及从uWSGI后面看到不同的行为。这只是一个通用列表,因为更具体的答案取决于您所包含的文件之外的一些内容。 uWSGI和nginx都有一个相当复杂的配置选项,我在这里从内存中飞出来,所以我可能不完全准确......:)

  1. 应用程序是有条件设置的,可能使用环境变量或类似if __name__ == '__main__':之类的内容,因此在这两种情况下您都无法运行完全相同的代码。

  2. uWSGI没有使用相同的callable / module。在配置文件和命令行之间,我记得有几种方法告诉uWSGI要加载哪个python文件,以及应用程序的python对象的名称是什么,但我不记得是能够从nginx这样做。我在配置文件中没有看到任何这些内容,但是您提供的日志片段表明当uWSGI尝试查找可调用内容(通常为appapplication)时会发生导入异常run.py。您是否从CLI运行相同的代码?