国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

安全無憂!在 Spring Boot 3.3 中輕松實現(xiàn) TOTP 雙因素認證

開發(fā) 架構(gòu)
本 TOTP 注冊系統(tǒng)通過結(jié)合現(xiàn)代前端技術(shù)與穩(wěn)健的后端架構(gòu),成功實現(xiàn)了高效、安全的用戶注冊流程。系統(tǒng)的設計充分考慮了安全性與用戶體驗,確保用戶在注冊過程中能夠快速獲取所需信息,而不影響安全標準。

隨著互聯(lián)網(wǎng)的快速發(fā)展,網(wǎng)絡安全問題日益嚴峻。傳統(tǒng)的用戶名和密碼認證方式已經(jīng)無法滿足現(xiàn)代應用對安全性的要求,因此雙因素認證(2FA)成為了提升安全性的有效手段。雙因素認證不僅要求用戶輸入密碼,還需通過第二種方式進行身份驗證,例如手機生成的動態(tài)驗證碼。

時間同步一次性密碼(TOTP)是一種基于時間的雙因素認證方式,它通過算法生成短期有效的驗證碼。用戶在登錄時,需要輸入從手機應用(如 Google Authenticator)獲取的 TOTP 代碼。由于 TOTP 代碼每 30 秒更新一次,即使攻擊者獲取了用戶的密碼,沒有有效的 TOTP 代碼,也無法登錄賬戶。

本文將詳細介紹如何在 Spring Boot 3.3 中實現(xiàn)基于 TOTP 的雙因素認證,涵蓋從依賴配置、服務實現(xiàn)到前端展示的完整過程。

什么是 TOTP?

TOTP(Time-based One-Time Password)是一種用于雙因素認證的算法,它基于當前時間和用戶的共享秘密(密鑰)生成一次性密碼。TOTP 主要遵循以下步驟:

  1. 密鑰生成:在用戶賬戶創(chuàng)建時生成一個共享密鑰,并與用戶的身份綁定。該密鑰通常以 Base32 編碼格式存儲。
  2. 時間戳使用:TOTP 使用當前時間戳,將時間分成固定的時間段(例如,30 秒)。每個時間段生成一個唯一的 TOTP 密碼。
  3. 動態(tài)密碼生成:通過將共享密鑰和當前時間戳作為輸入,使用 HMAC-SHA1 或類似算法生成一次性密碼。
  4. 驗證過程:在用戶登錄時,服務器端也使用相同的共享密鑰和當前時間戳生成 TOTP 密碼,并與用戶輸入的密碼進行比對。

這種機制保證了每次登錄時生成的密碼都是唯一且短暫的,極大地提升了賬戶的安全性。

運行效果:

圖片

若想獲取項目完整代碼以及其他文章的項目源碼,且在代碼編寫時遇到問題需要咨詢交流,歡迎加入下方的知識星球。

項目依賴配置

首先,在 pom.xml 中添加所需的依賴:

<?xml versinotallow="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
	<modelVersion>4.0.0</modelVersion>
	<parent>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-parent</artifactId>
		<version>3.3.4</version>
		<relativePath/> <!-- lookup parent from repository -->
	</parent>
	<groupId>com.icoderoad</groupId>
	<artifactId>totp-authentication</artifactId>
	<version>0.0.1-SNAPSHOT</version>
	<name>totp-authentication</name>
	<description>Demo project for Spring Boot</description>
	
	<properties>
		<java.version>17</java.version>
	</properties>
	<dependencies>
		<dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-security</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-configuration-processor</artifactId>
        </dependency>
        <dependency>
            <groupId>dev.samstevens.totp</groupId>
            <artifactId>totp</artifactId>
            <version>1.7.1</version>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <scope>provided</scope>
        </dependency>

		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-test</artifactId>
			<scope>test</scope>
		</dependency>
	</dependencies>

	<build>
		<plugins>
			<plugin>
				<groupId>org.springframework.boot</groupId>
				<artifactId>spring-boot-maven-plugin</artifactId>
			</plugin>
		</plugins>
	</build>

