强化用户体验与安全性:前端单点登录和统一认证的最佳实践与区别

简介: 互联网发展了这么多年,各种更新皆为了提供更好更安全的上网环境。同时为了提供更好的用户体验、减少用户反复输入用户名和密码的繁琐操作,并确保账户安全,前端领域中的单点登录(SSO)和统一认证(Unified Authentication)成为了重要概念。

单点登录原理

单点登录(Single Sign-On,简称SSO)是一种身份验证机制,允许用户只需一次登录,便能够访问多个相关系统或应用,而无需重复输入用户名和密码。下面将详细解释单点登录的原理:

image.png

  1. 用户访问主应用:用户在客户端通过浏览器或移动应用程序访问主应用,并尝试进行登录操作。
  2. 主应用认证:主应用接收到用户的登录请求后,验证用户提供的用户名和密码是否正确。如果认证成功,则生成一个用于标识用户身份的特定凭证或令牌。
  3. 凭证颁发:主应用将生成的凭证或令牌发送给用户浏览器,通常以Cookie或Token的形式进行存储。

  4. 子应用访问:当用户尝试访问其他子应用时,子应用会检测用户是否已登录。如果用户未登录,则重定向至主应用的登录页面。

  5. 单点登录验证:在重定向至主应用的登录页面后,子应用将用户重定向URL添加一个参数,包含子应用的标识信息。用户浏览器携带凭证或令牌访问主应用。

  6. 凭证验证:主应用接收到带有凭证或令牌的请求后,会解析验证凭证的有效性和合法性。主应用可能会使用加密算法验证凭证的完整性和真实性,确保凭证有效。

  7. 用户身份确认:一旦主应用确认凭证有效,则将用户视为已登录,并生成一个会话或认证令牌。该令牌将作为响应返回给用户浏览器。

  8. 子应用授权:用户浏览器将获取到的会话或认证令牌发送给子应用。子应用接收到令牌后,可以向主应用验证其有效性。主应用确认令牌有效后,子应用即可通过该令牌完成用户身份确认,并授权用户访问其资源。

统一认证原理

统一认证(Unified Authentication)是一种集中管理多个子系统的用户身份验证和授权过程的机制,以实现用户在不同系统中的一致登录和权限管理。下面将详细解释统一认证的原理:
image.png

  1. 用户访问应用系统:用户在客户端通过浏览器或移动应用程序访问某个应用系统,并尝试进行登录操作。
  2. 应用系统认证:应用系统接收到用户的登录请求后,验证用户提供的用户名和密码是否正确。如果认证成功,则生成一个用于标识用户身份的特定凭证或令牌。
  3. 统一认证中心:在统一认证环境中,存在一个专门负责用户身份验证和授权的统一认证中心。应用系统将用户的认证请求转发至统一认证中心进行处理。
  4. 用户身份认证:统一认证中心接收到应用系统的认证请求后,对用户提供的用户名和密码进行认证。认证中心可能会通过与用户存储的身份信息进行比对来验证用户身份的真实性。
  5. 状态维护:一旦用户身份认证成功,统一认证中心会为用户生成一个会话或认证令牌,并记录用户的登录状态。可以使用Cookie、Token或其他方式将会话或认证令牌发送给用户浏览器。
  6. 令牌传递:经过认证后,统一认证中心将用户的会话或认证令牌返回给应用系统。这个令牌是应用系统与统一认证中心之间进行交互的凭证,也是应用系统验证用户身份的依据。
  7. 子系统授权:应用系统接收到统一认证中心返回的会话或认证令牌后,可以向统一认证中心验证令牌的有效性和合法性。一旦验证通过,应用系统即可确认用户身份,并为用户授予相应的权限

两者的区别

单点登录 (SSO) 统一认证 (Unified Authentication)
定义 用户只需登录一次,即可访问多个应用系统。 用户通过一个中心化的认证系统进行身份验证和授权。
流程 用户登录后,通过认证中心颁发的令牌实现免密登录其他应用系统。 用户直接与统一认证中心进行交互,由认证中心验证身份并授权。
身份验证 认证中心负责验证用户身份,并生成令牌用于后续访问应用系统。 统一认证中心负责验证用户身份,并生成令牌用于后续访问应用系统。
应用系统集成 应用系统之间需要与认证中心建立信任关系,以验证令牌的有效性。 应用系统之间需要与统一认证中心建立信任关系,以验证令牌的有效性。
可扩展性 可支持多种不同的认证方式和协议。 可支持多种不同的认证方式和协议。
用户体验 用户只需进行一次登录,无需重复输入用户名和密码。 用户只需进行一次登录,无需重复输入用户名和密码。
隐私保护 用户个人信息在不同应用系统间共享较少,隐私更易得到保护。 用户个人信息在统一认证中心集中管理,需要注意隐私保护。

