使用特定于租户的端点或将应用程序配置为多租户

如何解决使用特定于租户的端点或将应用程序配置为多租户

我正在将Spring Boot应用程序从ADAL迁移到MSAL。

我收到如下所示的错误:

java.util.concurrent.ExecutionException: com.microsoft.aad.msal4j.MsalServiceException: 
AADSTS50194: Application 'fd0ac989-0246-4999-b562-6d42d3636c22'(primadollardev_solanapi) is not configured as a multi-tenant application. 
Usage of the /common endpoint is not supported for such applications created after '10/15/2018'. 
Use a tenant-specific endpoint or configure the application to be multi-tenant.
Trace ID: 49905cdc-df8e-4bbb-9f48-daa2fe893000
Correlation ID: d6957f88-b418-4351-b141-4a5fbb6c6a99
Timestamp: 2020-08-30 06:05:40Z

ADAL代码

/**
 * Copyright (c) Microsoft Corporation. All rights reserved.
 * Licensed under the MIT License. See LICENSE in the project root for
 * license information.
 */

    package com.microsoft.azure.spring.autoconfigure.aad;
    
    import com.microsoft.aad.adal4j.ClientCredential;
    import com.nimbusds.jose.JOSEException;
    import com.nimbusds.jose.proc.BadJOSEException;
    import com.nimbusds.jose.util.ResourceRetriever;
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.security.core.Authentication;
    import org.springframework.security.core.context.SecurityContextHolder;
    import org.springframework.security.web.authentication.preauth.PreAuthenticatedAuthenticationToken;
    import org.springframework.web.filter.OncePerRequestFilter;
    
    import javax.naming.ServiceUnavailableException;
    import javax.servlet.FilterChain;
    import javax.servlet.ServletException;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    import java.io.IOException;
    import java.net.MalformedURLException;
    import java.text.ParseException;
    import java.util.concurrent.ExecutionException;
    
    public class AADAuthenticationFilter extends OncePerRequestFilter {
        private static final Logger log = LoggerFactory.getLogger(AADAuthenticationFilter.class);
    
        private static final String CURRENT_USER_PRINCIPAL = "CURRENT_USER_PRINCIPAL";
        private static final String CURRENT_USER_PRINCIPAL_GRAPHAPI_TOKEN = "CURRENT_USER_PRINCIPAL_GRAPHAPI_TOKEN";
        private static final String CURRENT_USER_PRINCIPAL_JWT_TOKEN = "CURRENT_USER_PRINCIPAL_JWT_TOKEN";
    
        private static final String TOKEN_HEADER = "Authorization";
        private static final String TOKEN_TYPE = "Bearer ";
    
        private AADAuthenticationProperties aadAuthProps;
        private ServiceEndpointsProperties serviceEndpointsProps;
        private UserPrincipalManager principalManager;
    
        public AADAuthenticationFilter(AADAuthenticationProperties aadAuthProps,ServiceEndpointsProperties serviceEndpointsProps,ResourceRetriever resourceRetriever) {
            this.aadAuthProps = aadAuthProps;
            this.serviceEndpointsProps = serviceEndpointsProps;
            this.principalManager = new UserPrincipalManager(serviceEndpointsProps,aadAuthProps,resourceRetriever,false);
        }
    
        @Override
        protected void doFilterInternal(HttpServletRequest request,HttpServletResponse response,FilterChain filterChain) throws ServletException,IOException {
    
            final String authHeader = request.getHeader(TOKEN_HEADER);
    
            if (authHeader != null && authHeader.startsWith(TOKEN_TYPE)) {
                try {
                    final String idToken = authHeader.replace(TOKEN_TYPE,"");
                    UserPrincipal principal = (UserPrincipal) request
                            .getSession().getAttribute(CURRENT_USER_PRINCIPAL);
                    String graphApiToken = (String) request
                            .getSession().getAttribute(CURRENT_USER_PRINCIPAL_GRAPHAPI_TOKEN);
                    final String currentToken = (String) request
                            .getSession().getAttribute(CURRENT_USER_PRINCIPAL_JWT_TOKEN);
    
                    final ClientCredential credential =
                            new ClientCredential(aadAuthProps.getClientId(),aadAuthProps.getClientSecret());
    
                    final AzureADGraphClient client =
                            new AzureADGraphClient(credential,serviceEndpointsProps);
    
                    if (principal == null ||
                        graphApiToken == null ||
                        graphApiToken.isEmpty() ||
                        !idToken.equals(currentToken)
                    ) {
                        principal = principalManager.buildUserPrincipal(idToken);
    
                        final String tenantId = principal.getClaim().toString();
                        graphApiToken = client.acquireTokenForGraphApi(idToken,tenantId).getAccessToken();
    
                        principal.setUserGroups(client.getGroups(graphApiToken));
    
                        request.getSession().setAttribute(CURRENT_USER_PRINCIPAL,principal);
                        request.getSession().setAttribute(CURRENT_USER_PRINCIPAL_GRAPHAPI_TOKEN,graphApiToken);
                        request.getSession().setAttribute(CURRENT_USER_PRINCIPAL_JWT_TOKEN,idToken);
                    }
    
                    final Authentication authentication = new PreAuthenticatedAuthenticationToken(
                            principal,null,client.convertGroupsToGrantedAuthorities(principal.getUserGroups()));
    
                    authentication.setAuthenticated(true);
                    log.info("Request token verification success. {}",authentication);
                    SecurityContextHolder.getContext().setAuthentication(authentication);
                } catch (MalformedURLException | ParseException | BadJOSEException | JOSEException ex) {
                    log.error("Failed to initialize UserPrincipal.",ex);
                    throw new ServletException(ex);
                } catch (ServiceUnavailableException | InterruptedException | ExecutionException ex) {
                    log.error("Failed to acquire graph api token.",ex);
                    throw new ServletException(ex);
                }
            }
    
            filterChain.doFilter(request,response);
        }
    }