</project>

配置文件

接下來,我們在 application.yml 中配置所需的屬性:

server:
  port: 8080
  
totp:
  time-step: 30
  length: 6

生成和配置密鑰

生成密鑰服務類

package com.icoderoad.totp.service;

import org.springframework.stereotype.Service;

import dev.samstevens.totp.secret.DefaultSecretGenerator;
import dev.samstevens.totp.secret.SecretGenerator;

@Service
public class SecretService {
    private final SecretGenerator secretGenerator = new DefaultSecretGenerator();

    public String generateSecret() {
        // 生成安全的隨機 base32 編碼字符串
        return secretGenerator.generate();
    }
}

屬性配置類

package com.icoderoad.totp.config;

import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;

import lombok.Data;

@Data
@Component
@ConfigurationProperties(prefix = "totp")
public class TotpProperties {
    private int timeStep = 30; // 默認值為 30 秒
    private int length = 6;     // 默認值為 6 位
}

配置 TOTP 生成器

package com.icoderoad.totp.service;

import com.icoderoad.totp.config.TotpProperties;
import dev.samstevens.totp.time.TimeProvider;
import dev.samstevens.totp.time.SystemTimeProvider;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class TotpConfiguration {

    private final TotpProperties totpProperties;

    public TotpConfiguration(TotpProperties totpProperties) {
        this.totpProperties = totpProperties;
    }

    @Bean
    public TimeProvider timeProvider() {
        return new SystemTimeProvider(); // 使用系統(tǒng)時間提供者
    }

    @Bean
    public int getTotpLength() {
        return totpProperties.getLength();
    }

    public int getTimeStepInSeconds() {
        return totpProperties.getTimeStep();
    }
}

TOTP 生成和驗證

TOTP 生成服務

package com.icoderoad.totp.service;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import com.icoderoad.totp.config.TotpProperties;

import dev.samstevens.totp.code.CodeGenerator;
import dev.samstevens.totp.code.DefaultCodeGenerator;
import dev.samstevens.totp.exceptions.CodeGenerationException;
import dev.samstevens.totp.time.SystemTimeProvider;
import dev.samstevens.totp.time.TimeProvider;

@Service
public class TotpGeneratorService {
	
	@Autowired
	private TotpProperties totpProperties;
	
    private final CodeGenerator codeGenerator;
    private final TimeProvider timeProvider;

    @Autowired
    public TotpGeneratorService(TimeProvider timeProvider) {
        this.timeProvider = timeProvider != null ? timeProvider : new SystemTimeProvider();
        this.codeGenerator = new DefaultCodeGenerator(); // 使用默認構(gòu)造函數(shù)
    }

    public String generateTotp(String secret) {
        long counter = getCounter();
        try {
			return codeGenerator.generate(secret, counter);
		} catch (CodeGenerationException e) {
			return "";
		}
    }

    private long getCounter() {
        long timeStep = totpProperties.getTimeStep();
        return timeProvider.getTime() / timeStep;
    }
}

TOTP 驗證服務

package com.icoderoad.totp.service;

import dev.samstevens.totp.code.CodeVerifier;
import dev.samstevens.totp.code.DefaultCodeVerifier;
import dev.samstevens.totp.code.DefaultCodeGenerator;
import dev.samstevens.totp.exceptions.CodeGenerationException;
import dev.samstevens.totp.time.TimeProvider;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import com.icoderoad.totp.config.TotpProperties;

@Service
public class TotpVerificationService {
    private final DefaultCodeVerifier codeVerifier;
    private final TimeProvider timeProvider;
    
	private final TotpProperties totpProperties;

