spring安全性_具有PreAuthorize的Spring方法安全性

spring安全性

朋友不允许朋友写用户身份验证。 厌倦了管理自己的用户? 立即尝试Okta的API和Java SDK。 数分钟之内即可在任何应用程序中对用户进行身份验证,管理和保护。

本教程将探讨使用Spring Security在Spring Boot中配置身份验证和授权的两种方法。 一种方法是创建WebSecurityConfigurerAdapter并使用流畅的API覆盖HttpSecurity对象上的默认设置。 另一个方法是在控制器方法上使用@PreAuthorize批注,称为方法级安全性或基于表达式的安全性。 后者将是本教程的重点。 但是,我将通过对比介绍一些HttpSecurity代码和想法。

第一种身份验证方法是HttpSecurity ,它是全局的,默认情况下应用于所有请求。 但是,可以使用端点的模式匹配来进行更细粒度的控制,并且HttpSecurity公开的流畅API十分强大。 在此处公开了OAuth 2.0,表单登录和HTTP Basic等配置选项。 这是设置全局身份验证策略的好地方。

方法级别的安全性是通过将@PreAuthorize批注放置在控制器方法上(实际上是一组可用的批注之一,但最常用)来实现。 该批注包含一个Spring Expression Language(SpEL)代码段,该代码段经过评估以确定是否应该对该请求进行身份验证。 如果未授予访问权限,则不执行该方法,并返回HTTP Unauthorized。 实际上,在控制器方法上使用@PreAuthorize注释与在特定端点上使用HttpSecurity模式匹配器非常相似。 但是有一些区别。

区分Spring Security的@PreAuthorize和HttpSecurity

第一个区别是微妙的,但值得一提。 在发生控制器映射之前, HttpSecurity方法会在Web请求筛选器中更早地拒绝该请求。 相比之下, @PreAuthorize评估在控制器方法执行之前@PreAuthorize进行。 这意味着在HttpSecurity 之前应用 @PreAuthorize中的配置。

其次, HttpSecurity绑定到URL端点,而@PreAuthorize绑定到控制器方法,并且实际上位于与控制器定义相邻的代码内 。 将所有安全性集中在一处并由Web终结点定义具有一定的整洁性,尤其是在较小的项目或更全局的设置中; 但是,随着项目的扩大,使授权策略靠近受保护的代码可能更有意义,这就是基于注释的方法所允许的。

另一个优点是@PreAuthorize呈现在HttpSecurity就是利用规划环境地政司的。 使用Spring Expression Language,您可以基于复杂的表达式来做出授权决策,这些复杂的表达式可以访问内置的身份验证对象(例如authenticationprincipal ),注入依赖项的方法参数和查询参数。 在本教程中,您将主要看两个表达式: hasAuthority()hasRole() 。 Spring文档再次是深入研究的好地方。

在深入研究项目之前,我还要提到Spring还提供了@PostAuthorize批注。 毫不奇怪,这是在方法执行评估的方法级授权注释。 我们为什么要这样做? 它允许方法在评估注释之前根据喜欢的任何控制器逻辑执行自己的授权检查。 缺点是,由于控制器方法是在评估注释之前执行的,因此可能会导致效率低下,具体取决于实现方式。

依存关系

本教程的依赖项非常简单。 您需要:1)已安装Java 8+,以及2)Okta开发人员帐户。

如果没有安装Java,请转到AdoptOpenJDK 。 在* nix系统上,您也可以使用SDKMAN 。

如果您还没有免费的Okta开发者帐户,请访问我们的网站并注册 。

使用Spring Initializr启动示例项目

要启动项目,可以使用Spring Initializr 。 但是,尽管值得一游,但您甚至不必费心去那里创建项目。 您可以使用REST API和curl

打开终端,然后将cd转到您希望项目文件.zip结束的任何位置。 运行以下命令,该命令将下载压缩的Spring Boot项目。

curl https://start.spring.io/starter.zip \-d dependencies=web,security \-d type=gradle-project \-d bootVersion=2.1.5.RELEASE \-d groupId=com.okta.preauthorize \-d artifactId=application \-o PreAuthorizeProject.zip
unzip PreAuthorizeProject.zip

除了build.gradle文件和DemoApplication.java类文件之外,该项目没有多少其他内容。 但是,已经为您设置了整个项目结构。

对于本示例, build.gradle文件还具有两个Spring依赖项:

dependencies {implementation 'org.springframework.boot:spring-boot-starter-security'implementation 'org.springframework.boot:spring-boot-starter-web'
}

添加一个WebController

处于当前状态的示例应用程序执行的操作不多。 您需要添加一个控制器来定义一些端点和响应。

添加一个新文件src/main/java/com/okta/preauthorize/application/WebController.java

package com.okta.preauthorize.application;  import org.springframework.stereotype.Controller;  
import org.springframework.web.bind.annotation.RequestMapping;  
import org.springframework.web.bind.annotation.ResponseBody;  @Controller  
public class WebController {  @RequestMapping("/")  @ResponseBody  public String home() {  return "Welcome home!";  }  @RequestMapping("/restricted")  @ResponseBody  public String restricted() {  return "You found the secret lair!";  }
}

