0

0

使用 OCI Java SDK 签名自定义 REST API 请求

碧海醫心

碧海醫心

发布时间:2025-10-17 10:16:29

|

317人浏览过

|

来源于php中文网

原创

使用 oci java sdk 签名自定义 rest api 请求

本文旨在指导读者如何利用 Oracle OCI Java SDK 的请求签名功能,为自定义 REST API 调用生成符合 OCI 认证要求的 `Authorization` 和 `Date` 头。不同于直接获取“令牌”,OCI 认证基于请求签名机制。我们将详细介绍如何配置身份验证提供程序,获取并使用 `RequestSigner` 接口,从而确保您的自定义 HTTP 请求能够被 OCI 服务正确验证。

理解 OCI 认证机制与 Java SDK

Oracle Cloud Infrastructure (OCI) 的 API 认证机制主要基于请求签名。这意味着,当您向 OCI 服务发送请求时,SDK 或您的应用程序需要使用预配置的身份验证凭据(如 API 密钥)对请求的特定部分(包括 HTTP 方法、路径、查询参数、请求体哈希等)进行加密签名。生成的签名连同其他认证信息(如密钥指纹、租户 OCID 等)被放置在 Authorization HTTP 头中。此外,请求的 Date 头也扮演着关键角色,用于防止重放攻击,并与签名过程紧密关联。

OCI Java SDK 通常会自动处理这些复杂的签名过程。当您使用 SDK 提供的服务客户端(例如 ObjectStorageClient 或 ComputeClient)时,SDK 会透明地管理请求的构建、签名和发送。然而,在某些特定场景下,您可能需要使用自定义的 HTTP 客户端(如 java.net.http.HttpClient、Apache HttpClient 或 OkHttp)来直接与 OCI REST API 交互,此时就需要手动生成符合 OCI 规范的 Authorization 和 Date 头。

用户常见的误解是试图从 SDK 中直接获取一个“令牌”(token)来放入 Authorization 头。实际上,OCI 的认证并非基于短期的 bearer token 机制,而是每次请求都需要重新计算签名的过程。

立即学习Java免费学习笔记(深入)”;

OCI Java SDK 的请求签名器(RequestSigner)

为了支持自定义 HTTP 客户端的认证需求,OCI Java SDK 提供了一个名为 RequestSigner 的接口。这个接口的目的是将 OCI 复杂的请求签名逻辑暴露给开发者,以便他们可以对任何自定义的 HTTP 请求进行签名,而无需深入了解签名算法的细节。

RequestSigner 的核心功能是接收一个表示 HTTP 请求的对象,并返回一个包含所有必要认证头的映射。这些头通常包括 Authorization 和 Date,以及其他可能的签名相关头。通过使用 RequestSigner,您可以:

  • 避免手动实现签名逻辑: OCI 的签名规范复杂,手动实现容易出错。RequestSigner 封装了所有细节。
  • 确保兼容性: 生成的签名与 OCI 服务期望的格式完全一致。
  • 集成自定义客户端: 即使您不使用 OCI SDK 提供的服务客户端,也能安全地调用 OCI REST API。

配置身份验证提供程序

在使用 RequestSigner 之前,您需要配置一个 AuthenticationDetailsProvider。这是 OCI Java SDK 中用于提供认证凭据的核心组件。根据您的运行环境,可以选择不同的实现:

  1. ConfigFileAuthenticationDetailsProvider

    • 用途: 主要用于本地开发、测试或任何可以访问 OCI 配置文件 (~/.oci/config) 的环境。

    • 配置: 从 OCI 配置文件中读取用户 OCID、租户 OCID、区域、API 密钥路径和指纹。

    • 示例:

      import com.oracle.bmc.auth.ConfigFileAuthenticationDetailsProvider;
      import java.io.IOException;
      
      // 默认会查找 ~/.oci/config 文件和 [DEFAULT] 配置项
      // 您也可以指定配置文件的路径和配置项的 profile
      ConfigFileAuthenticationDetailsProvider provider = new ConfigFileAuthenticationDetailsProvider("~/.oci/config", "DEFAULT");
  2. InstancePrincipalsAuthenticationDetailsProvider

    • 用途: 当您的应用程序运行在 OCI 计算实例上时,推荐使用此提供程序。它利用实例主体(Instance Principal)的身份验证,无需在实例上存储 API 密钥。

      海螺视频
      海螺视频

      海螺AI推出的AI视频生成工具,可以生成高质量的视频内容。

      下载
    • 配置: 自动从实例元数据服务获取凭据。

    • 示例:

      import com.oracle.bmc.auth.InstancePrincipalsAuthenticationDetailsProvider;
      
      InstancePrincipalsAuthenticationDetailsProvider provider =
          InstancePrincipalsAuthenticationDetailsProvider.builder().build();
  3. ResourcePrincipalAuthenticationDetailsProvider

    • 用途: 当您的应用程序运行在 OCI 函数、容器实例或其他资源主体(Resource Principal)支持的服务中时,推荐使用此提供程序。

    • 配置: 自动从环境或元数据服务获取凭据。

    • 示例:

      import com.oracle.bmc.auth.ResourcePrincipalAuthenticationDetailsProvider;
      
      ResourcePrincipalAuthenticationDetailsProvider provider =
          ResourcePrincipalAuthenticationDetailsProvider.builder().build();

