Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/python/360.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Python 如何将Flask登录与Dash应用程序合并?_Python_Flask_Web Applications_Plotly Dash - Fatal编程技术网

Python 如何将Flask登录与Dash应用程序合并?

Python 如何将Flask登录与Dash应用程序合并?,python,flask,web-applications,plotly-dash,Python,Flask,Web Applications,Plotly Dash,我必须设计一个提供Flask服务和Dash服务的web应用程序。例如,我想在Flask中创建一个登录名,并结合Dash应用程序。问题是我不能用dash绑定flask登录名。我需要一个像“@require_login”这样的方法来过滤对Dash服务的访问。 代码如下: app_flask = Flask(__name__) app_flask.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:////login.db' app_flask.config['S

我必须设计一个提供Flask服务和Dash服务的web应用程序。例如,我想在Flask中创建一个登录名,并结合Dash应用程序。问题是我不能用dash绑定flask登录名。我需要一个像“@require_login”这样的方法来过滤对Dash服务的访问。 代码如下:

app_flask = Flask(__name__)

app_flask.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:////login.db'
app_flask.config['SECRET_KEY'] = 'thisissecret'

db = SQLAlchemy(app_flask)
login_manager = LoginManager()
login_manager.init_app(app_flask)

class User(UserMixin, db.Model):
    id = db.Column(db.Integer, primary_key=True)
    username = db.Column(db.String(30), unique=True)

@login_manager.user_loader
def load_user(user_id):
    return User.query.get(int(user_id))

@app_flask.route('/')
def index():
    user = User.query.filter_by(username='admin').first()
    login_user(user)
    return 'You are now logged in!'

@app_flask.route('/logout')
@login_required
def logout():
    logout_user()
    return 'You are now logged out!'

@app_flask.route('/home')
@login_required
def home():
    return 'The current FLASK user is ' + current_user.username

# TODO how to add login_required for dash? 
app_dash = Dash(server=app_flask, url_base_pathname='/dash/')
app_dash.layout = html.H1('MY DASH APP')


if __name__ == '__main__':
    app_dash.run_server(debug=True)

解决方案:烧瓶会话(使用cookie)

来自烧瓶导入会话

这是一个例子:

@login\u manager.user\u loader
def加载用户(用户id):
#我觉得这里很好
会话[“uid”]=用户id
返回User.query.get(int(User\u id))

#TODO如何添加dash所需的登录名?
如果会话中出现“uid”:
app_dash=dash(服务器=app_flask,url_base_路径名='/dash/'))

app_dash.layout=html.H1(“我的dash应用程序”)

这一行,
app_dash=dash(server=app_flask,url_base_pathname='/dash/')
app_flask
中创建新的
视图_函数
,由其
url_base_路径名
标识

您可以调试和检查
app\u flask的值。在创建
app\u dash
之前和之后查看功能

现在我们知道哪些
查看功能是由
app\u dash
创建的,我们可以手动对它们应用
登录所需的功能

for view_func in app_flask.view_functions:
    if view_func.startswith(app_dash.url_base_pathname):
        app_flask.view_functions[view_func] = login_required(app_flask.view_functions[view_func])

“app_dash”端点现在将受到保护。最好使用
@app.before_request
阻止所有请求,并且仅在登录或端点标记为public时才允许请求

def check_route_access():
    if request.endpoint is None:
        return redirect("/login")
 
    func = app.view_functions[request.endpoint]
    if (getattr(func, "is_public", False)):
        return  # Access granted

    # check if user is logged in (using session variable)
    user = session.get("user", None)
    if not user:
        redirect("/login")
    else:
        return  # Access granted```

现在将检查所有端点,甚至dash应用程序端点

添加名为
public\u route
的装饰器:

def public_route(function):
    function.is_public = True
    return function
并将decorator添加到公共方法中,如登录、错误页面等

@public_route
@app.route("/login")
def login():
   # show/handle login page
   # call did_login(username) when somebody successfully logged in


def did_login(username):
    session["user"] = username

这样,您就不再需要
@login\u required
,因为除非
@public\u route
另有说明,否则所有端点都需要登录。谢谢您的回复。然而,通过这种方式,会话被断章取义地使用。“RuntimeError:在请求上下文之外工作。这通常意味着您试图在活动HTTP请求中使用该功能。有关如何避免此问题的信息,请参阅测试文档。”可能的解决方案是将会话保存在数据库中。你认为这是最好的做法吗?或者我做错了什么?这是一个解决方案,但我认为它太多了,如果它是一个小应用程序,可能不使用Flask登录,但只使用会话?如果你想像这样使用Flask登录,也许可以找到另一种方法来实现它?我也有同样的问题。我在flask应用程序中捕获了数据,我想将其传递到dash route。会话对象生成与上面相同的错误。我正在访问一个sqlite数据库,我将插入数据,然后在dash应用程序中查询数据。我想不出更好的办法了?向你致敬!它不再工作了:用
app\u-dash.url\u-base\u-pathname['url\u-base\u-pathname']替换
app\u-dash.config['url\u-base\u-pathname']
工作。谢谢你的解决方案+我花了2天时间解决这个问题,这个解决方案对我来说很有效。谢谢