该控制器定义两个端点: //restricted 。 您将稍后向/restricted端点添加方法级安全性。 但是,目前尚未配置安全性。

继续运行该应用程序。 从项目根目录运行:

./gradlew bootRun

Spring Boot完成启动后,导航至http://localhost:8080

您会注意到出现了一个登录表单。 哇! 那个是从哪里来的?!

该表单是由Spring Boot自动生成的。 看一下Spring类WebSecurityConfigurerAdapter和方法configure(HttpSecurity http) 。 这是配置默认身份验证设置的位置。

/**  * Override this method to configure the {@link HttpSecurity}. Typically subclasses  * should not invoke this method by calling super as it may override their * configuration. The default configuration is: 
* 
*
  
* http.authorizeRequests().anyRequest().authenticated().and().formLogin().and().httpBasic(); 
* 
* * @param http the {@link HttpSecurity} to modify * @throws Exception if an error occurs */ protected void configure(HttpSecurity http) throws Exception { ... http .authorizeRequests() .anyRequest().authenticated() .and() .formLogin().and() .httpBasic(); }

默认情况下,所有请求都需要身份验证。 对所有请求都启用了基于表单的身份验证和HTTP Basic身份验证。

如果要登录,请在控制台输出中查找类似于以下内容的行,以告诉您所生成的密码:

Using generated security password: 9c299bb9-f561-4c12-9810-c9a2bc1dca08

用户名就是“用户”。 每次运行Spring Boot时都会重新生成此密码。

身份验证与授权

在继续进行之前,我想快速确保两个术语明确:身份验证和授权。 身份验证回答问题:谁在发出请求? 授权回答了这个问题:他们被允许做什么?

通常,首先进行身份验证,除非为匿名用户设置了特定的权限(在某些方面这是隐式身份验证)。 授权基于经过身份验证的用户的价值。 认证的实体可以是人类用户或自动化服务,也可以是代表人类用户运行的服务。

两种常见的授权方案基于组和角色。 这两个术语通常在网络上不太知名的地方被混用并互换使用,但是在官方上有区别。 组定义用户组,并向这些用户组分配权限。 用户可以是多个组的成员。 角色定义可以分配给用户的权限集(允许的操作或资源)。 在实践中,组往往是一种更静态,更不灵活的控制器访问方式,而角色通常是精细划分的,甚至在会话中也可以是动态的,为特定任务分配角色,并在不再需要它们时将其撤销。 任何使用过Amazon AWS的人都已经看到了这种效果,通常会产生令人困惑的效果。

为Spring @PreAuthorize启用方法级安全性

您现在想要做的是配置Spring Boot以允许home端点上的请求,同时将请求限制为/restricted端点。

您最初可能以为可以将@PreAuthorize("permitAll()")到本地端点,这将允许所有请求。 但是,如果尝试使用它,您会发现它什么也没做。 这是因为默认的HttpBuilder实现仍处于活动状态,并且因为它是在Web请求过滤器链中进行评估的,因此它具有优先权。 您还必须显式启用方法级安全注释,否则,它们将被忽略。

添加以下SecurityConfig类,该类将实现以上两个目标。

src/main/java/com/okta/preauthorize/application/SecurityConfig.java

package com.okta.preauthorize.application;  import org.springframework.context.annotation.Configuration;  
import org.springframework.security.config.annotation.method.configuration.EnableGlobalMethodSecurity;  
import org.springframework.security.config.annotation.web.builders.HttpSecurity;  
import org.springframework.security.config.annotation.web.configuration.WebSecurityConfigurerAdapter;  @Configuration  
@EnableGlobalMethodSecurity(prePostEnabled = true)  
public class SecurityConfig extends WebSecurityConfigurerAdapter {  protected void configure(final HttpSecurity http) throws Exception {}  
}

@EnableGlobalMethodSecurity(prePostEnabled = true)注释启用了@PreAuthorize注释。 可以使用@Configuration批注将其添加到任何类。 在这里,我将不做任何深入的介绍,但是您也可以启用@Secured ,较旧的Spring Security注释和JSR-250注释。

configure(final HttpSecurity http)方法将覆盖默认的HttpBuilder配置。 因为它是空的,所以它使应用程序未经授权或认证。

使用./gradlew bootRun再次运行该应用程序,您会发现两个端点现已完全打开。

实施全球安全政策

应用程序通常必须选择要围绕其构建安全性的全局安全策略:“默认为允许”或“默认为已验证”。 该应用程序默认情况下是否打开? 还是默认情况下受限制? 我通常默认为受限,并明确允许任何公共端点。 对于我所使用的专有Web应用程序类型,这些类型通常不是公开的,或者具有相对较小的公开面Kong,因此该方案有意义。 但是,如果您要进行的工作大部分是公开的,并且要有离散的访问控制支持,例如带有管理面板的网站,那么可能会采用更宽松的方案。 您将在此处查看两者。

