Merge pull request #835 from HorstKakuschke/master
[mono.git] / mcs / class / System / Test / System.Net / WebUtilityTest.cs
1 //
2 // System.Net.WebUtilityTest.cs
3 //
4 // copied and edited from System.Web.HttpUtilityTest.cs
5 //
6 // Author:
7 //      Sebastien Pouliot  <sebastien@ximian.com>
8 //      Mike Kestner <mkestner@novell.com>
9 //
10 // Copyright (C) 2005, 2010 Novell, Inc (http://www.novell.com)
11 //
12 // Permission is hereby granted, free of charge, to any person obtaining
13 // a copy of this software and associated documentation files (the
14 // "Software"), to deal in the Software without restriction, including
15 // without limitation the rights to use, copy, modify, merge, publish,
16 // distribute, sublicense, and/or sell copies of the Software, and to
17 // permit persons to whom the Software is furnished to do so, subject to
18 // the following conditions:
19 // 
20 // The above copyright notice and this permission notice shall be
21 // included in all copies or substantial portions of the Software.
22 // 
23 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
24 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
25 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
26 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
27 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
28 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
29 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
30 //
31
32 #if NET_4_0 && !MOBILE
33
34 using System;
35 using System.Text;
36 using System.Net;
37 using System.IO;
38 using System.Collections.Specialized;
39
40 using NUnit.Framework;
41
42 namespace MonoTests.System.Net {
43
44         [TestFixture]
45         public class WebUtilityTest {
46
47                 [Test]
48                 public void HtmlEncode_LtGt ()
49                 {
50                         Assert.AreEqual ("&lt;script&gt;", WebUtility.HtmlEncode ("<script>"));
51                 }
52
53                 // Notes:
54                 // * this is to avoid a regression that would cause Mono to 
55                 //   fail item #3 of the XSS vulnerabilities listed at:
56                 //   http://it-project.ru/andir/docs/aspxvuln/aspxvuln.en.xml
57                 //   we didn't fall the first time so let's ensure we never will
58                 // * The author notes that Microsoft has decided not to fix 
59                 //   this issue (hence the NotDotNet category).
60
61                 [Test]
62                 [Category ("NotDotNet")]
63 #if TARGET_JVM
64                 [Ignore ("TD #6954")]
65 #endif
66                 public void HtmlEncode_XSS ()
67                 {
68                         string problem = "\xff1cscript\xff1e";  // unicode looks alike <script>
69                         byte[] utf8data = Encoding.UTF8.GetBytes (problem);
70                         Encoding win1251 = Encoding.GetEncoding ("windows-1251");
71                         byte[] windata = Encoding.Convert (Encoding.UTF8, win1251, utf8data);
72                         // now it's a real problem
73                         Assert.AreEqual ("<script>", Encoding.ASCII.GetString (windata), "<script>");
74
75                         string encoded = WebUtility.HtmlEncode (problem);
76                         Assert.AreEqual ("&#65308;script&#65310;", encoded, "&#65308;script&#65310;");
77                         
78                         utf8data = Encoding.UTF8.GetBytes (encoded);
79                         windata = Encoding.Convert (Encoding.UTF8, win1251, utf8data);
80                         Assert.AreEqual ("&#65308;script&#65310;", Encoding.ASCII.GetString (windata), "ok");
81                 }
82                 
83                 string HtmlEncode (string s) {
84                         if (s == null)
85                                 return null;
86
87                         bool needEncode = false;
88                         for (int i = 0; i < s.Length; i++) {
89                                 char c = s [i];
90                                 if (c == '&' || c == '"' || c == '<' || c == '>' || c > 159) {
91                                         needEncode = true;
92                                         break;
93                                 }
94                         }
95
96                         if (!needEncode)
97                                 return s;
98
99                         StringBuilder output = new StringBuilder ();
100
101                         int len = s.Length;
102                         for (int i = 0; i < len; i++)
103                                 switch (s [i]) {
104                                 case '&':
105                                         output.Append ("&amp;");
106                                         break;
107                                 case '>':
108                                         output.Append ("&gt;");
109                                         break;
110                                 case '<':
111                                         output.Append ("&lt;");
112                                         break;
113                                 case '"':
114                                         output.Append ("&quot;");
115                                         break;
116                                 default:
117                                         // MS starts encoding with &# from 160 and stops at 255.
118                                         // We don't do that. One reason is the 65308/65310 unicode
119                                         // characters that look like '<' and '>'.
120                                         if (s [i] > 159 && s [i] < 256) {
121                                                 output.Append ("&#");
122                                                 output.Append (((int) s [i]).ToString ());
123                                                 output.Append (";");
124                                         }
125                                         else {
126                                                 output.Append (s [i]);
127                                         }
128                                         break;
129                                 }
130                         return output.ToString ();
131                 }
132
133                 
134                 [Test]
135                 public void EncodeEscapedCharacters ()
136                 {
137                         for (int i = 0; i < 256; i++) {
138                                 string str = new string ((char) i, 1);
139                                 string encoded = WebUtility.HtmlEncode (str);
140                                 if ((i > 159 && i < 256 ) || i == '&' || i == '<' || i == '>' || i == '"' || i == '\'') {
141                                         if (encoded [0] != '&' || encoded [encoded.Length - 1] != ';')
142                                                 Assert.Fail ("Failed for i = " + i);
143                                 } else if (encoded.Length != 1) {
144                                         Assert.Fail ("Wrong length for i = " + i);
145                                 }
146                         }
147                 }
148
149                 [Test]
150                 public void Decode ()
151                 {
152                         Assert.AreEqual ("\xE9", WebUtility.HtmlDecode ("&#233;"), "#1");
153                         Assert.IsNull (WebUtility.HtmlDecode (null), "#2");
154                         Assert.AreEqual ("", WebUtility.HtmlDecode (""), "#3");
155                         Assert.AreEqual ("U", WebUtility.HtmlDecode ("&#x55;"), "#4");
156                         Assert.AreEqual ("&#x55G;", WebUtility.HtmlDecode ("&#x55G;"), "#5");
157                         Assert.AreEqual (char.ConvertFromUtf32 (144308), WebUtility.HtmlDecode ("&#144308;"), "#6");
158                 }
159
160                 [Test]
161                 public void RoundTrip ()
162                 {
163                         string x = "<html>& hello+= world!";
164                         string y = WebUtility.HtmlEncode (x);
165                         string z = WebUtility.HtmlDecode (y);
166                         Assert.AreEqual (x, z);
167                 }
168
169                 [Test]
170                 public void LooksLikeEntity ()
171                 {
172                         string str = "<%# \"hola\" + \"/somepage.aspx?ItemID=\" + DataBinder.Eval(Container.DataItem,\"Country\")" +
173                                         " + \"&mid=\" + ModuleID + \"&pageindex=\" + Request.Params.Get(\"pageindex\") %>";
174                         Assert.AreEqual (str, WebUtility.HtmlDecode (str));
175                 }
176
177                 [Test]
178                 public void EntityEncoding ()
179                 {
180                         var expected = "\u00A0\u00A1\u00A2\u00A3\u00A4\u00A5\u00A6\u00A7\u00A8\u00A9\u00AA\u00AB\u00AC\u00AD\u00AE\u00AF\u00B0\u00B1\u00B2\u00B3\u00B4\u00B5\u00B6\u00B7\u00B8\u00B9\u00BA\u00BB\u00BC\u00BD\u00BE\u00BF\u00C0\u00C1\u00C2\u00C3\u00C4\u00C5\u00C6\u00C7\u00C8\u00C9\u00CA\u00CB\u00CC\u00CD\u00CE\u00CF\u00D0\u00D1\u00D2\u00D3\u00D4\u00D5\u00D6\u00D7\u00D8\u00D9\u00DA\u00DB\u00DC\u00DD\u00DE\u00DF\u00E0\u00E1\u00E2\u00E3\u00E4\u00E5\u00E6\u00E7\u00E8\u00E9\u00EA\u00EB\u00EC\u00ED\u00EE\u00EF\u00F0\u00F1\u00F2\u00F3\u00F4\u00F5\u00F6\u00F7\u00F8\u00F9\u00FA\u00FB\u00FC\u00FD\u00FE\u00FF\u0192\u0391\u0392\u0393\u0394\u0395\u0396\u0397\u0398\u0399\u039A\u039B\u039C\u039D\u039E\u039F\u03A0\u03A1\u03A3\u03A4\u03A5\u03A6\u03A7\u03A8\u03A9\u03B1\u03B2\u03B3\u03B4\u03B5\u03B6\u03B7\u03B8\u03B9\u03BA\u03BB\u03BC\u03BD\u03BE\u03BF\u03C0\u03C1\u03C2\u03C3\u03C4\u03C5\u03C6\u03C7\u03C8\u03C9\u03D1\u03D2\u03D6\u2022\u2026\u2032\u2033\u203E\u2044\u2118\u2111\u211C\u2122\u2135\u2190\u2191\u2192\u2193\u2194\u21B5\u21D0\u21D1\u21D2\u21D3\u21D4\u2200\u2202\u2203\u2205\u2207\u2208\u2209\u220B\u220F\u2211\u2212\u2217\u221A\u221D\u221E\u2220\u2227\u2228\u2229\u222A\u222B\u2234\u223C\u2245\u2248\u2260\u2261\u2264\u2265\u2282\u2283\u2284\u2286\u2287\u2295\u2297\u22A5\u22C5\u2308\u2309\u230A\u230B\u2329\u232A\u25CA\u2660\u2663\u2665\u2666\u0022\u0026\u003C\u003E\u0152\u0153\u0160\u0161\u0178\u02C6\u02DC\u2002\u2003\u2009\u200C\u200D\u200E\u200F\u2013\u2014\u2018\u2019\u201A\u201C\u201D\u201E\u2020\u2021\u2030\u2039\u203A\u20AC\'";
181
182                         var htmlDecoded = WebUtility.HtmlDecode ("&nbsp;&iexcl;&cent;&pound;&curren;&yen;&brvbar;&sect;&uml;&copy;&ordf;&laquo;&not;&shy;&reg;&macr;&deg;&plusmn;&sup2;&sup3;&acute;&micro;&para;&middot;&cedil;&sup1;&ordm;&raquo;&frac14;&frac12;&frac34;&iquest;&Agrave;&Aacute;&Acirc;&Atilde;&Auml;&Aring;&AElig;&Ccedil;&Egrave;&Eacute;&Ecirc;&Euml;&Igrave;&Iacute;&Icirc;&Iuml;&ETH;&Ntilde;&Ograve;&Oacute;&Ocirc;&Otilde;&Ouml;&times;&Oslash;&Ugrave;&Uacute;&Ucirc;&Uuml;&Yacute;&THORN;&szlig;&agrave;&aacute;&acirc;&atilde;&auml;&aring;&aelig;&ccedil;&egrave;&eacute;&ecirc;&euml;&igrave;&iacute;&icirc;&iuml;&eth;&ntilde;&ograve;&oacute;&ocirc;&otilde;&ouml;&divide;&oslash;&ugrave;&uacute;&ucirc;&uuml;&yacute;&thorn;&yuml;&fnof;&Alpha;&Beta;&Gamma;&Delta;&Epsilon;&Zeta;&Eta;&Theta;&Iota;&Kappa;&Lambda;&Mu;&Nu;&Xi;&Omicron;&Pi;&Rho;&Sigma;&Tau;&Upsilon;&Phi;&Chi;&Psi;&Omega;&alpha;&beta;&gamma;&delta;&epsilon;&zeta;&eta;&theta;&iota;&kappa;&lambda;&mu;&nu;&xi;&omicron;&pi;&rho;&sigmaf;&sigma;&tau;&upsilon;&phi;&chi;&psi;&omega;&thetasym;&upsih;&piv;&bull;&hellip;&prime;&Prime;&oline;&frasl;&weierp;&image;&real;&trade;&alefsym;&larr;&uarr;&rarr;&darr;&harr;&crarr;&lArr;&uArr;&rArr;&dArr;&hArr;&forall;&part;&exist;&empty;&nabla;&isin;&notin;&ni;&prod;&sum;&minus;&lowast;&radic;&prop;&infin;&ang;&and;&or;&cap;&cup;&int;&there4;&sim;&cong;&asymp;&ne;&equiv;&le;&ge;&sub;&sup;&nsub;&sube;&supe;&oplus;&otimes;&perp;&sdot;&lceil;&rceil;&lfloor;&rfloor;&lang;&rang;&loz;&spades;&clubs;&hearts;&diams;&quot;&amp;&lt;&gt;&OElig;&oelig;&Scaron;&scaron;&Yuml;&circ;&tilde;&ensp;&emsp;&thinsp;&zwnj;&zwj;&lrm;&rlm;&ndash;&mdash;&lsquo;&rsquo;&sbquo;&ldquo;&rdquo;&bdquo;&dagger;&Dagger;&permil;&lsaquo;&rsaquo;&euro;&apos;");
183                         
184                         Assert.AreEqual (expected, htmlDecoded);
185                 }
186
187                 [Test]
188                 public void UrlEncoding ()
189                 {
190                         Assert.AreEqual ("%27", WebUtility.UrlEncode("'"), "#1");
191                 }
192         }
193 }
194 #endif
195