378 lines
132 KiB
Go
378 lines
132 KiB
Go
|
// Copyright (c) 2014, David Kitchen <david@buro9.com>
|
||
|
//
|
||
|
// All rights reserved.
|
||
|
//
|
||
|
// Redistribution and use in source and binary forms, with or without
|
||
|
// modification, are permitted provided that the following conditions are met:
|
||
|
//
|
||
|
// * Redistributions of source code must retain the above copyright notice, this
|
||
|
// list of conditions and the following disclaimer.
|
||
|
//
|
||
|
// * Redistributions in binary form must reproduce the above copyright notice,
|
||
|
// this list of conditions and the following disclaimer in the documentation
|
||
|
// and/or other materials provided with the distribution.
|
||
|
//
|
||
|
// * Neither the name of the organisation (Microcosm) nor the names of its
|
||
|
// contributors may be used to endorse or promote products derived from
|
||
|
// this software without specific prior written permission.
|
||
|
//
|
||
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||
|
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||
|
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||
|
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||
|
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||
|
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||
|
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||
|
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||
|
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
|
||
|
package bluemonday
|
||
|
|
||
|
import (
|
||
|
"sync"
|
||
|
"testing"
|
||
|
)
|
||
|
|
||
|
func TestRegexpVars(t *testing.T) {
|
||
|
// CellAlign
|
||
|
if !CellAlign.MatchString("CENTER") {
|
||
|
t.Error("CellAlign did not match: CENTER")
|
||
|
}
|
||
|
if !CellAlign.MatchString("justIFY") {
|
||
|
t.Error("CellAlign did not match: justIFY")
|
||
|
}
|
||
|
if !CellAlign.MatchString("left") {
|
||
|
t.Error("CellAlign did not match: left")
|
||
|
}
|
||
|
if !CellAlign.MatchString("right") {
|
||
|
t.Error("CellAlign did not match: right")
|
||
|
}
|
||
|
if !CellAlign.MatchString("char") {
|
||
|
t.Error("CellAlign did not match: char")
|
||
|
}
|
||
|
if CellAlign.MatchString("char char") {
|
||
|
t.Error("CellAlign matched: char char")
|
||
|
}
|
||
|
|
||
|
// CellVerticalAlign
|
||
|
if !CellVerticalAlign.MatchString("BASELINE") {
|
||
|
t.Error("CellVerticalAlign did not match: BASELINE")
|
||
|
}
|
||
|
if !CellVerticalAlign.MatchString("boTtOM") {
|
||
|
t.Error("CellVerticalAlign did not match: boTtOM")
|
||
|
}
|
||
|
if !CellVerticalAlign.MatchString("middle") {
|
||
|
t.Error("CellVerticalAlign did not match: middle")
|
||
|
}
|
||
|
if !CellVerticalAlign.MatchString("top") {
|
||
|
t.Error("CellVerticalAlign did not match: top")
|
||
|
}
|
||
|
if CellVerticalAlign.MatchString("top top") {
|
||
|
t.Error("CellVerticalAlign matched: top top")
|
||
|
}
|
||
|
|
||
|
// Direction
|
||
|
if !Direction.MatchString("RTL") {
|
||
|
t.Error("Direction did not match: RTL")
|
||
|
}
|
||
|
if !Direction.MatchString("ltr") {
|
||
|
t.Error("Direction did not match: ltr")
|
||
|
}
|
||
|
if Direction.MatchString("") {
|
||
|
t.Error("Direction matched: ")
|
||
|
}
|
||
|
if Direction.MatchString("rtltr") {
|
||
|
t.Error("Direction matched: rtltr")
|
||
|
}
|
||
|
if Direction.MatchString("rtl rtl") {
|
||
|
t.Error("Direction matched: rtl rtl")
|
||
|
}
|
||
|
|
||
|
// ImageAlign
|
||
|
if !ImageAlign.MatchString("LEFT") {
|
||
|
t.Error("CellAlign did not match: LEFT")
|
||
|
}
|
||
|
if !ImageAlign.MatchString("right") {
|
||
|
t.Error("CellAlign did not match: right")
|
||
|
}
|
||
|
if !ImageAlign.MatchString("tOP") {
|
||
|
t.Error("CellAlign did not match: tOP")
|
||
|
}
|
||
|
if !ImageAlign.MatchString("texttop") {
|
||
|
t.Error("CellAlign did not match: texttop")
|
||
|
}
|
||
|
if !ImageAlign.MatchString("middle") {
|
||
|
t.Error("CellAlign did not match: middle")
|
||
|
}
|
||
|
if !ImageAlign.MatchString("absmiddle") {
|
||
|
t.Error("CellAlign did not match: absmiddle")
|
||
|
}
|
||
|
if !ImageAlign.MatchString("baseline") {
|
||
|
t.Error("CellAlign did not match: baseline")
|
||
|
}
|
||
|
if !ImageAlign.MatchString("bottom") {
|
||
|
t.Error("CellAlign did not match: bottom")
|
||
|
}
|
||
|
if !ImageAlign.MatchString("absbottom") {
|
||
|
t.Error("CellAlign did not match: absbottom")
|
||
|
}
|
||
|
if ImageAlign.MatchString("left right") {
|
||
|
t.Error("CellAlign matched: left right")
|
||
|
}
|
||
|
if ImageAlign.MatchString("left left") {
|
||
|
t.Error("CellAlign matched: left left")
|
||
|
}
|
||
|
if ImageAlign.MatchString("char") {
|
||
|
t.Error("CellAlign matched: char")
|
||
|
}
|
||
|
if ImageAlign.MatchString("char") {
|
||
|
t.Error("CellAlign matched: char")
|
||
|
}
|
||
|
|
||
|
// Integer
|
||
|
if !Integer.MatchString("123") {
|
||
|
t.Error("Integer did not match: 123")
|
||
|
}
|
||
|
if !Integer.MatchString("0") {
|
||
|
t.Error("Integer did not match: 0")
|
||
|
}
|
||
|
if Integer.MatchString("-1") {
|
||
|
t.Error("Integer matched: -1")
|
||
|
}
|
||
|
if Integer.MatchString("1abc") {
|
||
|
t.Error("Integer matched: 1abc")
|
||
|
}
|
||
|
|
||
|
// ISO8601
|
||
|
if !ISO8601.MatchString("2014") {
|
||
|
t.Error("ISO8601 did not match: 2014")
|
||
|
}
|
||
|
if !ISO8601.MatchString("2014-02") {
|
||
|
t.Error("ISO8601 did not match: 2014-02")
|
||
|
}
|
||
|
if !ISO8601.MatchString("2014-02-28") {
|
||
|
t.Error("ISO8601 did not match: 2014-02-28")
|
||
|
}
|
||
|
if !ISO8601.MatchString("2014-02-28T23:59") {
|
||
|
t.Error("ISO8601 did not match: 2014-02-28T23:59")
|
||
|
}
|
||
|
if !ISO8601.MatchString("2014-02-28T23:59-05:00") {
|
||
|
t.Error("ISO8601 did not match: 2014-02-28T23:59-05:00")
|
||
|
}
|
||
|
if !ISO8601.MatchString("2014-02-28T23:59-05:00") {
|
||
|
t.Error("ISO8601 did not match: 2014-02-28T23:59+01:00")
|
||
|
}
|
||
|
if !ISO8601.MatchString("2014-02-28T23:59:59") {
|
||
|
t.Error("ISO8601 did not match: 2014-02-28T23:59:59")
|
||
|
}
|
||
|
if !ISO8601.MatchString("2014-02-28T23:59:59-05:00") {
|
||
|
t.Error("ISO8601 did not match: 2014-02-28T23:59:59-05:00")
|
||
|
}
|
||
|
if !ISO8601.MatchString("2014-02-28T23:59:59-05:00") {
|
||
|
t.Error("ISO8601 did not match: 2014-02-28T23:59:59+01:00")
|
||
|
}
|
||
|
if ISO8601.MatchString("201-") {
|
||
|
t.Error("ISO8601 matched: 201-")
|
||
|
}
|
||
|
if ISO8601.MatchString("2014-0") {
|
||
|
t.Error("ISO8601 matched: 2014-0")
|
||
|
}
|
||
|
if ISO8601.MatchString("2014-02-28-") {
|
||
|
t.Error("ISO8601 matched: 2014-02-28-")
|
||
|
}
|
||
|
if ISO8601.MatchString("20-02-28T23:59-05:00") {
|
||
|
t.Error("ISO8601 matched: 20-02-28T23:59+01:00")
|
||
|
}
|
||
|
if ISO8601.MatchString(" 2014-02-28T23:59:59") {
|
||
|
t.Error("ISO8601 matched: 2014-02-28T23:59:59")
|
||
|
}
|
||
|
|
||
|
// ListType
|
||
|
if !ListType.MatchString("CIRCLE") {
|
||
|
t.Error("ListType did not match: CIRCLE")
|
||
|
}
|
||
|
if !ListType.MatchString("disc") {
|
||
|
t.Error("ListType did not match: disc")
|
||
|
}
|
||
|
if !ListType.MatchString("square") {
|
||
|
t.Error("ListType did not match: square")
|
||
|
}
|
||
|
if !ListType.MatchString("a") {
|
||
|
t.Error("ListType did not match: a")
|
||
|
}
|
||
|
if !ListType.MatchString("A") {
|
||
|
t.Error("ListType did not match: A")
|
||
|
}
|
||
|
if !ListType.MatchString("i") {
|
||
|
t.Error("ListType did not match: i")
|
||
|
}
|
||
|
if !ListType.MatchString("I") {
|
||
|
t.Error("ListType did not match: I")
|
||
|
}
|
||
|
if !ListType.MatchString("1") {
|
||
|
t.Error("ListType did not match: 1")
|
||
|
}
|
||
|
if ListType.MatchString("circle circle") {
|
||
|
t.Error("ListType matched: circle circle")
|
||
|
}
|
||
|
if ListType.MatchString("aa") {
|
||
|
t.Error("ListType matched: aa")
|
||
|
}
|
||
|
|
||
|
// SpaceSeparatedTokens
|
||
|
if !SpaceSeparatedTokens.MatchString("nofollow") {
|
||
|
t.Error("SpaceSeparatedTokens did not match: nofollow")
|
||
|
}
|
||
|
if !SpaceSeparatedTokens.MatchString("nofollow person") {
|
||
|
t.Error("SpaceSeparatedTokens did not match: nofollow person")
|
||
|
}
|
||
|
if !SpaceSeparatedTokens.MatchString("header") {
|
||
|
t.Error("SpaceSeparatedTokens did not match: header")
|
||
|
}
|
||
|
if !SpaceSeparatedTokens.MatchString("bläh") {
|
||
|
t.Error("SpaceSeparatedTokens did not match: bläh")
|
||
|
}
|
||
|
if !SpaceSeparatedTokens.MatchString("blah bläh") {
|
||
|
t.Error("SpaceSeparatedTokens did not match: blah bläh")
|
||
|
}
|
||
|
if SpaceSeparatedTokens.MatchString("bläh blah ☃") {
|
||
|
t.Error("SpaceSeparatedTokens matched: bläh blah ☃")
|
||
|
}
|
||
|
if SpaceSeparatedTokens.MatchString("header javascript:alert(1)") {
|
||
|
t.Error("SpaceSeparatedTokens matched: header javascript:alert(1)")
|
||
|
}
|
||
|
if SpaceSeparatedTokens.MatchString("header >") {
|
||
|
t.Error("SpaceSeparatedTokens matched: header >")
|
||
|
}
|
||
|
|
||
|
// Number
|
||
|
if !Number.MatchString("0") {
|
||
|
t.Error("Number did not match: 0")
|
||
|
}
|
||
|
if !Number.MatchString("1") {
|
||
|
t.Error("Number did not match: 1")
|
||
|
}
|
||
|
if !Number.MatchString("+1") {
|
||
|
t.Error("Number did not match: +1")
|
||
|
}
|
||
|
if !Number.MatchString("-1") {
|
||
|
t.Error("Number did not match: -1")
|
||
|
}
|
||
|
if !Number.MatchString("1.1") {
|
||
|
t.Error("Number did not match: 1.1")
|
||
|
}
|
||
|
if !Number.MatchString("1.2e3") {
|
||
|
t.Error("Number did not match: 1.2e3")
|
||
|
}
|
||
|
if !Number.MatchString("7E-10") {
|
||
|
t.Error("Number did not match: 7E-10")
|
||
|
}
|
||
|
if Number.MatchString("e7.13") {
|
||
|
t.Error("Number matched: e7.13")
|
||
|
}
|
||
|
if Number.MatchString(`7E`) {
|
||
|
t.Error(`Number matched: 7E`)
|
||
|
}
|
||
|
|
||
|
// NumberOrPercent
|
||
|
if !NumberOrPercent.MatchString("0") {
|
||
|
t.Error("NumberOrPercent did not match: 0")
|
||
|
}
|
||
|
if !NumberOrPercent.MatchString("1") {
|
||
|
t.Error("NumberOrPercent did not match: 1")
|
||
|
}
|
||
|
if !NumberOrPercent.MatchString(`1%`) {
|
||
|
t.Error(`NumberOrPercent did not match: 1(percent)`)
|
||
|
}
|
||
|
if NumberOrPercent.MatchString("-1") {
|
||
|
t.Error("NumberOrPercent matched: -1")
|
||
|
}
|
||
|
if NumberOrPercent.MatchString("1.1") {
|
||
|
t.Error("NumberOrPercent matched: 1.1")
|
||
|
}
|
||
|
if NumberOrPercent.MatchString("1.2e3") {
|
||
|
t.Error("NumberOrPercent matched: 1.2e3")
|
||
|
}
|
||
|
if NumberOrPercent.MatchString("7E-10") {
|
||
|
t.Error("NumberOrPercent matched: 7E-10")
|
||
|
}
|
||
|
if NumberOrPercent.MatchString("e7.13") {
|
||
|
t.Error("NumberOrPercent matched: e7.13")
|
||
|
}
|
||
|
if NumberOrPercent.MatchString(`7E`) {
|
||
|
t.Error(`NumberOrPercent matched: 7E`)
|
||
|
}
|
||
|
|
||
|
// Paragraph
|
||
|
if !Paragraph.MatchString("hello world") {
|
||
|
t.Error("Paragraph did not match: hello world")
|
||
|
}
|
||
|
if !Paragraph.MatchString("blah bläh blah") {
|
||
|
t.Error("Paragraph did not match: blah bläh blah")
|
||
|
}
|
||
|
if Paragraph.MatchString("bläh blah ☃") {
|
||
|
t.Error("Paragraph matched: bläh blah ☃")
|
||
|
}
|
||
|
if Paragraph.MatchString("javascript:alert(1)") {
|
||
|
t.Error("Paragraph matched: javascript:alert(1)")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestAllowDataURIImages(t *testing.T) {
|
||
|
|
||
|
p := NewPolicy()
|
||
|
p.AllowDataURIImages()
|
||
|
p.AllowImages()
|
||
|
|
||
|
tests := []test{
|
||
|
{
|
||
|
in: `<img src="">`,
|
||
|
expected: `<img src="">`,
|
||
|
},
|
||
|
{
|
||
|
in: `<img src="
|
||
|
expected: `<img src="
|
||
|
},
|
||
|
{
|
||
|
in: `<img src="">`,
|
||
|
expected: `<img src="">`,
|
||
|
},
|
||
|
{
|
||
|
in: `<img src="">`,
|
||
|
expected: `<img src="">`,
|
||
|
},
|
||
|
{
|
||
|
in: `<img src="data:text/javascript;charset=utf-8,alert('hi');">`,
|
||
|
expected: ``,
|
||
|
},
|
||
|
{
|
||
|
in: `<img src="-8,alert('hi');">`,
|
||
|
expected: ``,
|
||
|
},
|
||
|
{
|
||
|
in: `<img src="-_8/w38GIAXDIBKE0DHxgljNBAAO9TXL0Y4OHwAAAABJRU5ErkJggg==">`,
|
||
|
expected: ``,
|
||
|
},
|
||
|
}
|
||
|
|
||
|
wg := sync.WaitGroup{}
|
||
|
wg.Add(len(tests))
|
||
|
for ii, tt := range tests {
|
||
|
go func(ii int, tt test) {
|
||
|
out := p.Sanitize(tt.in)
|
||
|
if out != tt.expected {
|
||
|
t.Errorf(
|
||
|
"test %d failed;\ninput : %s\noutput : %s\nexpected: %s",
|
||
|
ii,
|
||
|
tt.in,
|
||
|
out,
|
||
|
tt.expected,
|
||
|
)
|
||
|
}
|
||
|
wg.Done()
|
||
|
}(ii, tt)
|
||
|
}
|
||
|
wg.Wait()
|
||
|
}
|