快試試用 API Key 來保護你的 SpringBoot 介面安全吧!

来源:https://www.cnblogs.com/javastack/archive/2023/07/10/17540392.html
-Advertisement-
Play Games

來源:baeldung.com/spring-boot-api-key-secret ## 1、概述 安全性在REST API開發中扮演著重要的角色。一個不安全的REST API可以直接訪問到後臺系統中的敏感數據。因此,企業組織需要關註API安全性。 Spring Security 提供了各種機制來 ...


來源:baeldung.com/spring-boot-api-key-secret

1、概述

安全性在REST API開發中扮演著重要的角色。一個不安全的REST API可以直接訪問到後臺系統中的敏感數據。因此,企業組織需要關註API安全性。

Spring Security 提供了各種機制來保護我們的 REST API。其中之一是 API 密鑰。API 密鑰是客戶端在調用 API 調用時提供的令牌。

在本教程中,我們將討論如何在Spring Security中實現基於API密鑰的身份驗證。

2、REST API Security

Spring Security可以用來保護REST API的安全性。REST API是無狀態的,因此不應該使用會話或cookie。相反,應該使用Basic authentication,API Keys,JWT或OAuth2-based tokens來確保其安全性。

2.1. Basic Authentication

Basic authentication是一種簡單的認證方案。客戶端發送HTTP請求,其中包含Authorization標頭的值為Basic base64_url編碼的用戶名:密碼。Basic authentication僅在HTTPS / SSL等其他安全機制下才被認為是安全的。

2.2. OAuth2

OAuth2是REST API安全的行業標準。它是一種開放的認證和授權標準,允許資源所有者通過訪問令牌將授權委托給客戶端,以獲得對私有數據的訪問許可權。

2.3. API Keys

一些REST API使用API密鑰進行身份驗證。API密鑰是一個標記,用於向API客戶端標識API,而無需引用實際用戶。標記可以作為查詢字元串或在請求頭中發送。

3、用API Keys保護REST API

Spring Boot 基礎就不介紹了,推薦看這個實戰項目:

https://github.com/javastacks/spring-boot-best-practice

3.1 添加Maven 依賴

讓我們首先在我們的pom.xml中聲明spring-boot-starter-security依賴關係:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-security</artifactId>
</dependency>

3.2 創建自定義過濾器(Filter)

實現思路是從請求頭中獲取API Key,然後使用我們的配置檢查秘鑰。在這種情況下,我們需要在Spring Security 配置類中添加一個自定義的Filter。

我們將從實現GenericFilterBean開始。GenericFilterBean是一個基於javax.servlet.Filter介面的簡單Spring實現。

讓我們創建AuthenticationFilter類:

public class AuthenticationFilter extends GenericFilterBean {

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain)
      throws IOException, ServletException {
        try {
            Authentication authentication = AuthenticationService.getAuthentication((HttpServletRequest) request);
            SecurityContextHolder.getContext().setAuthentication(authentication);
        } catch (Exception exp) {
            HttpServletResponse httpResponse = (HttpServletResponse) response;
            httpResponse.setStatus(HttpServletResponse.SC_UNAUTHORIZED);
            httpResponse.setContentType(MediaType.APPLICATION_JSON_VALUE);
            PrintWriter writer = httpResponse.getWriter();
            writer.print(exp.getMessage());
            writer.flush();
            writer.close();
        }

        filterChain.doFilter(request, response);
    }
}

我們只需要實現doFilter()方法,在這個方法中我們從請求頭中獲取API Key,並將生成的Authentication對象設置到當前的SecurityContext實例中。

然後請求被傳遞給其餘的過濾器處理,接著轉發給DispatcherServlet最後到達我們的控制器。

在AuthenticationService類中,實現從Header中獲取API Key並構造Authentication對象,代碼如下:

public class AuthenticationService {
    private static final String AUTH_TOKEN_HEADER_NAME = "X-API-KEY";
    private static final String AUTH_TOKEN = "Baeldung";

    public static Authentication getAuthentication(HttpServletRequest request) {
        String apiKey = request.getHeader(AUTH_TOKEN_HEADER_NAME);

        if ((apiKey == null) || !apiKey.equals(AUTH_TOKEN)) {
            throw new BadCredentialsException("Invalid API Key");
        }

        return new ApiKeyAuthentication(apiKey, AuthorityUtils.NO_AUTHORITIES);
    }
}

在這裡,我們檢查請求頭是否包含 API Key,如果為空 或者Key值不等於密鑰,那麼就拋出一個 BadCredentialsException。如果請求頭包含 API Key,並且驗證通過,則將密鑰添加到安全上下文中,然後調用下一個安全過濾器。getAuthentication 方法非常簡單,我們只是比較 API Key 頭部和密鑰是否相等。

為了構建 Authentication 對象,我們必須使用 Spring Security 為了標準身份驗證而構建對象時使用的相同方法。所以,需要擴展 AbstractAuthenticationToken 類並手動觸發身份驗證。

3.3. 擴展AbstractAuthenticationToken

為了成功地實現我們應用的身份驗證功能,我們需要將傳入的API Key轉換為AbstractAuthenticationToken類型的身份驗證對象。AbstractAuthenticationToken類實現了Authentication介面,表示一個認證請求的主體和認證信息。

讓我們創建ApiKeyAuthentication類:

public class ApiKeyAuthentication extends AbstractAuthenticationToken {
    private final String apiKey;

    public ApiKeyAuthentication(String apiKey,
        Collection<?extends GrantedAuthority> authorities) {
        super(authorities);
        this.apiKey = apiKey;
        setAuthenticated(true);
    }

