更新时间:2023-11-21 GMT+08:00
分享

C

操作场景

使用C语言调用APP认证的API时,您需要先获取SDK,参考API调用示例调用API。

准备环境

  1. 已获取API的域名、请求url、请求方法、AppKey和AppSecret等信息,具体参见认证前准备
  2. 安装openssl库。
    apt-get install libssl-dev
  3. 安装curl库。
    apt-get install libcurl4-openssl-dev

获取SDK

  1. 登录DataArts Studio控制台。
  2. 单击“数据服务”模块。
  3. 单击左侧菜单“共享版> SDK”或“专享版 > SDK”。
  4. 单击SDK使用引导区域里对应语言的SDK,下载SDK包到本地。
  5. 进行SDK包完整性校验。Windows操作系统下,打开本地命令提示符框,输入如下命令,在本地生成已下载SDK包的SHA256值,其中,“D:\java-sdk.zip”为SDK包的本地存放路径和SDK包名,请根据实际情况修改。

    certutil -hashfile D:\java-sdk.zip SHA256

    命令执行结果示例,如下所示:

    SHA256 的 D:\java-sdk.zip 哈希:
    becff4310645f3734344897ffdcabb1853d4b7d93b59a6ea187c5ae40543b36b
    CertUtil: -hashfile 命令成功完成。
    becff4310645f3734344897ffdcabb1853d4b7d93b59a6ea187c5ae40543b36b

    对比所下载SDK包的SHA256值和下表中对应语言SDK包的SHA256值。如果一致,则表示下载过程不存在篡改和丢包。

    不同语言SDK包

    SHA256

    Java

    becff4310645f3734344897ffdcabb1853d4b7d93b59a6ea187c5ae40543b36b

    Go

    bcf8cf19a21226e247195f2e584c8414da39b8d05840fb02948e1375d9bbb7e6

    Python

    c3da3b5814f828d6217963e856563d558d938b3da28993a8a13c8a7ebff5b95d

    C#

    a880b47e63ab35bfe216592e340a8135b866aef8f756ef7738fff3287885f33a

    JavaScript

    53261387f5fcf46e61d0bef5e890bea97952717f327c356412c3128389e848d6

    PHP

    29bf711144e77a4adaea1257cd6dedd2220e57b729a8fd000c51e68ccb42ad4b

    C++

    f604c6386c62cccb7c358007778037d5b15480987dc2860eef1b7bad37cb21d7

    C

    7086012c2d0569d5938830926b19fbea0d46682a983e04e52924978e8720c2f8

    Android

    89962b186707828b06b0c9f50c010b2f4cefd6a8e7ca9bdefb616bbbf6e739c8

获取“ApiGateway-c-sdk.zip”压缩包,解压后目录结构如下:

名称

说明

signer_common.c

SDK代码

signer_common.h

signer.c

signer.h

Makefile

Makefile文件

main.c

示例代码

调用API示例

  1. 在main.c中加入以下引用。

    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <curl/curl.h>
    #include "signer.h"

  2. 生成一个sig_params_t类型的变量, 填入AppKey和AppSecret。

    sig_params_t params;
    sig_params_init(&params);
    
    // 认证用的ak和sk硬编码到代码中或者明文存储都有很大的安全风险,建议在配置文件或者环境变量中密文存放,使用时解密,确保安全;
    // 本示例以ak和sk保存在环境变量中来实现身份验证为例,运行本示例前请先在本地环境中设置环境变量SDK_AK和SDK_SK。
    char* ak = getenv("SDK_AK");
    char* sk = getenv("SDK_SK");
    
    sig_str_t app_key = sig_str(ak);
    sig_str_t app_secret = sig_str(sk);
    params.key = app_key;
    params.secret = app_secret;

  3. 指定方法名、域名、请求uri、查询字符串和body。

    sig_str_t host = sig_str("c967a237-cd6c-470e-906f-a8655461897e.apigw.cn-north-1.huaweicloud.com");
    sig_str_t method = sig_str("GET");
    sig_str_t uri = sig_str("/app1");
    sig_str_t query_str = sig_str("a=1&b=2");
    sig_str_t payload = sig_str("");
    params.host = host;
    params.method = method;
    params.uri = uri;
    params.query_str = query_str;
    params.payload = payload;

  4. 给请求添加header头,内容为具体参数数据。如果有需要,添加需要签名的其他头域。

    sig_headers_add(&params.headers, "x-stage", "RELEASE");
    sig_headers_add(&params.headers, "name","value");

  5. 进行签名,执行此函数会将生成的签名头加入request变量中。然后为请求添加x-Authorization头,值与Authorization头相同。

    sig_sign(&params);
    char* authorization = sig_headers_get(&params.headers, "Authorization")->value.data;
    sig_headers_add(&params.headers, "x-Authorization", authorization);

  6. 使用curl库访问API,查看访问结果。

    static size_t
    WriteMemoryCallback(void *contents, size_t size, size_t nmemb, void *userp)
    {
        size_t realsize = size * nmemb;
        struct MemoryStruct *mem = (struct MemoryStruct *)userp;
    
        mem->memory = (char*)realloc(mem->memory, mem->size + realsize + 1);
        if (mem->memory == NULL) {
            /* out of memory! */
            printf("not enough memory (realloc returned NULL)\n");
            return 0;
        }
    
        memcpy(&(mem->memory[mem->size]), contents, realsize);
        mem->size += realsize;
        mem->memory[mem->size] = 0;
    
        return realsize;
    }
    
    //send http request using curl library
    int perform_request(RequestParams* request)
    {
        CURL *curl;
        CURLcode res;
        struct MemoryStruct resp_header;
        resp_header.memory = malloc(1);
        resp_header.size = 0;
        struct MemoryStruct resp_body;
        resp_body.memory = malloc(1);
        resp_body.size = 0;
    
        curl_global_init(CURL_GLOBAL_ALL);
        curl = curl_easy_init();
    
        curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, params.method.data);
        char url[1024];
        sig_snprintf(url, 1024, "http://%V%V?%V", &params.host, &params.uri, &params.query_str);
        curl_easy_setopt(curl, CURLOPT_URL, url);
        struct curl_slist *chunk = NULL;
        for (int i = 0; i < params.headers.len; i++) {
            char header[1024];
            sig_snprintf(header, 1024, "%V: %V", &params.headers.data[i].name, &params.headers.data[i].value);
            printf("%s\n", header);
            chunk = curl_slist_append(chunk, header);
        }
        printf("-------------\n");
        curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
        curl_easy_setopt(curl, CURLOPT_POSTFIELDS, params.payload.data);
        curl_easy_setopt(curl, CURLOPT_NOBODY, 0L);
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteMemoryCallback);
        curl_easy_setopt(curl, CURLOPT_HEADERDATA, (void *)&resp_header);
        curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *)&resp_body);
        //curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L);
        res = curl_easy_perform(curl);
        if (res != CURLE_OK) {
            fprintf(stderr, "curl_easy_perform() failed: %s\n", curl_easy_strerror(res));
        }
        else {
            long status;
            curl_easy_getinfo(curl, CURLINFO_HTTP_CODE, &status);
            printf("status %d\n", status);
            printf(resp_header.memory);
            printf(resp_body.memory);
        }
        free(resp_header.memory);
        free(resp_body.memory);
        curl_easy_cleanup(curl);
    
        curl_global_cleanup();
    
        //free signature params
        sig_params_free(&params);
        return 0;
    }

  7. 运行make命令编译,得到可执行文件main,执行main文件,查看结果。
分享:

    相关文档

    相关产品