由于该应用已经开放,因此我将首先向您展示如何限制特定方法。 在那之后,您将研究几种方法来实施更具全局限制的访问策略。

WebController类中,将@PreAuthorize批注添加到/restricted端点,如下所示:

@PreAuthorize("isAuthenticated()")  
@RequestMapping("/restricted")  
@ResponseBody  
public String restricted() {  return "You found the secret lair!";  
}

运行应用程序( ./gradlew bootRun )。

这次,您将能够导航到主页,但是转到/restricted端点将为您提供(非常丑陋的)whitelabel错误页面。

在生产应用程序中,您需要重写此代码以返回更好的自定义错误页面,或者以其他方式处理错误(如果您要创建自定义错误页面,请查看主题的Spring文档 )。 但是您可以看到该应用返回的是403 /未经授权 ,这就是您想要的。

大。 现在,改为更改您的WebController类以匹配以下内容:

@Controller
@PreAuthorize("isAuthenticated()")    
public class WebController {  @PreAuthorize("permitAll()")  @RequestMapping("/")  @ResponseBody  public String home() {  return "Welcome home!";  }  @RequestMapping("/restricted")  @ResponseBody  public String restricted() {  return "You found the secret lair!";  }  }

在这里,您已使用@PreAuthorize批注将整个控制器类限制为经过身份验证的用户,并明确允许所有请求(无论身份验证状态如何)到本地端点。

我知道我们一直将其称为“方法级”安全性,但是实际上,也可以将这些@PreAuthorize批注添加到控制器类中,以为整个类设置默认值。 这也是@PreAuthorize("permitAll()")派上用场的地方,因为它可以覆盖类级别的注释。

如果运行该应用程序( ./gradlew bootRun )并尝试使用端点,则会发现home端点处于打开状态,而/restricted端点处于关闭状态。

请注意,如果添加了另一个单独的Web控制器,则默认情况下,其所有方法仍将处于打开状态,并且不需要身份验证。

第三种选择(我最喜欢的是大多数中小型应用程序)是使用HttpBuilder在默认情况下要求对所有请求进行身份验证并显式允许任何公共端点。 这使您可以使用@PreAuthorize来基于用户或角色或组来优化对特定方法的访问控制,但要明确指出, 除非明确允许否则所有路径都将应用一些基本安全性。 这也意味着公共路径定义在中心位置。 同样,这适用于某种类型的项目,但可能不是所有项目的最佳结构。

要实现此目的,请将WebController类更改为此(删除所有@PreAuthorize批注):

@Controller  
public class WebController {  @RequestMapping("/")  @ResponseBody  public String home() {  return "Welcome home!";  }  @RequestMapping("/restricted")  @ResponseBody  public String restricted() {  return "You found the secret lair!";  }  }

并将WebSecurity类更改为:

Configuration  
@EnableGlobalMethodSecurity(prePostEnabled = true)  
public class SecurityConfig extends WebSecurityConfigurerAdapter {  protected void configure(final HttpSecurity http) throws Exception {  http.antMatcher("/**")  .authorizeRequests()  .antMatchers("/").permitAll()  .anyRequest().authenticated().and().formLogin();  }  
}

您要做的是使用SecurityConfig类显式允许本地终结点计算机上的所有请求,同时要求在所有其他终结点计算机上进行身份验证。 这为您的应用设置了全面的最低身份验证要求。 您还重新启用了基于表单的身份验证。

试试吧!

使用以下./gradlew bootRun运行应用程序: ./gradlew bootRun

导航到Home端点,该端点已打开: http://localhost:8080

以及需要认证的受限端点: http://localhost:8080/restricted

