ラディアスリーの加藤です。
先日『Unity2017.3.0p4 で AWS SDK が例外を吐くようになったので原因を調べてた』というエントリーを書いたのですが、テストを行ってみるとどうもiOS端末上で lambda の実行が失敗する事に気が付きました。
アプリ起動時にユーザデータの確認のため lambda を叩いてデータを取得するという処理を行っていたので、当初は「起動直後に lambda を叩いた場合 Credential なんかが確定する前に lambda を叩いてしまうケールがあるのかな?」なんてことを考えていました。
成功するケースもあるので、「リトライ処理を組み込めば良いのか?」なんてことも考えたのですが、原因がはっきりとしないまま変なリトライ処理を埋め込むのも気持ちが悪いと思い調査を行ってみました。
結構、軽い気持ちで調査を始めてみたのですが、これが意外と混沌としてしまっていて、思いの外時間を取られてしまったのですが…
まぁ、せっかくなので足跡を残しておこうと思います。
…Webでの情報も少ないので、レアケースなんですかね?
問題
Unity2017.3.0p4 で AWS SDK から Lambda の function を呼び出そうとすると、iOS端末(実機)でエラーが返ってきてしまうことがある。
Unity エディタ上では、現象が再現しない。
返ってくるエラーは以下の内容となっていた。
The request signature we calculated does not match the signature you provided. Check your AWS Secret Access Key and signing method. Consult the service documentation for details.
Unity Editor 上では状況が再現せずに、正常に lambda の実行が行えている。
iOS端末(iPhone7)では、体感的にだが6〜7割程度の確率でエラーが返ってきているようだ。
(正常に lambda の実行が行えているときもある)
ビルド環境
macOS High Sierra(10.13.3)
Unity2017.3.0p4
AWSSDK.Lambda.3.3.12.0.unitypackage
問題が発生する環境
iPhone7(MNCR2J/A)
iOS 11.2.5
原因?
AWS SDK の内部では UnityWebRequest を利用して通信が行われているが、その際にリクエストヘッダに ”Transfer-Encoding:chunked” が含まれるケースがあり、その時に上記エラーが返ってきてしまうようだ。
前回のポストで、AWS SDK が例外を吐いてしまう問題の対応として、
1 2 3 4 5 6 7 8 9 |
void Start() { UnityInitializer.AttachToGameObject(this.gameObject); // この行を追加 AWSConfigs.HttpClient = AWSConfigs.HttpClientOption.UnityWebRequest; InvokeButton.onClick.AddListener(() => { Invoke(); }); ListFunctionsButton.onClick.AddListener(() => { ListFunctions(); }); } |
のように UnityWebRequest を利用するように設定を行ったが、ここから今回の問題に派生したようだ。
対応方法
github で公開されている CustomUnityMainThreadDispatcher コンポーネントを採用させてもらった。
次のようなコードを追加した。
1 2 3 4 5 6 |
void Awake() { UnityInitializer.AttachToGameObject (this.gameObject); gameObject.AddComponent<CustomUnityMainThreadDispatcher>(); AWSConfigs.HttpClient = AWSConfigs.HttpClientOption.UnityWebRequest; } |
原因調査の記録
色々とハマったので、以下に原因調査の記録を残しておく。
(単なる備忘録なので、この先には有益な情報は無い。)
うまく lambda が実行できるときと、エラーが返ってしまう時があるので、原因を探りたかったのだがどうにも腑に落ちるような結論が得られずに困った。
エラーメッセージから、Quiitaで公開されていた『Unity2017.3 AWSSDKエラーの回避』のエントリーと、そこで紹介されていた AWSSDK.NET の github issue ( https://github.com/aws/aws-sdk-net/issues/820 ) にたどり着いたのだが、同じエラーが発生しているとは言え、発生確率が異なる(私のケースでは、うまく動くときもある)ので今ひとつ納得感ないのだが、上記対応を入れてみた所エラーが発生しなくなっているようなので、ひとまずはここが問題だったと結論づけておく。
CustomUnityMainThreadDispatcher では、UnityWebRequest の Transfer-Encoding を 無理やり false にすることで回避を行っているようなので、逆に Transfer-Encoding を常に true に設定することでエラーになる場合の再現が行えるかと考えたのだが、うまくいかなかった。(成功する場合も、失敗する場合もある)
件の Quiita エントリーでは、
UnityEditorからUnityWebRequest(AWSSDKを使用せずシンプルなコードで)を使って送信内容を確認してみると、Unity2017.3.0p4にのみTransfer-Encoding: chunkedのHeader情報が追加されていました。どうやらこの影響で、エラーが発生しているようなのです。
と書かれていたので、エラーになるケースではリクエストヘッダに “Transfer-Encoding:chunked” が含まれており、逆にうまく動作するケースでは含まれていないという仮説を立てて再現を行おうとしたが、AWS SDK を利用している状況でリクエストヘッダを確認する手段がどうにも思いつかなかった。
当初は、 CustomUnityMainThreadDispatcher の内部で、リクエストヘッダの中身を確認できないかとごにょごにょとコードをいじってみたのだが、思ったような成果は得られなかった。
仕方がないので、 UnityWebRequest を使ったシンプルなシーンとリクエストを受け取りヘッダをそのまま返すだけのシンプルなサーバを作成してリクエストヘッダの内容を確認することにした。
サーバは node.js で以下のような簡単な処理を書いた。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 |
var http = require('http'); var server = http.createServer(function(req, res) { console.log("================"); console.log(req.method); var body = ''; req.on('data', function(data) { body += data; }); req.on('end', function() { console.log("body=[" + body + "]"); }); console.log(JSON.stringify(req.rawHeaders)); console.log("---"); for (var i = 0; i < req.rawHeaders.length / 2; i++) { console.log(req.rawHeaders[(i*2)] + ":" + req.rawHeaders[(i*2)+1]); } console.log("---"); res.writeHeader(200, { 'Content-Type': 'text/plane' }); res.write(JSON.stringify(req.rawHeaders)); res.end(); return ; }).listen(80); |
対するクライアント側は、以下のようなコンポーネントを作成した。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 |
using System.Collections; using System.Collections.Generic; using UnityEngine; using UnityEngine.UI; using UnityEngine.Networking; public class WebRequestTest : MonoBehaviour { [SerializeField] Text _text; string uri = "http://xxx.xxx.xxx.xxx”; // 接続先のIPアドレス // Use this for initialization void Start () { } IEnumerator HttpRequestGet() { var request = UnityWebRequest.Get(uri); //request.chunkedTransfer = false; yield return request.Send(); Debug.Log(request.responseCode.ToString() + ":" + request.downloadHandler.text); _text.text = request.downloadHandler.text; } IEnumerator HttpRequestPost(string data) { var request = UnityWebRequest.Post(uri, data); //request.chunkedTransfer = false; yield return request.Send(); Debug.Log(request.responseCode.ToString() + ":" + request.downloadHandler.text); _text.text = request.downloadHandler.text; } public void OnSendGet() { _text.text = ""; StartCoroutine(HttpRequestGet()); } public void OnSendPost01() { _text.text = ""; StartCoroutine(HttpRequestPost("")); } public void OnSendPost02() { _text.text = ""; StartCoroutine(HttpRequestPost("hello")); } public void OnSendPost03() { _text.text = ""; StartCoroutine(HttpRequestPost("6\nHelloWorld\n5\n 0\n")); } } |
ボタンを押すとリクエストを投げて、結果を画面表示するだけの簡単なものだ。
調べていくとどうも UnityWebRequest で Post する時のデータに応じて、リクエストヘッダに Transfer-Encoding が含まれるのか、含まれないのかが決まるようだ。
これを実行してみると次のような結果が得られた。
Getの場合
1 2 3 4 5 6 7 |
Host:xxx.xxx.xxx.xxx(ここにはホストのIPアドレスが入っている) Connection:keep-alive Accept:*/* User-Agent:testapp/0 CFNetwork/894 Darwin/17.4.0 Accept-Language:ja-jp Accept-Encoding:gzip, deflate X-Unity-Version:2017.3.0p4 |
Post(データなし)の場合
1 2 3 4 5 6 7 8 9 |
Host:xxx.xxx.xxx.xxx Content-Type:application/x-www-form-urlencoded Content-Length:0 Connection:keep-alive Accept:*/* User-Agent:testapp/0 CFNetwork/894 Darwin/17.4.0 Accept-Language:ja-jp Accept-Encoding:gzip, deflate X-Unity-Version:2017.3.0p4 |
Post(データ=‘hello’)の場合
1 2 3 4 5 6 7 8 9 |
Host:xxx.xxx.xxx.xxx Content-Type:application/x-www-form-urlencoded X-Unity-Version:2017.3.0p4 Connection:keep-alive Accept:*/* Accept-Language:ja-jp User-Agent:testapp/0 CFNetwork/894 Darwin/17.4.0 Accept-Encoding:gzip, deflate Transfer-Encoding:chunked |
Post(データ=’6\nHelloWorld\n5\n 0\n’)の場合
1 2 3 4 5 6 7 8 9 |
Host:xxx.xxx.xxx.xxx Content-Type:application/x-www-form-urlencoded X-Unity-Version:2017.3.0p4 Connection:keep-alive Accept:*/* Accept-Language:ja-jp User-Agent:testapp/0 CFNetwork/894 Darwin/17.4.0 Accept-Encoding:gzip, deflate Transfer-Encoding:chunked |
結果から、Post 時に何らかのデータが含まれていると、リクエストヘッダに Transfer-Encoding が含まれることがわかった。
推察するに、AWD SDK が何らかの条件で、Post と Get を使い分けているのか、Post 時にデータを含めたり、含めなかったりしているのではないかと思う。
また、 iOS端末上では問題が発生することがあるがUnityエディタ上からでは問題が発生しないことから、iOS端末での実行とUnityエディタ上での実行で、レスポンスヘッダに変化があるのかと推察していたが、調査を行ってみた所、どちらも同じ条件(Post時になんらかのデータが含まれている場合のみ)で Transfer-Encoding が含まれることがわかった。
まとめ
なんとなく釈然とはしないが、
「iOS端末上でレスポンスヘッダに Transfer-Encoding が含まれている場合にlambda の実行が失敗する」
として、AWS SDK 側で対応が行われるまでの間は、 CustomUnityMainThreadDispatcher を利用させてもらうことにしようと思う。