选择合适的 AuthenticationDetailsProvider 是第一步,它将作为创建 RequestSigner 的基础。

获取并使用 RequestSigner

一旦您有了 AuthenticationDetailsProvider 实例,就可以使用它来构建一个 RequestSigner。RequestSigner 接口定义了 signRequest() 方法,该方法接受一个 com.oracle.bmc.http.internal.ResponseHelper.Request 对象(一个 SDK 内部定义的请求抽象)并返回一个包含签名头的 Map

由于 RequestSigner 期望的是 SDK 内部的 Request 对象,您需要将自定义 HTTP 客户端的请求转换为这种格式。SDK 提供了一个辅助类 com.oracle.bmc.http.signing.internal.RequestSignerImpl,它实现了 RequestSigner 接口。

以下是使用 RequestSigner 的基本步骤和示例代码:

  1. 添加必要的 Maven 依赖 确保您的 pom.xml 中包含 OCI Java SDK 的通用模块,它包含了 RequestSigner 及其相关类:

    
        com.oracle.oci.sdk
        oci-java-sdk-common
        2.x.x 
    
    
        com.oracle.oci.sdk
        oci-java-sdk-addons-jackson-databind
        2.x.x 
    
    
  2. 示例代码:使用 RequestSigner 签名自定义 HTTP 请求

    import com.oracle.bmc.auth.AuthenticationDetailsProvider;
    import com.oracle.bmc.auth.ConfigFileAuthenticationDetailsProvider;
    import com.oracle.bmc.http.internal.ResponseHelper;
    import com.oracle.bmc.http.signing.RequestSigner;
    import com.oracle.bmc.http.signing.internal.RequestSignerImpl;
    import com.oracle.bmc.http.signing.internal.Constants;
    
    import java.io.ByteArrayInputStream;
    import java.io.IOException;
    import java.io.InputStream;
    import java.net.URI;
    import java.net.http.HttpClient;
    import java.net.http.HttpRequest;
    import java.net.http.HttpResponse;
    import java.util.HashMap;
    import java.util.Map;
    import java.util.Optional;
    
    public class OciCustomApiSigner {
    
        public static void main(String[] args) throws IOException, InterruptedException {
            // 1. 配置身份验证提供程序
            // 假设您已在 ~/.oci/config 中配置了 [DEFAULT] profile
            AuthenticationDetailsProvider provider;
            try {
                provider = new ConfigFileAuthenticationDetailsProvider("~/.oci/config", "DEFAULT");
            } catch (IOException e) {
                System.err.println("无法加载 OCI 配置文件或 profile。请确保文件存在且配置正确。");
                e.printStackTrace();
                return;
            }
    
            // 2. 创建 RequestSigner 实例
            // RequestSignerImpl 是 RequestSigner 接口的默认实现
            RequestSigner requestSigner = new RequestSignerImpl(provider);
    
            // 3. 定义要签名的自定义 HTTP 请求的详细信息
            String method = "GET"; // HTTP 方法
            URI uri = URI.create("https://objectstorage.us-ashburn-1.oraclecloud.com/n/your_namespace/b/your_bucket_name/"); // 目标 URI
            Map headers = new HashMap<>();
            // OCI API 通常需要 Host 头
            headers.put("Host", uri.getHost());
            // 如果有请求体,需要计算其 SHA256 摘要并添加到 headers 中
            // 对于 GET 请求,通常没有请求体
    
            // 4. 将自定义请求转换为 RequestSigner 期望的 Request 对象
            // 注意:ResponseHelper.Request 是 SDK 内部类,但在此场景下是必需的
            ResponseHelper.Request.Builder requestBuilder = ResponseHelper.Request.builder();
            requestBuilder.method(method);
            requestBuilder.uri(uri);
            requestBuilder.headers(headers);
    
            // 如果是 PUT/POST 请求且有请求体,需要设置 body 和 content-length
            // byte[] requestBodyBytes = "{\"key\": \"value\"}".getBytes(java.nio.charset.StandardCharsets.UTF_8);
            // requestBuilder.body(new ByteArrayInputStream(requestBodyBytes));
            // requestBuilder.header("Content-Length", String.valueOf(requestBodyBytes.length));
            // requestBuilder.header("Content-Type", "application/json");
    
            ResponseHelper.Request ociRequest = requestBuilder.build();
    
            // 5. 调用 RequestSigner 对请求进行签名
            Map signedHeaders = requestSigner.signRequest(ociRequest);
    
            System.out.println("生成的签名头信息:");
            signedHeaders.forEach((key, value) -> System.out.println(key + ": " + value));
    
            // 6. 使用签名的头信息构建并发送实际的 HTTP 请求
            HttpClient httpClient = HttpClient.newBuilder().build();
            HttpRequest.Builder httpRequestBuilder = HttpRequest.newBuilder()
                    .uri(uri)
                    .method(method, HttpRequest.BodyPublishers.noBody()); // 对于GET请求,没有请求体
    
            // 将签名后的头添加到实际的 HttpRequest 中
            signedHeaders.forEach(httpRequestBuilder::header);
    
            HttpRequest finalHttpRequest = httpRequestBuilder.build();
    
            System.out.println("\n发送签名的 HTTP 请求...");
            HttpResponse response = httpClient.send(finalHttpRequest, HttpResponse.BodyHandlers.ofString());
    
            System.out.println("HTTP 响应状态码: " + response.statusCode());
            System.out.println("HTTP 响应体:\n" + response.body());
    
            // 检查响应头中的 Opc-Request-Id,用于调试
            Optional opcRequestId = response.headers().firstValue("opc-request-id");
            opcRequestId.ifPresent(id -> System.out.println("Opc-Request-Id: " + id));
        }
    }

