[system.net.http] Convert default request headers to string values using correct...
[mono.git] / mcs / class / System.Net.Http / Test / System.Net.Http / HttpClientTest.cs
1 //
2 // HttpClientTest.cs
3 //
4 // Authors:
5 //      Marek Safar  <marek.safar@gmail.com>
6 //
7 // Copyright (C) 2011 Xamarin Inc (http://www.xamarin.com)
8 //
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
16 //
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
19 //
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
27 //
28
29 using System;
30 using System.Collections;
31 using System.Collections.Generic;
32 using NUnit.Framework;
33 using System.Net.Http;
34 using System.Net.Http.Headers;
35 using System.Threading;
36 using System.Threading.Tasks;
37 using System.Net;
38 using System.Linq;
39 using System.IO;
40
41 namespace MonoTests.System.Net.Http
42 {
43         [TestFixture]
44         public class HttpClientTest
45         {
46                 class HttpMessageHandlerMock : HttpMessageHandler
47                 {
48                         public Func<HttpRequestMessage, Task<HttpResponseMessage>> OnSend;
49                         public Func<HttpRequestMessage, CancellationToken, Task<HttpResponseMessage>> OnSendFull;
50
51                         public HttpMessageHandlerMock ()
52                         {
53                         }
54
55                         protected override Task<HttpResponseMessage> SendAsync (HttpRequestMessage request, CancellationToken cancellationToken)
56                         {
57                                 if (OnSend != null)
58                                         return OnSend (request);
59
60                                 if (OnSendFull != null)
61                                         return OnSendFull (request, cancellationToken);
62
63                                 Assert.Fail ("Send");
64                                 return null;
65                         }
66                 }
67
68                 class HttpClientHandlerMock : HttpClientHandler
69                 {
70                         public Func<HttpRequestMessage, Task<HttpResponseMessage>> OnSend;
71                         public Func<HttpRequestMessage, CancellationToken, Task<HttpResponseMessage>> OnSendFull;
72
73                         protected override Task<HttpResponseMessage> SendAsync (HttpRequestMessage request, CancellationToken cancellationToken)
74                         {
75                                 if (OnSend != null)
76                                         return OnSend (request);
77
78                                 if (OnSendFull != null)
79                                         return OnSendFull (request, cancellationToken);
80
81                                 Assert.Fail ("Send");
82                                 return null;
83                         }
84                 }
85
86                 class CustomStream : Stream
87                 {
88                         public override void Flush ()
89                         {
90                                 throw new NotImplementedException ();
91                         }
92
93                         int pos;
94
95                         public override int Read (byte[] buffer, int offset, int count)
96                         {
97                                 ++pos;
98                                 if (pos > 4)
99                                         return 0;
100
101                                 return 11;
102                         }
103
104                         public override long Seek (long offset, SeekOrigin origin)
105                         {
106                                 throw new NotImplementedException ();
107                         }
108
109                         public override void SetLength (long value)
110                         {
111                                 throw new NotImplementedException ();
112                         }
113
114                         public override void Write (byte[] buffer, int offset, int count)
115                         {
116                                 throw new NotImplementedException ();
117                         }
118
119                         public override bool CanRead {
120                                 get {
121                                         return true;
122                                 }
123                         }
124
125                         public override bool CanSeek {
126                                 get {
127                                         return false;
128                                 }
129                         }
130
131                         public override bool CanWrite {
132                                 get {
133                                         throw new NotImplementedException ();
134                                 }
135                         }
136
137                         public override long Length {
138                                 get {
139                                         throw new NotImplementedException ();
140                                 }
141                         }
142
143                         public override long Position {
144                                 get {
145                                         throw new NotImplementedException ();
146                                 }
147                                 set {
148                                         throw new NotImplementedException ();
149                                 }
150                         }
151                 }
152
153                 const int WaitTimeout = 5000;
154
155                 string port, TestHost, LocalServer;
156
157                 [SetUp]
158                 public void SetupFixture ()
159                 {
160                         if (Environment.OSVersion.Platform == PlatformID.Win32NT) {
161                                 port = "810";
162                         } else {
163                                 port = "8810";
164                         }
165
166                         TestHost = "localhost:" + port;
167                         LocalServer = string.Format ("http://{0}/", TestHost);
168                 }
169
170                 [Test]
171                 public void Ctor_Default ()
172                 {
173                         var client = new HttpClient ();
174                         Assert.IsNull (client.BaseAddress, "#1");
175                         Assert.IsNotNull (client.DefaultRequestHeaders, "#2");  // TODO: full check
176                         Assert.AreEqual (int.MaxValue, client.MaxResponseContentBufferSize, "#3");
177                         Assert.AreEqual (TimeSpan.FromSeconds (100), client.Timeout, "#4");
178                 }
179
180                 [Test]
181                 public void CancelPendingRequests ()
182                 {
183                         var mh = new HttpMessageHandlerMock ();
184
185                         var client = new HttpClient (mh);
186                         var request = new HttpRequestMessage (HttpMethod.Get, "http://xamarin.com");
187                         var mre = new ManualResetEvent (false);
188
189                         mh.OnSendFull = (l, c) => {
190                                 mre.Set ();
191                                 Assert.IsTrue (c.WaitHandle.WaitOne (1000), "#20");
192                                 Assert.IsTrue (c.IsCancellationRequested, "#21");
193                                 mre.Set ();
194                                 return Task.FromResult (new HttpResponseMessage ());
195                         };
196
197                         var t = Task.Factory.StartNew (() => {
198                                 client.SendAsync (request).Wait (WaitTimeout);
199                         });
200
201                         Assert.IsTrue (mre.WaitOne (500), "#1");
202                         mre.Reset ();
203                         client.CancelPendingRequests ();
204                         Assert.IsTrue (t.Wait (500), "#2");
205
206                         request = new HttpRequestMessage (HttpMethod.Get, "http://xamarin.com");
207                         mh.OnSendFull = (l, c) => {
208                                 Assert.IsFalse (c.IsCancellationRequested, "#30");
209                                 return Task.FromResult (new HttpResponseMessage ());
210                         };
211
212                         client.SendAsync (request).Wait (WaitTimeout);
213                 }
214
215                 [Test]
216                 public void CancelPendingRequests_BeforeSend ()
217                 {
218                         var ct = new CancellationTokenSource ();
219                         ct.Cancel ();
220                         var rr = CancellationTokenSource.CreateLinkedTokenSource (new CancellationToken (), ct.Token);
221
222
223                         var mh = new HttpMessageHandlerMock ();
224
225                         var client = new HttpClient (mh);
226                         var request = new HttpRequestMessage (HttpMethod.Get, "http://xamarin.com");
227                         client.CancelPendingRequests ();
228
229                         mh.OnSendFull = (l, c) => {
230                                 Assert.IsFalse (c.IsCancellationRequested, "#30");
231                                 return Task.FromResult (new HttpResponseMessage ());
232                         };
233
234                         client.SendAsync (request).Wait (WaitTimeout);
235
236                         request = new HttpRequestMessage (HttpMethod.Get, "http://xamarin.com");
237                         client.SendAsync (request).Wait (WaitTimeout);
238                 }
239
240
241                 [Test]
242                 public void CancelRequestViaProxy ()
243                 {
244                         var handler = new HttpClientHandler {
245                                 Proxy = new WebProxy ("192.168.10.25:8888/"), // proxy that doesn't exist
246                                 UseProxy = true,
247                                 AutomaticDecompression = DecompressionMethods.GZip | DecompressionMethods.Deflate
248                         };
249
250                         var httpClient = new HttpClient (handler) {
251                                 BaseAddress = new Uri ("https://google.com"),
252                                 Timeout = TimeSpan.FromMilliseconds (1)
253                         };
254
255                         try {
256                                 var restRequest = new HttpRequestMessage {
257                                         Method = HttpMethod.Post,
258                                         RequestUri = new Uri("foo", UriKind.Relative),
259                                         Content = new StringContent("", null, "application/json")
260                                 };
261
262                                 httpClient.PostAsync (restRequest.RequestUri, restRequest.Content).Wait (WaitTimeout);
263                                 Assert.Fail ("#1");
264                         } catch (AggregateException e) {
265                                 Assert.IsTrue (e.InnerException is TaskCanceledException, "#2");
266                         }
267                 }
268
269                 [Test]
270                 public void Properties ()
271                 {
272                         var client = new HttpClient ();
273                         client.BaseAddress = null;
274                         client.MaxResponseContentBufferSize = int.MaxValue;
275                         client.Timeout = Timeout.InfiniteTimeSpan;
276
277                         Assert.IsNull (client.BaseAddress, "#1");
278                         Assert.AreEqual (int.MaxValue, client.MaxResponseContentBufferSize, "#2");
279                         Assert.AreEqual (Timeout.InfiniteTimeSpan, client.Timeout, "#3");
280                 }
281
282                 [Test]
283                 public void Properties_Invalid ()
284                 {
285                         var client = new HttpClient ();
286                         try {
287                                 client.MaxResponseContentBufferSize = 0;
288                                 Assert.Fail ("#1");
289                         } catch (ArgumentOutOfRangeException) {
290                         }
291
292                         try {
293                                 client.Timeout = TimeSpan.MinValue;
294                                 Assert.Fail ("#2");
295                         } catch (ArgumentOutOfRangeException) {
296                         }
297                 }
298
299                 [Test]
300                 public void Send ()
301                 {
302                         var mh = new HttpMessageHandlerMock ();
303
304                         var client = new HttpClient (mh);
305                         client.BaseAddress = new Uri ("http://xamarin.com");
306                         var request = new HttpRequestMessage ();
307                         var response = new HttpResponseMessage ();
308
309                         mh.OnSend = l => {
310                                 Assert.AreEqual (l, request, "#2");
311                                 Assert.AreEqual (client.BaseAddress, l.RequestUri, "#2");
312                                 return Task.FromResult (response);
313                         };
314
315                         Assert.AreEqual (response, client.SendAsync (request).Result, "#1");
316                 }
317
318                 [Test]
319                 public void Send_BaseAddress ()
320                 {
321                         var mh = new HttpMessageHandlerMock ();
322
323                         var client = new HttpClient (mh);
324                         client.BaseAddress = new Uri ("http://localhost/");
325                         var response = new HttpResponseMessage ();
326
327                         mh.OnSend = l => {
328                                 Assert.AreEqual ("http://localhost/relative", l.RequestUri.ToString (), "#2");
329                                 return Task.FromResult (response);
330                         };
331
332                         Assert.AreEqual (response, client.GetAsync ("relative").Result, "#1");
333                         Assert.AreEqual (response, client.GetAsync ("/relative").Result, "#2");
334                 }
335
336                 [Test]
337                 public void Send_DefaultRequestHeaders ()
338                 {
339                         var mh = new HttpMessageHandlerMock ();
340
341                         var client = new HttpClient (mh);
342                         client.DefaultRequestHeaders.Referrer = new Uri ("http://google.com");
343
344                         var request = new HttpRequestMessage (HttpMethod.Get, "http://xamarin.com");
345                         var response = new HttpResponseMessage ();
346
347                         mh.OnSend = l => {
348                                 Assert.AreEqual (client.DefaultRequestHeaders.Referrer, l.Headers.Referrer, "#2");
349                                 Assert.IsNotNull (l.Headers.Referrer, "#3");
350                                 return Task.FromResult (response);
351                         };
352
353                         Assert.AreEqual (response, client.SendAsync (request).Result, "#1");
354                 }
355
356                 [Test]
357                 public void Send_Complete_Default ()
358                 {
359                         bool? failed = null;
360                         var listener = CreateListener (l => {
361                                 try {
362                                         var request = l.Request;
363         
364                                         Assert.IsNull (request.AcceptTypes, "#1");
365                                         Assert.AreEqual (0, request.ContentLength64, "#2");
366                                         Assert.IsNull (request.ContentType, "#3");
367                                         Assert.AreEqual (0, request.Cookies.Count, "#4");
368                                         Assert.IsFalse (request.HasEntityBody, "#5");
369                                         Assert.AreEqual (TestHost, request.Headers["Host"], "#6b");
370                                         Assert.AreEqual ("GET", request.HttpMethod, "#7");
371                                         Assert.IsFalse (request.IsAuthenticated, "#8");
372                                         Assert.IsTrue (request.IsLocal, "#9");
373                                         Assert.IsFalse (request.IsSecureConnection, "#10");
374                                         Assert.IsFalse (request.IsWebSocketRequest, "#11");
375                                         Assert.IsTrue (request.KeepAlive, "#12");
376                                         Assert.AreEqual (HttpVersion.Version11, request.ProtocolVersion, "#13");
377                                         Assert.IsNull (request.ServiceName, "#14");
378                                         Assert.IsNull (request.UrlReferrer, "#15");
379                                         Assert.IsNull (request.UserAgent, "#16");
380                                         Assert.IsNull (request.UserLanguages, "#17");
381                                         failed = false;
382                                 } catch {
383                                         failed = true;
384                                 }
385                         });
386
387                         try {
388                                 var client = new HttpClient ();
389                                 var request = new HttpRequestMessage (HttpMethod.Get, LocalServer);
390                                 var response = client.SendAsync (request, HttpCompletionOption.ResponseHeadersRead).Result;
391
392                                 Assert.AreEqual ("", response.Content.ReadAsStringAsync ().Result, "#100");
393                                 Assert.AreEqual (HttpStatusCode.OK, response.StatusCode, "#101");
394                                 Assert.AreEqual (false, failed, "#102");
395                         } finally {
396                                 listener.Close ();
397                         }
398                 }
399
400                 [Test]
401                 public void Send_Complete_Version_1_0 ()
402                 {
403                         bool? failed = null;
404                         
405                         var listener = CreateListener (l => {
406                                 try {
407                                         var request = l.Request;
408         
409                                         Assert.IsNull (request.AcceptTypes, "#1");
410                                         Assert.AreEqual (0, request.ContentLength64, "#2");
411                                         Assert.IsNull (request.ContentType, "#3");
412                                         Assert.AreEqual (0, request.Cookies.Count, "#4");
413                                         Assert.IsFalse (request.HasEntityBody, "#5");
414                                         Assert.AreEqual (1, request.Headers.Count, "#6");
415                                         Assert.AreEqual (TestHost, request.Headers["Host"], "#6a");
416                                         Assert.AreEqual ("GET", request.HttpMethod, "#7");
417                                         Assert.IsFalse (request.IsAuthenticated, "#8");
418                                         Assert.IsTrue (request.IsLocal, "#9");
419                                         Assert.IsFalse (request.IsSecureConnection, "#10");
420                                         Assert.IsFalse (request.IsWebSocketRequest, "#11");
421                                         Assert.IsFalse (request.KeepAlive, "#12");
422                                         Assert.AreEqual (HttpVersion.Version10, request.ProtocolVersion, "#13");
423                                         Assert.IsNull (request.ServiceName, "#14");
424                                         Assert.IsNull (request.UrlReferrer, "#15");
425                                         Assert.IsNull (request.UserAgent, "#16");
426                                         Assert.IsNull (request.UserLanguages, "#17");
427                                         failed = false;
428                                 } catch {
429                                         failed = true;
430                                 }
431                         });
432
433                         try {
434                                 var client = new HttpClient ();
435                                 var request = new HttpRequestMessage (HttpMethod.Get, LocalServer);
436                                 request.Version = HttpVersion.Version10;
437                                 var response = client.SendAsync (request, HttpCompletionOption.ResponseHeadersRead).Result;
438
439                                 Assert.AreEqual ("", response.Content.ReadAsStringAsync ().Result, "#100");
440                                 Assert.AreEqual (HttpStatusCode.OK, response.StatusCode, "#101");
441                                 Assert.AreEqual (false, failed, "#102");
442                         } finally {
443                                 listener.Close ();
444                         }
445                 }
446
447                 [Test]
448                 public void Send_Complete_ClientHandlerSettings ()
449                 {
450                         bool? failed = null;
451                         
452                         var listener = CreateListener (l => {
453                                 var request = l.Request;
454                                 
455                                 try {
456                                         Assert.IsNull (request.AcceptTypes, "#1");
457                                         Assert.AreEqual (0, request.ContentLength64, "#2");
458                                         Assert.IsNull (request.ContentType, "#3");
459                                         Assert.AreEqual (1, request.Cookies.Count, "#4");
460                                         Assert.AreEqual (new Cookie ("mycookie", "vv"), request.Cookies[0], "#4a");
461                                         Assert.IsFalse (request.HasEntityBody, "#5");
462                                         Assert.AreEqual (4, request.Headers.Count, "#6");
463                                         Assert.AreEqual (TestHost, request.Headers["Host"], "#6a");
464                                         Assert.AreEqual ("gzip", request.Headers["Accept-Encoding"], "#6b");
465                                         Assert.AreEqual ("mycookie=vv", request.Headers["Cookie"], "#6c");
466                                         Assert.AreEqual ("GET", request.HttpMethod, "#7");
467                                         Assert.IsFalse (request.IsAuthenticated, "#8");
468                                         Assert.IsTrue (request.IsLocal, "#9");
469                                         Assert.IsFalse (request.IsSecureConnection, "#10");
470                                         Assert.IsFalse (request.IsWebSocketRequest, "#11");
471                                         Assert.IsTrue (request.KeepAlive, "#12");
472                                         Assert.AreEqual (HttpVersion.Version10, request.ProtocolVersion, "#13");
473                                         Assert.IsNull (request.ServiceName, "#14");
474                                         Assert.IsNull (request.UrlReferrer, "#15");
475                                         Assert.IsNull (request.UserAgent, "#16");
476                                         Assert.IsNull (request.UserLanguages, "#17");
477                                         failed = false;
478                                 } catch {
479                                         failed = true;
480                                 }
481                         });
482
483                         try {
484                                 var chandler = new HttpClientHandler ();
485                                 chandler.AllowAutoRedirect = true;
486                                 chandler.AutomaticDecompression = DecompressionMethods.GZip;
487                                 chandler.MaxAutomaticRedirections = 33;
488                                 chandler.MaxRequestContentBufferSize = 5555;
489                                 chandler.PreAuthenticate = true;
490                                 chandler.CookieContainer.Add (new Uri (LocalServer), new Cookie ( "mycookie", "vv"));
491                                 chandler.UseCookies = true;
492                                 chandler.UseDefaultCredentials = true;
493                                 chandler.Proxy = new WebProxy ("ee");
494                                 chandler.UseProxy = true;
495
496                                 var client = new HttpClient (chandler);
497                                 var request = new HttpRequestMessage (HttpMethod.Get, LocalServer);
498                                 request.Version = HttpVersion.Version10;
499                                 request.Headers.Add ("Keep-Alive", "false");
500                                 var response = client.SendAsync (request, HttpCompletionOption.ResponseHeadersRead).Result;
501
502                                 Assert.AreEqual ("", response.Content.ReadAsStringAsync ().Result, "#100");
503                                 Assert.AreEqual (HttpStatusCode.OK, response.StatusCode, "#101");
504                                 Assert.AreEqual (false, failed, "#102");
505                         } finally {
506                                 listener.Abort ();
507                                 listener.Close ();
508                         }
509                 }
510
511                 [Test]
512                 public void Send_Complete_CustomHeaders ()
513                 {
514                         bool? failed = null;
515                         
516                         var listener = CreateListener (l => {
517                                 var request = l.Request;
518                                 try {
519                                         Assert.AreEqual ("vv", request.Headers["aa"], "#1");
520         
521                                         var response = l.Response;
522                                         response.Headers.Add ("rsp", "rrr");
523                                         response.Headers.Add ("upgrade", "vvvvaa");
524                                         response.Headers.Add ("Date", "aa");
525                                         response.Headers.Add ("cache-control", "audio");
526         
527                                         response.StatusDescription = "test description";
528                                         response.ProtocolVersion = HttpVersion.Version10;
529                                         response.SendChunked = true;
530                                         response.RedirectLocation = "w3.org";
531                                         
532                                         failed = false;
533                                 } catch {
534                                         failed = true;
535                                 }
536                         });
537
538                         try {
539                                 var client = new HttpClient ();
540                                 var request = new HttpRequestMessage (HttpMethod.Get, LocalServer);
541                                 Assert.IsTrue (request.Headers.TryAddWithoutValidation ("aa", "vv"), "#0");
542                                 var response = client.SendAsync (request, HttpCompletionOption.ResponseHeadersRead).Result;
543
544                                 Assert.AreEqual ("", response.Content.ReadAsStringAsync ().Result, "#100");
545                                 Assert.AreEqual (HttpStatusCode.OK, response.StatusCode, "#101");
546                                 
547                                 IEnumerable<string> values;
548                                 Assert.IsTrue (response.Headers.TryGetValues ("rsp", out values), "#102");
549                                 Assert.AreEqual ("rrr", values.First (), "#102a");
550
551                                 Assert.IsTrue (response.Headers.TryGetValues ("Transfer-Encoding", out values), "#103");
552                                 Assert.AreEqual ("chunked", values.First (), "#103a");
553                                 Assert.AreEqual (true, response.Headers.TransferEncodingChunked, "#103b");
554
555                                 Assert.IsTrue (response.Headers.TryGetValues ("Date", out values), "#104");
556                                 Assert.AreEqual (1, values.Count (), "#104b");
557                                 // .NET overwrites Date, Mono does not
558                                 // Assert.IsNotNull (response.Headers.Date, "#104c");
559
560                                 Assert.AreEqual (new ProductHeaderValue ("vvvvaa"), response.Headers.Upgrade.First (), "#105");
561
562                                 Assert.AreEqual ("audio", response.Headers.CacheControl.Extensions.First ().Name, "#106");
563
564                                 Assert.AreEqual ("w3.org", response.Headers.Location.OriginalString, "#107");
565
566                                 Assert.AreEqual ("test description", response.ReasonPhrase, "#110");
567                                 Assert.AreEqual (HttpVersion.Version11, response.Version, "#111");
568                                 
569                                 Assert.AreEqual (false, failed, "#112");
570                         } finally {
571                                 listener.Close ();
572                         }
573                 }
574
575                 [Test]
576                 public void Send_Complete_CustomHeaders_SpecialSeparators ()
577                 {
578                         bool? failed = null;
579
580                         var listener = CreateListener (l => {
581                                 var request = l.Request;
582
583                                 try {
584                                         Assert.AreEqual ("MLK Android Phone 1.1.9", request.UserAgent, "#1");
585                                         failed = false;
586                                 } catch {
587                                         failed = true;
588                                 }
589                         });
590
591                         try {
592                                 var client = new HttpClient ();
593
594                                 client.DefaultRequestHeaders.Add("User-Agent", "MLK Android Phone 1.1.9");
595
596                                 var request = new HttpRequestMessage (HttpMethod.Get, LocalServer);
597
598                                 var response = client.SendAsync (request, HttpCompletionOption.ResponseHeadersRead).Result;
599
600                                 Assert.AreEqual ("", response.Content.ReadAsStringAsync ().Result, "#100");
601                                 Assert.AreEqual (HttpStatusCode.OK, response.StatusCode, "#101");
602                                 Assert.AreEqual (false, failed, "#102");
603                         } finally {
604                                 listener.Abort ();
605                                 listener.Close ();
606                         }
607                 }
608
609
610                 [Test]
611                 public void Send_Complete_Content ()
612                 {
613                         var listener = CreateListener (l => {
614                                 var request = l.Request;
615                                 l.Response.OutputStream.WriteByte (55);
616                                 l.Response.OutputStream.WriteByte (75);
617                         });
618
619                         try {
620                                 var client = new HttpClient ();
621                                 var request = new HttpRequestMessage (HttpMethod.Get, LocalServer);
622                                 Assert.IsTrue (request.Headers.TryAddWithoutValidation ("aa", "vv"), "#0");
623                                 var response = client.SendAsync (request, HttpCompletionOption.ResponseHeadersRead).Result;
624
625                                 Assert.AreEqual ("7K", response.Content.ReadAsStringAsync ().Result, "#100");
626                                 Assert.AreEqual (HttpStatusCode.OK, response.StatusCode, "#101");
627
628                                 IEnumerable<string> values;
629                                 Assert.IsTrue (response.Headers.TryGetValues ("Transfer-Encoding", out values), "#102");
630                                 Assert.AreEqual ("chunked", values.First (), "#102a");
631                                 Assert.AreEqual (true, response.Headers.TransferEncodingChunked, "#102b");
632                         } finally {
633                                 listener.Close ();
634                         }
635                 }
636
637                 [Test]
638                 public void Send_Complete_Content_MaxResponseContentBufferSize ()
639                 {
640                         var listener = CreateListener (l => {
641                                 var request = l.Request;
642                                 var b = new byte[4000];
643                                 l.Response.OutputStream.Write (b, 0, b.Length);
644                         });
645
646                         try {
647                                 var client = new HttpClient ();
648                                 client.MaxResponseContentBufferSize = 1000;
649                                 var request = new HttpRequestMessage (HttpMethod.Get, LocalServer);
650                                 var response = client.SendAsync (request, HttpCompletionOption.ResponseHeadersRead).Result;
651
652                                 Assert.AreEqual (4000, response.Content.ReadAsStringAsync ().Result.Length, "#100");
653                                 Assert.AreEqual (HttpStatusCode.OK, response.StatusCode, "#101");
654                         } finally {
655                                 listener.Close ();
656                         }
657                 }
658
659                 [Test]
660                 public void Send_Complete_Content_MaxResponseContentBufferSize_Error ()
661                 {
662                         var listener = CreateListener (l => {
663                                 var request = l.Request;
664                                 var b = new byte[4000];
665                                 l.Response.OutputStream.Write (b, 0, b.Length);
666                         });
667
668                         try {
669                                 var client = new HttpClient ();
670                                 client.MaxResponseContentBufferSize = 1000;
671                                 var request = new HttpRequestMessage (HttpMethod.Get, LocalServer);
672
673                                 try {
674                                         client.SendAsync (request, HttpCompletionOption.ResponseContentRead).Wait (WaitTimeout);
675                                         Assert.Fail ("#2");
676                                 } catch (AggregateException e) {
677                                         Assert.IsTrue (e.InnerException is HttpRequestException, "#3");
678                                 }
679
680                         } finally {
681                                 listener.Close ();
682                         }
683                 }
684
685                 [Test]
686                 public void Send_Complete_NoContent ()
687                 {
688                         foreach (var method in new HttpMethod[] { HttpMethod.Post, HttpMethod.Put, HttpMethod.Delete }) {
689                                 bool? failed = null;
690                                 var listener = CreateListener (l => {
691                                         try {
692                                                 var request = l.Request;
693
694                                                 Assert.AreEqual (2, request.Headers.Count, "#1");
695                                                 Assert.AreEqual ("0", request.Headers ["Content-Length"], "#1b");
696                                                 Assert.AreEqual (method.Method, request.HttpMethod, "#2");
697                                                 failed = false;
698                                         } catch {
699                                                 failed = true;
700                                         }
701                                 });
702
703                                 try {
704                                         var client = new HttpClient ();
705                                         var request = new HttpRequestMessage (method, LocalServer);
706                                         var response = client.SendAsync (request, HttpCompletionOption.ResponseHeadersRead).Result;
707
708                                         Assert.AreEqual ("", response.Content.ReadAsStringAsync ().Result, "#100");
709                                         Assert.AreEqual (HttpStatusCode.OK, response.StatusCode, "#101");
710                                         Assert.AreEqual (false, failed, "#102");
711                                 } finally {
712                                         listener.Close ();
713                                 }
714                         }
715                 }
716
717                 [Test]
718                 public void Send_Complete_Error ()
719                 {
720                         var listener = CreateListener (l => {
721                                 var response = l.Response;
722                                 response.StatusCode = 500;
723                         });
724
725                         try {
726                                 var client = new HttpClient ();
727                                 var request = new HttpRequestMessage (HttpMethod.Get, LocalServer);
728                                 var response = client.SendAsync (request, HttpCompletionOption.ResponseHeadersRead).Result;
729
730                                 Assert.AreEqual ("", response.Content.ReadAsStringAsync ().Result, "#100");
731                                 Assert.AreEqual (HttpStatusCode.InternalServerError, response.StatusCode, "#101");
732                         } finally {
733                                 listener.Close ();
734                         }
735                 }
736
737                 [Test]
738                 public void Send_Content_Get ()
739                 {
740                         var listener = CreateListener (l => {
741                                 var request = l.Request;
742                                 l.Response.OutputStream.WriteByte (72);
743                         });
744
745                         try {
746                                 var client = new HttpClient ();
747                                 var r = new HttpRequestMessage (HttpMethod.Get, LocalServer);
748                                 var response = client.SendAsync (r).Result;
749
750                                 Assert.AreEqual ("H", response.Content.ReadAsStringAsync ().Result);
751                         } finally {
752                                 listener.Close ();
753                         }
754                 }
755
756                 [Test]
757                 public void Send_Content_BomEncoding ()
758                 {
759                         var listener = CreateListener (l => {
760                                 var request = l.Request;
761
762                                 var str = l.Response.OutputStream;
763                                 str.WriteByte (0xEF);
764                                 str.WriteByte (0xBB);
765                                 str.WriteByte (0xBF);
766                                 str.WriteByte (71);
767                         });
768
769                         try {
770                                 var client = new HttpClient ();
771                                 var r = new HttpRequestMessage (HttpMethod.Get, LocalServer);
772                                 var response = client.SendAsync (r).Result;
773
774                                 Assert.AreEqual ("G", response.Content.ReadAsStringAsync ().Result);
775                         } finally {
776                                 listener.Close ();
777                         }
778                 }
779
780                 [Test]
781                 public void Send_Content_Put ()
782                 {
783                         bool passed = false;
784                         var listener = CreateListener (l => {
785                                 var request = l.Request;
786                                 passed = 7 == request.ContentLength64;
787                                 passed &= request.ContentType == "text/plain; charset=utf-8";
788                                 passed &= request.InputStream.ReadByte () == 'm';
789                         });
790
791                         try {
792                                 var client = new HttpClient ();
793                                 var r = new HttpRequestMessage (HttpMethod.Put, LocalServer);
794                                 r.Content = new StringContent ("my text");
795                                 var response = client.SendAsync (r).Result;
796
797                                 Assert.AreEqual (HttpStatusCode.OK, response.StatusCode, "#1");
798                                 Assert.IsTrue (passed, "#2");
799                         } finally {
800                                 listener.Abort ();
801                                 listener.Close ();
802                         }
803                 }
804
805                 [Test]
806                 public void Send_Content_Put_CustomStream ()
807                 {
808                         bool passed = false;
809                         var listener = CreateListener (l => {
810                                 var request = l.Request;
811                                 passed = 44 == request.ContentLength64;
812                                 passed &= request.ContentType == null;
813                         });
814
815                         try {
816                                 var client = new HttpClient ();
817                                 var r = new HttpRequestMessage (HttpMethod.Put, LocalServer);
818                                 r.Content = new StreamContent (new CustomStream ());
819                                 var response = client.SendAsync (r).Result;
820
821                                 Assert.AreEqual (HttpStatusCode.OK, response.StatusCode, "#1");
822                                 Assert.IsTrue (passed, "#2");
823                         } finally {
824                                 listener.Abort ();
825
826                                 listener.Close ();
827                         }
828                 }
829
830                 [Test]
831                 public void Send_Timeout ()
832                 {
833                         var mh = new HttpMessageHandlerMock ();
834
835                         var client = new HttpClient (mh);
836                         client.Timeout = TimeSpan.FromMilliseconds (100);
837                         var request = new HttpRequestMessage (HttpMethod.Get, "http://xamarin.com");
838                         var response = new HttpResponseMessage ();
839
840                         mh.OnSendFull = (l, c) => {
841                                 Assert.IsTrue (c.WaitHandle.WaitOne (500), "#2");
842                                 return Task.FromResult (response);
843                         };
844
845                         Assert.AreEqual (response, client.SendAsync (request).Result, "#1");
846                 }
847
848                 [Test]
849                 public void Send_Invalid ()
850                 {
851                         var client = new HttpClient ();
852                         try {
853                                 client.SendAsync (null).Wait (WaitTimeout);
854                                 Assert.Fail ("#1");
855                         } catch (ArgumentNullException) {
856                         }
857
858                         try {
859                                 var request = new HttpRequestMessage ();
860                                 client.SendAsync (request).Wait (WaitTimeout);
861                                 Assert.Fail ("#2");
862                         } catch (InvalidOperationException) {
863                         }
864                 }
865
866                 [Test]
867                 public void Send_InvalidHandler ()
868                 {
869                         var mh = new HttpMessageHandlerMock ();
870
871                         var client = new HttpClient (mh);
872                         client.BaseAddress = new Uri ("http://xamarin.com");
873                         var request = new HttpRequestMessage ();
874
875                         mh.OnSend = l => {
876                                 Assert.AreEqual (l, request, "#1");
877                                 return null;
878                         };
879
880                         try {
881                                 // Broken by design
882                                 client.SendAsync (request).Wait (WaitTimeout);
883                                 Assert.Fail ("#2");
884                         } catch (Exception) {
885                         }
886                 }
887
888                 [Test]
889                 public void Send_SameMessage ()
890                 {
891                         var mh = new HttpMessageHandlerMock ();
892
893                         var client = new HttpClient (mh);
894                         var request = new HttpRequestMessage (HttpMethod.Get, "http://xamarin.com");
895
896                         mh.OnSend = l => Task.FromResult (new HttpResponseMessage ());
897
898                         client.SendAsync (request).Wait (WaitTimeout);
899                         try {
900                                 client.SendAsync (request).Wait (WaitTimeout);
901                                 Assert.Fail ("#1");
902                         } catch (InvalidOperationException) {
903                         }
904                 }
905
906                 [Test]
907                 [Category ("MobileNotWorking")] // Missing encoding
908                 public void GetString_Many ()
909                 {
910                         var client = new HttpClient ();
911                         var t1 = client.GetStringAsync ("http://www.google.com");
912                         var t2 = client.GetStringAsync ("http://www.google.com");
913                         Assert.IsTrue (Task.WaitAll (new [] { t1, t2 }, WaitTimeout));          
914                 }
915
916                 [Test]
917                 public void GetByteArray_ServerError ()
918                 {
919                         var listener = CreateListener (l => {
920                                 var response = l.Response;
921                                 response.StatusCode = 500;
922                                 l.Response.OutputStream.WriteByte (72);
923                         });
924
925                         try {
926                                 var client = new HttpClient ();
927                                 try {
928                                         client.GetByteArrayAsync (LocalServer).Wait (WaitTimeout);
929                                         Assert.Fail ("#1");
930                                 } catch (AggregateException e) {
931                                         Assert.IsTrue (e.InnerException is HttpRequestException , "#2");
932                                 }
933                         } finally {
934                                 listener.Close ();
935                         }
936                 }
937
938                 [Test]
939                 public void DisallowAutoRedirect ()
940                 {
941                         var listener = CreateListener (l => {
942                                 var request = l.Request;
943                                 var response = l.Response;
944                                 
945                                 response.StatusCode = (int)HttpStatusCode.Moved;
946                                 response.RedirectLocation = "http://xamarin.com/";
947                         });
948
949                         try {
950                                 var chandler = new HttpClientHandler ();
951                                 chandler.AllowAutoRedirect = false;
952                                 var client = new HttpClient (chandler);
953
954                                 try {
955                                         client.GetStringAsync (LocalServer).Wait (WaitTimeout);
956                                         Assert.Fail ("#1");
957                                 } catch (AggregateException e) {
958                                         Assert.IsTrue (e.InnerException is HttpRequestException, "#2");
959                                 }
960                         } finally {
961                                 listener.Abort ();
962                                 listener.Close ();
963                         }
964                 }
965
966                 [Test]
967                 public void RequestUriAfterRedirect ()
968                 {
969                         var listener = CreateListener (l => {
970                                 var request = l.Request;
971                                 var response = l.Response;
972
973                                 response.StatusCode = (int)HttpStatusCode.Moved;
974                                 response.RedirectLocation = "http://xamarin.com/";
975                         });
976
977                         try {
978                                 var chandler = new HttpClientHandler ();
979                                 chandler.AllowAutoRedirect = true;
980                                 var client = new HttpClient (chandler);
981
982                                 var r = client.GetAsync (LocalServer);
983                                 Assert.IsTrue (r.Wait (WaitTimeout), "#1");
984                                 var resp = r.Result;
985                                 Assert.AreEqual ("http://xamarin.com/", resp.RequestMessage.RequestUri.AbsoluteUri, "#2");
986                         } finally {
987                                 listener.Abort ();
988                                 listener.Close ();
989                         }
990                 }
991
992                 [Test]
993                 /*
994                  * Properties may only be modified before sending the first request.
995                  */
996                 public void ModifyHandlerAfterFirstRequest ()
997                 {
998                         var chandler = new HttpClientHandler ();
999                         chandler.AllowAutoRedirect = true;
1000                         var client = new HttpClient (chandler, true);
1001
1002                         var listener = CreateListener (l => {
1003                                 var response = l.Response;
1004                                 response.StatusCode = 200;
1005                                 response.OutputStream.WriteByte (55);
1006                         });
1007
1008                         try {
1009                                 client.GetStringAsync (LocalServer).Wait (WaitTimeout);
1010                                 try {
1011                                         chandler.AllowAutoRedirect = false;
1012                                         Assert.Fail ("#1");
1013                                 } catch (InvalidOperationException) {
1014                                         ;
1015                                 }
1016                         } finally {
1017                                 listener.Abort ();
1018                                 listener.Close ();
1019                         }
1020                 }
1021
1022                 [Test]
1023                 /*
1024                  * However, this policy is not enforced for custom handlers and there
1025                  * is also no way a derived class could tell its HttpClientHandler parent
1026                  * that it just sent a request.
1027                  * 
1028                  */
1029                 public void ModifyHandlerAfterFirstRequest_Mock ()
1030                 {
1031                         var ch = new HttpClientHandlerMock ();
1032                         ch.AllowAutoRedirect = true;
1033
1034                         var client = new HttpClient (ch);
1035
1036                         ch.OnSend = (l) => {
1037                                 return Task.FromResult (new HttpResponseMessage ());
1038                         };
1039
1040                         client.GetAsync ("http://xamarin.com").Wait (WaitTimeout);
1041                         ch.AllowAutoRedirect = false;
1042                 }
1043
1044                 HttpListener CreateListener (Action<HttpListenerContext> contextAssert)
1045                 {
1046                         var l = new HttpListener ();
1047                         l.Prefixes.Add (string.Format ("http://+:{0}/", port));
1048                         l.Start ();
1049                         l.BeginGetContext (ar => {
1050                                 var ctx = l.EndGetContext (ar);
1051
1052                                 try {
1053                                         if (contextAssert != null)
1054                                                 contextAssert (ctx);
1055                                 } finally {
1056                                         ctx.Response.Close ();
1057                                 }
1058                         }, null);
1059
1060                         return l;
1061                 }
1062         }
1063 }