我正在尝试使用 JWT 令牌“保护”我的应用程序登录,但似乎我的编码或解码错误。为什么说我的签名无效,而 JWT.io 说它是有效的?
我正在我的应用程序初始化中设置密钥,并使用它对 JWT 令牌进行编码和解码:
#skey = os.urandom(16)
#app.secret_key = f"{skey}"
app.config['SECRET_KEY'] = 'abe535ed6a554fe48e09e111dad2dcbc' #temporary for testing
app.debug = True
print(app.secret_key)
我正在使用此功能保护 dash
路线:
def token_required(f):
@wraps(f)
def decorated(*args, **kwargs):
token = request.args.get('token') #http://localhost:5000/route?token={jwttoken}
if not token:
return jsonify({'message': 'Token is missing'}), 403
try:
data = jwt.decode(token, app.secret_key)
except jwt.ExpiredSignatureError:
return jsonify({'message': 'Token expired, log in again'}), 403
except jwt.InvalidTokenError:
return jsonify({'message': 'Invalid token. Please log in again.'}), 403
return f(*args, **kwargs)
return decorated
这里是破折号路线:
@app.route('/dash', methods=["GET", "POST"])
@token_required
def dash():
if 'logged_in' in session:
if session['logged_in']:
username = session['username']
return render_template("dash.html", username=username)
else:
return redirect(url_for('home'))
else:
return redirect(url_for('home'))
使用正确的详细信息登录时,我的登录路由重定向到以令牌为参数的破折号:
@app.route('/login', methods=["GET", "POST"])
def login():
if request.method == "POST":
uname = request.form["uname"]
pswd = request.form["pswd"]
valid = auth.login(uname, pswd)
if valid[0]:
if valid[1]:
session['username'] = uname
session['logged_in'] = True
token = jwt.encode({
'user' : uname,
'exp' : datetime.datetime.utcnow() + datetime.timedelta(minutes=60)
},
app.secret_key)
return redirect(url_for('dash', token=token))
else:
return redirect(url_for('login'))
else:
return redirect(url_for('home'))
else:
return render_template("login.html")
但是当我成功登录时,它重定向到 dash,但总是说我有一个无效的令牌,尽管我已经在 jwt.io 上验证了它:
答案 0 :(得分:1)
您需要在解码令牌时指定算法。
data = jwt.decode(token, app.secret_key, algorithms=["HS256"])
试试这个就行了
答案 1 :(得分:0)
一个 HTML 页面只能是一种编码。创建令牌时编码可能不同。 UTF-8 可以支持多种语言,可以容纳页面和表单。
在创建令牌时添加 .decode('utf-8')
:
token = jwt.encode({
'user' : uname,
'exp' : datetime.datetime.utcnow() + datetime.timedelta(minutes=60)
},
app.secret_key).decode('utf-8')
请注意这里的解码不是来自 jwt 库。它只是用于将字符串编码为 UTF-8。