要实现单点登录,通常会使用统一认证作为基础,将用户的登录状态在不同系统间进行共享。因此,在某些情况下,可以将统一认证看作是实现单点登录的一种方式、

前端如何实现

以下代码均为简单示例,实际情况将更为复杂
要在前端实现单点登录 (SSO),需要利用后端提供的认证中心或身份提供者来处理用户的认证和生成令牌。

实现单点登录

首先我们需要使用axios来进行后端请求。

import React, {
   
    useEffect } from 'react';
import axios from 'axios';

const SSO = () => {
   
   
  useEffect(() => {
   
   
    // 发送登录请求到认证中心
    axios.post('http://认证中心地址/sso/login', {
   
   
      username: '用户名',
      password: '密码'
    })
    .then(response => {
   
   
      // 登录成功,得到认证中心返回的令牌
      const token = response.data.token;

      // 将令牌保存在本地存储或 cookie 中
      localStorage.setItem('token', token);

      // 跳转到其他应用系统
      window.location.href = 'http://其他应用系统地址';
    })
    .catch(error => {
   
   
      // 登录失败,处理错误逻辑
      console.error('登录失败:', error);
    });
  }, []);

  return (
    <div>
      正在进行单点登录...
    </div>
  );
};

export default SSO;

通过 useEffect 钩子函数,在组件加载时发送登录请求到认证中心。认证中心验证用户名和密码,并返回令牌。然后,将令牌保存在本地存储或 cookie 中(根据实际需求),最后跳转到其他应用系统。

实现统一认证

在前端代码中,如果我们想要实现统一认证登录,那么就可能要安装第三方库 oidc-client,它是一个用于客户端实现 OpenID Connect 的 JavaScript 库

npm install oidc-client

然后,在前端代码中引入 oidc-client,并编写统一认证登录的逻辑:

import React, {
   
    useEffect } from 'react';
import {
   
    UserManager } from 'oidc-client';

const Login = () => {
   
   
  useEffect(() => {
   
   
    const userManager = new UserManager({
   
   
      authority: '认证中心地址',
      client_id: '客户端ID',
      redirect_uri: '重定向URI',
      response_type: 'code',
      scope: 'openid profile',
      silent_redirect_uri: '静默刷新URI',
      automaticSilentRenew: true,
      filterProtocolClaims: true
    });

    // 检查是否已经登录,如果已经登录则自动跳转到其他页面
    userManager.getUser().then(user => {
   
   
      if (user) {
   
   
        window.location.href = '其他页面地址';
      }
    });

    // 处理认证回调
    userManager.signinRedirectCallback().then(user => {
   
   
      // 登录成功,跳转到其他页面
      window.location.href = '其他页面地址';
    }).catch(error => {
   
   
      // 登录失败,处理错误逻辑
      console.error('登录失败:', error);
    });
  }, []);

  const handleLogin = () => {
   
   
    // 触发认证中心登录流程
    userManager.signinRedirect();
  };

  return (
    <div>
      <button onClick={
   
   handleLogin}>统一认证登录</button>
    </div>
  );
};

export default Login;

通过 oidc-client 创建一个 UserManager 实例,并根据实际情况配置认证中心的地址、客户端ID、重定向URI等参数。然后,使用 getUser() 方法检查用户是否已经登录,如果已经登录,则自动跳转到其他页面。

当点击 "统一认证登录" 按钮时,调用 signinRedirect() 方法触发认证中心的登录流程,并将用户重定向到认证中心进行身份验证。在认证回调页面中,调用 signinRedirectCallback() 方法处理认证结果,并获取用户信息。如果登录成功,可以将用户重定向到其他页面。

后端如何实现

经过对朋友的“拷打”终于让我逼问出Java是如何实现这两者的功能。

在Java中实现统一认证登录时,可以使用不同的框架和库,如Spring SecurityOAuth 2.0OpenID

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.security.config.annotation.web.configuration.EnableWebSecurity;
import org.springframework.security.oauth2.config.annotation.web.configuration.EnableOAuth2Client;

@SpringBootApplication
@EnableWebSecurity // 启用Web安全功能
@EnableOAuth2Client // 启用OAuth 2.0客户端功能
public class Application {
   
   
  public static void main(String[] args) {
   
   
    SpringApplication.run(Application.class, args);
  }
}

