2019-10-13 19:29:10 -03:00
// Copyright 2019 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package references
import (
2020-11-09 22:57:47 +00:00
"regexp"
2019-10-13 19:29:10 -03:00
"testing"
"code.gitea.io/gitea/modules/setting"
"github.com/stretchr/testify/assert"
)
2019-10-30 09:43:59 -03:00
type testFixture struct {
input string
expected [ ] testResult
}
2019-10-13 19:29:10 -03:00
2019-10-30 09:43:59 -03:00
type testResult struct {
Index int64
Owner string
Name string
Issue string
2019-12-01 10:57:05 -03:00
IsPull bool
2019-10-30 09:43:59 -03:00
Action XRefAction
RefLocation * RefSpan
ActionLocation * RefSpan
2020-09-04 18:37:37 +03:00
TimeLog string
2019-10-30 09:43:59 -03:00
}
2019-10-13 19:29:10 -03:00
2020-11-09 22:57:47 +00:00
func TestConvertFullHTMLReferencesToShortRefs ( t * testing . T ) {
re := regexp . MustCompile ( ` (\s|^|\(|\[) ` +
regexp . QuoteMeta ( "https://ourgitea.com/git/" ) +
` ([0-9a-zA-Z-_\.]+/[0-9a-zA-Z-_\.]+)/ ` +
` ((?:issues)|(?:pulls))/([0-9]+)(?:\s|$|\)|\]|[:;,.?!]\s|[:;,.?!]$) ` )
test := ` this is a https : //ourgitea.com/git/owner/repo/issues/123456789, foo
https : //ourgitea.com/git/owner/repo/pulls/123456789
And https : //ourgitea.com/git/owner/repo/pulls/123
`
expect := ` this is a owner / repo # 123456789 , foo
owner / repo ! 123456789
And owner / repo ! 123
`
contentBytes := [ ] byte ( test )
convertFullHTMLReferencesToShortRefs ( re , & contentBytes )
result := string ( contentBytes )
assert . EqualValues ( t , expect , result )
}
2019-10-30 09:43:59 -03:00
func TestFindAllIssueReferences ( t * testing . T ) {
2019-10-13 19:29:10 -03:00
fixtures := [ ] testFixture {
{
"Simply closes: #29 yes" ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 29 , "" , "" , "29" , false , XRefActionCloses , & RefSpan { Start : 15 , End : 18 } , & RefSpan { Start : 7 , End : 13 } , "" } ,
2019-12-01 10:57:05 -03:00
} ,
} ,
{
"Simply closes: !29 yes" ,
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 29 , "" , "" , "29" , true , XRefActionCloses , & RefSpan { Start : 15 , End : 18 } , & RefSpan { Start : 7 , End : 13 } , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
" #124 yes, this is a reference." ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 124 , "" , "" , "124" , false , XRefActionNone , & RefSpan { Start : 0 , End : 4 } , nil , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
"```\nThis is a code block.\n#723 no, it's a code block.```" ,
2019-10-30 09:43:59 -03:00
[ ] testResult { } ,
2019-10-13 19:29:10 -03:00
} ,
{
"This `#724` no, it's inline code." ,
2019-10-30 09:43:59 -03:00
[ ] testResult { } ,
2019-10-13 19:29:10 -03:00
} ,
{
"This user3/repo4#200 yes." ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 200 , "user3" , "repo4" , "200" , false , XRefActionNone , & RefSpan { Start : 5 , End : 20 } , nil , "" } ,
2019-12-01 10:57:05 -03:00
} ,
} ,
{
"This user3/repo4!200 yes." ,
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 200 , "user3" , "repo4" , "200" , true , XRefActionNone , & RefSpan { Start : 5 , End : 20 } , nil , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
"This [one](#919) no, this is a URL fragment." ,
2019-10-30 09:43:59 -03:00
[ ] testResult { } ,
2019-10-13 19:29:10 -03:00
} ,
{
"This [two](/user2/repo1/issues/921) yes." ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 921 , "user2" , "repo1" , "921" , false , XRefActionNone , nil , nil , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
"This [three](/user2/repo1/pulls/922) yes." ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 922 , "user2" , "repo1" , "922" , true , XRefActionNone , nil , nil , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
"This [four](http://gitea.com:3000/user3/repo4/issues/203) yes." ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 203 , "user3" , "repo4" , "203" , false , XRefActionNone , nil , nil , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
"This [five](http://github.com/user3/repo4/issues/204) no." ,
2019-10-30 09:43:59 -03:00
[ ] testResult { } ,
2019-10-13 19:29:10 -03:00
} ,
{
"This http://gitea.com:3000/user4/repo5/201 no, bad URL." ,
2019-10-30 09:43:59 -03:00
[ ] testResult { } ,
2019-10-13 19:29:10 -03:00
} ,
{
"This http://gitea.com:3000/user4/repo5/pulls/202 yes." ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 202 , "user4" , "repo5" , "202" , true , XRefActionNone , nil , nil , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
2020-11-09 22:57:47 +00:00
{
"This http://gitea.com:3000/user4/repo5/pulls/202 yes. http://gitea.com:3000/user4/repo5/pulls/203 no" ,
[ ] testResult {
{ 202 , "user4" , "repo5" , "202" , true , XRefActionNone , nil , nil , "" } ,
{ 203 , "user4" , "repo5" , "203" , true , XRefActionNone , nil , nil , "" } ,
} ,
} ,
2019-10-13 19:29:10 -03:00
{
"This http://GiTeA.COM:3000/user4/repo6/pulls/205 yes." ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 205 , "user4" , "repo6" , "205" , true , XRefActionNone , nil , nil , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
"Reopens #15 yes" ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 15 , "" , "" , "15" , false , XRefActionReopens , & RefSpan { Start : 8 , End : 11 } , & RefSpan { Start : 0 , End : 7 } , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
"This closes #20 for you yes" ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 20 , "" , "" , "20" , false , XRefActionCloses , & RefSpan { Start : 12 , End : 15 } , & RefSpan { Start : 5 , End : 11 } , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
"Do you fix user6/repo6#300 ? yes" ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 300 , "user6" , "repo6" , "300" , false , XRefActionCloses , & RefSpan { Start : 11 , End : 26 } , & RefSpan { Start : 7 , End : 10 } , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
"For 999 #1235 no keyword, but yes" ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 1235 , "" , "" , "1235" , false , XRefActionNone , & RefSpan { Start : 8 , End : 13 } , nil , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
2020-02-01 15:01:30 -03:00
{
"For [!123] yes" ,
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 123 , "" , "" , "123" , true , XRefActionNone , & RefSpan { Start : 5 , End : 9 } , nil , "" } ,
2020-02-01 15:01:30 -03:00
} ,
} ,
{
"For (#345) yes" ,
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 345 , "" , "" , "345" , false , XRefActionNone , & RefSpan { Start : 5 , End : 9 } , nil , "" } ,
2020-02-01 15:01:30 -03:00
} ,
} ,
{
"For #22,#23 no, neither #28:#29 or !30!31#32;33 should" ,
[ ] testResult { } ,
} ,
{
"For #24, and #25. yes; also #26; #27? #28! and #29: should" ,
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 24 , "" , "" , "24" , false , XRefActionNone , & RefSpan { Start : 4 , End : 7 } , nil , "" } ,
{ 25 , "" , "" , "25" , false , XRefActionNone , & RefSpan { Start : 13 , End : 16 } , nil , "" } ,
{ 26 , "" , "" , "26" , false , XRefActionNone , & RefSpan { Start : 28 , End : 31 } , nil , "" } ,
{ 27 , "" , "" , "27" , false , XRefActionNone , & RefSpan { Start : 33 , End : 36 } , nil , "" } ,
{ 28 , "" , "" , "28" , false , XRefActionNone , & RefSpan { Start : 38 , End : 41 } , nil , "" } ,
{ 29 , "" , "" , "29" , false , XRefActionNone , & RefSpan { Start : 47 , End : 50 } , nil , "" } ,
2020-02-01 15:01:30 -03:00
} ,
} ,
{
"This user3/repo4#200, yes." ,
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 200 , "user3" , "repo4" , "200" , false , XRefActionNone , & RefSpan { Start : 5 , End : 20 } , nil , "" } ,
2020-02-01 15:01:30 -03:00
} ,
} ,
2021-05-09 20:25:23 +02:00
{
"Merge pull request '#12345 My fix for a bug' (!1337) from feature-branch into main" ,
[ ] testResult {
{ 12345 , "" , "" , "12345" , false , XRefActionNone , & RefSpan { Start : 20 , End : 26 } , nil , "" } ,
{ 1337 , "" , "" , "1337" , true , XRefActionNone , & RefSpan { Start : 46 , End : 51 } , nil , "" } ,
} ,
} ,
2019-10-13 19:29:10 -03:00
{
"Which abc. #9434 same as above" ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 9434 , "" , "" , "9434" , false , XRefActionNone , & RefSpan { Start : 11 , End : 16 } , nil , "" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
{
"This closes #600 and reopens #599" ,
2019-10-30 09:43:59 -03:00
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 600 , "" , "" , "600" , false , XRefActionCloses , & RefSpan { Start : 12 , End : 16 } , & RefSpan { Start : 5 , End : 11 } , "" } ,
{ 599 , "" , "" , "599" , false , XRefActionReopens , & RefSpan { Start : 29 , End : 33 } , & RefSpan { Start : 21 , End : 28 } , "" } ,
} ,
} ,
{
"This fixes #100 spent @40m and reopens #101, also fixes #102 spent @4h15m" ,
[ ] testResult {
{ 100 , "" , "" , "100" , false , XRefActionCloses , & RefSpan { Start : 11 , End : 15 } , & RefSpan { Start : 5 , End : 10 } , "40m" } ,
{ 101 , "" , "" , "101" , false , XRefActionReopens , & RefSpan { Start : 39 , End : 43 } , & RefSpan { Start : 31 , End : 38 } , "" } ,
{ 102 , "" , "" , "102" , false , XRefActionCloses , & RefSpan { Start : 56 , End : 60 } , & RefSpan { Start : 50 , End : 55 } , "4h15m" } ,
2019-10-13 19:29:10 -03:00
} ,
} ,
}
2019-10-30 09:43:59 -03:00
testFixtures ( t , fixtures , "default" )
2019-10-13 19:29:10 -03:00
type alnumFixture struct {
input string
issue string
refLocation * RefSpan
action XRefAction
actionLocation * RefSpan
}
alnumFixtures := [ ] alnumFixture {
{
"This ref ABC-123 is alphanumeric" ,
"ABC-123" , & RefSpan { Start : 9 , End : 16 } ,
XRefActionNone , nil ,
} ,
{
"This closes ABCD-1234 alphanumeric" ,
"ABCD-1234" , & RefSpan { Start : 12 , End : 21 } ,
XRefActionCloses , & RefSpan { Start : 5 , End : 11 } ,
} ,
}
for _ , fixture := range alnumFixtures {
found , ref := FindRenderizableReferenceAlphanumeric ( fixture . input )
if fixture . issue == "" {
assert . False ( t , found , "Failed to parse: {%s}" , fixture . input )
} else {
assert . True ( t , found , "Failed to parse: {%s}" , fixture . input )
assert . Equal ( t , fixture . issue , ref . Issue , "Failed to parse: {%s}" , fixture . input )
assert . Equal ( t , fixture . refLocation , ref . RefLocation , "Failed to parse: {%s}" , fixture . input )
assert . Equal ( t , fixture . action , ref . Action , "Failed to parse: {%s}" , fixture . input )
assert . Equal ( t , fixture . actionLocation , ref . ActionLocation , "Failed to parse: {%s}" , fixture . input )
}
}
}
2019-10-30 09:43:59 -03:00
func testFixtures ( t * testing . T , fixtures [ ] testFixture , context string ) {
// Save original value for other tests that may rely on it
prevURL := setting . AppURL
setting . AppURL = "https://gitea.com:3000/"
for _ , fixture := range fixtures {
expraw := make ( [ ] * rawReference , len ( fixture . expected ) )
for i , e := range fixture . expected {
expraw [ i ] = & rawReference {
index : e . Index ,
owner : e . Owner ,
name : e . Name ,
2019-12-01 10:57:05 -03:00
isPull : e . IsPull ,
2019-10-30 09:43:59 -03:00
action : e . Action ,
issue : e . Issue ,
refLocation : e . RefLocation ,
actionLocation : e . ActionLocation ,
2020-09-04 18:37:37 +03:00
timeLog : e . TimeLog ,
2019-10-30 09:43:59 -03:00
}
}
expref := rawToIssueReferenceList ( expraw )
refs := FindAllIssueReferencesMarkdown ( fixture . input )
assert . EqualValues ( t , expref , refs , "[%s] Failed to parse: {%s}" , context , fixture . input )
rawrefs := findAllIssueReferencesMarkdown ( fixture . input )
assert . EqualValues ( t , expraw , rawrefs , "[%s] Failed to parse: {%s}" , context , fixture . input )
}
// Restore for other tests that may rely on the original value
setting . AppURL = prevURL
}
2020-02-01 15:01:30 -03:00
func TestFindAllMentions ( t * testing . T ) {
res := FindAllMentionsBytes ( [ ] byte ( "@tasha, @mike; @lucy: @john" ) )
assert . EqualValues ( t , [ ] RefSpan {
{ Start : 0 , End : 6 } ,
{ Start : 8 , End : 13 } ,
{ Start : 15 , End : 20 } ,
{ Start : 22 , End : 27 } ,
} , res )
}
2019-10-13 19:29:10 -03:00
func TestRegExp_mentionPattern ( t * testing . T ) {
2019-11-09 19:12:05 -03:00
trueTestCases := [ ] struct {
pat string
exp string
} {
{ "@Unknwon" , "@Unknwon" } ,
{ "@ANT_123" , "@ANT_123" } ,
{ "@xxx-DiN0-z-A..uru..s-xxx" , "@xxx-DiN0-z-A..uru..s-xxx" } ,
{ " @lol " , "@lol" } ,
{ " @Te-st" , "@Te-st" } ,
{ "(@gitea)" , "@gitea" } ,
{ "[@gitea]" , "@gitea" } ,
{ "@gitea! this" , "@gitea" } ,
{ "@gitea? this" , "@gitea" } ,
{ "@gitea. this" , "@gitea" } ,
{ "@gitea, this" , "@gitea" } ,
{ "@gitea; this" , "@gitea" } ,
{ "@gitea!\nthis" , "@gitea" } ,
{ "\n@gitea?\nthis" , "@gitea" } ,
{ "\t@gitea.\nthis" , "@gitea" } ,
{ "@gitea,\nthis" , "@gitea" } ,
{ "@gitea;\nthis" , "@gitea" } ,
{ "@gitea!" , "@gitea" } ,
{ "@gitea?" , "@gitea" } ,
{ "@gitea." , "@gitea" } ,
{ "@gitea," , "@gitea" } ,
{ "@gitea;" , "@gitea" } ,
2020-12-21 23:39:28 +08:00
{ "@gitea/team1;" , "@gitea/team1" } ,
2019-10-13 19:29:10 -03:00
}
falseTestCases := [ ] string {
"@ 0" ,
"@ " ,
"@" ,
"" ,
"ABC" ,
2019-11-09 19:12:05 -03:00
"@.ABC" ,
2019-10-13 19:29:10 -03:00
"/home/gitea/@gitea" ,
"\"@gitea\"" ,
2019-11-09 19:12:05 -03:00
"@@gitea" ,
"@gitea!this" ,
"@gitea?this" ,
"@gitea,this" ,
"@gitea;this" ,
2020-12-21 23:39:28 +08:00
"@gitea/team1/more" ,
2019-10-13 19:29:10 -03:00
}
for _ , testCase := range trueTestCases {
2019-11-09 19:12:05 -03:00
found := mentionPattern . FindStringSubmatch ( testCase . pat )
assert . Len ( t , found , 2 )
assert . Equal ( t , testCase . exp , found [ 1 ] )
2019-10-13 19:29:10 -03:00
}
for _ , testCase := range falseTestCases {
res := mentionPattern . MatchString ( testCase )
2019-11-09 19:12:05 -03:00
assert . False ( t , res , "[%s] should be false" , testCase )
2019-10-13 19:29:10 -03:00
}
}
func TestRegExp_issueNumericPattern ( t * testing . T ) {
trueTestCases := [ ] string {
"#1234" ,
"#0" ,
"#1234567890987654321" ,
" #12" ,
"#12:" ,
"ref: #12: msg" ,
}
falseTestCases := [ ] string {
"# 1234" ,
"# 0" ,
"# " ,
"#" ,
"#ABC" ,
"#1A2B" ,
"" ,
"ABC" ,
}
for _ , testCase := range trueTestCases {
assert . True ( t , issueNumericPattern . MatchString ( testCase ) )
}
for _ , testCase := range falseTestCases {
assert . False ( t , issueNumericPattern . MatchString ( testCase ) )
}
}
func TestRegExp_issueAlphanumericPattern ( t * testing . T ) {
trueTestCases := [ ] string {
"ABC-1234" ,
"A-1" ,
"RC-80" ,
"ABCDEFGHIJ-1234567890987654321234567890" ,
"ABC-123." ,
"(ABC-123)" ,
"[ABC-123]" ,
"ABC-123:" ,
}
falseTestCases := [ ] string {
"RC-08" ,
"PR-0" ,
"ABCDEFGHIJK-1" ,
"PR_1" ,
"" ,
"#ABC" ,
"" ,
"ABC" ,
"GG-" ,
"rm-1" ,
"/home/gitea/ABC-1234" ,
"MY-STRING-ABC-123" ,
}
for _ , testCase := range trueTestCases {
assert . True ( t , issueAlphanumericPattern . MatchString ( testCase ) )
}
for _ , testCase := range falseTestCases {
assert . False ( t , issueAlphanumericPattern . MatchString ( testCase ) )
}
}
2019-10-30 09:43:59 -03:00
func TestCustomizeCloseKeywords ( t * testing . T ) {
fixtures := [ ] testFixture {
{
"Simplemente cierra: #29 yes" ,
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 29 , "" , "" , "29" , false , XRefActionCloses , & RefSpan { Start : 20 , End : 23 } , & RefSpan { Start : 12 , End : 18 } , "" } ,
2019-10-30 09:43:59 -03:00
} ,
} ,
{
"Closes: #123 no, this English." ,
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 123 , "" , "" , "123" , false , XRefActionNone , & RefSpan { Start : 8 , End : 12 } , nil , "" } ,
2019-10-30 09:43:59 -03:00
} ,
} ,
{
"Cerró user6/repo6#300 yes" ,
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 300 , "user6" , "repo6" , "300" , false , XRefActionCloses , & RefSpan { Start : 7 , End : 22 } , & RefSpan { Start : 0 , End : 6 } , "" } ,
2019-10-30 09:43:59 -03:00
} ,
} ,
{
"Reabre user3/repo4#200 yes" ,
[ ] testResult {
2020-09-04 18:37:37 +03:00
{ 200 , "user3" , "repo4" , "200" , false , XRefActionReopens , & RefSpan { Start : 7 , End : 22 } , & RefSpan { Start : 0 , End : 6 } , "" } ,
2019-10-30 09:43:59 -03:00
} ,
} ,
}
issueKeywordsOnce . Do ( func ( ) { } )
doNewKeywords ( [ ] string { "cierra" , "cerró" } , [ ] string { "reabre" } )
testFixtures ( t , fixtures , "spanish" )
// Restore default settings
doNewKeywords ( setting . Repository . PullRequest . CloseKeywords , setting . Repository . PullRequest . ReopenKeywords )
}
func TestParseCloseKeywords ( t * testing . T ) {
// Test parsing of CloseKeywords and ReopenKeywords
assert . Len ( t , parseKeywords ( [ ] string { "" } ) , 0 )
assert . Len ( t , parseKeywords ( [ ] string { " aa " , " bb " , "99" , "#" , "" , "this is" , "cc" } ) , 3 )
for _ , test := range [ ] struct {
pattern string
match string
expected string
} {
{ "close" , "This PR will close " , "close" } ,
{ "cerró" , "cerró " , "cerró" } ,
{ "cerró" , "AQUÍ SE CERRÓ: " , "CERRÓ" } ,
{ "закрывается" , "закрывается " , "закрывается" } ,
{ "κλείνει" , "κλείνει: " , "κλείνει" } ,
{ "关闭" , "关闭 " , "关闭" } ,
{ "閉じます" , "閉じます " , "閉じます" } ,
{ ",$!" , "" , "" } ,
{ "1234" , "" , "" } ,
} {
2021-07-08 07:38:13 -04:00
// The pattern only needs to match the part that precedes the reference.
2019-10-30 09:43:59 -03:00
// getCrossReference() takes care of finding the reference itself.
pat := makeKeywordsPat ( [ ] string { test . pattern } )
if test . expected == "" {
assert . Nil ( t , pat )
} else {
assert . NotNil ( t , pat )
res := pat . FindAllStringSubmatch ( test . match , - 1 )
assert . Len ( t , res , 1 )
assert . Len ( t , res [ 0 ] , 2 )
assert . EqualValues ( t , test . expected , res [ 0 ] [ 1 ] )
}
}
}