当前位置: 首页 > article >正文

Flask 实现登录状态持久化:让用户 1 天内无需重新登录

在开发 Web 应用时,用户登录状态的管理至关重要。默认情况下,Session 是临时的,一旦用户关闭浏览器,Session 通常会被清除,这意味着用户需要重新登录。然而,在许多应用场景中,我们希望实现持久化登录,即用户即使关闭了浏览器,在一段时间内仍然不需要重新登录。本文将介绍如何在 Flask 中实现这一功能,包括使用 Session 持久化、JWT Token、以及Redis 存储 Session 的方法。


目录

  1. 使用 Flask-Session 持久化登录状态
  2. 使用 Cookies 存储 JWT Token 实现持久化登录
  3. Redis 持久化 Session,支持分布式架构
  4. 总结

1. 使用 Flask-Session 持久化登录状态

Flask 使用 Session 管理用户的登录状态。默认情况下,Session 是临时的,通常在用户关闭浏览器后失效。为了让用户的登录状态可以持续一段时间(如 1 天),我们可以将 Session 配置为“永久”,并设置一个有效期。

代码实现
from flask import Flask, session, redirect, url_for, request
import datetime

app = Flask(__name__)
app.secret_key = 'your_secret_key'

# 设置 Session 的过期时间为1天
app.config['PERMANENT_SESSION_LIFETIME'] = datetime.timedelta(days=1)

@app.route('/login', methods=['GET', 'POST'])
def login():
    if request.method == 'POST':
        username = request.form['username']
        password = request.form['password']
        # 假设用户名和密码正确
        if username == 'test_user' and password == 'test_password':
            session.permanent = True  # 使 session 永久化
            session['username'] = username  # 存储用户信息在 session 中
            return redirect(url_for('dashboard'))
        return 'Invalid credentials'
    return '''
        <form method="post">
            <p><input type=text name=username>
            <p><input type=password name=password>
            <p><input type=submit value=Login>
        </form>
    '''

@app.route('/dashboard')
def dashboard():
    if 'username' in session:
        return f'Hello, {session["username"]}! You are logged in.'
    return redirect(url_for('login'))

@app.route('/logout')
def logout():
    session.pop('username', None)  # 删除 session 中的用户信息
    return redirect(url_for('login'))

if __name__ == '__main__':
    app.run(debug=True)
解释
  • Session 持久化:通过 session.permanent = True 将 Session 设置为永久,即使关闭浏览器,Session 仍然会保持。
  • PERMANENT_SESSION_LIFETIME:设置为 1 天,使得用户在登录后的 1 天内保持登录状态。

2. 使用 Cookies 存储 JWT Token 实现持久化登录

另一种实现持久化登录的方式是通过 JWT (JSON Web Token),将生成的 Token 存储在 Cookies 中,并设置过期时间。在用户访问受保护的路由时,服务器可以通过解码 Token 验证用户身份,从而实现 1 天内无需重新登录。

代码实现
from flask import Flask, request, redirect, url_for, render_template, make_response
import datetime
import jwt

app = Flask(__name__)
app.secret_key = 'your_secret_key'

# JWT 秘钥
JWT_SECRET = 'your_jwt_secret'

def create_jwt_token(username):
    payload = {
        'username': username,
        'exp': datetime.datetime.utcnow() + datetime.timedelta(days=1)  # 1天后过期
    }
    token = jwt.encode(payload, JWT_SECRET, algorithm='HS256')
    return token

def verify_jwt_token(token):
    try:
        decoded = jwt.decode(token, JWT_SECRET, algorithms=['HS256'])
        return decoded['username']
    except (jwt.ExpiredSignatureError, jwt.InvalidTokenError):
        return None

@app.route('/login', methods=['GET', 'POST'])
def login():
    if request.method == 'POST':
        username = request.form['username']
        password = request.form['password']
        if username == 'test_user' and password == 'test_password':
            token = create_jwt_token(username)
            resp = make_response(redirect(url_for('dashboard')))
            # 设置 Cookie,保存1天
            resp.set_cookie('auth_token', token, max_age=60*60*24)  # 1天有效期
            return resp
        return 'Invalid username or password', 401
    return render_template('login.html')

@app.route('/dashboard')
def dashboard():
    token = request.cookies.get('auth_token')
    if token:
        username = verify_jwt_token(token)
        if username:
            return f'Hello, {username}! You are logged in.'
    return redirect(url_for('login'))