    @Override
    public Object getCredentials() {
        return null;
    }

    @Override
    public Object getPrincipal() {
        return apiKey;
    }
}

ApiKeyAuthentication 類是類型為 AbstractAuthenticationToken 的對象,其中包含從 HTTP 請求中獲取的 apiKey 信息。在構造方法中使用 setAuthenticated(true) 方法。因此,Authentication對象包含 apiKey 和authenticated欄位:

3.4. Security Config

通過創建建一個SecurityFilterChain bean,可以通過編程方式把我們上面編寫的自定義過濾器(Filter)進行註冊。

我們需要在 HttpSecurity 實例上使用 addFilterBefore() 方法在 UsernamePasswordAuthenticationFilter 類之前添加 AuthenticationFilter。

創建SecurityConfig 類:

@Configuration
@EnableWebSecurity
public class SecurityConfig {

    @Bean
    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
        http.csrf()
          .disable()
          .authorizeRequests()
          .antMatchers("/**")
          .authenticated()
          .and()
          .httpBasic()
          .and()
          .sessionManagement()
          .sessionCreationPolicy(SessionCreationPolicy.STATELESS)
          .and()
          .addFilterBefore(new AuthenticationFilter(), UsernamePasswordAuthenticationFilter.class);

        return http.build();
    }

}

此外註意代碼中我們吧繪畫策略(session policy)設置為無狀態(STATELESS),因為我們使用的是REST。

3.5. ResourceController

最後,我們創建ResourceController,實現一個Get請求 /home

@RestController
public class ResourceController {
    @GetMapping("/home")
    public String homeEndpoint() {
        return "Baeldung !";
    }
}

3.6. 禁用 Auto-Configuration

@SpringBootApplication(exclude = {SecurityAutoConfiguration.class, UserDetailsServiceAutoConfiguration.class})
public class ApiKeySecretAuthApplication {

    public static void main(String[] args) {
        SpringApplication.run(ApiKeySecretAuthApplication.class, args);
    }
}

4. 測試

我們先不提供API Key進行測試

curl --location --request GET 'http://localhost:8080/home'

返回 401 未經授權錯誤。

請求頭中加上API Key後,再次請求

curl --location --request GET 'http://localhost:8080/home' \
--header 'X-API-KEY: Baeldung'

請求返回狀態200

代碼:

https://github.com/eugenp/tutorials/tree/master/spring-security-modules/spring-security-web-boot-4

近期熱文推薦:

1.1,000+ 道 Java面試題及答案整理(2022最新版)

2.勁爆!Java 協程要來了。。。

3.Spring Boot 2.x 教程,太全了!

4.別再寫滿屏的爆爆爆炸類了,試試裝飾器模式,這才是優雅的方式!!

5.《Java開發手冊(嵩山版)》最新發佈,速速下載!

覺得不錯,別忘了隨手點贊+轉發哦!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 引自:https://blog.csdn.net/weixin_43795921/article/details/127224633 template <typename IdentifierType, class AbstractProduct, class ProductCreator = Ab ...
  • 將bytes 轉換為long類型: 第一種方式: String 接收 bytes 的構造器轉成 String,再 Long.parseLong; 但此種情況需要註意:位元組數組中的每個位元組都必須是有效的數字字元。如果位元組數組包含非數字字元,則會引發NumberFormatException異常。確保在 ...
  • 一. 介紹 String 是Java.long包下的String類,是一個特殊的引用類型,用於表示字元串。它提供了許多方法來操作和處理字元串,比如連接、截取、查找、替換等。String類內部使用字元數組( char[] ) 來存儲字元串的內容,value欄位被final修飾,String對象一旦創建 ...
  • # 1. 配置文件 ## 1.1 外部載入順序 1. 命令行參數 java -jar spring-boot-02-config-02.0.0.1-SNAPSHOT.jar --server.port=8087 java -jar spring-boot-02-config-02.0.0.1-SNA ...
  • 電商支付實戰項目與相關面試題總結 接下來我將用一篇萬字長文,總結好這個項目以達到可以正面硬鋼面試官的水平,如果作為一個毫無頭緒的大學生的你,簡歷中需要一個還算拿得出手的項目,那麼在2023年的今天,足矣作為一個還算OK的項目寫進你的簡歷。當然,這隻能算簡歷中的第一個項目,你還需要一個更好一些的項目作 ...
  • 當聲明枚舉類型或定義一組相關常量時,Go語言中的`iota`關鍵字可以幫助我們簡化代碼並自動生成遞增的值。本文檔將詳細介紹`iota`的用法和行為。 ### `iota`關鍵字 `iota`是Go語言中的一個預定義標識符,它用於創建自增的無類型整數常量。`iota`的行為類似於一個計數器,每次在常量 ...
  • **深度探討Django ORM的概念、基礎使用、進階操作以及詳細解析在實際使用中如何處理資料庫操作。同時,我們還討論了模型深入理解,如何進行CRUD操作,並且深化理解到資料庫遷移等高級主題。為了全面解讀Django ORM,我們也討論了其存在的不足,並對其未來發展進行了展望。這篇文章旨在幫助讀者全 ...
  • 在我們的日常編程任務中,對於集合的製造和處理是必不可少的。當我們需要對於集合進行分組或查找的操作時,需要用迭代器對於集合進行操作,而當我們需要處理的數據量很大的時候,為了提高性能,就需要使用到並行處理,這樣的處理方式是很複雜的。流可以幫助開發者節約寶貴的時間,讓以上的事情變得輕鬆。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...