import org.springframework.beans.factory.annotation.Autowired;
// import ...


@Configuration
@EnableGlobalMethodSecurity(prePostEnabled = true) // 启用全局方法级安全注解,如@PreAuthorize和@PostAuthorize
public class SecurityConfig extends WebSecurityConfigurerAdapter {
   
   

  @Autowired
  private OAuth2AuthorizedClientService clientService;

  @Override
  protected void configure(HttpSecurity http) throws Exception {
   
   
    http
      .authorizeRequests()
        .antMatchers("/login**", "/error**")
        .permitAll() // 允许/login和/error路径的任意请求
        .anyRequest()
        .authenticated() // 其他路径需要认证后才能访问
      .and()
      .oauth2Login() // 配置OAuth 2.0登录支持
        .loginPage("/login") // 指定自定义的登录页面路径
        .defaultSuccessUrl("/success") // 登录成功后的默认跳转路径
        .failureUrl("/error") // 登录失败后的跳转路径
      .and()
      .logout() // 配置退出登录的处理
        .logoutRequestMatcher(new AntPathRequestMatcher("/logout")) // 指定退出登录的URL匹配规则
        .logoutSuccessUrl("/login") // 退出登录后的跳转路径
        .deleteCookies("JSESSIONID"); // 退出登录时删除指定的cookie
  }

  @Autowired
  public void configureGlobal(AuthenticationManagerBuilder auth) throws Exception {
   
   
    // 根据需要添加用户认证逻辑,例如从数据库中验证用户信息、集成第三方身份提供者等
  }
}

在上面代码里定义了一个名为SecurityConfig的配置类,继承自WebSecurityConfigurerAdapter,用于配置Spring Security的安全策略。

通过@EnableGlobalMethodSecurity(prePostEnabled = true)注解,启用了全局方法级安全注解,如@PreAuthorize@PostAuthorize,以便在代码中使用注解控制方法的访问权限。

configure(HttpSecurity http)方法中,配置了URL的访问权限。其中,.antMatchers("/login**", "/error**").permitAll()表示允许/login/error路径的任意请求,.anyRequest().authenticated()表示其他路径需要经过认证才能访问。然后配置了OAuth 2.0的登录支持,设置了登录页面路径、默认成功跳转路径和失败跳转路径。最后,我们配置了退出登录的处理,包括退出URL匹配规则、退出成功后的跳转路径和要删除的cookie。

configureGlobal(AuthenticationManagerBuilder auth)方法中,可以根据需要添加用户认证逻辑,例如从数据库中验证用户信息、集成第三方身份提供者等,以实现具体的认证功能。

总结

总结起来,与前端代码相比,Java后端代码中的主要区别在于使用了Spring Security库来处理认证和授权的逻辑,通过配置类来定义URL的访问权限、认证方式等。

本文同步我的技术文档

