mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-31 21:28:11 +09:00 
			
		
		
		
	Accept punctuation after simple+cross repository issue references (#10091)
* Support references ending in , . and ; * Accept :;, in simple refs; fix 2+ consecutive refs * Include cross-repository references * Add ?!, fix spacing problem
This commit is contained in:
		| @@ -29,12 +29,14 @@ var ( | |||||||
| 	// mentionPattern matches all mentions in the form of "@user" | 	// mentionPattern matches all mentions in the form of "@user" | ||||||
| 	mentionPattern = regexp.MustCompile(`(?:\s|^|\(|\[)(@[0-9a-zA-Z-_]+|@[0-9a-zA-Z-_][0-9a-zA-Z-_.]+[0-9a-zA-Z-_])(?:\s|[:,;.?!]\s|[:,;.?!]?$|\)|\])`) | 	mentionPattern = regexp.MustCompile(`(?:\s|^|\(|\[)(@[0-9a-zA-Z-_]+|@[0-9a-zA-Z-_][0-9a-zA-Z-_.]+[0-9a-zA-Z-_])(?:\s|[:,;.?!]\s|[:,;.?!]?$|\)|\])`) | ||||||
| 	// issueNumericPattern matches string that references to a numeric issue, e.g. #1287 | 	// issueNumericPattern matches string that references to a numeric issue, e.g. #1287 | ||||||
| 	issueNumericPattern = regexp.MustCompile(`(?:\s|^|\(|\[)([#!][0-9]+)(?:\s|$|\)|\]|:|\.(\s|$))`) | 	issueNumericPattern = regexp.MustCompile(`(?:\s|^|\(|\[)([#!][0-9]+)(?:\s|$|\)|\]|[:;,.?!]\s|[:;,.?!]$)`) | ||||||
| 	// issueAlphanumericPattern matches string that references to an alphanumeric issue, e.g. ABC-1234 | 	// issueAlphanumericPattern matches string that references to an alphanumeric issue, e.g. ABC-1234 | ||||||
| 	issueAlphanumericPattern = regexp.MustCompile(`(?:\s|^|\(|\[)([A-Z]{1,10}-[1-9][0-9]*)(?:\s|$|\)|\]|:|\.(\s|$))`) | 	issueAlphanumericPattern = regexp.MustCompile(`(?:\s|^|\(|\[)([A-Z]{1,10}-[1-9][0-9]*)(?:\s|$|\)|\]|:|\.(\s|$))`) | ||||||
| 	// crossReferenceIssueNumericPattern matches string that references a numeric issue in a different repository | 	// crossReferenceIssueNumericPattern matches string that references a numeric issue in a different repository | ||||||
| 	// e.g. gogits/gogs#12345 | 	// e.g. gogits/gogs#12345 | ||||||
| 	crossReferenceIssueNumericPattern = regexp.MustCompile(`(?:\s|^|\(|\[)([0-9a-zA-Z-_\.]+/[0-9a-zA-Z-_\.]+[#!][0-9]+)(?:\s|$|\)|\]|\.(\s|$))`) | 	crossReferenceIssueNumericPattern = regexp.MustCompile(`(?:\s|^|\(|\[)([0-9a-zA-Z-_\.]+/[0-9a-zA-Z-_\.]+[#!][0-9]+)(?:\s|$|\)|\]|[:;,.?!]\s|[:;,.?!]$)`) | ||||||
|  | 	// spaceTrimmedPattern let's us find the trailing space | ||||||
|  | 	spaceTrimmedPattern = regexp.MustCompile(`(?:.*[0-9a-zA-Z-_])\s`) | ||||||
|  |  | ||||||
| 	issueCloseKeywordsPat, issueReopenKeywordsPat *regexp.Regexp | 	issueCloseKeywordsPat, issueReopenKeywordsPat *regexp.Regexp | ||||||
| 	issueKeywordsOnce                             sync.Once | 	issueKeywordsOnce                             sync.Once | ||||||
| @@ -172,10 +174,24 @@ func FindAllMentionsMarkdown(content string) []string { | |||||||
| // FindAllMentionsBytes matches mention patterns in given content | // FindAllMentionsBytes matches mention patterns in given content | ||||||
| // and returns a list of locations for the unvalidated user names, including the @ prefix. | // and returns a list of locations for the unvalidated user names, including the @ prefix. | ||||||
| func FindAllMentionsBytes(content []byte) []RefSpan { | func FindAllMentionsBytes(content []byte) []RefSpan { | ||||||
| 	mentions := mentionPattern.FindAllSubmatchIndex(content, -1) | 	// Sadly we can't use FindAllSubmatchIndex because our pattern checks for starting and | ||||||
| 	ret := make([]RefSpan, len(mentions)) | 	// trailing spaces (\s@mention,\s), so if we get two consecutive references, the space | ||||||
| 	for i, val := range mentions { | 	// from the second reference will be "eaten" by the first one: | ||||||
| 		ret[i] = RefSpan{Start: val[2], End: val[3]} | 	// ...\s@mention1\s@mention2\s...	--> ...`\s@mention1\s`, (not) `@mention2,\s...` | ||||||
|  | 	ret := make([]RefSpan, 0, 5) | ||||||
|  | 	pos := 0 | ||||||
|  | 	for { | ||||||
|  | 		match := mentionPattern.FindSubmatchIndex(content[pos:]) | ||||||
|  | 		if match == nil { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		ret = append(ret, RefSpan{Start: match[2] + pos, End: match[3] + pos}) | ||||||
|  | 		notrail := spaceTrimmedPattern.FindSubmatchIndex(content[match[2]+pos : match[3]+pos]) | ||||||
|  | 		if notrail == nil { | ||||||
|  | 			pos = match[3] + pos | ||||||
|  | 		} else { | ||||||
|  | 			pos = match[3] + pos + notrail[1] - notrail[3] | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	return ret | 	return ret | ||||||
| } | } | ||||||
| @@ -252,19 +268,44 @@ func FindRenderizableReferenceAlphanumeric(content string) (bool, *RenderizableR | |||||||
| func findAllIssueReferencesBytes(content []byte, links []string) []*rawReference { | func findAllIssueReferencesBytes(content []byte, links []string) []*rawReference { | ||||||
|  |  | ||||||
| 	ret := make([]*rawReference, 0, 10) | 	ret := make([]*rawReference, 0, 10) | ||||||
|  | 	pos := 0 | ||||||
|  |  | ||||||
| 	matches := issueNumericPattern.FindAllSubmatchIndex(content, -1) | 	// Sadly we can't use FindAllSubmatchIndex because our pattern checks for starting and | ||||||
| 	for _, match := range matches { | 	// trailing spaces (\s#ref,\s), so if we get two consecutive references, the space | ||||||
| 		if ref := getCrossReference(content, match[2], match[3], false, false); ref != nil { | 	// from the second reference will be "eaten" by the first one: | ||||||
|  | 	// ...\s#ref1\s#ref2\s...	--> ...`\s#ref1\s`, (not) `#ref2,\s...` | ||||||
|  | 	for { | ||||||
|  | 		match := issueNumericPattern.FindSubmatchIndex(content[pos:]) | ||||||
|  | 		if match == nil { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		if ref := getCrossReference(content, match[2]+pos, match[3]+pos, false, false); ref != nil { | ||||||
| 			ret = append(ret, ref) | 			ret = append(ret, ref) | ||||||
| 		} | 		} | ||||||
|  | 		notrail := spaceTrimmedPattern.FindSubmatchIndex(content[match[2]+pos : match[3]+pos]) | ||||||
|  | 		if notrail == nil { | ||||||
|  | 			pos = match[3] + pos | ||||||
|  | 		} else { | ||||||
|  | 			pos = match[3] + pos + notrail[1] - notrail[3] | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	matches = crossReferenceIssueNumericPattern.FindAllSubmatchIndex(content, -1) | 	pos = 0 | ||||||
| 	for _, match := range matches { |  | ||||||
| 		if ref := getCrossReference(content, match[2], match[3], false, false); ref != nil { | 	for { | ||||||
|  | 		match := crossReferenceIssueNumericPattern.FindSubmatchIndex(content[pos:]) | ||||||
|  | 		if match == nil { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		if ref := getCrossReference(content, match[2]+pos, match[3]+pos, false, false); ref != nil { | ||||||
| 			ret = append(ret, ref) | 			ret = append(ret, ref) | ||||||
| 		} | 		} | ||||||
|  | 		notrail := spaceTrimmedPattern.FindSubmatchIndex(content[match[2]+pos : match[3]+pos]) | ||||||
|  | 		if notrail == nil { | ||||||
|  | 			pos = match[3] + pos | ||||||
|  | 		} else { | ||||||
|  | 			pos = match[3] + pos + notrail[1] - notrail[3] | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	localhost := getGiteaHostName() | 	localhost := getGiteaHostName() | ||||||
|   | |||||||
| @@ -135,6 +135,39 @@ func TestFindAllIssueReferences(t *testing.T) { | |||||||
| 				{1235, "", "", "1235", false, XRefActionNone, &RefSpan{Start: 8, End: 13}, nil}, | 				{1235, "", "", "1235", false, XRefActionNone, &RefSpan{Start: 8, End: 13}, nil}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"For [!123] yes", | ||||||
|  | 			[]testResult{ | ||||||
|  | 				{123, "", "", "123", true, XRefActionNone, &RefSpan{Start: 5, End: 9}, nil}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"For (#345) yes", | ||||||
|  | 			[]testResult{ | ||||||
|  | 				{345, "", "", "345", false, XRefActionNone, &RefSpan{Start: 5, End: 9}, nil}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"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{ | ||||||
|  | 				{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}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"This user3/repo4#200, yes.", | ||||||
|  | 			[]testResult{ | ||||||
|  | 				{200, "user3", "repo4", "200", false, XRefActionNone, &RefSpan{Start: 5, End: 20}, nil}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
| 		{ | 		{ | ||||||
| 			"Which abc. #9434 same as above", | 			"Which abc. #9434 same as above", | ||||||
| 			[]testResult{ | 			[]testResult{ | ||||||
| @@ -217,6 +250,16 @@ func testFixtures(t *testing.T, fixtures []testFixture, context string) { | |||||||
| 	setting.AppURL = prevURL | 	setting.AppURL = prevURL | ||||||
| } | } | ||||||
|  |  | ||||||
|  | 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) | ||||||
|  | } | ||||||
|  |  | ||||||
| func TestRegExp_mentionPattern(t *testing.T) { | func TestRegExp_mentionPattern(t *testing.T) { | ||||||
| 	trueTestCases := []struct { | 	trueTestCases := []struct { | ||||||
| 		pat string | 		pat string | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user