AWS TranslateText REST API вызов добавления подписи v4 - PullRequest
0 голосов
/ 24 мая 2018

На самом деле это не вопрос, а ответ для тех, кто пытается использовать AWS TranslateText без SDK.У меня было много проблем, пока я не получил работающую версию.

По моему мнению, документация AWS для этого сервиса не завершена и не так много примеров для проверки (по крайней мере, для .Net).

Сначала я подумал, что не генерирую правильную сигнатуру V4, но после очередной проверки шагов и значений я решил использовать Postman для вызова службы.Это было очень полезно.

Почтальон может генерировать подпись AWS!(да, я не знал), поэтому, наконец, я заметил, что значение подписи не было проблемой.

Проверяя запрос, я мог видеть необходимые заголовки и некоторые из его значений.

Проблема вВ моем случае я не отправлял заголовок "x-amz-target".И случайно я обнаружил, что значение для этого заголовка должно быть «AWSShineFrontendService_20170701.TranslateText» (я видел подобное значение здесь https://rdrr.io/cran/aws.translate/src/R/translateHTTP.R)

Эта другая ссылка также была полезна Проблема подписания запроса Ivona- подпись не совпадает (AWS Signature Version 4)

Итак, у меня есть работающая версия, я хочу поделиться своим .Net кодом. Надеюсь, это поможет :) !!

using Newtonsoft.Json.Linq;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net;
using System.Security.Cryptography;
using System.Text;
using System.Web.Script.Serialization;

namespace AWS_TranslateTextTest
{
    class AWS_TranslateText
    {

        // Replace this with your own values
        const string AccessKey = "AKI_ADD_YOUR_ACCESSKEY";
        const string SecretKey = "ADD_YOUR_SECRETKEY";

        static void Main()
        {
                try
            {
                    string text = "Translate this text from English to German.";
                    string sourceLang = "en";
                    string targetLang = "de";

                    string responseText = TranslateText(text, sourceLang, targetLang);

                    JObject json = JObject.Parse(responseText);

                    string translatedText = ""; // to do read response  from json or responseText

                    if (json.ToString().Contains("TranslatedText")){
                        //To access to the properties in "dot" notation use a dynamic object
                        dynamic obj = json;
                        translatedText = obj.TranslatedText.Value;
                        Console.WriteLine("TranslatedText is: {0}", translatedText);
                    }
                    else{
                        Console.WriteLine("TranslatedText not found in response.");
                        throw new Exception(json.ToString());
                    }
            }
            catch (WebException ex)
            {
               Console.WriteLine(ex.Message);
                    if (ex.Response != null){
                foreach (string header in ex.Response.Headers)
                {
                    Console.WriteLine("{0}: {1}", header, ex.Response.Headers[header]);
                }
                using (var responseStream = ex.Response.GetResponseStream())
                {
                    if (responseStream != null)
                    {
                        using (var streamReader = new StreamReader(responseStream))
                        {
                            Console.WriteLine(streamReader.ReadToEnd());
                        }
                    }
                }    }
            }
            catch (Exception ex)
            {
                Console.WriteLine(ex.Message);
            }           
        }