相关文章
|
25天前
|
前端开发 UED 开发者
颠覆你的前端知识:防抖与节流的区别及实战解析!
【8月更文挑战第23天】在Web前端开发中,处理用户界面交互产生的事件可能会影响性能。为此,我们有两种优化方法:防抖(debounce)和节流(throttle)。防抖确保函数仅在事件停止触发一段时间后执行一次,适用于如搜索自动补全场景。而节流则确保函数按固定时间间隔执行,不管用户操作频率如何。本篇技术博客将深入解析两者差异并提供示例代码,帮助开发者更好地理解和应用这些技巧以提升应用性能和用户体验。
54 0
|
16天前
|
机器学习/深度学习 存储 前端开发
实战揭秘:如何借助TensorFlow.js的强大力量,轻松将高效能的机器学习模型无缝集成到Web浏览器中,从而打造智能化的前端应用并优化用户体验
【8月更文挑战第31天】将机器学习模型集成到Web应用中,可让用户在浏览器内体验智能化功能。TensorFlow.js作为在客户端浏览器中运行的库,提供了强大支持。本文通过问答形式详细介绍如何使用TensorFlow.js将机器学习模型带入Web浏览器,并通过具体示例代码展示最佳实践。首先,需在HTML文件中引入TensorFlow.js库;接着,可通过加载预训练模型如MobileNet实现图像分类;然后,编写代码处理图像识别并显示结果;此外,还介绍了如何训练自定义模型及优化模型性能的方法,包括模型量化、剪枝和压缩等。
25 1
|
29天前
|
缓存 前端开发 JavaScript
优化前端性能的十大最佳实践
在现代网页开发中,前端性能优化不仅仅是为了提升用户体验,还能显著提高网站的加载速度和响应时间。本文探讨了十大最佳实践,从优化资源加载到减少网络请求,再到提高页面渲染效率,每个实践都旨在解决常见的性能瓶颈。通过实现这些策略,开发者可以显著提升前端性能,提升用户满意度,并确保网站在各种设备上的流畅运行。
|
29天前
|
缓存 前端开发 JavaScript
高效开发现代 Web 应用:从前端到后端的最佳实践
在开发现代 Web 应用时,前端和后端技术的选择对项目的性能、可维护性和用户体验至关重要。本文将探讨如何通过现代工具和框架来优化前端和后端开发流程。我们将分析前端技术(如 React 和 Vue.js)与后端技术(如 Node.js 和 Django)的集成,并提供实际案例来展示如何实现高效开发。无论是对新手还是经验丰富的开发者,本指南都提供了宝贵的洞见和实用的技巧,以帮助提高开发效率并构建出色的 Web 应用。
|
29天前
|
缓存 编解码 前端开发
优化Web应用性能的前端技巧:从加载时间到用户体验
在现代Web开发中,提升前端性能不仅仅是为了缩短页面加载时间,更是为了提供更流畅的用户体验。本文将探讨几种有效的前端优化技术,包括懒加载、代码拆分、资源压缩以及浏览器缓存策略。通过具体实例和最佳实践,读者将能够掌握如何系统地提高Web应用的响应速度,减少资源消耗,并最终改善用户的整体体验。
|
1月前
|
前端开发 Java 编译器
【前端学java】java中的Object类和前端中的Object有什么区别(9)
【8月更文挑战第10天】java中的Object类和前端中的Object有什么区别
31 0
【前端学java】java中的Object类和前端中的Object有什么区别(9)
|
16天前
|
前端开发 开发者 UED
数据校验的艺术:揭秘JSF如何将前端与后端验证合二为一,打造无缝用户体验
【8月更文挑战第31天】JavaServer Faces(JSF)是构建企业级Web应用的Java规范,提供了丰富的组件和API,便于快速搭建用户界面。JSF验证框架基于JavaBean验证API(JSR 303/JSR 380),利用注解如`@NotNull`、`@Size`等在模型类上定义验证规则,结合前端的`&lt;h:inputText&gt;`和`&lt;h:message&gt;`标签展示错误信息。
22 0
|
16天前
|
开发者 容器 Docker
JSF与Docker,引领容器化浪潮!让你的Web应用如虎添翼,轻松应对高并发!
【8月更文挑战第31天】在现代Web应用开发中,JSF框架因其实用性和灵活性被广泛应用。随着云计算及微服务架构的兴起,容器化技术变得日益重要,Docker作为该领域的佼佼者,为JSF应用提供了便捷的部署和管理方案。本文通过基础概念讲解及示例代码展示了如何利用Docker容器化JSF应用,帮助开发者实现高效、便携的应用部署。同时也提醒开发者注意JSF与Docker结合使用时可能遇到的限制,并根据实际情况做出合理选择。
26 0
|
16天前
|
API UED 开发者
如何在Uno Platform中轻松实现流畅动画效果——从基础到优化,全方位打造用户友好的动态交互体验!
【8月更文挑战第31天】在开发跨平台应用时,确保用户界面流畅且具吸引力至关重要。Uno Platform 作为多端统一的开发框架,不仅支持跨系统应用开发,还能通过优化实现流畅动画,增强用户体验。本文探讨了Uno Platform中实现流畅动画的多个方面,包括动画基础、性能优化、实践技巧及问题排查,帮助开发者掌握具体优化策略,提升应用质量与用户满意度。通过合理利用故事板、减少布局复杂性、使用硬件加速等技术,结合异步方法与预设缓存技巧,开发者能够创建美观且流畅的动画效果。
38 0
|
16天前
|
前端开发 UED 开发者
深度剖析Angular表单控件:从模板驱动到响应式表单的最佳实践,带你全面掌握Angular表单处理机制,提升前端开发效率与用户体验的终极指南
【8月更文挑战第31天】本文通过代码示例详细介绍了 Angular 中两种主要的表单处理方式:模板驱动表单和响应式表单。模板驱动表单适用于简单场景,可在 HTML 模板中直接定义表单控件并实现数据绑定和验证。响应式表单基于 RxJS,提供更灵活的表单管理和复杂的逻辑处理。通过具体示例展示了每种方式的最佳实践,帮助开发者简化表单处理,提高开发效率和用户体验。
20 0