1
0
mirror of https://github.com/matrix-org/matrix-authentication-service.git synced 2025-08-07 17:03:01 +03:00

JWT response wrapper

Helps replying with a JWT to a request, with a
`Content-Type: application/jwt` header
This commit is contained in:
Quentin Gliech
2022-09-01 17:47:12 +02:00
parent 6818316a11
commit 1f0e273ac3
5 changed files with 45 additions and 7 deletions

View File

@@ -0,0 +1,31 @@
// Copyright 2022 The Matrix.org Foundation C.I.C.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
use axum::{
response::{IntoResponse, Response},
TypedHeader,
};
use headers::ContentType;
use mas_jose::jwt::Jwt;
use mime::Mime;
pub struct JwtResponse<T>(pub Jwt<'static, T>);
impl<T> IntoResponse for JwtResponse<T> {
fn into_response(self) -> Response {
let application_jwt: Mime = "application/jwt".parse().unwrap();
let content_type = ContentType::from(application_jwt);
(TypedHeader(content_type), self.0.into_string()).into_response()
}
}

View File

@@ -21,6 +21,7 @@ pub mod client_authorization;
pub mod cookies; pub mod cookies;
pub mod csrf; pub mod csrf;
pub mod fancy_error; pub mod fancy_error;
pub mod jwt;
pub mod session; pub mod session;
pub mod user_authorization; pub mod user_authorization;

View File

@@ -16,17 +16,15 @@ use anyhow::Context;
use axum::{ use axum::{
extract::Extension, extract::Extension,
response::{IntoResponse, Response}, response::{IntoResponse, Response},
Json, TypedHeader, Json,
}; };
use headers::ContentType; use mas_axum_utils::{jwt::JwtResponse, user_authorization::UserAuthorization, FancyError};
use mas_axum_utils::{user_authorization::UserAuthorization, FancyError};
use mas_jose::{ use mas_jose::{
constraints::Constrainable, constraints::Constrainable,
jwt::{JsonWebSignatureHeader, Jwt}, jwt::{JsonWebSignatureHeader, Jwt},
}; };
use mas_keystore::Keystore; use mas_keystore::Keystore;
use mas_router::UrlBuilder; use mas_router::UrlBuilder;
use mime::Mime;
use oauth2_types::scope; use oauth2_types::scope;
use serde::Serialize; use serde::Serialize;
use serde_with::skip_serializing_none; use serde_with::skip_serializing_none;
@@ -91,9 +89,7 @@ pub async fn get(
}; };
let token = Jwt::sign(header, user_info, &signer)?; let token = Jwt::sign(header, user_info, &signer)?;
let application_jwt: Mime = "application/jwt".parse().unwrap(); Ok(JwtResponse(token).into_response())
let content_type = ContentType::from(application_jwt);
Ok((TypedHeader(content_type), token.as_str().to_owned()).into_response())
} else { } else {
Ok(Json(user_info).into_response()) Ok(Json(user_info).into_response())
} }

View File

@@ -85,6 +85,12 @@ pub enum DecodeError {
TooManyDots, TooManyDots,
} }
impl<'a> From<RawJwt<'a>> for String {
fn from(val: RawJwt<'a>) -> Self {
val.inner.into()
}
}
impl<'a> TryFrom<&'a str> for RawJwt<'a> { impl<'a> TryFrom<&'a str> for RawJwt<'a> {
type Error = DecodeError; type Error = DecodeError;
fn try_from(value: &'a str) -> Result<Self, Self::Error> { fn try_from(value: &'a str) -> Result<Self, Self::Error> {

View File

@@ -256,6 +256,10 @@ impl<'a, T> Jwt<'a, T> {
pub fn as_str(&'a self) -> &'a str { pub fn as_str(&'a self) -> &'a str {
&self.raw &self.raw
} }
pub fn into_string(self) -> String {
self.raw.into()
}
} }
#[derive(Debug, Error)] #[derive(Debug, Error)]