#签名 #aws #头部 #生成 #服务 #版本 #认证

aws-sign-v4

轻松生成 AWS Signature 4 头部

5 个不稳定版本

0.3.0 2024年4月1日
0.2.0 2023年1月18日
0.1.1 2019年12月30日
0.1.0 2019年11月26日

#748 in 网络编程

Download history 238/week @ 2024-04-14 236/week @ 2024-04-21 188/week @ 2024-04-28 397/week @ 2024-05-05 363/week @ 2024-05-12 279/week @ 2024-05-19 159/week @ 2024-05-26 231/week @ 2024-06-02 266/week @ 2024-06-09 316/week @ 2024-06-16 534/week @ 2024-06-23 316/week @ 2024-06-30 348/week @ 2024-07-07 286/week @ 2024-07-14 405/week @ 2024-07-21 294/week @ 2024-07-28

1,350 每月下载量

BSD-2-Clause

17KB
237

aws-sign-v4

Crates.io

使用此 crate 生成 AWS Signature Version 4 服务的 AUTHORIZATION 头部。

示例

示例 AWS S3 GET 请求

使用 reqwest 的示例使用方法

const S3_ACCESS: &str = "my-access-key";
const S3_SECRET: &str = "my-secret-key";

#[tokio::main]
async fn main() -> Result<(), reqwest::Error> {
    let datetime = chrono::Utc::now();
    let url = "https://myHost/s3-bucket-url";
    let mut headers = reqwest::header::HeaderMap::new();

    headers.insert(
        "X-Amz-Date",
        datetime
            .format("%Y%m%dT%H%M%SZ")
            .to_string()
            .parse()
            .unwrap(),
    );
    headers.insert("host", "myHost".parse().unwrap());

    let s = aws_sign_v4::AwsSign::new(
        "GET",
        url,
        &datetime,
        &headers,
        "us-east-1",
        &S3_ACCESS,
        &S3_SECRET,
        "s3",
        ""
    );
    let signature = s.sign();
    println!("{:#?}", signature);
    headers.insert(reqwest::header::AUTHORIZATION, signature.parse().unwrap());

    let client = reqwest::Client::new();
    let res = client
        .get(url)
        .headers(headers.to_owned())
        .body("")
        .send()
        .await?;

    println!("Status: {}", res.status());
    let body = res.text().await?;
    println!("Body:\n\n{}", body);
    Ok(())
}

示例 AWS graphql POST 请求

使用 reqwest 的示例使用方法

const S3_ACCESS: &str = "my-access-key";
const S3_SECRET: &str = "my-secret-key";

#[tokio::main]
async fn main() -> Result<(), reqwest::Error> {
    let datetime = chrono::Utc::now();
    let url = "https://myHost/s3-bucket-url";
    let mut headers = reqwest::header::HeaderMap::new();

    headers.insert(
        "X-Amz-Date",
        datetime
            .format("%Y%m%dT%H%M%SZ")
            .to_string()
            .parse()
            .unwrap(),
    );
    headers.insert("host", "myHost".parse().unwrap());

    let s = aws_sign_v4::AwsSign::new(
        "POST",
        url,
        &datetime,
        &headers,
        "us-east-1",
        &S3_ACCESS,
        &S3_SECRET,
        "execute-api",
        ""
    );
    let signature = s.sign();
    println!("{:#?}", signature);
    headers.insert(reqwest::header::AUTHORIZATION, signature.parse().unwrap());

    let client = reqwest::Client::new();
    let res = client
        .get(url)
        .headers(headers.to_owned())
        .body(r#"{"query":"query test_q { mynode { id }} ","variables":{},"operationName":"test_q"}"#)
        .send()
        .await?;

    println!("Status: {}", res.status());
    let body = res.text().await?;
    println!("Response Body:\n{}", body);
    Ok(())
}

示例 AWS SNS 发布请求

使用 reqwest 的示例使用方法

const S3_ACCESS: &str = "my-access-key";
const S3_SECRET: &str = "my-secret-key";
const S3_TOPIC_ARN: &str = "arn:aws:sns:eu-west-1:xxx:xxx";
const S3_REGION: &str = "eu-west-1";

#[tokio::main]
async fn main() {
    let hostname = format!("sns.{}.amazonaws.com", S3_REGION);
    let url = format!("https://{}/", hostname);
    let ts = chrono::Utc::now();

    let mut headers = reqwest::header::HeaderMap::new();
    headers.insert("host", hostname.parse().unwrap());
    headers.insert(
        "X-Amz-Date",
        ts.format("%Y%m%dT%H%M%SZ").to_string().parse().unwrap(),
    );
    headers.insert(
        reqwest::header::CONTENT_TYPE,
        "application/x-www-form-urlencoded".parse().unwrap(),
    );

    let body = [
        ("Action", "Publish"),
        ("TopicArn", S3_TOPIC_ARN),
        ("MessageAttributes.entry.1.Name", "AttributeExample"),
        ("MessageAttributes.entry.1.Value.DataType", "String"),
        ("MessageAttributes.entry.1.Value.StringValue", "AttributeExampleValue"),
        ("Message", "Hello world!"),
    ];
    let body = serde_urlencoded::to_string(body).unwrap();

    let s = aws_sign_v4::AwsSign::new(
        "POST", &url, &ts, &headers, &S3_REGION, &S3_ACCESS, &S3_SECRET, "sns", &body,
    )
    .sign();

    headers.insert(reqwest::header::AUTHORIZATION, s.parse().unwrap());

    let client = reqwest::Client::new();
    let res = client
        .post(url)
        .headers(headers)
        .body(body)
        .send()
        .await
        .unwrap();
    res.error_for_status().unwrap();
}

从 0.1.x 迁移到 0.2.x 的代码

# Before (0.1.1):
let s = aws_sign_v4::AwsSign::new(
        "GET",
        url,
        &datetime,
        &headers,
        "us-east-1",
        &S3_ACCESS,
        &S3_SECRET,
    );

# After (0.2.0):
let s = aws_sign_v4::AwsSign::new(
        "GET",
        url,
        &datetime,
        &headers,
        "us-east-1",
        &S3_ACCESS,
        &S3_SECRET,
        "s3", // <- explicitly add "s3", since 0.1.x only supported "s3"
        "", // <-- body can be ignored for "s3" service GETs
    );

参考

依赖项

~8–17MB
~331K SLoC