@app.route('/logout')
def logout():
    resp = make_response(redirect(url_for('login')))
    resp.delete_cookie('auth_token')  # 删除 Cookie
    return resp

if __name__ == '__main__':
    app.run(debug=True)
解释
  • JWT:登录成功后生成 JWT Token,将其存储在 Cookie 中,并设置有效期为 1 天。
  • Token 验证:在用户访问受保护的路由时,通过解码 Token 来验证用户身份。

3. Redis 持久化 Session,支持分布式架构

在更复杂的应用场景中,可能需要跨多个服务器或进程保持用户的登录状态。这时,可以使用 Redis 或其他数据库来持久化 Session。Redis 允许将 Session 存储在服务器端,并支持集群化和分布式应用场景。

代码实现
from flask import Flask, session
from flask_session import Session
import redis

app = Flask(__name__)

# 配置 Redis 作为 Session 后端
app.config['SESSION_TYPE'] = 'redis'
app.config['SESSION_PERMANENT'] = True
app.config['PERMANENT_SESSION_LIFETIME'] = datetime.timedelta(days=1)
app.config['SESSION_USE_SIGNER'] = True
app.config['SESSION_KEY_PREFIX'] = 'myapp_'
app.config['SESSION_REDIS'] = redis.StrictRedis(host='localhost', port=6379, db=0)

# 初始化 Session
Session(app)

@app.route('/login')
def login():
    session['username'] = 'test_user'  # 存储用户名到 Session
    return 'Logged in successfully!'

@app.route('/dashboard')
def dashboard():
    username = session.get('username')
    if username:
        return f'Hello, {username}!'
    return 'You need to log in first.', 401

if __name__ == '__main__':
    app.run(debug=True)
解释
  • Redis 后端:Session 数据保存在 Redis 中,支持多服务器之间的 Session 同步和持久化。

4. 总结

通过本文,我们介绍了如何在 Flask 应用中实现持久化登录。无论是使用 Session 还是 JWT Token,都可以让用户在 1 天内无需重新登录。如果您的应用需要分布式架构支持,Redis 是一个很好的选择,用于持久化存储 Session 数据。根据应用的需求,您可以选择适合的方案来优化用户体验。



http://www.kler.cn/news/321737.html

相关文章:

  • SpringBoot实现自定义Redis的连接
  • 如何将二氧化碳“封”入海底?
  • 顶象滑块、顶象验证码就这?2024-09-27 最新版(持续更新)确定不点进来看看?看到就是赚到
  • 【心灵解药】面对烦躁不安,这几招让你瞬间找回宁静与平和!
  • scrapy之setting文件详解
  • 更新 Git 软件
  • modbus 的float与uint_16的转换
  • 双十一有哪些数码家电值得入手?双十一五款必入手名单大曝光
  • Python 入门(一、使用 VSCode 开发 Python 环境搭建)
  • 数学建模练习小题目
  • 嵌入式项目:STM32平衡车详解 (基础知识篇) (基于STM32F103C8T6)
  • 基于Ambari搭建hadoop生态圈+Centos7安装教程V2.0优化版(本篇博客写的较为详细,可能比较多,请耐心看)
  • Android在外部存储上读写图片文件
  • 【python】range 语句
  • NLP 生成式任务核心梳理
  • react通过下拉框选择多个,并展示在下方的方式
  • 看Threejs好玩示例,学习创新与技术(React-three-fiber)
  • 【C++篇】从零实现 C++ Vector:深度剖析 STL 的核心机制与优化
  • SpringCloud源码:客户端分析(二)- 客户端源码分析
  • ArduSub程序学习(11)--EKF实现逻辑①
  • [AI问答] Auto-sklearn和Auto-Keras对比
  • Ubuntu20.04.6 环境下docker设置proxy
  • SpringBoot-Starter2.7.3自动装配Redisson升级版本运行时的问题
  • 自动驾驶技术:人工智能驾驶的未来
  • tauri程序加载本地图片或者文件在前端页面展示
  • ModStartCMS v8.9.0 图片上传优化,富文本编辑器修复
  • Spring Boot 实战:使用观察者模式实现实时库存管理
  • localectl 命令:系统语言、键盘布局和区域设置
  • CORE 中间件、wwwroot
  • C++11中引入的thread