package goldmark_test import ( "bytes" "os" "strconv" "strings" "testing" "time" . "github.com/yuin/goldmark" "github.com/yuin/goldmark/ast" "github.com/yuin/goldmark/parser" "github.com/yuin/goldmark/renderer/html" "github.com/yuin/goldmark/testutil" ) var testTimeoutMultiplier = 1.0 func init() { m, err := strconv.ParseFloat(os.Getenv("GOLDMARK_TEST_TIMEOUT_MULTIPLIER"), 64) if err == nil { testTimeoutMultiplier = m } } func TestExtras(t *testing.T) { markdown := New(WithRendererOptions( html.WithXHTML(), html.WithUnsafe(), )) testutil.DoTestCaseFile(markdown, "_test/extra.txt", t, testutil.ParseCliCaseArg()...) } func TestEndsWithNonSpaceCharacters(t *testing.T) { markdown := New(WithRendererOptions( html.WithXHTML(), html.WithUnsafe(), )) source := []byte("```\na\n```") var b bytes.Buffer err := markdown.Convert(source, &b) if err != nil { t.Error(err.Error()) } if b.String() != "
a\n
\n" { t.Errorf("%s \n---------\n %s", source, b.String()) } } func TestWindowsNewLine(t *testing.T) { markdown := New(WithRendererOptions( html.WithXHTML(), )) source := []byte("a \r\nb\n") var b bytes.Buffer err := markdown.Convert(source, &b) if err != nil { t.Error(err.Error()) } if b.String() != "

a
\nb

\n" { t.Errorf("%s\n---------\n%s", source, b.String()) } source = []byte("a\\\r\nb\r\n") var b2 bytes.Buffer err = markdown.Convert(source, &b2) if err != nil { t.Error(err.Error()) } if b2.String() != "

a
\nb

\n" { t.Errorf("\n%s\n---------\n%s", source, b2.String()) } } type myIDs struct { } func (s *myIDs) Generate(value []byte, kind ast.NodeKind) []byte { return []byte("my-id") } func (s *myIDs) Put(value []byte) { } func TestAutogeneratedIDs(t *testing.T) { ctx := parser.NewContext(parser.WithIDs(&myIDs{})) markdown := New(WithParserOptions(parser.WithAutoHeadingID())) source := []byte("# Title1\n## Title2") var b bytes.Buffer err := markdown.Convert(source, &b, parser.WithContext(ctx)) if err != nil { t.Error(err.Error()) } if b.String() != `

Title1

Title2

` { t.Errorf("%s\n---------\n%s", source, b.String()) } } func nowMillis() int64 { // TODO: replace UnixNano to UnixMillis(drops Go1.16 support) return time.Now().UnixNano() / 1000000 } func TestDeepNestedLabelPerformance(t *testing.T) { if testing.Short() { t.Skip("skipping performance test in short mode") } markdown := New(WithRendererOptions( html.WithXHTML(), html.WithUnsafe(), )) started := nowMillis() n := 50000 source := []byte(strings.Repeat("[", n) + strings.Repeat("]", n)) var b bytes.Buffer _ = markdown.Convert(source, &b) finished := nowMillis() if (finished - started) > int64(5000*testTimeoutMultiplier) { t.Error("Parsing deep nested labels took too long") } } func TestManyProcessingInstructionPerformance(t *testing.T) { if testing.Short() { t.Skip("skipping performance test in short mode") } markdown := New(WithRendererOptions( html.WithXHTML(), html.WithUnsafe(), )) started := nowMillis() n := 50000 source := []byte("a " + strings.Repeat(" int64(5000*testTimeoutMultiplier) { t.Error("Parsing processing instructions took too long") } } func TestManyCDATAPerformance(t *testing.T) { if testing.Short() { t.Skip("skipping performance test in short mode") } markdown := New(WithRendererOptions( html.WithXHTML(), html.WithUnsafe(), )) started := nowMillis() n := 50000 source := []byte(strings.Repeat("a int64(5000*testTimeoutMultiplier) { t.Error("Parsing processing instructions took too long") } } func TestManyDeclPerformance(t *testing.T) { if testing.Short() { t.Skip("skipping performance test in short mode") } markdown := New(WithRendererOptions( html.WithXHTML(), html.WithUnsafe(), )) started := nowMillis() n := 50000 source := []byte(strings.Repeat("a int64(5000*testTimeoutMultiplier) { t.Error("Parsing processing instructions took too long") } } func TestManyCommentPerformance(t *testing.T) { if testing.Short() { t.Skip("skipping performance test in short mode") } markdown := New(WithRendererOptions( html.WithXHTML(), html.WithUnsafe(), )) started := nowMillis() n := 50000 source := []byte(strings.Repeat("a