          private static string TranslateText(string text, string sourceLang, string targetLang)
        {
            var date = DateTime.UtcNow;

            const string algorithm = "AWS4-HMAC-SHA256";
            const string regionName = "eu-west-1";
            const string serviceName = "translate";
            const string method = "POST";
            const string canonicalUri = "/";    
            const string canonicalQueryString = "";  
            const string x_amz_target_header = "AWSShineFrontendService_20170701.TranslateText";

                const string contentType = "application/x-amz-json-1.1";


            const string host = serviceName + "." + regionName + ".amazonaws.com";

            var obj = new
            {
                     SourceLanguageCode = sourceLang,
                     TargetLanguageCode = targetLang,
                     Text = text
            };
            var requestPayload = new JavaScriptSerializer().Serialize(obj);

            var hashedRequestPayload = HexEncode(Hash(ToBytes(requestPayload)));

            var dateStamp = date.ToString("yyyyMMdd");
            var requestDate = date.ToString("yyyyMMddTHHmmss") + "Z";
            var credentialScope = string.Format("{0}/{1}/{2}/aws4_request", dateStamp, regionName, serviceName);

                var bytes = ToBytes(requestPayload);


                var headers = new SortedDictionary<string, string>
            {

                     {"content-length", bytes.Length.ToString()},
                     {"content-type", contentType},
                {"host", host},
                     {"x-amz-date", requestDate},
                     {"x-amz-target", x_amz_target_header}
            };

            string canonicalHeaders =
                string.Join("\n", headers.Select(x => x.Key.ToLowerInvariant() + ":" + x.Value.Trim())) + "\n";

                string signedHeaders =
                string.Join(";", headers.Select(x => x.Key.ToLowerInvariant() ));

            // Task 1: Create a Canonical Request For Signature Version 4
            var canonicalRequest = method + '\n' + canonicalUri + '\n' + canonicalQueryString +
                                   '\n' + canonicalHeaders + '\n' + signedHeaders + '\n' + hashedRequestPayload;

            var hashedCanonicalRequest = HexEncode(Hash(ToBytes(canonicalRequest)));


            // Task 2: Create a String to Sign for Signature Version 4
            // StringToSign  = Algorithm + '\n' + RequestDate + '\n' + CredentialScope + '\n' + HashedCanonicalRequest

            var stringToSign = string.Format("{0}\n{1}\n{2}\n{3}", algorithm, requestDate, credentialScope,
                hashedCanonicalRequest);


            // Task 3: Calculate the AWS Signature Version 4

            // HMAC(HMAC(HMAC(HMAC("AWS4" + kSecret,"20130913"),"eu-west-1"),"tts"),"aws4_request")
            byte[] signingKey = GetSignatureKey(SecretKey, dateStamp, regionName, serviceName);

            // signature = HexEncode(HMAC(derived-signing-key, string-to-sign))
            var signature = HexEncode(HmacSha256(stringToSign, signingKey));


            // Task 4: Prepare a signed request
            // Authorization: algorithm Credential=access key ID/credential scope, SignedHeadaers=SignedHeaders, Signature=signature

            var authorization =
                string.Format("{0} Credential={1}/{2}/{3}/{4}/aws4_request, SignedHeaders={5}, Signature={6}",
                    algorithm, AccessKey, dateStamp, regionName, serviceName, signedHeaders, signature);

            // Send the request
                string endpoint = "https://" + host; // + canonicalUri ;


            var webRequest = WebRequest.Create(endpoint);

            webRequest.Method = method;
            webRequest.Timeout = 20000;
            webRequest.ContentType = contentType;
            webRequest.Headers.Add("X-Amz-Date", requestDate);
            webRequest.Headers.Add("Authorization", authorization);
                webRequest.Headers.Add("X-Amz-Target", x_amz_target_header);    

            webRequest.ContentLength = bytes.Length; 

            using (Stream newStream = webRequest.GetRequestStream())
            {
                newStream.Write(bytes, 0, bytes.Length);
                newStream.Flush();
            }

            var response = (HttpWebResponse)webRequest.GetResponse();

            using (Stream responseStream = response.GetResponseStream())
            {
                if (responseStream != null)
                {

                          using (var streamReader = new StreamReader(responseStream))
                          {
                                string res = streamReader.ReadToEnd();
                                return res;
                          }
                }
            }



            return null;
        }

        private static byte[] GetSignatureKey(String key, String dateStamp, String regionName, String serviceName)
        {
            byte[] kDate = HmacSha256(dateStamp, ToBytes("AWS4" + key));
            byte[] kRegion = HmacSha256(regionName, kDate);
            byte[] kService = HmacSha256(serviceName, kRegion);
            return HmacSha256("aws4_request", kService);
        }

        private static byte[] ToBytes(string str)
        {
            return Encoding.UTF8.GetBytes(str.ToCharArray());
        }

        private static string HexEncode(byte[] bytes)
        {
            return BitConverter.ToString(bytes).Replace("-", string.Empty).ToLowerInvariant();
        }

        private static byte[] Hash(byte[] bytes)
        {
            return SHA256.Create().ComputeHash(bytes);
        }

        private static byte[] HmacSha256(String data, byte[] key)
        {
            return new HMACSHA256(key).ComputeHash(ToBytes(data));
        }
    }
}
...