MSAL代码

    public class MSALAuthenticationFilter extends OncePerRequestFilter {
        private static final Logger log = LoggerFactory.getLogger(MSALAuthenticationFilter.class);
    
        private static final String TOKEN_HEADER = "Authorization";
        private static final String TOKEN_TYPE = "Bearer ";
    
        // Properties from the application.properties file like clientId,tenant and stuff.
    
        private static final String CURRENT_USER_PRINCIPAL = "CURRENT_USER_PRINCIPAL";
        private static final String CURRENT_USER_ACCESS_TOKEN = "CURRENT_USER_ACCESS_TOKEN";
    
        @Override
        protected void doFilterInternal(HttpServletRequest request,IOException {
    
            final String authHeader = request.getHeader(TOKEN_HEADER);
            UserPrincipal principal = (UserPrincipal) request.getSession().getAttribute(CURRENT_USER_PRINCIPAL);
    
            if (authHeader != null && authHeader.startsWith(TOKEN_TYPE)) {
                try {
                    final String idToken = authHeader.replace(TOKEN_TYPE,"");
    
                    ConfidentialClientApplication clientApplication = ConfidentialClientApplication.builder(
                            clientId,ClientCredentialFactory.create(clientSecret))
                            .authority(authority)
                            .build();
    
                    Set<String> scopes = new HashSet<>(Arrays.asList(scope.split(" ")));
                    UserAssertion assertion = new UserAssertion(idToken);
    
                    OnBehalfOfParameters params = OnBehalfOfParameters.builder(scopes,assertion).build();
                    CompletableFuture<AuthenticationResult> future = clientApplication.acquireToken(params);
    
                    AuthenticationResult accessToken = future.get();
    
                    if (principal == null) {
                        principal = principalManager.buildUserPrincipal(idToken,accessToken);
    
                        request.getSession().setAttribute(CURRENT_USER_PRINCIPAL,principal);
                        request.getSession().setAttribute(CURRENT_USER_ACCESS_TOKEN,accessToken);
                    }
                    final Authentication authentication = new PreAuthenticatedAuthenticationToken(
                            principal,convertGroupsToGrantedAuthorities(principal.getUserGroups()));
    
                    authentication.setAuthenticated(true);
                    log.info("Request token verification success. {}",authentication);
                    SecurityContextHolder.getContext().setAuthentication(authentication);
    
    
                }
                catch (MalformedURLException | InterruptedException | ExecutionException ex) {
                    log.error("Failed to authenticate",ex);
                    throw new ServletException(ex);
                }
    
            }    
            filterChain.doFilter(request,response);
        }    
    }

请给我一个解决方案...

解决方法

这似乎是版本问题,请参考此GitHub issue。请更新您的版本为2.2.4。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 &lt;select id=&quot;xxx&quot;&gt; SELECT di.id, di.name, di.work_type, di.updated... &lt;where&gt; &lt;if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 &lt;property name=&quot;dynamic.classpath&quot; value=&quot;tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-