    @Autowired
    public TotpVerificationService(TimeProvider timeProvider, TotpProperties totpProperties) {
    	this.totpProperties = totpProperties;
        this.timeProvider = timeProvider;
        this.codeVerifier = new DefaultCodeVerifier(new DefaultCodeGenerator(), timeProvider);
        this.codeVerifier.setTimePeriod(this.totpProperties.getTimeStep()); // 從配置文件中讀取或設置
        this.codeVerifier.setAllowedTimePeriodDiscrepancy( this.totpProperties.getLength() ); // 可配置的時間誤差
    }

    public boolean verifyTotp(String secret, String code) {
        return codeVerifier.isValidCode(secret, code);
    }
}

用戶注冊與 TOTP 集成

UserService 類

package com.icoderoad.totp.service;

import org.springframework.stereotype.Service;

import java.util.HashMap;
import java.util.Map;

@Service
public class UserService {
    // 使用 HashMap 模擬用戶存儲(可以替換為數(shù)據(jù)庫實現(xiàn))
    private final Map<String, String> userSecrets = new HashMap<>();

    /**
     * 保存用戶的 TOTP 秘密
     *
     * @param username 用戶名
     * @param secret   用戶的 TOTP 秘密
     */
    public void saveUserSecret(String username, String secret) {
        userSecrets.put(username, secret);
    }

    /**
     * 根據(jù)用戶名獲取 TOTP 秘密
     *
     * @param username 用戶名
     * @return TOTP 秘密
     */
    public String findSecretByUsername(String username) {
        return userSecrets.get(username);
    }

    // 可以添加更多與用戶相關的方法,如驗證用戶、獲取用戶信息等
}

QRCodeGenerator類

package com.icoderoad.totp.generator;

import org.springframework.stereotype.Component;

import dev.samstevens.totp.exceptions.QrGenerationException;
import dev.samstevens.totp.qr.QrData;
import dev.samstevens.totp.qr.ZxingPngQrGenerator;

@Component
public class QRCodeGenerator {

    private final ZxingPngQrGenerator qrGenerator;

    public QRCodeGenerator() {
        this.qrGenerator = new ZxingPngQrGenerator();
    }

    public byte[] generate(String secret, String username, String issuer, int digits, int period) throws QrGenerationException {
        // 創(chuàng)建 QR 數(shù)據(jù)
        QrData qrData = new QrData.Builder()
                .label(username)
                .secret(secret)
                .issuer(issuer)
                .digits(digits)
                .period(period)
                .build();

        // 生成 QR 代碼
        return qrGenerator.generate(qrData);
    }

    public String generateQrCodeUrl(String secret, String username, String issuer, int digits, int period) throws QrGenerationException {
        byte[] qrCodeBytes = generate(secret, username, issuer, digits, period);
        
        // 將生成的 QR 代碼轉(zhuǎn)換為 Base64 URL,便于在 HTML 中顯示
        return "data:image/png;base64," + java.util.Base64.getEncoder().encodeToString(qrCodeBytes);
    }
}

RegistrationResponse類

package com.icoderoad.totp.controller;

public class RegistrationResponse {
    private final String secret;
    private final String qrCodeUrl;

    public RegistrationResponse(String secret, String qrCodeUrl) {
        this.secret = secret;
        this.qrCodeUrl = qrCodeUrl;
    }

    public String getSecret() {
        return secret;
    }

    public String getQrCodeUrl() {
        return qrCodeUrl;
    }
}

注冊控制器

package com.icoderoad.totp.controller;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RestController;

import com.icoderoad.totp.dto.UserDto;
import com.icoderoad.totp.generator.QRCodeGenerator;
import com.icoderoad.totp.service.SecretService;
import com.icoderoad.totp.service.UserService;

import dev.samstevens.totp.exceptions.QrGenerationException;

