mirror of
				https://github.com/usememos/memos.git
				synced 2025-06-05 22:09:59 +02:00 
			
		
		
		
	chore: update access token ui
This commit is contained in:
		| @@ -98,7 +98,7 @@ func (s *APIV1Service) SignIn(c echo.Context) error { | ||||
| 		return echo.NewHTTPError(http.StatusUnauthorized, "Incorrect login credentials, please try again") | ||||
| 	} | ||||
|  | ||||
| 	accessToken, err := auth.GenerateAccessToken(user.Email, user.ID, time.Now().Add(auth.AccessTokenDuration), s.Secret) | ||||
| 	accessToken, err := auth.GenerateAccessToken(user.Username, user.ID, time.Now().Add(auth.AccessTokenDuration), s.Secret) | ||||
| 	if err != nil { | ||||
| 		return echo.NewHTTPError(http.StatusInternalServerError, fmt.Sprintf("failed to generate tokens, err: %s", err)).SetInternal(err) | ||||
| 	} | ||||
| @@ -222,7 +222,7 @@ func (s *APIV1Service) SignInSSO(c echo.Context) error { | ||||
| 		return echo.NewHTTPError(http.StatusForbidden, fmt.Sprintf("User has been archived with username %s", userInfo.Identifier)) | ||||
| 	} | ||||
|  | ||||
| 	accessToken, err := auth.GenerateAccessToken(user.Email, user.ID, time.Now().Add(auth.AccessTokenDuration), s.Secret) | ||||
| 	accessToken, err := auth.GenerateAccessToken(user.Username, user.ID, time.Now().Add(auth.AccessTokenDuration), s.Secret) | ||||
| 	if err != nil { | ||||
| 		return echo.NewHTTPError(http.StatusInternalServerError, fmt.Sprintf("failed to generate tokens, err: %s", err)).SetInternal(err) | ||||
| 	} | ||||
| @@ -318,7 +318,7 @@ func (s *APIV1Service) SignUp(c echo.Context) error { | ||||
| 	if err != nil { | ||||
| 		return echo.NewHTTPError(http.StatusInternalServerError, "Failed to create user").SetInternal(err) | ||||
| 	} | ||||
| 	accessToken, err := auth.GenerateAccessToken(user.Email, user.ID, time.Now().Add(auth.AccessTokenDuration), s.Secret) | ||||
| 	accessToken, err := auth.GenerateAccessToken(user.Username, user.ID, time.Now().Add(auth.AccessTokenDuration), s.Secret) | ||||
| 	if err != nil { | ||||
| 		return echo.NewHTTPError(http.StatusInternalServerError, fmt.Sprintf("failed to generate tokens, err: %s", err)).SetInternal(err) | ||||
| 	} | ||||
|   | ||||
| @@ -10,6 +10,7 @@ import ( | ||||
| 	"github.com/pkg/errors" | ||||
| 	"github.com/usememos/memos/api/auth" | ||||
| 	"github.com/usememos/memos/common/util" | ||||
| 	storepb "github.com/usememos/memos/proto/gen/store" | ||||
| 	"github.com/usememos/memos/store" | ||||
| ) | ||||
|  | ||||
| @@ -66,13 +67,15 @@ func JWTMiddleware(server *APIV1Service, next echo.HandlerFunc, secret string) e | ||||
| 			return next(c) | ||||
| 		} | ||||
|  | ||||
| 		println("path", path) | ||||
|  | ||||
| 		// Skip validation for server status endpoints. | ||||
| 		if util.HasPrefixes(path, "/api/v1/ping", "/api/v1/idp", "/api/v1/status", "/api/v1/user") && path != "/api/v1/user/me" && method == http.MethodGet { | ||||
| 			return next(c) | ||||
| 		} | ||||
|  | ||||
| 		token := findAccessToken(c) | ||||
| 		if token == "" { | ||||
| 		accessToken := findAccessToken(c) | ||||
| 		if accessToken == "" { | ||||
| 			// Allow the user to access the public endpoints. | ||||
| 			if util.HasPrefixes(path, "/o") { | ||||
| 				return next(c) | ||||
| @@ -85,7 +88,7 @@ func JWTMiddleware(server *APIV1Service, next echo.HandlerFunc, secret string) e | ||||
| 		} | ||||
|  | ||||
| 		claims := &auth.ClaimsMessage{} | ||||
| 		_, err := jwt.ParseWithClaims(token, claims, func(t *jwt.Token) (any, error) { | ||||
| 		_, err := jwt.ParseWithClaims(accessToken, claims, func(t *jwt.Token) (any, error) { | ||||
| 			if t.Method.Alg() != jwt.SigningMethodHS256.Name { | ||||
| 				return nil, errors.Errorf("unexpected access token signing method=%v, expect %v", t.Header["alg"], jwt.SigningMethodHS256) | ||||
| 			} | ||||
| @@ -98,6 +101,7 @@ func JWTMiddleware(server *APIV1Service, next echo.HandlerFunc, secret string) e | ||||
| 		}) | ||||
|  | ||||
| 		if err != nil { | ||||
| 			RemoveTokensAndCookies(c) | ||||
| 			return echo.NewHTTPError(http.StatusUnauthorized, errors.Wrap(err, "Invalid or expired access token")) | ||||
| 		} | ||||
| 		if !audienceContains(claims.Audience, auth.AccessTokenAudienceName) { | ||||
| @@ -110,6 +114,15 @@ func JWTMiddleware(server *APIV1Service, next echo.HandlerFunc, secret string) e | ||||
| 			return echo.NewHTTPError(http.StatusUnauthorized, "Malformed ID in the token.") | ||||
| 		} | ||||
|  | ||||
| 		accessTokens, err := server.Store.GetUserAccessTokens(ctx, userID) | ||||
| 		if err != nil { | ||||
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to get user access tokens.").WithInternal(err) | ||||
| 		} | ||||
| 		if !validateAccessToken(accessToken, accessTokens) { | ||||
| 			RemoveTokensAndCookies(c) | ||||
| 			return echo.NewHTTPError(http.StatusUnauthorized, "Invalid access token.") | ||||
| 		} | ||||
|  | ||||
| 		// Even if there is no error, we still need to make sure the user still exists. | ||||
| 		user, err := server.Store.GetUser(ctx, &store.FindUser{ | ||||
| 			ID: &userID, | ||||
| @@ -127,13 +140,16 @@ func JWTMiddleware(server *APIV1Service, next echo.HandlerFunc, secret string) e | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (s *APIV1Service) defaultAuthSkipper(c echo.Context) bool { | ||||
| func (*APIV1Service) defaultAuthSkipper(c echo.Context) bool { | ||||
| 	path := c.Path() | ||||
| 	return util.HasPrefixes(path, "/api/v1/auth") | ||||
| } | ||||
|  | ||||
| 	// Skip auth. | ||||
| 	if util.HasPrefixes(path, "/api/v1/auth") { | ||||
| 		return true | ||||
| func validateAccessToken(accessTokenString string, userAccessTokens []*storepb.AccessTokensUserSetting_AccessToken) bool { | ||||
| 	for _, userAccessToken := range userAccessTokens { | ||||
| 		if accessTokenString == userAccessToken.AccessToken { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return false | ||||
| } | ||||
|   | ||||
| @@ -8,6 +8,7 @@ import ( | ||||
| 	"github.com/golang-jwt/jwt/v4" | ||||
| 	"github.com/pkg/errors" | ||||
| 	"github.com/usememos/memos/api/auth" | ||||
| 	storepb "github.com/usememos/memos/proto/gen/store" | ||||
| 	"github.com/usememos/memos/store" | ||||
| 	"google.golang.org/grpc" | ||||
| 	"google.golang.org/grpc/codes" | ||||
| @@ -44,12 +45,12 @@ func (in *GRPCAuthInterceptor) AuthenticationInterceptor(ctx context.Context, re | ||||
| 	if !ok { | ||||
| 		return nil, status.Errorf(codes.Unauthenticated, "failed to parse metadata from incoming context") | ||||
| 	} | ||||
| 	accessTokenStr, err := getTokenFromMetadata(md) | ||||
| 	accessToken, err := getTokenFromMetadata(md) | ||||
| 	if err != nil { | ||||
| 		return nil, status.Errorf(codes.Unauthenticated, err.Error()) | ||||
| 	} | ||||
|  | ||||
| 	username, err := in.authenticate(ctx, accessTokenStr) | ||||
| 	username, err := in.authenticate(ctx, accessToken) | ||||
| 	if err != nil { | ||||
| 		if isUnauthorizeAllowedMethod(serverInfo.FullMethod) { | ||||
| 			return handler(ctx, request) | ||||
| @@ -74,12 +75,12 @@ func (in *GRPCAuthInterceptor) AuthenticationInterceptor(ctx context.Context, re | ||||
| 	return handler(childCtx, request) | ||||
| } | ||||
|  | ||||
| func (in *GRPCAuthInterceptor) authenticate(ctx context.Context, accessTokenStr string) (string, error) { | ||||
| 	if accessTokenStr == "" { | ||||
| func (in *GRPCAuthInterceptor) authenticate(ctx context.Context, accessToken string) (string, error) { | ||||
| 	if accessToken == "" { | ||||
| 		return "", status.Errorf(codes.Unauthenticated, "access token not found") | ||||
| 	} | ||||
| 	claims := &auth.ClaimsMessage{} | ||||
| 	_, err := jwt.ParseWithClaims(accessTokenStr, claims, func(t *jwt.Token) (any, error) { | ||||
| 	_, err := jwt.ParseWithClaims(accessToken, claims, func(t *jwt.Token) (any, error) { | ||||
| 		if t.Method.Alg() != jwt.SigningMethodHS256.Name { | ||||
| 			return nil, status.Errorf(codes.Unauthenticated, "unexpected access token signing method=%v, expect %v", t.Header["alg"], jwt.SigningMethodHS256) | ||||
| 		} | ||||
| @@ -115,6 +116,14 @@ func (in *GRPCAuthInterceptor) authenticate(ctx context.Context, accessTokenStr | ||||
| 		return "", errors.Errorf("user %q is archived", username) | ||||
| 	} | ||||
|  | ||||
| 	accessTokens, err := in.Store.GetUserAccessTokens(ctx, user.ID) | ||||
| 	if err != nil { | ||||
| 		return "", errors.Wrapf(err, "failed to get user access tokens") | ||||
| 	} | ||||
| 	if !validateAccessToken(accessToken, accessTokens) { | ||||
| 		return "", status.Errorf(codes.Unauthenticated, "invalid access token") | ||||
| 	} | ||||
|  | ||||
| 	return username, nil | ||||
| } | ||||
|  | ||||
| @@ -148,3 +157,12 @@ func audienceContains(audience jwt.ClaimStrings, token string) bool { | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| func validateAccessToken(accessTokenString string, userAccessTokens []*storepb.AccessTokensUserSetting_AccessToken) bool { | ||||
| 	for _, userAccessToken := range userAccessTokens { | ||||
| 		if accessTokenString == userAccessToken.AccessToken { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|   | ||||
| @@ -167,7 +167,7 @@ func (s *UserService) CreateUserAccessToken(ctx context.Context, request *apiv2p | ||||
| 		return nil, status.Errorf(codes.Internal, "failed to get current user: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	accessToken, err := auth.GenerateAccessToken(user.Email, user.ID, request.UserAccessToken.ExpiresAt.AsTime(), s.Secret) | ||||
| 	accessToken, err := auth.GenerateAccessToken(user.Username, user.ID, request.UserAccessToken.ExpiresAt.AsTime(), s.Secret) | ||||
| 	if err != nil { | ||||
| 		return nil, status.Errorf(codes.Internal, "failed to generate access token: %v", err) | ||||
| 	} | ||||
|   | ||||
		Reference in New Issue
	
	Block a user