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 密钥。

      Build AI
      Build 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 Maven专题
Java Maven专题

本专题聚焦 Java 主流构建工具 Maven 的学习与应用,系统讲解项目结构、依赖管理、插件使用、生命周期与多模块项目配置。通过企业管理系统、Web 应用与微服务项目实战,帮助学员全面掌握 Maven 在 Java 项目构建与团队协作中的核心技能。

0

2025.09.15

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

483

2023.08.02

resource是什么文件
resource是什么文件

Resource文件是一种特殊类型的文件,它通常用于存储应用程序或操作系统中的各种资源信息。它们在应用程序开发中起着关键作用,并在跨平台开发和国际化方面提供支持。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

158

2023.12.20

pdf怎么转换成xml格式
pdf怎么转换成xml格式

将 pdf 转换为 xml 的方法:1. 使用在线转换器;2. 使用桌面软件(如 adobe acrobat、itext);3. 使用命令行工具(如 pdftoxml)。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1903

2024.04.01

xml怎么变成word
xml怎么变成word

步骤:1. 导入 xml 文件;2. 选择 xml 结构;3. 映射 xml 元素到 word 元素;4. 生成 word 文档。提示:确保 xml 文件结构良好,并预览 word 文档以验证转换是否成功。想了解更多xml的相关内容,可以阅读本专题下面的文章。

2094

2024.08.01

xml是什么格式的文件
xml是什么格式的文件

xml是一种纯文本格式的文件。xml指的是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。想了解更多相关的内容,可阅读本专题下面的相关文章。

1084

2024.11.28

登录token无效
登录token无效

登录token无效解决方法:1、检查token的有效期限,如果token已经过期,需要重新获取一个新的token;2、检查token的签名,如果签名不正确,需要重新获取一个新的token;3、检查密钥的正确性,如果密钥不正确,需要重新获取一个新的token;4、使用HTTPS协议传输token,建议使用HTTPS协议进行传输 ;5、使用双因素认证,双因素认证可以提高账户的安全性。

6220

2023.09.14

登录token无效怎么办
登录token无效怎么办

登录token无效的解决办法有检查Token是否过期、检查Token是否正确、检查Token是否被篡改、检查Token是否与用户匹配、清除缓存或Cookie、检查网络连接和服务器状态、重新登录或请求新的Token、联系技术支持或开发人员等。本专题为大家提供token相关的文章、下载、课程内容,供大家免费下载体验。

822

2023.09.14

2026赚钱平台入口大全
2026赚钱平台入口大全

2026年最新赚钱平台入口汇总,涵盖任务众包、内容创作、电商运营、技能变现等多类正规渠道,助你轻松开启副业增收之路。阅读专题下面的文章了解更多详细内容。

54

2026.01.31

热门下载

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

精品课程

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

共61课时 | 3.6万人学习

Java 教程
Java 教程

共578课时 | 54万人学习

oracle知识库
oracle知识库

共0课时 | 0人学习

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

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