@RestController
public class RegistrationController {

總結(jié)

本 TOTP 注冊系統(tǒng)通過結(jié)合現(xiàn)代前端技術(shù)與穩(wěn)健的后端架構(gòu),成功實現(xiàn)了高效、安全的用戶注冊流程。系統(tǒng)的設計充分考慮了安全性與用戶體驗,確保用戶在注冊過程中能夠快速獲取所需信息,而不影響安全標準。總體而言,該系統(tǒng)不僅提升了用戶賬戶的安全性,也通過友好的操作流程增強了用戶的信任感,為未來的擴展和優(yōu)化打下了堅實基礎。

責任編輯:武曉燕 來源: 路條編程
相關推薦

2020-05-25 07:00:00

雙因素認證身份認證密碼

2022-08-01 00:08:03

雙因素認證2FA

2011-10-31 14:57:37

2022-11-10 19:49:29

2024-09-05 09:35:58

CGLIBSpring動態(tài)代理

2022-12-19 16:17:21

CDN

2013-06-19 11:26:39

2011-08-15 09:31:55

2021-12-28 11:13:05

安全認證 Spring Boot

2021-06-10 10:26:37

網(wǎng)絡安全/智慧醫(yī)療

2011-03-16 16:00:06

內(nèi)網(wǎng)安全

2020-09-30 11:22:16

帳戶安全

2013-05-23 18:03:25

2023-04-25 10:59:56

2018-08-08 05:03:31

2024-10-07 08:18:05

SpringBOM管理

2020-12-24 17:12:29

賬戶安全雙因素認證Facebook

2025-05-13 07:13:25

點贊
收藏

51CTO技術(shù)棧公眾號

91免费看片网站| 一区二区三区的久久的视频| 国产1区在线| 国产一区久久久| 欧美一级高清免费播放| 久久香蕉一区| 亚洲成人综合网站| 国产日韩亚洲欧美在线| 欧美日本在线| 91禁外国网站| 亚洲乱码国产乱码精品精98午夜| 日韩激情视频在线| 日韩av网站电影| 亚洲色在线视频| 国产精品精品久久久| 免费日韩视频在线观看| 视频精品国内| 久久国产精品无码网站| 成人99免费视频| 亚洲精品视频在线看| 精品少妇一区二区| 成年网站免费| 一个色综合网| 久久久免费观看| 欧美色网在线| 日韩精品中文字幕在线不卡尤物 | 色综合久久天天| 久久久久国产一区| 成人免费观看在线| 国产精品一品| 欧美三级乱人伦电影| 国产一二三区av| 99久久精品情趣| 日本黄大片在线观看| 日韩专区中文字幕一区二区| 国产美女精品久久久| 成人中文在线| 日本国产欧美一区二区三区| 精品999日本久久久影院| 亚洲人成电影网| 91国拍精品国产粉嫩亚洲一区| 亚洲欧美制服中文字幕| 在线中文字幕播放| 亚洲人成伊人成综合网久久久| 男人的天堂免费在线视频| 337p日本欧洲亚洲大胆精品| caoporn免费在线| 欧美一区二区成人| 污网站在线免费看| 精品网站999www| 久久野战av| 草民午夜欧美限制a级福利片| 色狠狠一区二区三区| 北条麻妃一区二区三区中文字幕| ww久久综合久中文字幕| 欧美成人全部免费| 久久久久久毛片免费看 | 久久综合久久99| 欧美在线观看www| 国产亚洲精品超碰| 免费女人黄页| 亚洲欧美日韩成人高清在线一区| 狠狠操夜夜操| 日韩欧美中文在线| 日本高清视频在线观看| 精品日韩在线观看| 亚洲天堂1区| 日本精品va在线观看| 99精品在线观看| 久久av一区二区三区亚洲| 日本伊人精品一区二区三区观看方式| 亚洲国产精品www| 97精品国产露脸对白| 国产h色视频在线观看| 黑人巨大精品欧美一区二区| а√天堂中文在线资源bt在线| 日韩欧美国产综合| 91麻豆精品| 成人激情黄色网| 日本vs亚洲vs韩国一区三区| 欧美亚洲日本一区二区三区| 亚洲图片激情小说| caopon在线免费视频| 欧美日韩999| 精品动漫一区| 欧美日韩亚洲一| 欧美伊人精品成人久久综合97| 欧美性猛交xxx高清大费中文| 97av在线播放| 一本色道久久综合亚洲精品高清| 成年人看的毛片| 欧美色道久久88综合亚洲精品| 制服丝袜在线播放| 午夜精品蜜臀一区二区三区免费 | 亚洲色图美腿丝袜| 色婷婷精品视频| 视频在线99| 亚洲欧洲日韩女同| 免费在线播放电影| 欧美亚洲成人精品| 美国十次了思思久久精品导航| 成人毛片高清视频观看| 日韩一区二区电影在线| 91久久偷偷做嫩草影院电| 国产日韩在线一区二区三区| 成人黄色一级视频| yw视频在线观看| 欧美—级a级欧美特级ar全黄| 在线精品在线| 九九热精品在线播放| 亚洲第一中文字幕| 小处雏高清一区二区三区| 欧美国产综合在线| 欧美美女网站色| 九九精品在线| av免费观看网| 亚洲第一天堂av| 欧美日韩精品| 理论片播放午夜国外| 最近中文字幕日韩精品| 韩日欧美一区| 亚洲一级免费观看| 日韩经典中文字幕| 欧美人成在线| 国产特级毛片| 欧美激情女人20p| 高清av一区二区| 手机在线免费观看av| 114国产精品久久免费观看| 99久久久免费精品国产一区二区 | 欧美激情综合五月色丁香小说| 菠萝菠萝蜜在线观看| 国产suv精品一区二区| www.欧美色图| а√在线中文在线新版| 国产精品区一区二区三在线播放| 亚洲天堂成人在线观看| 国产精品色婷婷在线观看| 久久久天堂国产精品| 精品国产精品网麻豆系列| 亚洲美女色禁图| 麻豆av电影在线观看| 日本三级久久久| 国产精品每日更新在线播放网址| 美女网站视频一区| 性做爰过程免费播放| 欧美mv和日韩mv的网站| 亚洲啪啪91| 五月天婷婷在线视频| 古典武侠综合av第一页| 精品久久久香蕉免费精品视频| 亚洲欧美成人vr| 久草在线免费二| 97香蕉超级碰碰久久免费软件 | 91精品久久久久久久久久久| 国产精品理伦片| 亚洲一区二区三区免费| 国产精品免费观看久久| 国产一区二区三区在线看| 国产原创一区二区| 涩涩视频在线免费看| 性做爰过程免费播放| 亚洲美女精品久久| 国产黑丝在线一区二区三区| gogo亚洲高清大胆美女人体| 日韩专区第三页| 久久精品2019中文字幕| 91视频精品在这里| 日本一区精品视频| 国产又大又黄又猛| 欧美做受高潮1| 五月激情综合色| 亚洲黑丝一区二区| 成人三级小说| 国产亚洲精品久久久久久久| 一区二区中文字幕| 国产精品欧美久久久久无广告| 亚洲精品无吗| 欧美日韩国产综合视频| 极品校花啪啪激情久久| 亚洲第一免费播放区| 成+人+亚洲+综合天堂| 中文字幕一区二区三区四区久久| 亚洲欧美视频二区| 91麻豆国产语对白在线观看| 欧美日韩一二三| 经典一区二区三区| 日韩高清一区| 天堂av在线免费观看| 欧洲一区二区在线| 国产综合婷婷| 北条麻妃国产九九九精品小说| 涩涩视频在线| 色哟哟在线观看| 亚洲一区二区四区| 亚洲18私人小影院| 亚洲精品福利在线观看| 精品国产3级a| av在线播放网站| 亚洲三区在线|