続・アレな文字をWebClient.DownloadString(String)
に渡すとローカルのファイルが読める
ここ2つの記事でMonoのWebClient.DownloadString(string)
にアレな文字列渡すとローカルファイルを落としてきてしまうという挙動について調べてたわけですが、よくよくスタックトレースを見てみると、.NET FrameworkでもGetUri
というメソッドを経由してPath.GetFullPath
が呼ばれていたことがわかりました。
そんなわけでPath.GetFullPath
の挙動を見てみると、http://../../../../etc/passwd
といったような文字列を与えたときに.NETとMonoで次のような挙動の違いが見られました。
- .NET Frameworkは「URIフォーマットはサポートしていない」というメッセージと共にArgumentExceptionが投げられる
- Monoはそのままフルパスに変換する
.NET上で同じことをしてもDownloadString(string)
がここまで問題にしてきた挙動をしなかったのは、ここで例外を吐いて止まっていたから、というだけのことだったわけです。
スキーム
ところでスキームをhttpじゃなくて適当な何かに変えたらどうなるんだろうと思って試してみました。
与える文字列をabc://../../../../etc/passwd
といったように、実在しないような適当なスキームに変えて同じプログラムを動かしたところ、.NETではUriとしてのパースに成功しました。httpスキームのときは最初の部分をhostnameとして認識していたので、恐らくスキームを見てフォーマットを認識してるんでしょう。
さて、.NETにおいてはパースの結果abc://../etc/passwd
になりました。なぜか../
の部分がひとつにまとめられていましたがこれは一体どういう挙動なんでしょう。それはともかく、この結果を踏まえた上で同じ文字列をWebClient.DownloadString(string)
に渡すと、今度は「そんなURIプリフィックスは知らん」とNotSupportedExceptionを投げられました。WebRequest.Create
の中から呼ばれているようなので、前述のGetUri
などは成功しているようです(というか、new Uriが成功するんだからそら成功するだろう)。で、System.Net.WebRequest.Create(String)を見ると、やはり渡されたURIのスキームに対応するものがない、という例外でよさそうです。どんなURIが渡されてもこれが呼ばれるとするなら、http(s)://
、ftp://
、file://
のいずれにも該当しないURIは常に弾かれることになります。
Monoの場合、まずnew Uri(String)
がUriFormatException
で失敗します。ということは前の記事で述べた通りPath.GetFullPath
が呼ばれることになり、やはりこの場合でもローカルのファイルが読めてしまいます。
まとめ
結局何がこの挙動の違いを生み出していたのかというと以下の2点だと考えられます。
- Monoの
new Uri(String)
が未知のスキームを持つURIに対して失敗する(.NETは成功する) - Monoの
Path.GetFullPath(String)
がURIであるような文字列に対して成功する(.NETは失敗する)
とりあえずバグレポートも書いたのでこの件は一段落ということでいいんじゃないんでしょうか……(遠い目)
(バグレポート、type=“text"なフィールドでEnter叩いちゃって途中送信されて慌てて削除する方法を探してみるも見当も付かず、結局コメントで「途中送信しちゃった許してくださいお願いします何でもしますから」って言いながらレポート書き直したのは内緒だぞっ)
あと、整理のためにDownloadString(String)
の処理の流れを書いておきます。
.NET
既知のスキームを持つ不正なURI
given: http://../../etc/passwd
GetUri
が呼ばれる- たぶん内部的に
new Uri(String)
して失敗する - たぶんその結果
Path.GetFullPath(String)
が呼ばれる Path.GetFullPath(String)
がArgumentExceptionを投げる (“URI formats are not supported.”)
未知のスキームを持つ不正なURI
given: abc://../../etc/passwd
GetUri
が呼ばれる- たぶん内部的に
new Uri(String)
して成功する - 成功したのでその結果をそのまま返す
WebRequest.Create
が呼ばれるWebRequest.Create
が未知のスキームに対応できずNotSupportedExceptionを投げる
Mono
既知のスキームを持つ不正なURI
given: http://../../etc/passwd
CreateUri
が呼ばれるnew Uri(String)
して失敗するPath.GetFullPath(String)
が呼ばれるPath.GetFullPath(String)
が成功し、結果を返す (e.g./home/etc/passwd
)- その結果を
new Uri(String)
に渡す (結果file:///home/etc/passwd
なUriが返る) - それを取得する (この場合は(たぶん)そのファイルがないので例外を吐く)
未知のスキームを持つ不正なURI
given: abc://../../etc/passwd
CreateUri
が呼ばれるnew Uri(String)
して失敗するPath.GetFullPath(String)
が呼ばれるPath.GetFullPath(String)
が成功し、結果を返す (e.g./home/etc/passwd
)- その結果を
new Uri(String)
に渡す (結果file:///home/etc/passwd
なUriが返る) - それを取得する (この場合は(たぶん)そのファイルがないので例外を吐く)
※.NETとの対比で両方書いたが、Monoの場合いずれも処理の流れは全く同じ