Python 计算AWS4签名,Header鉴权与URL鉴权

AWS4 版本签名计算参考

#!/usr/bin/env python3
# -*- coding:utf-8 -*-
#  @Time: 2021/7/24 8:12
#  @Author:zhangmingda
#  @File: api_for_aws4_signature.py
#  @Software: PyCharm
#  Description:

from urllib.request import quote
import hashlib
import hmac
import datetime
import requests
import json
import base64

class KscClient(object):
    def __init__(self, ak, sk,service, domain, region, use_ssl=False):
        self.ak = ak
        self.sk = sk
        self.host = service + "." + domain
        self.region = region
        self.service = service
        self.protocol = 'https' if use_ssl else 'http'
        self.base_url = self.protocol + '://' + self.host + '/'

    # 创建规范化请求查询字符串方法
    def generate_canonical_querystring(self, **params):
        '''
        :param params: 查询字符串字典
        :return: url编码后的查询字符串key1=value1&key2=value2 ...
        '''
        sorted_params = ''
        sorted_param_keys = sorted(params.keys())
        for param in sorted_param_keys:
            # print(param)
            sorted_params += quote(param) + '=' + quote(params.get(param), safe='') + '&'  # 查询字符串中的值任何字符串都进行URL编码
            # print(sorted_params)
        return sorted_params[:-1]

    #  生成标准化请求头,请求头大写转小写,排序返回字符串
    def generate_canonical_headers(self, **headers):
        '''
        :param headers: 所有用来签名的请求头
        :return: 返回小写请求头排序后的key:value 字符串,每个key:value 之间\n换行
        '''
        sorted_headers = ''                                         # 准备标准请求头key+value字符串
        sorted_signed_headers = ''                                  # 准备签名的请求头key组合字符串
        header_keys = headers.keys()                                # 获取请求头所有key
        lower_headers = {}                                          # 准备存放小写请求头和值的字典
        for header in header_keys:                                  # 把大写请求头字典都转换为小写请求头字典
            lower_headers[header.lower()] = headers.get(header)
        sorted_lower_header_keys = sorted(lower_headers.keys())     # 把签名的请求头排序

        '''排序后的请求头和值进行组合字符串'''
        for lower_header in sorted_lower_header_keys:
            sorted_headers += lower_header + ':' + str(lower_headers[lower_header]).strip() + '\n'
            sorted_signed_headers += lower_header + ";"
        # print("============请求头排序==============")
        # print(sorted_headers)
        # print('------被签名的请求头----')
        # print(sorted_signed_headers)
        # print("==========请求头排序end============")
        sorted_lower_headers = {
            'sorted_headers': sorted_headers,                           # 签名的请求头key 和 value
            'signed_headers': sorted_signed_headers[:-1]                # 被签名的请求头字符串组合,最后一个分号; 不要
        }
        return sorted_lower_headers

    # 十六进制请求体Sha256Hash值--->请求体用
    def bytes_data_sha256_hex(self, bytes_data=''.encode()):
        '''
        :param binary_data: 字节码文件
        :return: 返回请求体内容的sha256 哈希值
        '''
        sha256 = hashlib.sha256()
        sha256.update(bytes_data)
        return sha256.hexdigest().lower()

    # 构建被签名字符串,把标准请求canonical_request 哈希取16进制用
    def str_sha256_hex(self, string=''):
        '''
        :param string:
        :return: 字符串的sha256哈希值
        '''
        sha256 = hashlib.sha256()
        sha256.update(string.encode())
        return sha256.hexdigest()

    # UTC时间字符串工具
    def get_utc_datetime(self):
        now_utc = datetime.datetime.utcnow()
        datetime_utc = now_utc.strftime('%Y%m%dT%H%M%SZ')
        date_utc = now_utc.strftime('%Y%m%d')
        return datetime_utc, date_utc

    # 生成规范化请求的字符串方法
    def generate_canonical_request(self, method, encode_uri, canonical_querystring, canonical_headers, signed_headers,payload_sha256_hex):
        ''' 1.1 规范化请求CanonicalRequest计算方法'''
        return method + '\n' \
               + encode_uri + '\n' \
               + canonical_querystring + '\n' \
               + canonical_headers + '\n' \
               + signed_headers + '\n' \
               + payload_sha256_hex  # 请求体不参与签名 时 : + 'UNSIGNED-PAYLOAD'

    # 生成被签名字符串的方法
    def generate_string_tosign(self, algorithm, datetime_utc, credentialscope, canonical_reques_sha256_hex ):
        ''' 1.2 被签名字符串计算方法'''
        return algorithm + '\n' \
                        + datetime_utc + '\n' \
                        + credentialscope + '\n' \
                        + canonical_reques_sha256_hex

    # 第二步构建签名key的工具方法
    def encode_string_to_hmac_256_digest(self, encode_salt, msg, digestmod=hashlib.sha256):
        '''
        :param encode_salt: 盐
        :param msg: 字符串信息
        :param digestmod: 摘要算法
        :return: HMAC-SHA256 加盐哈希后的字节
        '''
        digest_maker = hmac.new(encode_salt, msg.encode(), digestmod=digestmod)
        return digest_maker.digest()

    # 创建签名KEY的方法
    def generate_signing_key(self, signature_version,sk, date_utc, region, service, termchar='aws4_request'):
        k_secret = signature_version + sk
        k_date = self.encode_string_to_hmac_256_digest(k_secret.encode(), date_utc)
        k_region = self.encode_string_to_hmac_256_digest(k_date, region)
        k_service = self.encode_string_to_hmac_256_digest(k_region, service)
        k_signing = self.encode_string_to_hmac_256_digest(k_service, termchar)
        return k_signing

    # 创建签名的方法
    def generate_signature(self, signing_key, string_tosign):
        digest_maker = hmac.new(signing_key, string_tosign.encode(), digestmod=hashlib.sha256)
        hex_signature = digest_maker.hexdigest()
        # print('hex_signature:', digest_maker.hexdigest())
        return hex_signature

    # 构建带签名的请求头,或签名url查询字符串
    def get_auth_data(self, method, query_params, append_headers={}, binary_payload=''.encode()):
        # ========================================1. 创建被签名字符串======================================
        # 1.1 构建标准化请求
        method = method             # 1.1.1 HTTP方法
        canonical_uri = '/'         # 1.1.2 资源URI
        encode_uri = quote(canonical_uri, safe="/")  # 1.1.2 将URI编码成%字符串格式 不包含? 后的url查询参数
        payload_sha256_hex = self.bytes_data_sha256_hex(binary_payload)  # 1.1.5 请求体sha256 十六进制HASH值
        datetime_utc, date_utc = self.get_utc_datetime()
        # 准备请求头
        signature_headers = {
            'Host': self.host,
            'X-amz-date': datetime_utc
        }
        signature_headers.update(append_headers)
        # 请求头排序格式化
        sorted_lower_headers = self.generate_canonical_headers(**signature_headers)
        canonical_headers = sorted_lower_headers.get('sorted_headers')      # 1.1.4 小写排序后的请求头key:value
        signed_headers = sorted_lower_headers.get('signed_headers')         # 1.1.5 小写排序后的签名头
        # 1.2 创建信任状
        credentialscope = date_utc + "/" + self.region + "/" + self.service + "/aws4_request"
        # 查询参数字典
        auth_params = {}
        # 签名放在URL中时计算签名传递参数
        if query_params:
            auth_params = {
                'X-Amz-Algorithm': "AWS4-HMAC-SHA256",
                'X-Amz-Credential': self.ak + '/' + credentialscope,
                'X-Amz-Date': datetime_utc,
                'X-Amz-SignedHeaders': signed_headers,
            }
            auth_params.update(query_params)
            # print('auth_params: ', auth_params)
        auth_in_header_canonical_querystring = self.generate_canonical_querystring(**query_params)      # 1.1.3 通过header传递签名的查询字符串
        auth_in_queryparam_canonical_querystring = self.generate_canonical_querystring(**auth_params)   # 1.1.3 通过URL传递签名的查询字符串
        canonical_request_auth_in_header = self.generate_canonical_request(method, encode_uri, auth_in_header_canonical_querystring, canonical_headers, signed_headers, payload_sha256_hex)
        canonical_request_auth_in_url = self.generate_canonical_request(method, encode_uri, auth_in_queryparam_canonical_querystring, canonical_headers, signed_headers, payload_sha256_hex)
        print("Header传递签名的规范化请求".center(50, "="))
        print(canonical_request_auth_in_header)
        print("canonical_request_auth_in_url".center(50, "*"))
        print(canonical_request_auth_in_header)
        print("规范化化请求done".center(50, "="))

        # 1.2 创建被签名字的符串StringToSign
        canonical_request_auth_in_header_sha256_hex = self.str_sha256_hex(canonical_request_auth_in_header)  # 规范化请求的256哈希值
        canonical_request_auth_in_url_sha256_hex = self.str_sha256_hex(canonical_request_auth_in_url)        # 规范化请求的256哈希值
        algorithm = 'AWS4-HMAC-SHA256'
        string_tosign_auth_in_header = self.generate_string_tosign(algorithm, datetime_utc, credentialscope, canonical_request_auth_in_header_sha256_hex)
        string_tosign_auth_in_url = self.generate_string_tosign(algorithm, datetime_utc, credentialscope, canonical_request_auth_in_url_sha256_hex)

        print('Header传递签名被签名的字符串'.center(50, '='))
        print(string_tosign_auth_in_header)
        print('URL传递签名被签名的字符串'.center(50, '*'))
        print(string_tosign_auth_in_url)
        print('被签名的字符串Done!'.center(50,'='))

        # ========================================2. 创建签名 ======================================
        # 2.1创建签名签名秘钥
        signature_version = 'AWS4'
        signing_key = self.generate_signing_key(signature_version, self.sk, date_utc, self.region, self.service)
        print("签名秘钥:",signing_key)
        # 2.2 创建签名
        signature_for_header = self.generate_signature(signing_key, string_tosign_auth_in_header)
        signature_for_url = self.generate_signature(signing_key, string_tosign_auth_in_url)
        print('Header鉴权的签名:', signature_for_header)
        print('url鉴权的签名:', signature_for_url)
        # ========================================3 请求头或URL 传递签名鉴权和参数 ======================================
        authorization_headers = {
            'Authorization': "AWS4-HMAC-SHA256 Credential="
                             + self.ak + '/'
                             + credentialscope + ", "
                             + "SignedHeaders="
                             + signed_headers + ', '
                             + 'Signature=' + signature_for_header,
        }
        # 3.1请求头携带签名,返回带签名的请求头
        authorization_headers.update(signature_headers)

        # 3.2 url携带签名返回带签名的URL查询字符串
        request_query_params = {
            'X-Amz-Signature': signature_for_url
        }
        request_query_params.update(auth_params)
        authorization_params = self.generate_canonical_querystring(**request_query_params)
        # print('request_query_params: ', request_query_params)
        # print('authorization_params:',authorization_params)
        auth_data = {
            'authorization_headers': authorization_headers,
            'authorization_params': request_query_params,
            'signature_headers': signature_headers,  # get object 在URL里面传递签名 请求头要带的签名头
        }
        return auth_data

    #  通过标准请求头方式传递签名GET文件
    def get_test_auth_in_header(self, query_params={}, append_headers={}):
        method = 'GET'                                      # 1.1.1 HTTP方法
        auth_data = self.get_auth_data(method, query_params, append_headers)
        authorization_headers = auth_data.get('authorization_headers')
        req = requests.get(url=self.base_url, headers=authorization_headers, params=query_params)
        print('Header鉴权的URL:', req.url)
        print(req.status_code)
        print(req.text)

    def get_test_auth_in_query_param(self,query_params={}, append_headers={}):
        method = 'GET'  # 1.1.1 HTTP方法
        auth_data = self.get_auth_data(method, query_params, append_headers)
        auth_query_params = auth_data.get('authorization_params')
        signature_headers = auth_data.get('signature_headers')
        canonical_querystring = self.generate_canonical_querystring(**auth_query_params)

        # 通过URL传递鉴权=requests 传参方式1
        url_for_auth = self.base_url + "?" + canonical_querystring
        req2 = requests.get(url=url_for_auth, headers=signature_headers)
        print("手工拼接鉴权的URL:", url_for_auth)
        print(req2.status_code)
        print(req2.text)
        # 通过URL传递鉴权=requests 传参方式2
        req3 = requests.get(self.base_url, headers=signature_headers, params=auth_query_params)
        # req3 = requests.get(url=self.base_url, headers=signature_headers, params=canonical_querystring)
        print("requests自动拼接鉴权的URL:", req3.url)
        print(req3.status_code)
        print(req3.text)

if __name__ == '__main__':
    ak = 'XXXXXXXXXXXXXXX'
    sk = 'XXXXXXXXXXXXXXXXXXXXXXXXX'
    service = 'kcm'
    domain = 'api.ksyun.com'
    region = 'cn-beijing-6'
    ksc = KscClient(ak, sk, service, domain, region)
    query_params = {
        'Action': 'GetDownloadLink',
        'Version': '2016-03-04',
        'CertificateId': 'kcm2021022216204501'
    }
    ksc.get_test_auth_in_header(query_params=query_params)
    # ksc.get_test_auth_in_query_param(query_params=query_params)

Header 传递鉴权测试

 

 URL传递鉴权测试

 

posted on 2021-07-24 10:34  zhangmingda  阅读(677)  评论(1编辑  收藏  举报

导航