当Spring的登录表单出现时,别忘了您可以使用默认凭据。 用户是“用户”,并且在控制台输出中找到了密码(查找Using generated security password:

前进到OAuth 2.0登录

这些天,基于表单的身份验证感觉非常破旧。 用户越来越希望能够使用第三方站点登录,由于安全威胁的增加,在您自己的服务器上管理用户凭据的动机也越来越少。 Okta是一家提供服务的软件即服务身份和访问管理公司。 在本部分中,您将使用它们通过OAuth 2.0和OIDC(OpenID Connect)快速实现登录表单。

非常非常简短:OAuth 2.0是一种行业标准的授权协议,而OIDC是OAuth之上的另一个开放标准,它添加了一个身份层(身份验证)。 它们共同为程序提供了一种结构化的方式来管理身份验证和授权,以及跨网络和Internet进行通信。 但是,OAuth和OIDC均未提供实现。 它们只是规格或协议。 这就是Okta的用处。Okta实现了OAuth 2.0和OIDC规范的实现,该规范允许程序使用其服务来快速提供登录,注册和单点登录(或社交登录)服务。 在本教程中,您将仅实现登录功能,但是在本教程的最后,您可以找到指向其他资源的链接,以向您展示如何实现社交登录和注册。

首先,注册一个免费的Okta Developer帐户: https : //developer.okta.com/signup/ 。

如果这是您首次登录或刚刚注册,则可能需要单击Admin按钮才能进入开发人员控制台。

接下来,您需要配置OIDC应用程序。

在Okta开发人员仪表板的顶部菜单中,单击应用程序

  • 单击绿色的“ 添加应用程序”按钮
  • 单击“ Web应用程序类型”,然后单击“下一步”。
  • 为应用命名。 任何名字。
  • 登录重定向URI设置为http://localhost:8080/login/oauth2/code/okta
  • 单击完成

记下页面底部的客户端ID客户端密钥 。 在下一部分中,您将需要这些。

在Okta方面就是这样。

现在,您需要配置Spring Boot以将Okta用作OAuth 2.0提供程序。

为OAuth 2.0配置Spring Boot应用

使用OAuth 2.0和Okta使Spring Boot变得非常简单。 第一步是添加Okta Spring Boot Starter依赖项。 无需使用我们的启动程序就可以完全使用Okta OAuth 2.0 / OIDC; 但是,启动器简化了配置。 它还处理从JSON Web令牌中提取组声明,并将其转变为Spring Security授权(稍后将介绍)。 查看Okta Spring Boot Starter GitHub页面以获取更多信息。

更新build.gradle文件的依赖项部分:

dependencies {  implementation 'com.okta.spring:okta-spring-boot-starter:1.2.1'  // <-- ADDEDimplementation 'org.springframework.boot:spring-boot-starter-security'  implementation 'org.springframework.boot:spring-boot-starter-web'  testImplementation 'org.springframework.boot:spring-boot-starter-test'  testImplementation 'org.springframework.security:spring-security-test'  
}

src/main/resources目录中,有一个application.properties文件。 将其重命名为application.yml 。 添加以下内容:

okta:oauth2:  issuer: https://{yourOktaDomain}/oauth2/default  client-id: {yourClientId}client-secret: {yourClientSecret}

不要忘记更新client-idclient-secretissuer值以匹配Okta开发人员帐户和OIDC应用程序中的值。 Okta发行者的外观应类似于https://dev-123456.okta.com/oauth2/default

最后,更新SecurityConfiguration.java文件:

@Configuration  
@EnableGlobalMethodSecurity(prePostEnabled = true)  
public class SecurityConfig extends WebSecurityConfigurerAdapter {  protected void configure(final HttpSecurity http) throws Exception {  http.antMatcher("/**")  .authorizeRequests()  .antMatchers("/").permitAll()  .anyRequest().authenticated()  .and().oauth2Login();  // <-- THIS WAS CHANGED}  
}

请注意,您在这里真正进行的更改只是从formLogin()oauth2Login()

运行应用程序: ./gradlew bootRun (您可能需要退出Okta开发人员仪表板,或者使用隐身窗口查看登录屏幕)。

/端点仍然打开,但是当您访问受限端点时: http://localhost:8080/restricted

您将看到Okta登录屏幕。

使用您的Okta凭据登录,即可通过身份验证!

检查OAuth 2.0用户属性

开发OAuth应用程序时,我发现能够检查Spring Boot关于客户端和已认证用户的信息会很有帮助。 为此,添加一个名为UserInfoController.java的新控制器。

src/main/java/com/okta/preauthorize/application/UserInfoController.java

package com.okta.preauthorize.application;  import org.springframework.security.core.annotation.AuthenticationPrincipal;  
import org.springframework.security.oauth2.client.OAuth2AuthorizedClient;  
import org.springframework.security.oauth2.client.annotation.RegisteredOAuth2AuthorizedClient;  
import org.springframework.security.oauth2.core.user.OAuth2User;  
import org.springframework.stereotype.Controller;  
import org.springframework.web.bind.annotation.RequestMapping;  
import org.springframework.web.bind.annotation.ResponseBody;  import java.util.Map;  @RequestMapping("/user")  
@Controller  
public class UserInfoController {  @RequestMapping("/oauthinfo")  @ResponseBody  public String oauthUserInfo(@RegisteredOAuth2AuthorizedClient OAuth2AuthorizedClient authorizedClient,  @AuthenticationPrincipal OAuth2User oauth2User) {  return  "User Name: " + oauth2User.getName() + "<br/>" +  "User Authorities: " + oauth2User.getAuthorities() + "<br/>" +  "Client Name: " + authorizedClient.getClientRegistration().getClientName() + "<br/>" +  this.prettyPrintAttributes(oauth2User.getAttributes());  }  private String prettyPrintAttributes(Map<String, Object> attributes) {  String acc = "User Attributes: <br/><div style='padding-left:20px'>";  for (String key : attributes.keySet()){  Object value = attributes.get(key);  acc += "<div>"+key + ":&nbsp" + value.toString() + "</div>";  }  return acc + "</div>";  }  
}

请注意,此类定义了全类请求映射路径/user ,从而形成了实际的oauthUserInfo()端点/user/oauthinfo

第二种方法prettyPrintAttributes()只是一些格式化用户属性的糖,使它们更具可读性。

运行应用程序: ./gradlew bootRun

导航到http://localhost:8080/user/oauthinfo

您会看到以下内容:

User Name: 00ab834zk7eJ18e8Y0h7  
User Authorities: [ROLE_USER, SCOPE_email, SCOPE_openid, SCOPE_profile]  
Client Name: Okta  
User Attributes:  at_hash: 1yq0lbHDupcb8AhBNShkeQsub: 00ue9mlzk7eW24e8Y0h7zoneinfo: America/Los_Angelesver: 1email_verified: trueamr: ["pwd"]iss: https://dev-123456.oktapreview.com/oauth2/defaultpreferred_username: andrew.hughes@mail.comlocale: en-USgiven_name: Andrewaud: [0oakz4teswoV7sDZI0h7]updated_at: 1558380884idp: 00oe9mlzh0xuqOT5z0h7auth_time: 1558454889name: Andrew Hughesexp: 2019-05-21T17:46:28Zfamily_name: Hughesiat: 2019-05-21T16:46:28Zemail: andrew.hughes@mail.comjti: ID.CnwVJ_h1Dq5unqkwherWyf8ZFTETX_X4TP39ythQ-ZE

我想指出几件事。 首先,请注意, 用户名实际上是Okta的OIDC应用程序的客户端ID。 这是因为,从Spring Boot OAuth的角度来看,客户端应用程序是“用户”。 要找到实际的用户名和信息,您必须在User Attributes查看 。 请记住,这些属性的实际内容在OAuth提供程序之间会有所不同,因此,例如,如果您支持Okta,GitHub和Twitter,则需要检查每个OAuth提供程序的这些属性以查看它们返回的内容。

另一个要点是用户权限 。 如Spring在这里所使用的,授权机构是授权信息的元术语。 它们只是字符串。 它们的值是从OAuth / OIDC信息中提取的。 客户端应用程序可以正确使用它们。 它们可以是角色,作用域,组等。就OAuth / OIDC而言,它们的用法基本上是任意的。

要查看其工作原理,在接下来的几节中,您将在Okta中添加一个Admin组,将用户分配给该组,并使用@PreAuthorize批注将方法限制为Admin组。

ROLE_USER :您会注意到,Spring已为所有经过身份验证的用户分配了ROLE_USER 。 这是自动分配的默认最低级别角色。

SCOPE_ :还应注意,OAuth范围已映射到Spring授权,并且可以用于授权,例如,在@PreAuthorize@PostAuthorize批注中,您将在@PostAuthorize秒钟后看到。

激活对Okta的团体声明

Okta默认情况下不将组声明包括在JSON Web令牌(JWT)中。 JWT是Okta用于将身份验证和授权信息传达到客户端应用程序的工具。 在本文章末尾链接的其他一些博客文章中,可以进行更深入的研究。

要将Okta配置为添加组声明,请转到Okta开发人员仪表板。

在顶部菜单中,转到API并选择授权服务器

选择默认授权服务器。

单击“ 索赔”选项卡。

您将要创建两个索赔映射。 您本身并不是在创建两个声明,而是指示Okta将组声明添加到访问令牌和ID令牌中。 您需要执行此操作,因为根据OAuth流的不同,可能会从任一组中提取组声明。 在我们的案例中,对于OIDC流程而言,实际上实际上是ID令牌,但最好将它们同时添加到两者中,以免日后沮丧。 资源服务器流要求这些组声明在访问令牌中。

首先,为令牌类型Access Token添加一个声明映射。

点击添加声明

更新以下值(其他默认值也可以):

  • 名称:团体
  • 包含在令牌类型中 :访问令牌
  • 值类型:
  • 过滤器:匹配正则表达式.*

其次,为令牌类型ID Token添加第二个索赔映射。

点击添加声明

更新以下值(除了令牌类型外,与上面的值相同):

  • 名称:团体
  • 包含在令牌类型中 :ID令牌
  • 值类型:
  • 过滤器:匹配正则表达式.*

大! 因此,现在Okta会将其所有组映射到有关访问令牌和ID令牌的groups声明。

在Spring,这些团体声称发生的事情不一定是显而易见的,也不是自动的。 Spring Boot启动程序的好处之一是,它会自动从JWT中提取组声明并将其映射到Spring授权机构。 否则,您将需要实现自己的GrantedAuthoritiesExtractor

仅供参考:可以使用application.yml文件中的okta.oauth2.groupsClaim字段来配置组声明的名称。 它默认为groups

使用组检查用户属性

运行应用程序: ./gradlew bootRun

导航到http://localhost:8080/user/oauthinfo

您将看到类似以下内容(为清楚起见,省略了许多多余的行):

User Name: 00ab834zk7eJ18e8Y0h7  
User Authorities: [Everyone, ROLE_USER, SCOPE_email, SCOPE_openid, SCOPE_profile]
Client Name: Okta  
User Attributes:  ...groups: ["Everyone"]...

请注意一个新的用户属性(映射的组声明)。 值Everyone是映射到Everyone的默认组。 这被映射到用户权限Everyone

这是基本思想。 添加管理员组后,下一步将更加令人兴奋。

在Okta中创建一个管理员组

现在,您想在Okta上添加一个管理员组。 登录到Okta开发人员仪表板。

在顶部菜单中,转到“ 用户”,然后选择“ 组”

单击添加组

在弹出窗口中:

  • 将组命名为“ Admin”。
  • 描述可以是您喜欢的任何内容。
  • 单击添加组

至此,您已经创建了Admin组, 但实际上尚未分配任何人!

使用方法级授权来限制端点

WebController类中,更新/restricted终结点方法。 您将在方法中添加以下注释:

@PreAuthorize("hasAuthority('Admin')")

像这样:

@PreAuthorize("hasAuthority('Admin')")  
@RequestMapping("/restricted")  
@ResponseBody  
public String restricted() {  return "You found the secret lair!";  
}

这告诉Spring检查经过身份验证的用户是否具有Admin权限,如果没有,则拒绝该请求。

运行应用程序: ./gradlew bootRun

导航到http://localhost:8080/restricted

您将收到403 /未经授权的白页错误。

将您的用户添加到管理员组

现在,您需要将Okta用户添加到Admin组。 从顶部菜单中,选择“ 用户” ,然后单击“ 组” 。 单击管理员组。 单击添加成员 。 在弹出窗口中搜索您的用户,然后单击添加

测试管理员组成员身份

做完了! 让我们看看这样做了。 再次运行Spring Boot应用程序: ./gradlew bootRun

导航到http://localhost/user/oauthinfo

这次,您将看到Admin组和权限。 (您可能需要一个新的浏览器会话才能看到更改,或使用隐身模式。)

User Name: 00ab834zk7eJ18e8Y0h7  
User Authorities: [Admin, Everyone, ROLE_USER, SCOPE_email, SCOPE_openid, SCOPE_profile]
Client Name: Okta  
User Attributes:  ...groups: ["Everyone", "Admin"]...

并且,如果您导航到http://localhost:8080/restricted ,则将被允许访问。

比较Spring Security角色和权限

最初让我感到困惑的一件事是hasRole()hasAuthority()

Spring中的角色是具有ROLE_前缀的授权机构(就像Spring中的所有其他事物一样,该前缀是可配置的)。 考虑这种情况的一种方法是,角色应用于大量权限,而权限可以用于更细粒度的控制。 但是,这只是一种可能的用法。 实际的实现取决于开发人员。 在本教程中,您实际上是在使用权限映射到授权组。

要记住的重要一点是,如果要使用hasRole() ,则需要声明中的授权名称以ROLE_ 。 例如,如果您添加了一个ROLE_ADMIN组,并将您的用户添加到该组,并将该组添加到OIDC应用程序,则可以使用hasRole('ADMIN')

基于OAuth 2.0范围的授权和Spring PreAuthorize

您还可以使用@PreAuthorize批注来限制基于OAuth范围的访问。 从OAuth 2.0范围文档中 :

范围是OAuth 2.0中的一种机制,用于限制应用程序对用户帐户的访问。 应用程序可以请求一个或多个范围,然后在同意屏幕中将此信息呈现给用户,并且颁发给该应用程序的访问令牌将限于所授予的范围。

如果查看从/user/oauthinfo端点返回的经过检查的User Authorities ,则会看到三个以SCOPE_开头的SCOPE_

  • SCOPE_email
  • SCOPE_openid
  • SCOPE_profile

这些对应于电子邮件,openid和配置文件范围。 要将方法限制为具有特定范围的用户,应使用诸如: @PreAuthorize("hasAuthority('SCOPE_email')")的注释。

我还将指出,您可以使用SecurityConfig类中的HttpSecurity完成完全一样的事情(或多或少完全一样):

protected void configure(final HttpSecurity http) throws Exception {  http.antMatcher("/**")  .authorizeRequests()  .antMatchers("/").permitAll()  .antMatchers("/restricted").hasAuthority("SCOPE_custom") // <- LOOK AT ME!  .anyRequest().authenticated()  .and().oauth2Login();  
}

通过将scopes属性添加到application.yml文件,可以自定义客户端应用程序从Okta授权服务器请求的scopes 。 例如,在下面,我将application.yml文件设置为仅请求openid范围,这对于OAuth是必需的。

okta:  oauth2:  ... scopes: openid...

如果在/user/oauthinfo端点上运行此请求,则会得到以下信息:

User Name: 00ab834zk7eJ18e8Y0h7  
User Authorities: [Admin, Everyone, ROLE_USER, SCOPE_openid]  
Client Name: Okta  
...

请注意,只有一个范围已分配给用户权限。

尝试添加自定义范围。 更改application.yml文件中的okta.oauth2.scopes属性以使其匹配:

okta:  oauth2:  ... scopes: openid email profile custom...

在运行该应用并尝试之前,您需要将自定义范围添加到Okta授权服务器(如果现在运行它,将会出现错误)。

打开您的Okta开发人员仪表板。

在顶部菜单中,转到API并选择授权服务器

选择默认授权服务器。

单击“ 作用域”选项卡。

单击添加范围按钮。

  • 名称custom
  • 描述Custom test scope

点击创建

您刚刚向默认的Okta授权服务器添加了一个自定义范围(俗称为custom )。

最后一次,运行Spring Boot应用程序: ./gradlew bootRun

导航到http://localhost:8080/user/oauthinfo

User Name: 00ab834zk7eJ18e8Y0h7  
User Authorities: [Admin, Everyone, ROLE_USER, SCOPE_custom, SCOPE_email, SCOPE_openid, SCOPE_profile]  
Client Name: Okta  
...

成功! 您应该在用户权限中看到新的SCOPE_custom

Spring Boot中的Spring PreAuthorize,HttpSecurity和Security

您覆盖了一大堆土地! 您已经使用@PreAuthorize很好地了解了Spring方法级别的安全性,并了解了它与HttpSecurity 。 您使用了一些基本的SpEL(Spring表达式语言)语句来配置授权。 您检查了授权和身份验证之间的区别。 您将Spring Boot配置为使用Okta作为OAuth 2.0 / OIDC单一登录提供程序,并向身份验证服务器和客户端应用添加了组声明。 您甚至还创建了一个新的Admin组,并看到了如何使用映射到Spring授权的组声明来限制访问。 最后,您了解了OAuth 2.0范围如何用于定义授权方案并在应用程序中实现它们。

下一站:火箭科学!

如果您想查看这个完整的项目,可以在GithHub上找到该仓库 。

如果您想了解有关Spring Boot,Spring Security或安全用户管理的更多信息,请查看以下任何出色的教程:

  • Spring Boot,OAuth 2.0和Okta入门
  • 15分钟内将单一登录添加到您的Spring Boot Web App
  • 使用多重身份验证保护您的Spring Boot应用程序安全
  • 使用Spring Boot和GraphQL构建安全的API

如果您想更深入地学习,请查看Okta Spring Boot Starter GitHub Project 。

如果您对此帖子有任何疑问,请在下面添加评论。 有关更多精彩内容, 请在Twitter上关注@oktadev , 在Facebook上关注我们,或订阅我们的YouTube频道 。

“带有PreAuthorize的Spring方法安全性”最初于2019年6月20日发布在Okta Developer博客上。

朋友不允许朋友写用户身份验证。 厌倦了管理自己的用户? 立即尝试Okta的API和Java SDK。 数分钟之内即可在任何应用程序中对用户进行身份验证,管理和保护。

翻译自: https://www.javacodegeeks.com/2019/09/spring-method-security-preauthorize.html

spring安全性

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/332148.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

animiz动画制作软件_AN动画制作软件

AN基础介绍我们先了解一些基本概念&#xff0c;才能帮助我们更好的进行下面一系列的操作。1.图层。图层就像是含有文字或图形等元素的胶片&#xff0c;一张张按顺序叠放在一起&#xff0c;组合起来形成页面的最终效果。图层可以将页面上的元素精确定位。图层中可以加入文本、图…

关于一致/非一致代码段与TSS 关系的个人看法

【0】概念定义 0.1&#xff09;一致代码段: 简单理解&#xff0c;就是操作系统拿出来被共享的代码段,可以被低特权级的用户直接调用访问的代码&#xff0c; 但是特权级高的程序不允许访问特权级低的数据. 通常这些共享代码&#xff0c;是”不访问”受保护的资源和某些类型异…

python预处理标准化_tensorflow预处理:数据标准化的几种方法

tensorflow预处理&#xff1a;数据标准化的几种方法发布时间&#xff1a;2018-08-09 19:39,浏览次数&#xff1a;1774, 标签&#xff1a;tensorflow数据归一化问题是数据挖掘中特征向量表达时的重要问题&#xff0c;当不同的特征成列在一起的时候&#xff0c;由于特征本身表达方…

三思笔记_使用反射前先三思

三思笔记介绍 有时&#xff0c;作为开发人员&#xff0c;您可能会遇到无法使用new运算符实例化对象的情况&#xff0c;因为其类名称存储在配置XML中的某个位置&#xff0c;或者您需要调用一个名称指定为注释属性的方法。 在这种情况下&#xff0c;您总会有一个答案&#xff1a;…

分页机制总结

【0】写在前面&#xff08;分页机制&#xff09; 0.0&#xff09; source code from orange’s implemention of a os and text description from Zhaojiong’s perfect analysis of Linux kernel and for complete code ,please visit https://github.com/pacosonTang/Orange…

python程序填空题参照代码模板、完善代码_python二级考试操作题11.pdf

综合应用题参照代码模板完善代码&#xff0c;实现下述功能。文件ngchina.html 保持了网页源代码&#xff0c;请将该页面中图片的URL 提取出来,并输出所有图像的URL。习题讲解#P301#读取HTML 文件内容def getHTMLlines(htmlpath):f open(htmlpath,"r",encoding utf-…

Struts2学习笔记

文章目录Struts2 的核心开发包Struts2 配置文件Struts2 域对象Struts2 编程流程Action 组件使用通配符配置 ActionAction 中如何访问 Session通过 ActionContext 对象访问 Session 对象&#xff08;不推荐&#xff09;通过实现 SessionAware 接口访问 sessionAction 如何访问 r…

如何从文件系统中读取文件内容

【0】写在前面 0.0&#xff09; text description from orange’s implemention of a os &#xff0c;文末总结系个人臆测出的干货 【1】intro to FAT12&#xff08;file allocation table 12&#xff09;文件系统格式&#xff08;from Baidu Baike&#xff09; &#xff08;…

java微服务,微在哪_Java:ChronicleMap第3部分,快速微服务

java微服务,微在哪标准Java Maps需要在启动时进行初始化。 了解如何利用可从文件初始化的ChronicleMaps并显着减少微服务启动时间&#xff0c;以及如何在JVM之间共享Maps。 内置的Map实现&#xff08;例如HashMap和ConcurrentHashMap速度很快&#xff0c;但是必须先使用映射进…

excel离散度图表怎么算_Excel数据分析——离散值排除-excel直方图

今天举例的数据继续沿用昨天做出来的结果&#xff0c;至于这组数据还要接着用多久~~可能要混到我讲不下去为止吧~~~我们通过两个不同的拟合公式得到了两组不同的残差值&#xff0c;数据情况如下&#xff1a;有没有觉得看上面那张散点图有点糊啊&#xff1f;没错&#xff0c;问题…

drools dmn_Drools DMN最新开源引擎性能改进

drools dmn我们一直在寻求改善Drools DMN开源引擎的性能。 我们最近审查了DMN用例&#xff0c;其中输入数据节点的实际输入总体有所不同。 这突出显示了引擎的次佳性能&#xff0c;我们在最新版本中对此进行了改进。 我想分享我们的发现&#xff01; 基准制定 当我们开始为此用…

制作FAT12软盘以查看软盘的根目录条目+文件属性+文件内容

【-1】Before for specific info , please visit http://wiki.osdev.org/Loopback_Device 【0】我们先上干货&#xff0c;看到效果后&#xff0c;我们再说明每个步骤的缘由&#xff1b; 【1】进入挂载目录&#xff0c;添加相关文件&#xff08;依个人意愿&#xff09; Attenti…

如何取消高亮显示重复项_如何将重复数据突出显示?

将表格中一列数据中重复的&#xff0c;使用特殊颜色突出显示或者使用一些符号标记出来。例如&#xff1a;一个供应商&#xff0c;可以邀请别的供应商加入成为联合体&#xff0c;报表要显示所有供应商&#xff0c;然后供应商最后一列显示所有联合体&#xff0c;当联合体供应商跟…

Maven的maven-source-plugin插件详解

maven-source-plugin 这个插件专门负责将项目源文件打成包的&#xff0c;该插件在 pom.xml 中的配置如下&#xff1a; <build><plugins><plugin><artifactId>maven-source-plugin</artifactId><version>3.0.1</version><configu…

Maven Java Web Project打包详解/如何打包

文章目录打包部署构件&#xff08;Artifacts&#xff09;打源码包方式一&#xff1a; 命令行方式方式二&#xff1a;使用 IDE将源码包发布到本地 Maven 仓库中涉及到案例项目的结构&#xff1a; 打包部署构件&#xff08;Artifacts&#xff09; war 格式的部署构件可以直接放…

四位共阳极数码管显示函数_数码管模块.doc

数码管模块数码管1、概 述数码管模块采用四位共阳极数码管&#xff0c;用于显示数字和少数特殊字符。可以在机器人项目中使用该模块&#xff0c;用于显示速度、时间、分数、温度、距离等传感器的值。同时&#xff0c;Makeblock提供易于编程的Arduino库&#xff0c;使用户能够方…

java中的可检查和不检查_检查Java测试中发生了什么

java中的可检查和不检查有人想到了在Java单元测试中使用try和catch块的想法&#xff1a; Test public void test() { try { callSomeCode(); } catch (Exception e) { assertEquals( "foo" , e.getMessage()); } } 上面的内容很诱人&#xff0c;但不起作用 。 如果…

FAT12中,如何定位大于一个扇区(512B)的文件内容

【0】README 0.1&#xff09;本文旨在于 演示在FAT12中&#xff0c; 如何取定位大于 512B 的文件内容&#xff0c;和查看它&#xff1b;0.2&#xff09;如何制作FAT12文件系统&#xff0c;以及如何向文件中添加temp.txt文件&#xff0c;参见&#xff1a; { http://blog.csdn.n…

通过Struts2的拦截器实现文件上传/上传文件功能

struts.xml配置内容如下&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.3//EN""http://struts.apache.org/dtds/struts-2.3.dtd&qu…

java获取ram_Java:ChronicleMap第2部分,超级RAM映射

java获取ram诸如无处不在的HashMap类的标准Java Map最终受到可用RAM的限制。 阅读本文并了解如何创建几乎无限大小甚至超过目标计算机RAM大小的Java Map。 内置的Map实现&#xff0c;例如HashMap和ConcurrentHashMap &#xff0c;只要它们相对较小&#xff0c;就可以正常工作。…