在上述示例中,RequestSigner 会自动生成 Authorization 和 Date 头,并将它们添加到 signedHeaders 映射中。您只需将这些头应用到您选择的 HTTP 客户端所构建的请求中即可。

注意事项与最佳实践

  • 优先使用 OCI SDK 服务客户端: 尽管 RequestSigner 提供了灵活性,但在大多数情况下,直接使用 OCI Java SDK 提供的服务客户端(如 ObjectStorageClient)更为简单和健壮。它们不仅处理认证,还处理请求/响应序列化、错误处理、重试机制等。
  • 请求体的处理: 如果您的请求包含请求体(如 POST 或 PUT 请求),您需要确保 RequestSigner 能够访问到请求体的内容,以便计算其 SHA256 摘要。在 ResponseHelper.Request.Builder 中,通过 body(InputStream) 方法设置请求体。同时,Content-Length 和 Content-Type 头也应正确设置。
  • URI 的精确性: 确保您传递给 RequestSigner 的 URI 与您实际发送请求的 URI 完全一致,包括路径和查询参数。任何不匹配都可能导致签名验证失败。
  • 错误处理: OCI API 调用可能会失败,例如由于权限不足、资源不存在或网络问题。务必在您的代码中添加适当的错误处理逻辑,捕获并处理 com.oracle.bmc.model.BmcException 或其他 IOException。
  • 安全性: 妥善保管您的 API 密钥和配置文件。避免将敏感信息硬编码到代码中。
  • 版本兼容性: 确保您使用的 oci-java-sdk-common 版本与您的其他 OCI SDK 模块版本兼容。

总结

通过 OCI Java SDK 提供的 RequestSigner 接口,开发者可以有效地为自定义 HTTP 请求生成符合 OCI 认证规范的 Authorization 和 Date 头。这消除了手动实现复杂签名逻辑的需要,并使得在不使用 SDK 服务客户端的情况下,也能安全、可靠地与 OCI REST API 进行交互。理解 OCI 基于签名的认证机制,并正确配置 AuthenticationDetailsProvider 和 RequestSigner,是成功实现这一目标的关键。在大多数情况下,SDK 提供的服务客户端是首选,但当您需要更细粒度的控制或集成现有 HTTP 客户端时,RequestSigner 是一个强大的工具

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

868

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

745

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

741

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

420

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

447

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

431

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16948

2023.08.03

拼多多赚钱的5种方法 拼多多赚钱的5种方法
拼多多赚钱的5种方法 拼多多赚钱的5种方法

在拼多多上赚钱主要可以通过无货源模式一件代发、精细化运营特色店铺、参与官方高流量活动、利用拼团机制社交裂变,以及成为多多进宝推广员这5种方法实现。核心策略在于通过低成本、高效率的供应链管理与营销,利用平台社交电商红利实现盈利。

31

2026.01.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
SQL 教程
SQL 教程

共61课时 | 3.6万人学习

Java 教程
Java 教程

共578课时 | 51.5万人学习

oracle知识库
oracle知识库

共0课时 | 0人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号