mirror of
https://github.com/go-gitea/gitea.git
synced 2025-01-12 16:02:38 +00:00
Merge branch 'main' into patch-20
This commit is contained in:
commit
497dce5c20
@ -63,3 +63,4 @@ Tim-Niclas Oelschläger <zokki.softwareschmiede@gmail.com> (@zokkis)
|
|||||||
Yu Liu <1240335630@qq.com> (@HEREYUA)
|
Yu Liu <1240335630@qq.com> (@HEREYUA)
|
||||||
Kemal Zebari <kemalzebra@gmail.com> (@kemzeb)
|
Kemal Zebari <kemalzebra@gmail.com> (@kemzeb)
|
||||||
Rowan Bohde <rowan.bohde@gmail.com> (@bohde)
|
Rowan Bohde <rowan.bohde@gmail.com> (@bohde)
|
||||||
|
hiifong <i@hiif.ong> (@hiifong)
|
||||||
|
8
Makefile
8
Makefile
@ -377,12 +377,12 @@ lint-backend-fix: lint-go-fix lint-go-vet lint-editorconfig
|
|||||||
.PHONY: lint-js
|
.PHONY: lint-js
|
||||||
lint-js: node_modules
|
lint-js: node_modules
|
||||||
npx eslint --color --max-warnings=0 --ext js,ts,vue $(ESLINT_FILES)
|
npx eslint --color --max-warnings=0 --ext js,ts,vue $(ESLINT_FILES)
|
||||||
# npx tsc
|
# npx vue-tsc
|
||||||
|
|
||||||
.PHONY: lint-js-fix
|
.PHONY: lint-js-fix
|
||||||
lint-js-fix: node_modules
|
lint-js-fix: node_modules
|
||||||
npx eslint --color --max-warnings=0 --ext js,ts,vue $(ESLINT_FILES) --fix
|
npx eslint --color --max-warnings=0 --ext js,ts,vue $(ESLINT_FILES) --fix
|
||||||
# npx tsc
|
# npx vue-tsc
|
||||||
|
|
||||||
.PHONY: lint-css
|
.PHONY: lint-css
|
||||||
lint-css: node_modules
|
lint-css: node_modules
|
||||||
@ -451,6 +451,10 @@ lint-templates: .venv node_modules
|
|||||||
lint-yaml: .venv
|
lint-yaml: .venv
|
||||||
@poetry run yamllint .
|
@poetry run yamllint .
|
||||||
|
|
||||||
|
.PHONY: tsc
|
||||||
|
tsc:
|
||||||
|
npx vue-tsc
|
||||||
|
|
||||||
.PHONY: watch
|
.PHONY: watch
|
||||||
watch:
|
watch:
|
||||||
@bash tools/watch.sh
|
@bash tools/watch.sh
|
||||||
|
4
assets/go-licenses.json
generated
4
assets/go-licenses.json
generated
@ -1090,8 +1090,8 @@
|
|||||||
"licenseText": "MIT License\n\nCopyright (c) 2017 Asher\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n"
|
"licenseText": "MIT License\n\nCopyright (c) 2017 Asher\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "github.com/stretchr/testify/assert",
|
"name": "github.com/stretchr/testify",
|
||||||
"path": "github.com/stretchr/testify/assert/LICENSE",
|
"path": "github.com/stretchr/testify/LICENSE",
|
||||||
"licenseText": "MIT License\n\nCopyright (c) 2012-2020 Mat Ryer, Tyler Bunnell and contributors.\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n"
|
"licenseText": "MIT License\n\nCopyright (c) 2012-2020 Mat Ryer, Tyler Bunnell and contributors.\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -1944,6 +1944,13 @@ LEVEL = Info
|
|||||||
;; Minio secretAccessKey to connect only available when STORAGE_TYPE is `minio`
|
;; Minio secretAccessKey to connect only available when STORAGE_TYPE is `minio`
|
||||||
;MINIO_SECRET_ACCESS_KEY =
|
;MINIO_SECRET_ACCESS_KEY =
|
||||||
;;
|
;;
|
||||||
|
;; Preferred IAM Endpoint to override Minio's default IAM Endpoint resolution only available when STORAGE_TYPE is `minio`.
|
||||||
|
;; If not provided and STORAGE_TYPE is `minio`, will search for and derive endpoint from known environment variables
|
||||||
|
;; (AWS_CONTAINER_AUTHORIZATION_TOKEN, AWS_CONTAINER_AUTHORIZATION_TOKEN_FILE, AWS_CONTAINER_CREDENTIALS_RELATIVE_URI,
|
||||||
|
;; AWS_CONTAINER_CREDENTIALS_FULL_URI, AWS_WEB_IDENTITY_TOKEN_FILE, AWS_ROLE_ARN, AWS_ROLE_SESSION_NAME, AWS_REGION),
|
||||||
|
;; or the DefaultIAMRoleEndpoint if not provided otherwise.
|
||||||
|
;MINIO_IAM_ENDPOINT =
|
||||||
|
;;
|
||||||
;; Minio bucket to store the attachments only available when STORAGE_TYPE is `minio`
|
;; Minio bucket to store the attachments only available when STORAGE_TYPE is `minio`
|
||||||
;MINIO_BUCKET = gitea
|
;MINIO_BUCKET = gitea
|
||||||
;;
|
;;
|
||||||
@ -2688,6 +2695,13 @@ LEVEL = Info
|
|||||||
;; Minio secretAccessKey to connect only available when STORAGE_TYPE is `minio`
|
;; Minio secretAccessKey to connect only available when STORAGE_TYPE is `minio`
|
||||||
;MINIO_SECRET_ACCESS_KEY =
|
;MINIO_SECRET_ACCESS_KEY =
|
||||||
;;
|
;;
|
||||||
|
;; Preferred IAM Endpoint to override Minio's default IAM Endpoint resolution only available when STORAGE_TYPE is `minio`.
|
||||||
|
;; If not provided and STORAGE_TYPE is `minio`, will search for and derive endpoint from known environment variables
|
||||||
|
;; (AWS_CONTAINER_AUTHORIZATION_TOKEN, AWS_CONTAINER_AUTHORIZATION_TOKEN_FILE, AWS_CONTAINER_CREDENTIALS_RELATIVE_URI,
|
||||||
|
;; AWS_CONTAINER_CREDENTIALS_FULL_URI, AWS_WEB_IDENTITY_TOKEN_FILE, AWS_ROLE_ARN, AWS_ROLE_SESSION_NAME, AWS_REGION),
|
||||||
|
;; or the DefaultIAMRoleEndpoint if not provided otherwise.
|
||||||
|
;MINIO_IAM_ENDPOINT =
|
||||||
|
;;
|
||||||
;; Minio bucket to store the attachments only available when STORAGE_TYPE is `minio`
|
;; Minio bucket to store the attachments only available when STORAGE_TYPE is `minio`
|
||||||
;MINIO_BUCKET = gitea
|
;MINIO_BUCKET = gitea
|
||||||
;;
|
;;
|
||||||
|
@ -200,7 +200,7 @@ func (a *Action) LoadActUser(ctx context.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *Action) loadRepo(ctx context.Context) {
|
func (a *Action) LoadRepo(ctx context.Context) {
|
||||||
if a.Repo != nil {
|
if a.Repo != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -250,7 +250,7 @@ func (a *Action) GetActDisplayNameTitle(ctx context.Context) string {
|
|||||||
|
|
||||||
// GetRepoUserName returns the name of the action repository owner.
|
// GetRepoUserName returns the name of the action repository owner.
|
||||||
func (a *Action) GetRepoUserName(ctx context.Context) string {
|
func (a *Action) GetRepoUserName(ctx context.Context) string {
|
||||||
a.loadRepo(ctx)
|
a.LoadRepo(ctx)
|
||||||
if a.Repo == nil {
|
if a.Repo == nil {
|
||||||
return "(non-existing-repo)"
|
return "(non-existing-repo)"
|
||||||
}
|
}
|
||||||
@ -265,7 +265,7 @@ func (a *Action) ShortRepoUserName(ctx context.Context) string {
|
|||||||
|
|
||||||
// GetRepoName returns the name of the action repository.
|
// GetRepoName returns the name of the action repository.
|
||||||
func (a *Action) GetRepoName(ctx context.Context) string {
|
func (a *Action) GetRepoName(ctx context.Context) string {
|
||||||
a.loadRepo(ctx)
|
a.LoadRepo(ctx)
|
||||||
if a.Repo == nil {
|
if a.Repo == nil {
|
||||||
return "(non-existing-repo)"
|
return "(non-existing-repo)"
|
||||||
}
|
}
|
||||||
@ -644,7 +644,7 @@ func NotifyWatchers(ctx context.Context, actions ...*Action) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if repoChanged {
|
if repoChanged {
|
||||||
act.loadRepo(ctx)
|
act.LoadRepo(ctx)
|
||||||
repo = act.Repo
|
repo = act.Repo
|
||||||
|
|
||||||
// check repo owner exist.
|
// check repo owner exist.
|
||||||
|
@ -1108,7 +1108,7 @@ func FindComments(ctx context.Context, opts *FindCommentsOptions) (CommentList,
|
|||||||
sess.Join("INNER", "issue", "issue.id = comment.issue_id")
|
sess.Join("INNER", "issue", "issue.id = comment.issue_id")
|
||||||
}
|
}
|
||||||
|
|
||||||
if opts.Page != 0 {
|
if opts.Page > 0 {
|
||||||
sess = db.SetSessionPagination(sess, opts)
|
sess = db.SetSessionPagination(sess, opts)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -7,8 +7,8 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
|
|
||||||
"code.gitea.io/gitea/models/db"
|
"code.gitea.io/gitea/models/db"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
|
|
||||||
"xorm.io/builder"
|
"xorm.io/builder"
|
||||||
@ -112,14 +112,8 @@ func findCodeComments(ctx context.Context, opts FindCommentsOptions, issue *Issu
|
|||||||
}
|
}
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
if comment.RenderedContent, err = markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, issue.Repo)
|
||||||
Ctx: ctx,
|
if comment.RenderedContent, err = markdown.RenderString(rctx, comment.Content); err != nil {
|
||||||
Repo: issue.Repo,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: issue.Repo.Link(),
|
|
||||||
},
|
|
||||||
Metas: issue.Repo.ComposeMetas(ctx),
|
|
||||||
}, comment.Content); err != nil {
|
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -641,7 +641,7 @@ func (issue *Issue) BlockedByDependencies(ctx context.Context, opts db.ListOptio
|
|||||||
Where("issue_id = ?", issue.ID).
|
Where("issue_id = ?", issue.ID).
|
||||||
// sort by repo id then created date, with the issues of the same repo at the beginning of the list
|
// sort by repo id then created date, with the issues of the same repo at the beginning of the list
|
||||||
OrderBy("CASE WHEN issue.repo_id = ? THEN 0 ELSE issue.repo_id END, issue.created_unix DESC", issue.RepoID)
|
OrderBy("CASE WHEN issue.repo_id = ? THEN 0 ELSE issue.repo_id END, issue.created_unix DESC", issue.RepoID)
|
||||||
if opts.Page != 0 {
|
if opts.Page > 0 {
|
||||||
sess = db.SetSessionPagination(sess, &opts)
|
sess = db.SetSessionPagination(sess, &opts)
|
||||||
}
|
}
|
||||||
err = sess.Find(&issueDeps)
|
err = sess.Find(&issueDeps)
|
||||||
|
@ -105,7 +105,7 @@ func GetIssueWatchers(ctx context.Context, issueID int64, listOptions db.ListOpt
|
|||||||
And("`user`.prohibit_login = ?", false).
|
And("`user`.prohibit_login = ?", false).
|
||||||
Join("INNER", "`user`", "`user`.id = `issue_watch`.user_id")
|
Join("INNER", "`user`", "`user`.id = `issue_watch`.user_id")
|
||||||
|
|
||||||
if listOptions.Page != 0 {
|
if listOptions.Page > 0 {
|
||||||
sess = db.SetSessionPagination(sess, &listOptions)
|
sess = db.SetSessionPagination(sess, &listOptions)
|
||||||
watches := make([]*IssueWatch, 0, listOptions.PageSize)
|
watches := make([]*IssueWatch, 0, listOptions.PageSize)
|
||||||
return watches, sess.Find(&watches)
|
return watches, sess.Find(&watches)
|
||||||
|
@ -390,7 +390,7 @@ func GetLabelsByRepoID(ctx context.Context, repoID int64, sortType string, listO
|
|||||||
sess.Asc("name")
|
sess.Asc("name")
|
||||||
}
|
}
|
||||||
|
|
||||||
if listOptions.Page != 0 {
|
if listOptions.Page > 0 {
|
||||||
sess = db.SetSessionPagination(sess, &listOptions)
|
sess = db.SetSessionPagination(sess, &listOptions)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -462,7 +462,7 @@ func GetLabelsByOrgID(ctx context.Context, orgID int64, sortType string, listOpt
|
|||||||
sess.Asc("name")
|
sess.Asc("name")
|
||||||
}
|
}
|
||||||
|
|
||||||
if listOptions.Page != 0 {
|
if listOptions.Page > 0 {
|
||||||
sess = db.SetSessionPagination(sess, &listOptions)
|
sess = db.SetSessionPagination(sess, &listOptions)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -163,7 +163,7 @@ func FindReactions(ctx context.Context, opts FindReactionsOptions) (ReactionList
|
|||||||
Where(opts.toConds()).
|
Where(opts.toConds()).
|
||||||
In("reaction.`type`", setting.UI.Reactions).
|
In("reaction.`type`", setting.UI.Reactions).
|
||||||
Asc("reaction.issue_id", "reaction.comment_id", "reaction.created_unix", "reaction.id")
|
Asc("reaction.issue_id", "reaction.comment_id", "reaction.created_unix", "reaction.id")
|
||||||
if opts.Page != 0 {
|
if opts.Page > 0 {
|
||||||
sess = db.SetSessionPagination(sess, &opts)
|
sess = db.SetSessionPagination(sess, &opts)
|
||||||
|
|
||||||
reactions := make([]*Reaction, 0, opts.PageSize)
|
reactions := make([]*Reaction, 0, opts.PageSize)
|
||||||
|
@ -96,7 +96,7 @@ func GetUIDsAndStopwatch(ctx context.Context) ([]*UserStopwatch, error) {
|
|||||||
func GetUserStopwatches(ctx context.Context, userID int64, listOptions db.ListOptions) ([]*Stopwatch, error) {
|
func GetUserStopwatches(ctx context.Context, userID int64, listOptions db.ListOptions) ([]*Stopwatch, error) {
|
||||||
sws := make([]*Stopwatch, 0, 8)
|
sws := make([]*Stopwatch, 0, 8)
|
||||||
sess := db.GetEngine(ctx).Where("stopwatch.user_id = ?", userID)
|
sess := db.GetEngine(ctx).Where("stopwatch.user_id = ?", userID)
|
||||||
if listOptions.Page != 0 {
|
if listOptions.Page > 0 {
|
||||||
sess = db.SetSessionPagination(sess, &listOptions)
|
sess = db.SetSessionPagination(sess, &listOptions)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -139,7 +139,7 @@ func (opts *FindTrackedTimesOptions) toSession(e db.Engine) db.Engine {
|
|||||||
|
|
||||||
sess = sess.Where(opts.ToConds())
|
sess = sess.Where(opts.ToConds())
|
||||||
|
|
||||||
if opts.Page != 0 {
|
if opts.Page > 0 {
|
||||||
sess = db.SetSessionPagination(sess, opts)
|
sess = db.SetSessionPagination(sess, opts)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,6 +9,7 @@ import (
|
|||||||
"code.gitea.io/gitea/models/db"
|
"code.gitea.io/gitea/models/db"
|
||||||
"code.gitea.io/gitea/models/perm"
|
"code.gitea.io/gitea/models/perm"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
|
"code.gitea.io/gitea/models/unit"
|
||||||
|
|
||||||
"xorm.io/builder"
|
"xorm.io/builder"
|
||||||
)
|
)
|
||||||
@ -83,3 +84,16 @@ func GetTeamsWithAccessToRepo(ctx context.Context, orgID, repoID int64, mode per
|
|||||||
OrderBy("name").
|
OrderBy("name").
|
||||||
Find(&teams)
|
Find(&teams)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetTeamsWithAccessToRepoUnit returns all teams in an organization that have given access level to the repository special unit.
|
||||||
|
func GetTeamsWithAccessToRepoUnit(ctx context.Context, orgID, repoID int64, mode perm.AccessMode, unitType unit.Type) ([]*Team, error) {
|
||||||
|
teams := make([]*Team, 0, 5)
|
||||||
|
return teams, db.GetEngine(ctx).Where("team_unit.access_mode >= ?", mode).
|
||||||
|
Join("INNER", "team_repo", "team_repo.team_id = team.id").
|
||||||
|
Join("INNER", "team_unit", "team_unit.team_id = team.id").
|
||||||
|
And("team_repo.org_id = ?", orgID).
|
||||||
|
And("team_repo.repo_id = ?", repoID).
|
||||||
|
And("team_unit.type = ?", unitType).
|
||||||
|
OrderBy("name").
|
||||||
|
Find(&teams)
|
||||||
|
}
|
||||||
|
31
models/organization/team_repo_test.go
Normal file
31
models/organization/team_repo_test.go
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package organization_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/models/db"
|
||||||
|
"code.gitea.io/gitea/models/organization"
|
||||||
|
"code.gitea.io/gitea/models/perm"
|
||||||
|
"code.gitea.io/gitea/models/repo"
|
||||||
|
"code.gitea.io/gitea/models/unit"
|
||||||
|
"code.gitea.io/gitea/models/unittest"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGetTeamsWithAccessToRepoUnit(t *testing.T) {
|
||||||
|
assert.NoError(t, unittest.PrepareTestDatabase())
|
||||||
|
|
||||||
|
org41 := unittest.AssertExistsAndLoadBean(t, &organization.Organization{ID: 41})
|
||||||
|
repo61 := unittest.AssertExistsAndLoadBean(t, &repo.Repository{ID: 61})
|
||||||
|
|
||||||
|
teams, err := organization.GetTeamsWithAccessToRepoUnit(db.DefaultContext, org41.ID, repo61.ID, perm.AccessModeRead, unit.TypePullRequests)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
if assert.Len(t, teams, 2) {
|
||||||
|
assert.EqualValues(t, 21, teams[0].ID)
|
||||||
|
assert.EqualValues(t, 22, teams[1].ID)
|
||||||
|
}
|
||||||
|
}
|
53
models/renderhelper/commit_checker.go
Normal file
53
models/renderhelper/commit_checker.go
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package renderhelper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/modules/git"
|
||||||
|
"code.gitea.io/gitea/modules/gitrepo"
|
||||||
|
"code.gitea.io/gitea/modules/log"
|
||||||
|
)
|
||||||
|
|
||||||
|
type commitChecker struct {
|
||||||
|
ctx context.Context
|
||||||
|
commitCache map[string]bool
|
||||||
|
gitRepoFacade gitrepo.Repository
|
||||||
|
|
||||||
|
gitRepo *git.Repository
|
||||||
|
gitRepoCloser io.Closer
|
||||||
|
}
|
||||||
|
|
||||||
|
func newCommitChecker(ctx context.Context, gitRepo gitrepo.Repository) *commitChecker {
|
||||||
|
return &commitChecker{ctx: ctx, commitCache: make(map[string]bool), gitRepoFacade: gitRepo}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *commitChecker) Close() error {
|
||||||
|
if c != nil && c.gitRepoCloser != nil {
|
||||||
|
return c.gitRepoCloser.Close()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *commitChecker) IsCommitIDExisting(commitID string) bool {
|
||||||
|
exist, inCache := c.commitCache[commitID]
|
||||||
|
if inCache {
|
||||||
|
return exist
|
||||||
|
}
|
||||||
|
|
||||||
|
if c.gitRepo == nil {
|
||||||
|
r, closer, err := gitrepo.RepositoryFromContextOrOpen(c.ctx, c.gitRepoFacade)
|
||||||
|
if err != nil {
|
||||||
|
log.Error("unable to open repository: %s Error: %v", gitrepo.RepoGitURL(c.gitRepoFacade), err)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
c.gitRepo, c.gitRepoCloser = r, closer
|
||||||
|
}
|
||||||
|
|
||||||
|
exist = c.gitRepo.IsReferenceExist(commitID) // Don't use IsObjectExist since it doesn't support short hashs with gogit edition.
|
||||||
|
c.commitCache[commitID] = exist
|
||||||
|
return exist
|
||||||
|
}
|
27
models/renderhelper/main_test.go
Normal file
27
models/renderhelper/main_test.go
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package renderhelper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/models/unittest"
|
||||||
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestMain(m *testing.M) {
|
||||||
|
unittest.MainTest(m, &unittest.TestOptions{
|
||||||
|
FixtureFiles: []string{"repository.yml", "user.yml"},
|
||||||
|
SetUp: func() error {
|
||||||
|
markup.RenderBehaviorForTesting.DisableAdditionalAttributes = true
|
||||||
|
markup.Init(&markup.RenderHelperFuncs{
|
||||||
|
IsUsernameMentionable: func(ctx context.Context, username string) bool {
|
||||||
|
return username == "user2"
|
||||||
|
},
|
||||||
|
})
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
73
models/renderhelper/repo_comment.go
Normal file
73
models/renderhelper/repo_comment.go
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package renderhelper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
"code.gitea.io/gitea/modules/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RepoComment struct {
|
||||||
|
ctx *markup.RenderContext
|
||||||
|
opts RepoCommentOptions
|
||||||
|
|
||||||
|
commitChecker *commitChecker
|
||||||
|
repoLink string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RepoComment) CleanUp() {
|
||||||
|
_ = r.commitChecker.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RepoComment) IsCommitIDExisting(commitID string) bool {
|
||||||
|
return r.commitChecker.IsCommitIDExisting(commitID)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RepoComment) ResolveLink(link string, likeType markup.LinkType) (finalLink string) {
|
||||||
|
switch likeType {
|
||||||
|
case markup.LinkTypeApp:
|
||||||
|
finalLink = r.ctx.ResolveLinkApp(link)
|
||||||
|
default:
|
||||||
|
finalLink = r.ctx.ResolveLinkRelative(r.repoLink, r.opts.CurrentRefPath, link)
|
||||||
|
}
|
||||||
|
return finalLink
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ markup.RenderHelper = (*RepoComment)(nil)
|
||||||
|
|
||||||
|
type RepoCommentOptions struct {
|
||||||
|
DeprecatedRepoName string // it is only a patch for the non-standard "markup" api
|
||||||
|
DeprecatedOwnerName string // it is only a patch for the non-standard "markup" api
|
||||||
|
CurrentRefPath string // eg: "branch/main" or "commit/11223344"
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewRenderContextRepoComment(ctx context.Context, repo *repo_model.Repository, opts ...RepoCommentOptions) *markup.RenderContext {
|
||||||
|
helper := &RepoComment{
|
||||||
|
repoLink: repo.Link(),
|
||||||
|
opts: util.OptionalArg(opts),
|
||||||
|
}
|
||||||
|
rctx := markup.NewRenderContext(ctx)
|
||||||
|
helper.ctx = rctx
|
||||||
|
if repo != nil {
|
||||||
|
helper.repoLink = repo.Link()
|
||||||
|
helper.commitChecker = newCommitChecker(ctx, repo)
|
||||||
|
rctx = rctx.WithMetas(repo.ComposeMetas(ctx))
|
||||||
|
} else {
|
||||||
|
// this is almost dead code, only to pass the incorrect tests
|
||||||
|
helper.repoLink = fmt.Sprintf("%s/%s", helper.opts.DeprecatedOwnerName, helper.opts.DeprecatedRepoName)
|
||||||
|
rctx = rctx.WithMetas(map[string]string{
|
||||||
|
"user": helper.opts.DeprecatedOwnerName,
|
||||||
|
"repo": helper.opts.DeprecatedRepoName,
|
||||||
|
|
||||||
|
"markdownLineBreakStyle": "comment",
|
||||||
|
"markupAllowShortIssuePattern": "true",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
rctx = rctx.WithHelper(helper)
|
||||||
|
return rctx
|
||||||
|
}
|
76
models/renderhelper/repo_comment_test.go
Normal file
76
models/renderhelper/repo_comment_test.go
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package renderhelper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
|
"code.gitea.io/gitea/models/unittest"
|
||||||
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestRepoComment(t *testing.T) {
|
||||||
|
unittest.PrepareTestEnv(t)
|
||||||
|
|
||||||
|
repo1 := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1})
|
||||||
|
|
||||||
|
t.Run("AutoLink", func(t *testing.T) {
|
||||||
|
rctx := NewRenderContextRepoComment(context.Background(), repo1).WithMarkupType(markdown.MarkupName)
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
65f1bf27bc3bf70f64657658635e66094edbcb4d
|
||||||
|
#1
|
||||||
|
@user2
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t,
|
||||||
|
`<p><a href="/user2/repo1/commit/65f1bf27bc3bf70f64657658635e66094edbcb4d" rel="nofollow"><code>65f1bf27bc</code></a><br/>
|
||||||
|
<a href="/user2/repo1/issues/1" class="ref-issue" rel="nofollow">#1</a><br/>
|
||||||
|
<a href="/user2" rel="nofollow">@user2</a></p>
|
||||||
|
`, rendered)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("AbsoluteAndRelative", func(t *testing.T) {
|
||||||
|
rctx := NewRenderContextRepoComment(context.Background(), repo1).WithMarkupType(markdown.MarkupName)
|
||||||
|
|
||||||
|
// It is Gitea's old behavior, the relative path is resolved to the repo path
|
||||||
|
// It is different from GitHub, GitHub resolves relative links to current page's path
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
[/test](/test)
|
||||||
|
[./test](./test)
|
||||||
|
![/image](/image)
|
||||||
|
![./image](./image)
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t,
|
||||||
|
`<p><a href="/user2/repo1/test" rel="nofollow">/test</a><br/>
|
||||||
|
<a href="/user2/repo1/test" rel="nofollow">./test</a><br/>
|
||||||
|
<a href="/user2/repo1/image" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/image" alt="/image"/></a><br/>
|
||||||
|
<a href="/user2/repo1/image" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/image" alt="./image"/></a></p>
|
||||||
|
`, rendered)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("WithCurrentRefPath", func(t *testing.T) {
|
||||||
|
rctx := NewRenderContextRepoComment(context.Background(), repo1, RepoCommentOptions{CurrentRefPath: "/commit/1234"}).
|
||||||
|
WithMarkupType(markdown.MarkupName)
|
||||||
|
|
||||||
|
// the ref path is only used to render commit message: a commit message is rendered at the commit page with its commit ID path
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
[/test](/test)
|
||||||
|
[./test](./test)
|
||||||
|
![/image](/image)
|
||||||
|
![./image](./image)
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, `<p><a href="/user2/repo1/test" rel="nofollow">/test</a><br/>
|
||||||
|
<a href="/user2/repo1/commit/1234/test" rel="nofollow">./test</a><br/>
|
||||||
|
<a href="/user2/repo1/image" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/image" alt="/image"/></a><br/>
|
||||||
|
<a href="/user2/repo1/commit/1234/image" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/commit/1234/image" alt="./image"/></a></p>
|
||||||
|
`, rendered)
|
||||||
|
})
|
||||||
|
}
|
77
models/renderhelper/repo_file.go
Normal file
77
models/renderhelper/repo_file.go
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package renderhelper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"path"
|
||||||
|
|
||||||
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
"code.gitea.io/gitea/modules/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RepoFile struct {
|
||||||
|
ctx *markup.RenderContext
|
||||||
|
opts RepoFileOptions
|
||||||
|
|
||||||
|
commitChecker *commitChecker
|
||||||
|
repoLink string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RepoFile) CleanUp() {
|
||||||
|
_ = r.commitChecker.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RepoFile) IsCommitIDExisting(commitID string) bool {
|
||||||
|
return r.commitChecker.IsCommitIDExisting(commitID)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RepoFile) ResolveLink(link string, likeType markup.LinkType) string {
|
||||||
|
finalLink := link
|
||||||
|
switch likeType {
|
||||||
|
case markup.LinkTypeApp:
|
||||||
|
finalLink = r.ctx.ResolveLinkApp(link)
|
||||||
|
case markup.LinkTypeDefault:
|
||||||
|
finalLink = r.ctx.ResolveLinkRelative(path.Join(r.repoLink, "src", r.opts.CurrentRefPath), r.opts.CurrentTreePath, link)
|
||||||
|
case markup.LinkTypeRaw:
|
||||||
|
finalLink = r.ctx.ResolveLinkRelative(path.Join(r.repoLink, "raw", r.opts.CurrentRefPath), r.opts.CurrentTreePath, link)
|
||||||
|
case markup.LinkTypeMedia:
|
||||||
|
finalLink = r.ctx.ResolveLinkRelative(path.Join(r.repoLink, "media", r.opts.CurrentRefPath), r.opts.CurrentTreePath, link)
|
||||||
|
}
|
||||||
|
return finalLink
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ markup.RenderHelper = (*RepoFile)(nil)
|
||||||
|
|
||||||
|
type RepoFileOptions struct {
|
||||||
|
DeprecatedRepoName string // it is only a patch for the non-standard "markup" api
|
||||||
|
DeprecatedOwnerName string // it is only a patch for the non-standard "markup" api
|
||||||
|
|
||||||
|
CurrentRefPath string // eg: "branch/main"
|
||||||
|
CurrentTreePath string // eg: "path/to/file" in the repo
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewRenderContextRepoFile(ctx context.Context, repo *repo_model.Repository, opts ...RepoFileOptions) *markup.RenderContext {
|
||||||
|
helper := &RepoFile{opts: util.OptionalArg(opts)}
|
||||||
|
rctx := markup.NewRenderContext(ctx)
|
||||||
|
helper.ctx = rctx
|
||||||
|
if repo != nil {
|
||||||
|
helper.repoLink = repo.Link()
|
||||||
|
helper.commitChecker = newCommitChecker(ctx, repo)
|
||||||
|
rctx = rctx.WithMetas(repo.ComposeDocumentMetas(ctx))
|
||||||
|
} else {
|
||||||
|
// this is almost dead code, only to pass the incorrect tests
|
||||||
|
helper.repoLink = fmt.Sprintf("%s/%s", helper.opts.DeprecatedOwnerName, helper.opts.DeprecatedRepoName)
|
||||||
|
rctx = rctx.WithMetas(map[string]string{
|
||||||
|
"user": helper.opts.DeprecatedOwnerName,
|
||||||
|
"repo": helper.opts.DeprecatedRepoName,
|
||||||
|
|
||||||
|
"markdownLineBreakStyle": "document",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
rctx = rctx.WithHelper(helper)
|
||||||
|
return rctx
|
||||||
|
}
|
83
models/renderhelper/repo_file_test.go
Normal file
83
models/renderhelper/repo_file_test.go
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package renderhelper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
|
"code.gitea.io/gitea/models/unittest"
|
||||||
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestRepoFile(t *testing.T) {
|
||||||
|
unittest.PrepareTestEnv(t)
|
||||||
|
repo1 := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1})
|
||||||
|
|
||||||
|
t.Run("AutoLink", func(t *testing.T) {
|
||||||
|
rctx := NewRenderContextRepoFile(context.Background(), repo1).WithMarkupType(markdown.MarkupName)
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
65f1bf27bc3bf70f64657658635e66094edbcb4d
|
||||||
|
#1
|
||||||
|
@user2
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t,
|
||||||
|
`<p><a href="/user2/repo1/commit/65f1bf27bc3bf70f64657658635e66094edbcb4d" rel="nofollow"><code>65f1bf27bc</code></a>
|
||||||
|
#1
|
||||||
|
<a href="/user2" rel="nofollow">@user2</a></p>
|
||||||
|
`, rendered)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("AbsoluteAndRelative", func(t *testing.T) {
|
||||||
|
rctx := NewRenderContextRepoFile(context.Background(), repo1, RepoFileOptions{CurrentRefPath: "branch/main"}).
|
||||||
|
WithMarkupType(markdown.MarkupName)
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
[/test](/test)
|
||||||
|
[./test](./test)
|
||||||
|
![/image](/image)
|
||||||
|
![./image](./image)
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t,
|
||||||
|
`<p><a href="/user2/repo1/src/branch/main/test" rel="nofollow">/test</a>
|
||||||
|
<a href="/user2/repo1/src/branch/main/test" rel="nofollow">./test</a>
|
||||||
|
<a href="/user2/repo1/media/branch/main/image" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/media/branch/main/image" alt="/image"/></a>
|
||||||
|
<a href="/user2/repo1/media/branch/main/image" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/media/branch/main/image" alt="./image"/></a></p>
|
||||||
|
`, rendered)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("WithCurrentRefPath", func(t *testing.T) {
|
||||||
|
rctx := NewRenderContextRepoFile(context.Background(), repo1, RepoFileOptions{CurrentRefPath: "/commit/1234"}).
|
||||||
|
WithMarkupType(markdown.MarkupName)
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
[/test](/test)
|
||||||
|
![/image](/image)
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, `<p><a href="/user2/repo1/src/commit/1234/test" rel="nofollow">/test</a>
|
||||||
|
<a href="/user2/repo1/media/commit/1234/image" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/media/commit/1234/image" alt="/image"/></a></p>
|
||||||
|
`, rendered)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("WithCurrentRefPathByTag", func(t *testing.T) {
|
||||||
|
rctx := NewRenderContextRepoFile(context.Background(), repo1, RepoFileOptions{
|
||||||
|
CurrentRefPath: "/commit/1234",
|
||||||
|
CurrentTreePath: "my-dir",
|
||||||
|
}).
|
||||||
|
WithMarkupType(markdown.MarkupName)
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
<img src="LINK">
|
||||||
|
<video src="LINK">
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, `<a href="/user2/repo1/media/commit/1234/my-dir/LINK" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/media/commit/1234/my-dir/LINK"/></a>
|
||||||
|
<video src="/user2/repo1/media/commit/1234/my-dir/LINK">
|
||||||
|
</video>`, rendered)
|
||||||
|
})
|
||||||
|
}
|
80
models/renderhelper/repo_wiki.go
Normal file
80
models/renderhelper/repo_wiki.go
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package renderhelper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"path"
|
||||||
|
|
||||||
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
|
"code.gitea.io/gitea/modules/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RepoWiki struct {
|
||||||
|
ctx *markup.RenderContext
|
||||||
|
opts RepoWikiOptions
|
||||||
|
|
||||||
|
commitChecker *commitChecker
|
||||||
|
repoLink string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RepoWiki) CleanUp() {
|
||||||
|
_ = r.commitChecker.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RepoWiki) IsCommitIDExisting(commitID string) bool {
|
||||||
|
return r.commitChecker.IsCommitIDExisting(commitID)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RepoWiki) ResolveLink(link string, likeType markup.LinkType) string {
|
||||||
|
finalLink := link
|
||||||
|
switch likeType {
|
||||||
|
case markup.LinkTypeApp:
|
||||||
|
finalLink = r.ctx.ResolveLinkApp(link)
|
||||||
|
case markup.LinkTypeDefault:
|
||||||
|
finalLink = r.ctx.ResolveLinkRelative(path.Join(r.repoLink, "wiki", r.opts.currentRefPath), r.opts.currentTreePath, link)
|
||||||
|
case markup.LinkTypeMedia:
|
||||||
|
finalLink = r.ctx.ResolveLinkRelative(path.Join(r.repoLink, "wiki/raw", r.opts.currentRefPath), r.opts.currentTreePath, link)
|
||||||
|
case markup.LinkTypeRaw: // wiki doesn't use it
|
||||||
|
}
|
||||||
|
|
||||||
|
return finalLink
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ markup.RenderHelper = (*RepoWiki)(nil)
|
||||||
|
|
||||||
|
type RepoWikiOptions struct {
|
||||||
|
DeprecatedRepoName string // it is only a patch for the non-standard "markup" api
|
||||||
|
DeprecatedOwnerName string // it is only a patch for the non-standard "markup" api
|
||||||
|
|
||||||
|
// these options are not used at the moment because Wiki doesn't support sub-path, nor branch
|
||||||
|
currentRefPath string // eg: "branch/main"
|
||||||
|
currentTreePath string // eg: "path/to/file" in the repo
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewRenderContextRepoWiki(ctx context.Context, repo *repo_model.Repository, opts ...RepoWikiOptions) *markup.RenderContext {
|
||||||
|
helper := &RepoWiki{opts: util.OptionalArg(opts)}
|
||||||
|
rctx := markup.NewRenderContext(ctx).WithMarkupType(markdown.MarkupName)
|
||||||
|
if repo != nil {
|
||||||
|
helper.repoLink = repo.Link()
|
||||||
|
helper.commitChecker = newCommitChecker(ctx, repo)
|
||||||
|
rctx = rctx.WithMetas(repo.ComposeWikiMetas(ctx))
|
||||||
|
} else {
|
||||||
|
// this is almost dead code, only to pass the incorrect tests
|
||||||
|
helper.repoLink = fmt.Sprintf("%s/%s", helper.opts.DeprecatedOwnerName, helper.opts.DeprecatedRepoName)
|
||||||
|
rctx = rctx.WithMetas(map[string]string{
|
||||||
|
"user": helper.opts.DeprecatedOwnerName,
|
||||||
|
"repo": helper.opts.DeprecatedRepoName,
|
||||||
|
|
||||||
|
"markdownLineBreakStyle": "document",
|
||||||
|
"markupAllowShortIssuePattern": "true",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
rctx = rctx.WithHelper(helper)
|
||||||
|
helper.ctx = rctx
|
||||||
|
return rctx
|
||||||
|
}
|
65
models/renderhelper/repo_wiki_test.go
Normal file
65
models/renderhelper/repo_wiki_test.go
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package renderhelper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
|
"code.gitea.io/gitea/models/unittest"
|
||||||
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestRepoWiki(t *testing.T) {
|
||||||
|
unittest.PrepareTestEnv(t)
|
||||||
|
repo1 := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1})
|
||||||
|
|
||||||
|
t.Run("AutoLink", func(t *testing.T) {
|
||||||
|
rctx := NewRenderContextRepoWiki(context.Background(), repo1).WithMarkupType(markdown.MarkupName)
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
65f1bf27bc3bf70f64657658635e66094edbcb4d
|
||||||
|
#1
|
||||||
|
@user2
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t,
|
||||||
|
`<p><a href="/user2/repo1/commit/65f1bf27bc3bf70f64657658635e66094edbcb4d" rel="nofollow"><code>65f1bf27bc</code></a>
|
||||||
|
<a href="/user2/repo1/issues/1" class="ref-issue" rel="nofollow">#1</a>
|
||||||
|
<a href="/user2" rel="nofollow">@user2</a></p>
|
||||||
|
`, rendered)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("AbsoluteAndRelative", func(t *testing.T) {
|
||||||
|
rctx := NewRenderContextRepoWiki(context.Background(), repo1).WithMarkupType(markdown.MarkupName)
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
[/test](/test)
|
||||||
|
[./test](./test)
|
||||||
|
![/image](/image)
|
||||||
|
![./image](./image)
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t,
|
||||||
|
`<p><a href="/user2/repo1/wiki/test" rel="nofollow">/test</a>
|
||||||
|
<a href="/user2/repo1/wiki/test" rel="nofollow">./test</a>
|
||||||
|
<a href="/user2/repo1/wiki/raw/image" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/wiki/raw/image" alt="/image"/></a>
|
||||||
|
<a href="/user2/repo1/wiki/raw/image" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/wiki/raw/image" alt="./image"/></a></p>
|
||||||
|
`, rendered)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("PathInTag", func(t *testing.T) {
|
||||||
|
rctx := NewRenderContextRepoWiki(context.Background(), repo1).WithMarkupType(markdown.MarkupName)
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
<img src="LINK">
|
||||||
|
<video src="LINK">
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, `<a href="/user2/repo1/wiki/raw/LINK" target="_blank" rel="nofollow noopener"><img src="/user2/repo1/wiki/raw/LINK"/></a>
|
||||||
|
<video src="/user2/repo1/wiki/raw/LINK">
|
||||||
|
</video>`, rendered)
|
||||||
|
})
|
||||||
|
}
|
29
models/renderhelper/simple_document.go
Normal file
29
models/renderhelper/simple_document.go
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package renderhelper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
)
|
||||||
|
|
||||||
|
type SimpleDocument struct {
|
||||||
|
*markup.SimpleRenderHelper
|
||||||
|
ctx *markup.RenderContext
|
||||||
|
baseLink string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *SimpleDocument) ResolveLink(link string, likeType markup.LinkType) string {
|
||||||
|
return r.ctx.ResolveLinkRelative(r.baseLink, "", link)
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ markup.RenderHelper = (*SimpleDocument)(nil)
|
||||||
|
|
||||||
|
func NewRenderContextSimpleDocument(ctx context.Context, baseLink string) *markup.RenderContext {
|
||||||
|
helper := &SimpleDocument{baseLink: baseLink}
|
||||||
|
rctx := markup.NewRenderContext(ctx).WithHelper(helper).WithMetas(markup.ComposeSimpleDocumentMetas())
|
||||||
|
helper.ctx = rctx
|
||||||
|
return rctx
|
||||||
|
}
|
40
models/renderhelper/simple_document_test.go
Normal file
40
models/renderhelper/simple_document_test.go
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package renderhelper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/models/unittest"
|
||||||
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestSimpleDocument(t *testing.T) {
|
||||||
|
unittest.PrepareTestEnv(t)
|
||||||
|
rctx := NewRenderContextSimpleDocument(context.Background(), "/base").WithMarkupType(markdown.MarkupName)
|
||||||
|
rendered, err := markup.RenderString(rctx, `
|
||||||
|
65f1bf27bc3bf70f64657658635e66094edbcb4d
|
||||||
|
#1
|
||||||
|
@user2
|
||||||
|
|
||||||
|
[/test](/test)
|
||||||
|
[./test](./test)
|
||||||
|
![/image](/image)
|
||||||
|
![./image](./image)
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t,
|
||||||
|
`<p>65f1bf27bc3bf70f64657658635e66094edbcb4d
|
||||||
|
#1
|
||||||
|
<a href="/base/user2" rel="nofollow">@user2</a></p>
|
||||||
|
<p><a href="/base/test" rel="nofollow">/test</a>
|
||||||
|
<a href="/base/test" rel="nofollow">./test</a>
|
||||||
|
<a href="/base/image" target="_blank" rel="nofollow noopener"><img src="/base/image" alt="/image"/></a>
|
||||||
|
<a href="/base/image" target="_blank" rel="nofollow noopener"><img src="/base/image" alt="./image"/></a></p>
|
||||||
|
`, rendered)
|
||||||
|
}
|
@ -617,10 +617,7 @@ func (repo *Repository) CanEnableEditor() bool {
|
|||||||
|
|
||||||
// DescriptionHTML does special handles to description and return HTML string.
|
// DescriptionHTML does special handles to description and return HTML string.
|
||||||
func (repo *Repository) DescriptionHTML(ctx context.Context) template.HTML {
|
func (repo *Repository) DescriptionHTML(ctx context.Context) template.HTML {
|
||||||
desc, err := markup.RenderDescriptionHTML(&markup.RenderContext{
|
desc, err := markup.RenderDescriptionHTML(markup.NewRenderContext(ctx), repo.Description)
|
||||||
Ctx: ctx,
|
|
||||||
// Don't use Metas to speedup requests
|
|
||||||
}, repo.Description)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("Failed to render description for %s (ID: %d): %v", repo.Name, repo.ID, err)
|
log.Error("Failed to render description for %s (ID: %d): %v", repo.Name, repo.ID, err)
|
||||||
return template.HTML(markup.SanitizeDescription(repo.Description))
|
return template.HTML(markup.SanitizeDescription(repo.Description))
|
||||||
|
@ -11,7 +11,6 @@ import (
|
|||||||
"code.gitea.io/gitea/models/unit"
|
"code.gitea.io/gitea/models/unit"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
"code.gitea.io/gitea/modules/container"
|
"code.gitea.io/gitea/modules/container"
|
||||||
api "code.gitea.io/gitea/modules/structs"
|
|
||||||
|
|
||||||
"xorm.io/builder"
|
"xorm.io/builder"
|
||||||
)
|
)
|
||||||
@ -146,57 +145,6 @@ func GetRepoAssignees(ctx context.Context, repo *Repository) (_ []*user_model.Us
|
|||||||
return users, nil
|
return users, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetReviewers get all users can be requested to review:
|
|
||||||
// * for private repositories this returns all users that have read access or higher to the repository.
|
|
||||||
// * for public repositories this returns all users that have read access or higher to the repository,
|
|
||||||
// all repo watchers and all organization members.
|
|
||||||
// TODO: may be we should have a busy choice for users to block review request to them.
|
|
||||||
func GetReviewers(ctx context.Context, repo *Repository, doerID, posterID int64) ([]*user_model.User, error) {
|
|
||||||
// Get the owner of the repository - this often already pre-cached and if so saves complexity for the following queries
|
|
||||||
if err := repo.LoadOwner(ctx); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
cond := builder.And(builder.Neq{"`user`.id": posterID}).
|
|
||||||
And(builder.Eq{"`user`.is_active": true})
|
|
||||||
|
|
||||||
if repo.IsPrivate || repo.Owner.Visibility == api.VisibleTypePrivate {
|
|
||||||
// This a private repository:
|
|
||||||
// Anyone who can read the repository is a requestable reviewer
|
|
||||||
|
|
||||||
cond = cond.And(builder.In("`user`.id",
|
|
||||||
builder.Select("user_id").From("access").Where(
|
|
||||||
builder.Eq{"repo_id": repo.ID}.
|
|
||||||
And(builder.Gte{"mode": perm.AccessModeRead}),
|
|
||||||
),
|
|
||||||
))
|
|
||||||
|
|
||||||
if repo.Owner.Type == user_model.UserTypeIndividual && repo.Owner.ID != posterID {
|
|
||||||
// as private *user* repos don't generate an entry in the `access` table,
|
|
||||||
// the owner of a private repo needs to be explicitly added.
|
|
||||||
cond = cond.Or(builder.Eq{"`user`.id": repo.Owner.ID})
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// This is a "public" repository:
|
|
||||||
// Any user that has read access, is a watcher or organization member can be requested to review
|
|
||||||
cond = cond.And(builder.And(builder.In("`user`.id",
|
|
||||||
builder.Select("user_id").From("access").
|
|
||||||
Where(builder.Eq{"repo_id": repo.ID}.
|
|
||||||
And(builder.Gte{"mode": perm.AccessModeRead})),
|
|
||||||
).Or(builder.In("`user`.id",
|
|
||||||
builder.Select("user_id").From("watch").
|
|
||||||
Where(builder.Eq{"repo_id": repo.ID}.
|
|
||||||
And(builder.In("mode", WatchModeNormal, WatchModeAuto))),
|
|
||||||
).Or(builder.In("`user`.id",
|
|
||||||
builder.Select("uid").From("org_user").
|
|
||||||
Where(builder.Eq{"org_id": repo.OwnerID}),
|
|
||||||
)))))
|
|
||||||
}
|
|
||||||
|
|
||||||
users := make([]*user_model.User, 0, 8)
|
|
||||||
return users, db.GetEngine(ctx).Where(cond).OrderBy(user_model.GetOrderByName()).Find(&users)
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetIssuePostersWithSearch returns users with limit of 30 whose username started with prefix that have authored an issue/pull request for the given repository
|
// GetIssuePostersWithSearch returns users with limit of 30 whose username started with prefix that have authored an issue/pull request for the given repository
|
||||||
// If isShowFullName is set to true, also include full name prefix search
|
// If isShowFullName is set to true, also include full name prefix search
|
||||||
func GetIssuePostersWithSearch(ctx context.Context, repo *Repository, isPull bool, search string, isShowFullName bool) ([]*user_model.User, error) {
|
func GetIssuePostersWithSearch(ctx context.Context, repo *Repository, isPull bool, search string, isShowFullName bool) ([]*user_model.User, error) {
|
||||||
|
@ -38,46 +38,3 @@ func TestRepoAssignees(t *testing.T) {
|
|||||||
assert.NotContains(t, []int64{users[0].ID, users[1].ID, users[2].ID}, 15)
|
assert.NotContains(t, []int64{users[0].ID, users[1].ID, users[2].ID}, 15)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRepoGetReviewers(t *testing.T) {
|
|
||||||
assert.NoError(t, unittest.PrepareTestDatabase())
|
|
||||||
|
|
||||||
// test public repo
|
|
||||||
repo1 := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1})
|
|
||||||
|
|
||||||
ctx := db.DefaultContext
|
|
||||||
reviewers, err := repo_model.GetReviewers(ctx, repo1, 2, 2)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
if assert.Len(t, reviewers, 3) {
|
|
||||||
assert.ElementsMatch(t, []int64{1, 4, 11}, []int64{reviewers[0].ID, reviewers[1].ID, reviewers[2].ID})
|
|
||||||
}
|
|
||||||
|
|
||||||
// should include doer if doer is not PR poster.
|
|
||||||
reviewers, err = repo_model.GetReviewers(ctx, repo1, 11, 2)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Len(t, reviewers, 3)
|
|
||||||
|
|
||||||
// should not include PR poster, if PR poster would be otherwise eligible
|
|
||||||
reviewers, err = repo_model.GetReviewers(ctx, repo1, 11, 4)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Len(t, reviewers, 2)
|
|
||||||
|
|
||||||
// test private user repo
|
|
||||||
repo2 := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 2})
|
|
||||||
|
|
||||||
reviewers, err = repo_model.GetReviewers(ctx, repo2, 2, 4)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Len(t, reviewers, 1)
|
|
||||||
assert.EqualValues(t, reviewers[0].ID, 2)
|
|
||||||
|
|
||||||
// test private org repo
|
|
||||||
repo3 := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 3})
|
|
||||||
|
|
||||||
reviewers, err = repo_model.GetReviewers(ctx, repo3, 2, 1)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Len(t, reviewers, 2)
|
|
||||||
|
|
||||||
reviewers, err = repo_model.GetReviewers(ctx, repo3, 2, 2)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Len(t, reviewers, 1)
|
|
||||||
}
|
|
||||||
|
@ -9,6 +9,7 @@ import (
|
|||||||
"code.gitea.io/gitea/models/db"
|
"code.gitea.io/gitea/models/db"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
"xorm.io/builder"
|
"xorm.io/builder"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -64,10 +65,10 @@ func BeanExists(t assert.TestingT, bean any, conditions ...any) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// AssertExistsAndLoadBean assert that a bean exists and load it from the test database
|
// AssertExistsAndLoadBean assert that a bean exists and load it from the test database
|
||||||
func AssertExistsAndLoadBean[T any](t assert.TestingT, bean T, conditions ...any) T {
|
func AssertExistsAndLoadBean[T any](t require.TestingT, bean T, conditions ...any) T {
|
||||||
exists, err := LoadBeanIfExists(bean, conditions...)
|
exists, err := LoadBeanIfExists(bean, conditions...)
|
||||||
assert.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.True(t, exists,
|
require.True(t, exists,
|
||||||
"Expected to find %+v (of type %T, with conditions %+v), but did not",
|
"Expected to find %+v (of type %T, with conditions %+v), but did not",
|
||||||
bean, bean, conditions)
|
bean, bean, conditions)
|
||||||
return bean
|
return bean
|
||||||
|
@ -152,7 +152,7 @@ func SearchUsers(ctx context.Context, opts *SearchUserOptions) (users []*User, _
|
|||||||
|
|
||||||
sessQuery := opts.toSearchQueryBase(ctx).OrderBy(opts.OrderBy.String())
|
sessQuery := opts.toSearchQueryBase(ctx).OrderBy(opts.OrderBy.String())
|
||||||
defer sessQuery.Close()
|
defer sessQuery.Close()
|
||||||
if opts.Page != 0 {
|
if opts.Page > 0 {
|
||||||
sessQuery = db.SetSessionPagination(sessQuery, opts)
|
sessQuery = db.SetSessionPagination(sessQuery, opts)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -330,7 +330,7 @@ func GetUserFollowers(ctx context.Context, u, viewer *User, listOptions db.ListO
|
|||||||
And("`user`.type=?", UserTypeIndividual).
|
And("`user`.type=?", UserTypeIndividual).
|
||||||
And(isUserVisibleToViewerCond(viewer))
|
And(isUserVisibleToViewerCond(viewer))
|
||||||
|
|
||||||
if listOptions.Page != 0 {
|
if listOptions.Page > 0 {
|
||||||
sess = db.SetSessionPagination(sess, &listOptions)
|
sess = db.SetSessionPagination(sess, &listOptions)
|
||||||
|
|
||||||
users := make([]*User, 0, listOptions.PageSize)
|
users := make([]*User, 0, listOptions.PageSize)
|
||||||
@ -352,7 +352,7 @@ func GetUserFollowing(ctx context.Context, u, viewer *User, listOptions db.ListO
|
|||||||
And("`user`.type IN (?, ?)", UserTypeIndividual, UserTypeOrganization).
|
And("`user`.type IN (?, ?)", UserTypeIndividual, UserTypeOrganization).
|
||||||
And(isUserVisibleToViewerCond(viewer))
|
And(isUserVisibleToViewerCond(viewer))
|
||||||
|
|
||||||
if listOptions.Page != 0 {
|
if listOptions.Page > 0 {
|
||||||
sess = db.SetSessionPagination(sess, &listOptions)
|
sess = db.SetSessionPagination(sess, &listOptions)
|
||||||
|
|
||||||
users := make([]*User, 0, listOptions.PageSize)
|
users := make([]*User, 0, listOptions.PageSize)
|
||||||
|
@ -7,7 +7,7 @@ import (
|
|||||||
"bytes"
|
"bytes"
|
||||||
stdcsv "encoding/csv"
|
stdcsv "encoding/csv"
|
||||||
"io"
|
"io"
|
||||||
"path/filepath"
|
"path"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
@ -53,7 +53,7 @@ func CreateReaderAndDetermineDelimiter(ctx *markup.RenderContext, rd io.Reader)
|
|||||||
func determineDelimiter(ctx *markup.RenderContext, data []byte) rune {
|
func determineDelimiter(ctx *markup.RenderContext, data []byte) rune {
|
||||||
extension := ".csv"
|
extension := ".csv"
|
||||||
if ctx != nil {
|
if ctx != nil {
|
||||||
extension = strings.ToLower(filepath.Ext(ctx.RelativePath))
|
extension = strings.ToLower(path.Ext(ctx.RenderOptions.RelativePath))
|
||||||
}
|
}
|
||||||
|
|
||||||
var delimiter rune
|
var delimiter rune
|
||||||
|
@ -5,13 +5,13 @@ package csv
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"context"
|
||||||
"encoding/csv"
|
"encoding/csv"
|
||||||
"io"
|
"io"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/git"
|
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
"code.gitea.io/gitea/modules/translation"
|
"code.gitea.io/gitea/modules/translation"
|
||||||
|
|
||||||
@ -231,10 +231,7 @@ John Doe john@doe.com This,note,had,a,lot,of,commas,to,test,delimiters`,
|
|||||||
}
|
}
|
||||||
|
|
||||||
for n, c := range cases {
|
for n, c := range cases {
|
||||||
delimiter := determineDelimiter(&markup.RenderContext{
|
delimiter := determineDelimiter(markup.NewRenderContext(context.Background()).WithRelativePath(c.filename), []byte(decodeSlashes(t, c.csv)))
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
RelativePath: c.filename,
|
|
||||||
}, []byte(decodeSlashes(t, c.csv)))
|
|
||||||
assert.EqualValues(t, c.expectedDelimiter, delimiter, "case %d: delimiter should be equal, expected '%c' got '%c'", n, c.expectedDelimiter, delimiter)
|
assert.EqualValues(t, c.expectedDelimiter, delimiter, "case %d: delimiter should be equal, expected '%c' got '%c'", n, c.expectedDelimiter, delimiter)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -44,10 +44,10 @@ func (Renderer) SanitizerRules() []setting.MarkupSanitizerRule {
|
|||||||
func (Renderer) Render(ctx *markup.RenderContext, _ io.Reader, output io.Writer) error {
|
func (Renderer) Render(ctx *markup.RenderContext, _ io.Reader, output io.Writer) error {
|
||||||
rawURL := fmt.Sprintf("%s/%s/%s/raw/%s/%s",
|
rawURL := fmt.Sprintf("%s/%s/%s/raw/%s/%s",
|
||||||
setting.AppSubURL,
|
setting.AppSubURL,
|
||||||
url.PathEscape(ctx.Metas["user"]),
|
url.PathEscape(ctx.RenderOptions.Metas["user"]),
|
||||||
url.PathEscape(ctx.Metas["repo"]),
|
url.PathEscape(ctx.RenderOptions.Metas["repo"]),
|
||||||
ctx.Metas["BranchNameSubURL"],
|
ctx.RenderOptions.Metas["BranchNameSubURL"],
|
||||||
url.PathEscape(ctx.RelativePath),
|
url.PathEscape(ctx.RenderOptions.RelativePath),
|
||||||
)
|
)
|
||||||
return ctx.RenderInternal.FormatWithSafeAttrs(output, `<div class="%s" %s="%s"></div>`, playerClassName, playerSrcAttr, rawURL)
|
return ctx.RenderInternal.FormatWithSafeAttrs(output, `<div class="%s" %s="%s"></div>`, playerClassName, playerSrcAttr, rawURL)
|
||||||
}
|
}
|
||||||
|
@ -4,10 +4,10 @@
|
|||||||
package console
|
package console
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/git"
|
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
@ -24,8 +24,7 @@ func TestRenderConsole(t *testing.T) {
|
|||||||
canRender := render.CanRender("test", strings.NewReader(k))
|
canRender := render.CanRender("test", strings.NewReader(k))
|
||||||
assert.True(t, canRender)
|
assert.True(t, canRender)
|
||||||
|
|
||||||
err := render.Render(&markup.RenderContext{Ctx: git.DefaultContext},
|
err := render.Render(markup.NewRenderContext(context.Background()), strings.NewReader(k), &buf)
|
||||||
strings.NewReader(k), &buf)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.EqualValues(t, v, buf.String())
|
assert.EqualValues(t, v, buf.String())
|
||||||
}
|
}
|
||||||
|
@ -133,10 +133,10 @@ func (r Renderer) Render(ctx *markup.RenderContext, input io.Reader, output io.W
|
|||||||
// Check if maxRows or maxSize is reached, and if true, warn.
|
// Check if maxRows or maxSize is reached, and if true, warn.
|
||||||
if (row >= maxRows && maxRows != 0) || (rd.InputOffset() >= maxSize && maxSize != 0) {
|
if (row >= maxRows && maxRows != 0) || (rd.InputOffset() >= maxSize && maxSize != 0) {
|
||||||
warn := `<table class="data-table"><tr><td>`
|
warn := `<table class="data-table"><tr><td>`
|
||||||
rawLink := ` <a href="` + ctx.Links.RawLink() + `/` + util.PathEscapeSegments(ctx.RelativePath) + `">`
|
rawLink := ` <a href="` + ctx.RenderHelper.ResolveLink(util.PathEscapeSegments(ctx.RenderOptions.RelativePath), markup.LinkTypeRaw) + `">`
|
||||||
|
|
||||||
// Try to get the user translation
|
// Try to get the user translation
|
||||||
if locale, ok := ctx.Ctx.Value(translation.ContextKey).(translation.Locale); ok {
|
if locale, ok := ctx.Value(translation.ContextKey).(translation.Locale); ok {
|
||||||
warn += locale.TrString("repo.file_too_large")
|
warn += locale.TrString("repo.file_too_large")
|
||||||
rawLink += locale.TrString("repo.file_view_raw")
|
rawLink += locale.TrString("repo.file_view_raw")
|
||||||
} else {
|
} else {
|
||||||
|
@ -4,10 +4,10 @@
|
|||||||
package markup
|
package markup
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/git"
|
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
@ -24,8 +24,7 @@ func TestRenderCSV(t *testing.T) {
|
|||||||
|
|
||||||
for k, v := range kases {
|
for k, v := range kases {
|
||||||
var buf strings.Builder
|
var buf strings.Builder
|
||||||
err := render.Render(&markup.RenderContext{Ctx: git.DefaultContext},
|
err := render.Render(markup.NewRenderContext(context.Background()), strings.NewReader(k), &buf)
|
||||||
strings.NewReader(k), &buf)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.EqualValues(t, v, buf.String())
|
assert.EqualValues(t, v, buf.String())
|
||||||
}
|
}
|
||||||
|
19
modules/markup/external/external.go
vendored
19
modules/markup/external/external.go
vendored
@ -12,7 +12,6 @@ import (
|
|||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/graceful"
|
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
"code.gitea.io/gitea/modules/process"
|
"code.gitea.io/gitea/modules/process"
|
||||||
@ -80,8 +79,8 @@ func envMark(envName string) string {
|
|||||||
func (p *Renderer) Render(ctx *markup.RenderContext, input io.Reader, output io.Writer) error {
|
func (p *Renderer) Render(ctx *markup.RenderContext, input io.Reader, output io.Writer) error {
|
||||||
var (
|
var (
|
||||||
command = strings.NewReplacer(
|
command = strings.NewReplacer(
|
||||||
envMark("GITEA_PREFIX_SRC"), ctx.Links.SrcLink(),
|
envMark("GITEA_PREFIX_SRC"), ctx.RenderHelper.ResolveLink("", markup.LinkTypeDefault),
|
||||||
envMark("GITEA_PREFIX_RAW"), ctx.Links.RawLink(),
|
envMark("GITEA_PREFIX_RAW"), ctx.RenderHelper.ResolveLink("", markup.LinkTypeRaw),
|
||||||
).Replace(p.Command)
|
).Replace(p.Command)
|
||||||
commands = strings.Fields(command)
|
commands = strings.Fields(command)
|
||||||
args = commands[1:]
|
args = commands[1:]
|
||||||
@ -113,22 +112,14 @@ func (p *Renderer) Render(ctx *markup.RenderContext, input io.Reader, output io.
|
|||||||
args = append(args, f.Name())
|
args = append(args, f.Name())
|
||||||
}
|
}
|
||||||
|
|
||||||
if ctx.Ctx == nil {
|
processCtx, _, finished := process.GetManager().AddContext(ctx, fmt.Sprintf("Render [%s] for %s", commands[0], ctx.RenderHelper.ResolveLink("", markup.LinkTypeDefault)))
|
||||||
if !setting.IsProd || setting.IsInTesting {
|
|
||||||
panic("RenderContext did not provide context")
|
|
||||||
}
|
|
||||||
log.Warn("RenderContext did not provide context, defaulting to Shutdown context")
|
|
||||||
ctx.Ctx = graceful.GetManager().ShutdownContext()
|
|
||||||
}
|
|
||||||
|
|
||||||
processCtx, _, finished := process.GetManager().AddContext(ctx.Ctx, fmt.Sprintf("Render [%s] for %s", commands[0], ctx.Links.SrcLink()))
|
|
||||||
defer finished()
|
defer finished()
|
||||||
|
|
||||||
cmd := exec.CommandContext(processCtx, commands[0], args...)
|
cmd := exec.CommandContext(processCtx, commands[0], args...)
|
||||||
cmd.Env = append(
|
cmd.Env = append(
|
||||||
os.Environ(),
|
os.Environ(),
|
||||||
"GITEA_PREFIX_SRC="+ctx.Links.SrcLink(),
|
"GITEA_PREFIX_SRC="+ctx.RenderHelper.ResolveLink("", markup.LinkTypeDefault),
|
||||||
"GITEA_PREFIX_RAW="+ctx.Links.RawLink(),
|
"GITEA_PREFIX_RAW="+ctx.RenderHelper.ResolveLink("", markup.LinkTypeRaw),
|
||||||
)
|
)
|
||||||
if !p.IsInputFile {
|
if !p.IsInputFile {
|
||||||
cmd.Stdin = input
|
cmd.Stdin = input
|
||||||
|
@ -260,7 +260,6 @@ func RenderEmoji(ctx *RenderContext, content string) (string, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func postProcess(ctx *RenderContext, procs []processor, input io.Reader, output io.Writer) error {
|
func postProcess(ctx *RenderContext, procs []processor, input io.Reader, output io.Writer) error {
|
||||||
defer ctx.Cancel()
|
|
||||||
// FIXME: don't read all content to memory
|
// FIXME: don't read all content to memory
|
||||||
rawHTML, err := io.ReadAll(input)
|
rawHTML, err := io.ReadAll(input)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -396,7 +395,7 @@ func createLink(ctx *RenderContext, href, content, class string) *html.Node {
|
|||||||
Data: atom.A.String(),
|
Data: atom.A.String(),
|
||||||
Attr: []html.Attribute{{Key: "href", Val: href}},
|
Attr: []html.Attribute{{Key: "href", Val: href}},
|
||||||
}
|
}
|
||||||
if !RenderBehaviorForTesting.DisableInternalAttributes {
|
if !RenderBehaviorForTesting.DisableAdditionalAttributes {
|
||||||
a.Attr = append(a.Attr, html.Attribute{Key: "data-markdown-generated-content"})
|
a.Attr = append(a.Attr, html.Attribute{Key: "data-markdown-generated-content"})
|
||||||
}
|
}
|
||||||
if class != "" {
|
if class != "" {
|
||||||
|
@ -38,7 +38,7 @@ func renderCodeBlock(ctx *RenderContext, node *html.Node) (urlPosStart, urlPosSt
|
|||||||
CommitID: node.Data[m[6]:m[7]],
|
CommitID: node.Data[m[6]:m[7]],
|
||||||
FilePath: node.Data[m[8]:m[9]],
|
FilePath: node.Data[m[8]:m[9]],
|
||||||
}
|
}
|
||||||
if !httplib.IsCurrentGiteaSiteURL(ctx.Ctx, opts.FullURL) {
|
if !httplib.IsCurrentGiteaSiteURL(ctx, opts.FullURL) {
|
||||||
return 0, 0, "", nil
|
return 0, 0, "", nil
|
||||||
}
|
}
|
||||||
u, err := url.Parse(opts.FilePath)
|
u, err := url.Parse(opts.FilePath)
|
||||||
@ -51,7 +51,7 @@ func renderCodeBlock(ctx *RenderContext, node *html.Node) (urlPosStart, urlPosSt
|
|||||||
lineStart, _ := strconv.Atoi(strings.TrimPrefix(lineStartStr, "L"))
|
lineStart, _ := strconv.Atoi(strings.TrimPrefix(lineStartStr, "L"))
|
||||||
lineStop, _ := strconv.Atoi(strings.TrimPrefix(lineStopStr, "L"))
|
lineStop, _ := strconv.Atoi(strings.TrimPrefix(lineStopStr, "L"))
|
||||||
opts.LineStart, opts.LineStop = lineStart, lineStop
|
opts.LineStart, opts.LineStop = lineStart, lineStop
|
||||||
h, err := DefaultProcessorHelper.RenderRepoFileCodePreview(ctx.Ctx, opts)
|
h, err := DefaultRenderHelperFuncs.RenderRepoFileCodePreview(ctx, opts)
|
||||||
return m[0], m[1], h, err
|
return m[0], m[1], h, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,7 +9,6 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/git"
|
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
|
|
||||||
@ -17,19 +16,16 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func TestRenderCodePreview(t *testing.T) {
|
func TestRenderCodePreview(t *testing.T) {
|
||||||
markup.Init(&markup.ProcessorHelper{
|
markup.Init(&markup.RenderHelperFuncs{
|
||||||
RenderRepoFileCodePreview: func(ctx context.Context, opts markup.RenderCodePreviewOptions) (template.HTML, error) {
|
RenderRepoFileCodePreview: func(ctx context.Context, options markup.RenderCodePreviewOptions) (template.HTML, error) {
|
||||||
return "<div>code preview</div>", nil
|
return "<div>code preview</div>", nil
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
buffer, err := markup.RenderString(&markup.RenderContext{
|
buffer, err := markup.RenderString(markup.NewTestRenderContext().WithMarkupType(markdown.MarkupName), input)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
MarkupType: markdown.MarkupName,
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
||||||
}
|
}
|
||||||
test("http://localhost:3000/owner/repo/src/commit/0123456789/foo/bar.md#L10-L20", "<p><div>code preview</div></p>")
|
test("http://localhost:3000/owner/repo/src/commit/0123456789/foo/bar.md#L10-L20", "<p><div>code preview</div></p>")
|
||||||
test("http://other/owner/repo/src/commit/0123456789/foo/bar.md#L10-L20", `<p><a href="http://other/owner/repo/src/commit/0123456789/foo/bar.md#L10-L20" data-markdown-generated-content="" rel="nofollow">http://other/owner/repo/src/commit/0123456789/foo/bar.md#L10-L20</a></p>`)
|
test("http://other/owner/repo/src/commit/0123456789/foo/bar.md#L10-L20", `<p><a href="http://other/owner/repo/src/commit/0123456789/foo/bar.md#L10-L20" rel="nofollow">http://other/owner/repo/src/commit/0123456789/foo/bar.md#L10-L20</a></p>`)
|
||||||
}
|
}
|
||||||
|
@ -4,13 +4,10 @@
|
|||||||
package markup
|
package markup
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io"
|
|
||||||
"slices"
|
"slices"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/base"
|
"code.gitea.io/gitea/modules/base"
|
||||||
"code.gitea.io/gitea/modules/gitrepo"
|
|
||||||
"code.gitea.io/gitea/modules/log"
|
|
||||||
"code.gitea.io/gitea/modules/util"
|
"code.gitea.io/gitea/modules/util"
|
||||||
|
|
||||||
"golang.org/x/net/html"
|
"golang.org/x/net/html"
|
||||||
@ -84,7 +81,7 @@ func anyHashPatternExtract(s string) (ret anyHashPatternResult, ok bool) {
|
|||||||
|
|
||||||
// fullHashPatternProcessor renders SHA containing URLs
|
// fullHashPatternProcessor renders SHA containing URLs
|
||||||
func fullHashPatternProcessor(ctx *RenderContext, node *html.Node) {
|
func fullHashPatternProcessor(ctx *RenderContext, node *html.Node) {
|
||||||
if ctx.Metas == nil {
|
if ctx.RenderOptions.Metas == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
nodeStop := node.NextSibling
|
nodeStop := node.NextSibling
|
||||||
@ -111,7 +108,7 @@ func fullHashPatternProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func comparePatternProcessor(ctx *RenderContext, node *html.Node) {
|
func comparePatternProcessor(ctx *RenderContext, node *html.Node) {
|
||||||
if ctx.Metas == nil {
|
if ctx.RenderOptions.Metas == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
nodeStop := node.NextSibling
|
nodeStop := node.NextSibling
|
||||||
@ -163,15 +160,12 @@ func comparePatternProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
// hashCurrentPatternProcessor renders SHA1 strings to corresponding links that
|
// hashCurrentPatternProcessor renders SHA1 strings to corresponding links that
|
||||||
// are assumed to be in the same repository.
|
// are assumed to be in the same repository.
|
||||||
func hashCurrentPatternProcessor(ctx *RenderContext, node *html.Node) {
|
func hashCurrentPatternProcessor(ctx *RenderContext, node *html.Node) {
|
||||||
if ctx.Metas == nil || ctx.Metas["user"] == "" || ctx.Metas["repo"] == "" || (ctx.Repo == nil && ctx.GitRepo == nil) {
|
if ctx.RenderOptions.Metas == nil || ctx.RenderOptions.Metas["user"] == "" || ctx.RenderOptions.Metas["repo"] == "" || ctx.RenderHelper == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
start := 0
|
start := 0
|
||||||
next := node.NextSibling
|
next := node.NextSibling
|
||||||
if ctx.ShaExistCache == nil {
|
|
||||||
ctx.ShaExistCache = make(map[string]bool)
|
|
||||||
}
|
|
||||||
for node != nil && node != next && start < len(node.Data) {
|
for node != nil && node != next && start < len(node.Data) {
|
||||||
m := globalVars().hashCurrentPattern.FindStringSubmatchIndex(node.Data[start:])
|
m := globalVars().hashCurrentPattern.FindStringSubmatchIndex(node.Data[start:])
|
||||||
if m == nil {
|
if m == nil {
|
||||||
@ -189,35 +183,12 @@ func hashCurrentPatternProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
// as used by git and github for linking and thus we have to do similar.
|
// as used by git and github for linking and thus we have to do similar.
|
||||||
// Because of this, we check to make sure that a matched hash is actually
|
// Because of this, we check to make sure that a matched hash is actually
|
||||||
// a commit in the repository before making it a link.
|
// a commit in the repository before making it a link.
|
||||||
|
if !ctx.RenderHelper.IsCommitIDExisting(hash) {
|
||||||
// check cache first
|
|
||||||
exist, inCache := ctx.ShaExistCache[hash]
|
|
||||||
if !inCache {
|
|
||||||
if ctx.GitRepo == nil {
|
|
||||||
var err error
|
|
||||||
var closer io.Closer
|
|
||||||
ctx.GitRepo, closer, err = gitrepo.RepositoryFromContextOrOpen(ctx.Ctx, ctx.Repo)
|
|
||||||
if err != nil {
|
|
||||||
log.Error("unable to open repository: %s Error: %v", gitrepo.RepoGitURL(ctx.Repo), err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ctx.AddCancel(func() {
|
|
||||||
_ = closer.Close()
|
|
||||||
ctx.GitRepo = nil
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// Don't use IsObjectExist since it doesn't support short hashs with gogit edition.
|
|
||||||
exist = ctx.GitRepo.IsReferenceExist(hash)
|
|
||||||
ctx.ShaExistCache[hash] = exist
|
|
||||||
}
|
|
||||||
|
|
||||||
if !exist {
|
|
||||||
start = m[3]
|
start = m[3]
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
link := util.URLJoin(ctx.Links.Prefix(), ctx.Metas["user"], ctx.Metas["repo"], "commit", hash)
|
link := ctx.RenderHelper.ResolveLink(util.URLJoin(ctx.RenderOptions.Metas["user"], ctx.RenderOptions.Metas["repo"], "commit", hash), LinkTypeApp)
|
||||||
replaceContent(node, m[2], m[3], createCodeLink(link, base.ShortSha(hash), "commit"))
|
replaceContent(node, m[2], m[3], createCodeLink(link, base.ShortSha(hash), "commit"))
|
||||||
start = 0
|
start = 0
|
||||||
node = node.NextSibling.NextSibling
|
node = node.NextSibling.NextSibling
|
||||||
|
@ -9,7 +9,6 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/git"
|
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
testModule "code.gitea.io/gitea/modules/test"
|
testModule "code.gitea.io/gitea/modules/test"
|
||||||
"code.gitea.io/gitea/modules/util"
|
"code.gitea.io/gitea/modules/util"
|
||||||
@ -34,8 +33,7 @@ func numericIssueLink(baseURL, class string, index int, marker string) string {
|
|||||||
|
|
||||||
// link an HTML link
|
// link an HTML link
|
||||||
func link(href, class, contents string) string {
|
func link(href, class, contents string) string {
|
||||||
extra := ` data-markdown-generated-content=""`
|
extra := util.Iif(class != "", ` class="`+class+`"`, "")
|
||||||
extra += util.Iif(class != "", ` class="`+class+`"`, "")
|
|
||||||
return fmt.Sprintf(`<a href="%s"%s>%s</a>`, href, extra, contents)
|
return fmt.Sprintf(`<a href="%s"%s>%s</a>`, href, extra, contents)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -69,22 +67,11 @@ var localMetas = map[string]string{
|
|||||||
"markupAllowShortIssuePattern": "true",
|
"markupAllowShortIssuePattern": "true",
|
||||||
}
|
}
|
||||||
|
|
||||||
var localWikiMetas = map[string]string{
|
|
||||||
"user": "test-owner",
|
|
||||||
"repo": "test-repo",
|
|
||||||
"markupContentMode": "wiki",
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestRender_IssueIndexPattern(t *testing.T) {
|
func TestRender_IssueIndexPattern(t *testing.T) {
|
||||||
// numeric: render inputs without valid mentions
|
// numeric: render inputs without valid mentions
|
||||||
test := func(s string) {
|
test := func(s string) {
|
||||||
testRenderIssueIndexPattern(t, s, s, &RenderContext{
|
testRenderIssueIndexPattern(t, s, s, NewTestRenderContext())
|
||||||
Ctx: git.DefaultContext,
|
testRenderIssueIndexPattern(t, s, s, NewTestRenderContext(numericMetas))
|
||||||
})
|
|
||||||
testRenderIssueIndexPattern(t, s, s, &RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: numericMetas,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// should not render anything when there are no mentions
|
// should not render anything when there are no mentions
|
||||||
@ -132,10 +119,7 @@ func TestRender_IssueIndexPattern2(t *testing.T) {
|
|||||||
links[i] = numericIssueLink(util.URLJoin(TestRepoURL, path), "ref-issue", index, marker)
|
links[i] = numericIssueLink(util.URLJoin(TestRepoURL, path), "ref-issue", index, marker)
|
||||||
}
|
}
|
||||||
expectedNil := fmt.Sprintf(expectedFmt, links...)
|
expectedNil := fmt.Sprintf(expectedFmt, links...)
|
||||||
testRenderIssueIndexPattern(t, s, expectedNil, &RenderContext{
|
testRenderIssueIndexPattern(t, s, expectedNil, NewTestRenderContext(TestAppURL, localMetas))
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: localMetas,
|
|
||||||
})
|
|
||||||
|
|
||||||
class := "ref-issue"
|
class := "ref-issue"
|
||||||
if isExternal {
|
if isExternal {
|
||||||
@ -146,10 +130,7 @@ func TestRender_IssueIndexPattern2(t *testing.T) {
|
|||||||
links[i] = numericIssueLink(prefix, class, index, marker)
|
links[i] = numericIssueLink(prefix, class, index, marker)
|
||||||
}
|
}
|
||||||
expectedNum := fmt.Sprintf(expectedFmt, links...)
|
expectedNum := fmt.Sprintf(expectedFmt, links...)
|
||||||
testRenderIssueIndexPattern(t, s, expectedNum, &RenderContext{
|
testRenderIssueIndexPattern(t, s, expectedNum, NewTestRenderContext(TestAppURL, numericMetas))
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: numericMetas,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// should render freestanding mentions
|
// should render freestanding mentions
|
||||||
@ -183,10 +164,7 @@ func TestRender_IssueIndexPattern3(t *testing.T) {
|
|||||||
|
|
||||||
// alphanumeric: render inputs without valid mentions
|
// alphanumeric: render inputs without valid mentions
|
||||||
test := func(s string) {
|
test := func(s string) {
|
||||||
testRenderIssueIndexPattern(t, s, s, &RenderContext{
|
testRenderIssueIndexPattern(t, s, s, NewTestRenderContext(alphanumericMetas))
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: alphanumericMetas,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
test("")
|
test("")
|
||||||
test("this is a test")
|
test("this is a test")
|
||||||
@ -216,10 +194,7 @@ func TestRender_IssueIndexPattern4(t *testing.T) {
|
|||||||
links[i] = externalIssueLink("https://someurl.com/someUser/someRepo/", "ref-issue ref-external-issue", name)
|
links[i] = externalIssueLink("https://someurl.com/someUser/someRepo/", "ref-issue ref-external-issue", name)
|
||||||
}
|
}
|
||||||
expected := fmt.Sprintf(expectedFmt, links...)
|
expected := fmt.Sprintf(expectedFmt, links...)
|
||||||
testRenderIssueIndexPattern(t, s, expected, &RenderContext{
|
testRenderIssueIndexPattern(t, s, expected, NewTestRenderContext(alphanumericMetas))
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: alphanumericMetas,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
test("OTT-1234 test", "%s test", "OTT-1234")
|
test("OTT-1234 test", "%s test", "OTT-1234")
|
||||||
test("test T-12 issue", "test %s issue", "T-12")
|
test("test T-12 issue", "test %s issue", "T-12")
|
||||||
@ -239,10 +214,7 @@ func TestRender_IssueIndexPattern5(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
expected := fmt.Sprintf(expectedFmt, links...)
|
expected := fmt.Sprintf(expectedFmt, links...)
|
||||||
testRenderIssueIndexPattern(t, s, expected, &RenderContext{
|
testRenderIssueIndexPattern(t, s, expected, NewTestRenderContext(metas))
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: metas,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
test("abc ISSUE-123 def", "abc %s def",
|
test("abc ISSUE-123 def", "abc %s def",
|
||||||
@ -263,10 +235,7 @@ func TestRender_IssueIndexPattern5(t *testing.T) {
|
|||||||
[]string{"ISSUE-123"},
|
[]string{"ISSUE-123"},
|
||||||
)
|
)
|
||||||
|
|
||||||
testRenderIssueIndexPattern(t, "will not match", "will not match", &RenderContext{
|
testRenderIssueIndexPattern(t, "will not match", "will not match", NewTestRenderContext(regexpMetas))
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: regexpMetas,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRender_IssueIndexPattern_NoShortPattern(t *testing.T) {
|
func TestRender_IssueIndexPattern_NoShortPattern(t *testing.T) {
|
||||||
@ -278,18 +247,9 @@ func TestRender_IssueIndexPattern_NoShortPattern(t *testing.T) {
|
|||||||
"style": IssueNameStyleNumeric,
|
"style": IssueNameStyleNumeric,
|
||||||
}
|
}
|
||||||
|
|
||||||
testRenderIssueIndexPattern(t, "#1", "#1", &RenderContext{
|
testRenderIssueIndexPattern(t, "#1", "#1", NewTestRenderContext(metas))
|
||||||
Ctx: git.DefaultContext,
|
testRenderIssueIndexPattern(t, "#1312", "#1312", NewTestRenderContext(metas))
|
||||||
Metas: metas,
|
testRenderIssueIndexPattern(t, "!1", "!1", NewTestRenderContext(metas))
|
||||||
})
|
|
||||||
testRenderIssueIndexPattern(t, "#1312", "#1312", &RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: metas,
|
|
||||||
})
|
|
||||||
testRenderIssueIndexPattern(t, "!1", "!1", &RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: metas,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRender_RenderIssueTitle(t *testing.T) {
|
func TestRender_RenderIssueTitle(t *testing.T) {
|
||||||
@ -300,20 +260,12 @@ func TestRender_RenderIssueTitle(t *testing.T) {
|
|||||||
"repo": "someRepo",
|
"repo": "someRepo",
|
||||||
"style": IssueNameStyleNumeric,
|
"style": IssueNameStyleNumeric,
|
||||||
}
|
}
|
||||||
actual, err := RenderIssueTitle(&RenderContext{
|
actual, err := RenderIssueTitle(NewTestRenderContext(metas), "#1")
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: metas,
|
|
||||||
}, "#1")
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "#1", actual)
|
assert.Equal(t, "#1", actual)
|
||||||
}
|
}
|
||||||
|
|
||||||
func testRenderIssueIndexPattern(t *testing.T, input, expected string, ctx *RenderContext) {
|
func testRenderIssueIndexPattern(t *testing.T, input, expected string, ctx *RenderContext) {
|
||||||
ctx.Links.AbsolutePrefix = true
|
|
||||||
if ctx.Links.Base == "" {
|
|
||||||
ctx.Links.Base = TestRepoURL
|
|
||||||
}
|
|
||||||
|
|
||||||
var buf strings.Builder
|
var buf strings.Builder
|
||||||
err := postProcess(ctx, []processor{issueIndexPatternProcessor}, strings.NewReader(input), &buf)
|
err := postProcess(ctx, []processor{issueIndexPatternProcessor}, strings.NewReader(input), &buf)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
@ -325,24 +277,12 @@ func TestRender_AutoLink(t *testing.T) {
|
|||||||
|
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
var buffer strings.Builder
|
var buffer strings.Builder
|
||||||
err := PostProcess(&RenderContext{
|
err := PostProcess(NewTestRenderContext(localMetas), strings.NewReader(input), &buffer)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: Links{
|
|
||||||
Base: TestRepoURL,
|
|
||||||
},
|
|
||||||
Metas: localMetas,
|
|
||||||
}, strings.NewReader(input), &buffer)
|
|
||||||
assert.Equal(t, err, nil)
|
assert.Equal(t, err, nil)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer.String()))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer.String()))
|
||||||
|
|
||||||
buffer.Reset()
|
buffer.Reset()
|
||||||
err = PostProcess(&RenderContext{
|
err = PostProcess(NewTestRenderContext(localMetas), strings.NewReader(input), &buffer)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: Links{
|
|
||||||
Base: TestRepoURL,
|
|
||||||
},
|
|
||||||
Metas: localWikiMetas,
|
|
||||||
}, strings.NewReader(input), &buffer)
|
|
||||||
assert.Equal(t, err, nil)
|
assert.Equal(t, err, nil)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer.String()))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer.String()))
|
||||||
}
|
}
|
||||||
@ -364,16 +304,10 @@ func TestRender_AutoLink(t *testing.T) {
|
|||||||
|
|
||||||
func TestRender_FullIssueURLs(t *testing.T) {
|
func TestRender_FullIssueURLs(t *testing.T) {
|
||||||
setting.AppURL = TestAppURL
|
setting.AppURL = TestAppURL
|
||||||
defer testModule.MockVariableValue(&RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer testModule.MockVariableValue(&RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
var result strings.Builder
|
var result strings.Builder
|
||||||
err := postProcess(&RenderContext{
|
err := postProcess(NewTestRenderContext(localMetas), []processor{fullIssuePatternProcessor}, strings.NewReader(input), &result)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: Links{
|
|
||||||
Base: TestRepoURL,
|
|
||||||
},
|
|
||||||
Metas: localMetas,
|
|
||||||
}, []processor{fullIssuePatternProcessor}, strings.NewReader(input), &result)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, expected, result.String())
|
assert.Equal(t, expected, result.String())
|
||||||
}
|
}
|
||||||
|
@ -19,7 +19,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func fullIssuePatternProcessor(ctx *RenderContext, node *html.Node) {
|
func fullIssuePatternProcessor(ctx *RenderContext, node *html.Node) {
|
||||||
if ctx.Metas == nil {
|
if ctx.RenderOptions.Metas == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
next := node.NextSibling
|
next := node.NextSibling
|
||||||
@ -36,14 +36,14 @@ func fullIssuePatternProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
link := node.Data[m[0]:m[1]]
|
link := node.Data[m[0]:m[1]]
|
||||||
if !httplib.IsCurrentGiteaSiteURL(ctx.Ctx, link) {
|
if !httplib.IsCurrentGiteaSiteURL(ctx, link) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
text := "#" + node.Data[m[2]:m[3]]
|
text := "#" + node.Data[m[2]:m[3]]
|
||||||
// if m[4] and m[5] is not -1, then link is to a comment
|
// if m[4] and m[5] is not -1, then link is to a comment
|
||||||
// indicate that in the text by appending (comment)
|
// indicate that in the text by appending (comment)
|
||||||
if m[4] != -1 && m[5] != -1 {
|
if m[4] != -1 && m[5] != -1 {
|
||||||
if locale, ok := ctx.Ctx.Value(translation.ContextKey).(translation.Locale); ok {
|
if locale, ok := ctx.Value(translation.ContextKey).(translation.Locale); ok {
|
||||||
text += " " + locale.TrString("repo.from_comment")
|
text += " " + locale.TrString("repo.from_comment")
|
||||||
} else {
|
} else {
|
||||||
text += " (comment)"
|
text += " (comment)"
|
||||||
@ -56,7 +56,7 @@ func fullIssuePatternProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
matchOrg := linkParts[len(linkParts)-4]
|
matchOrg := linkParts[len(linkParts)-4]
|
||||||
matchRepo := linkParts[len(linkParts)-3]
|
matchRepo := linkParts[len(linkParts)-3]
|
||||||
|
|
||||||
if matchOrg == ctx.Metas["user"] && matchRepo == ctx.Metas["repo"] {
|
if matchOrg == ctx.RenderOptions.Metas["user"] && matchRepo == ctx.RenderOptions.Metas["repo"] {
|
||||||
replaceContent(node, m[0], m[1], createLink(ctx, link, text, "ref-issue"))
|
replaceContent(node, m[0], m[1], createLink(ctx, link, text, "ref-issue"))
|
||||||
} else {
|
} else {
|
||||||
text = matchOrg + "/" + matchRepo + text
|
text = matchOrg + "/" + matchRepo + text
|
||||||
@ -67,14 +67,14 @@ func fullIssuePatternProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func issueIndexPatternProcessor(ctx *RenderContext, node *html.Node) {
|
func issueIndexPatternProcessor(ctx *RenderContext, node *html.Node) {
|
||||||
if ctx.Metas == nil {
|
if ctx.RenderOptions.Metas == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// crossLinkOnly: do not parse "#123", only parse "owner/repo#123"
|
// crossLinkOnly: do not parse "#123", only parse "owner/repo#123"
|
||||||
// if there is no repo in the context, then the "#123" format can't be parsed
|
// if there is no repo in the context, then the "#123" format can't be parsed
|
||||||
// old logic: crossLinkOnly := ctx.Metas["mode"] == "document" && !ctx.IsWiki
|
// old logic: crossLinkOnly := ctx.RenderOptions.Metas["mode"] == "document" && !ctx.IsWiki
|
||||||
crossLinkOnly := ctx.Metas["markupAllowShortIssuePattern"] != "true"
|
crossLinkOnly := ctx.RenderOptions.Metas["markupAllowShortIssuePattern"] != "true"
|
||||||
|
|
||||||
var (
|
var (
|
||||||
found bool
|
found bool
|
||||||
@ -84,20 +84,20 @@ func issueIndexPatternProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
next := node.NextSibling
|
next := node.NextSibling
|
||||||
|
|
||||||
for node != nil && node != next {
|
for node != nil && node != next {
|
||||||
_, hasExtTrackFormat := ctx.Metas["format"]
|
_, hasExtTrackFormat := ctx.RenderOptions.Metas["format"]
|
||||||
|
|
||||||
// Repos with external issue trackers might still need to reference local PRs
|
// Repos with external issue trackers might still need to reference local PRs
|
||||||
// We need to concern with the first one that shows up in the text, whichever it is
|
// We need to concern with the first one that shows up in the text, whichever it is
|
||||||
isNumericStyle := ctx.Metas["style"] == "" || ctx.Metas["style"] == IssueNameStyleNumeric
|
isNumericStyle := ctx.RenderOptions.Metas["style"] == "" || ctx.RenderOptions.Metas["style"] == IssueNameStyleNumeric
|
||||||
foundNumeric, refNumeric := references.FindRenderizableReferenceNumeric(node.Data, hasExtTrackFormat && !isNumericStyle, crossLinkOnly)
|
foundNumeric, refNumeric := references.FindRenderizableReferenceNumeric(node.Data, hasExtTrackFormat && !isNumericStyle, crossLinkOnly)
|
||||||
|
|
||||||
switch ctx.Metas["style"] {
|
switch ctx.RenderOptions.Metas["style"] {
|
||||||
case "", IssueNameStyleNumeric:
|
case "", IssueNameStyleNumeric:
|
||||||
found, ref = foundNumeric, refNumeric
|
found, ref = foundNumeric, refNumeric
|
||||||
case IssueNameStyleAlphanumeric:
|
case IssueNameStyleAlphanumeric:
|
||||||
found, ref = references.FindRenderizableReferenceAlphanumeric(node.Data)
|
found, ref = references.FindRenderizableReferenceAlphanumeric(node.Data)
|
||||||
case IssueNameStyleRegexp:
|
case IssueNameStyleRegexp:
|
||||||
pattern, err := regexplru.GetCompiled(ctx.Metas["regexp"])
|
pattern, err := regexplru.GetCompiled(ctx.RenderOptions.Metas["regexp"])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -121,9 +121,9 @@ func issueIndexPatternProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
var link *html.Node
|
var link *html.Node
|
||||||
reftext := node.Data[ref.RefLocation.Start:ref.RefLocation.End]
|
reftext := node.Data[ref.RefLocation.Start:ref.RefLocation.End]
|
||||||
if hasExtTrackFormat && !ref.IsPull {
|
if hasExtTrackFormat && !ref.IsPull {
|
||||||
ctx.Metas["index"] = ref.Issue
|
ctx.RenderOptions.Metas["index"] = ref.Issue
|
||||||
|
|
||||||
res, err := vars.Expand(ctx.Metas["format"], ctx.Metas)
|
res, err := vars.Expand(ctx.RenderOptions.Metas["format"], ctx.RenderOptions.Metas)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// here we could just log the error and continue the rendering
|
// here we could just log the error and continue the rendering
|
||||||
log.Error("unable to expand template vars for ref %s, err: %v", ref.Issue, err)
|
log.Error("unable to expand template vars for ref %s, err: %v", ref.Issue, err)
|
||||||
@ -136,9 +136,11 @@ func issueIndexPatternProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
// Gitea will redirect on click as appropriate.
|
// Gitea will redirect on click as appropriate.
|
||||||
issuePath := util.Iif(ref.IsPull, "pulls", "issues")
|
issuePath := util.Iif(ref.IsPull, "pulls", "issues")
|
||||||
if ref.Owner == "" {
|
if ref.Owner == "" {
|
||||||
link = createLink(ctx, util.URLJoin(ctx.Links.Prefix(), ctx.Metas["user"], ctx.Metas["repo"], issuePath, ref.Issue), reftext, "ref-issue")
|
linkHref := ctx.RenderHelper.ResolveLink(util.URLJoin(ctx.RenderOptions.Metas["user"], ctx.RenderOptions.Metas["repo"], issuePath, ref.Issue), LinkTypeApp)
|
||||||
|
link = createLink(ctx, linkHref, reftext, "ref-issue")
|
||||||
} else {
|
} else {
|
||||||
link = createLink(ctx, util.URLJoin(ctx.Links.Prefix(), ref.Owner, ref.Name, issuePath, ref.Issue), reftext, "ref-issue")
|
linkHref := ctx.RenderHelper.ResolveLink(util.URLJoin(ref.Owner, ref.Name, issuePath, ref.Issue), LinkTypeApp)
|
||||||
|
link = createLink(ctx, linkHref, reftext, "ref-issue")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -177,7 +179,8 @@ func commitCrossReferencePatternProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
reftext := ref.Owner + "/" + ref.Name + "@" + base.ShortSha(ref.CommitSha)
|
reftext := ref.Owner + "/" + ref.Name + "@" + base.ShortSha(ref.CommitSha)
|
||||||
link := createLink(ctx, util.URLJoin(ctx.Links.Prefix(), ref.Owner, ref.Name, "commit", ref.CommitSha), reftext, "commit")
|
linkHref := ctx.RenderHelper.ResolveLink(util.URLJoin(ref.Owner, ref.Name, "commit", ref.CommitSha), LinkTypeApp)
|
||||||
|
link := createLink(ctx, linkHref, reftext, "commit")
|
||||||
|
|
||||||
replaceContent(node, ref.RefLocation.Start, ref.RefLocation.End, link)
|
replaceContent(node, ref.RefLocation.Start, ref.RefLocation.End, link)
|
||||||
node = node.NextSibling.NextSibling
|
node = node.NextSibling.NextSibling
|
||||||
|
@ -6,37 +6,14 @@ package markup
|
|||||||
import (
|
import (
|
||||||
"net/url"
|
"net/url"
|
||||||
"path"
|
"path"
|
||||||
"path/filepath"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/markup/common"
|
"code.gitea.io/gitea/modules/markup/common"
|
||||||
"code.gitea.io/gitea/modules/util"
|
|
||||||
|
|
||||||
"golang.org/x/net/html"
|
"golang.org/x/net/html"
|
||||||
"golang.org/x/net/html/atom"
|
"golang.org/x/net/html/atom"
|
||||||
)
|
)
|
||||||
|
|
||||||
func ResolveLink(ctx *RenderContext, link, userContentAnchorPrefix string) (result string, resolved bool) {
|
|
||||||
isAnchorFragment := link != "" && link[0] == '#'
|
|
||||||
if !isAnchorFragment && !IsFullURLString(link) {
|
|
||||||
linkBase := ctx.Links.Base
|
|
||||||
if ctx.IsMarkupContentWiki() {
|
|
||||||
// no need to check if the link should be resolved as a wiki link or a wiki raw link
|
|
||||||
// just use wiki link here, and it will be redirected to a wiki raw link if necessary
|
|
||||||
linkBase = ctx.Links.WikiLink()
|
|
||||||
} else if ctx.Links.BranchPath != "" || ctx.Links.TreePath != "" {
|
|
||||||
// if there is no BranchPath, then the link will be something like "/owner/repo/src/{the-file-path}"
|
|
||||||
// and then this link will be handled by the "legacy-ref" code and be redirected to the default branch like "/owner/repo/src/branch/main/{the-file-path}"
|
|
||||||
linkBase = ctx.Links.SrcLink()
|
|
||||||
}
|
|
||||||
link, resolved = util.URLJoin(linkBase, link), true
|
|
||||||
}
|
|
||||||
if isAnchorFragment && userContentAnchorPrefix != "" {
|
|
||||||
link, resolved = userContentAnchorPrefix+link[1:], true
|
|
||||||
}
|
|
||||||
return link, resolved
|
|
||||||
}
|
|
||||||
|
|
||||||
func shortLinkProcessor(ctx *RenderContext, node *html.Node) {
|
func shortLinkProcessor(ctx *RenderContext, node *html.Node) {
|
||||||
next := node.NextSibling
|
next := node.NextSibling
|
||||||
for node != nil && node != next {
|
for node != nil && node != next {
|
||||||
@ -116,7 +93,7 @@ func shortLinkProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
|
|
||||||
name += tail
|
name += tail
|
||||||
image := false
|
image := false
|
||||||
ext := filepath.Ext(link)
|
ext := path.Ext(link)
|
||||||
switch ext {
|
switch ext {
|
||||||
// fast path: empty string, ignore
|
// fast path: empty string, ignore
|
||||||
case "":
|
case "":
|
||||||
@ -139,6 +116,7 @@ func shortLinkProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
if image {
|
if image {
|
||||||
link = strings.ReplaceAll(link, " ", "+")
|
link = strings.ReplaceAll(link, " ", "+")
|
||||||
} else {
|
} else {
|
||||||
|
// the hacky wiki name encoding: space to "-"
|
||||||
link = strings.ReplaceAll(link, " ", "-") // FIXME: it should support dashes in the link, eg: "the-dash-support.-"
|
link = strings.ReplaceAll(link, " ", "-") // FIXME: it should support dashes in the link, eg: "the-dash-support.-"
|
||||||
}
|
}
|
||||||
if !strings.Contains(link, "/") {
|
if !strings.Contains(link, "/") {
|
||||||
@ -146,9 +124,7 @@ func shortLinkProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if image {
|
if image {
|
||||||
if !absoluteLink {
|
link = ctx.RenderHelper.ResolveLink(link, LinkTypeMedia)
|
||||||
link = util.URLJoin(ctx.Links.ResolveMediaLink(ctx.IsMarkupContentWiki()), link)
|
|
||||||
}
|
|
||||||
title := props["title"]
|
title := props["title"]
|
||||||
if title == "" {
|
if title == "" {
|
||||||
title = props["alt"]
|
title = props["alt"]
|
||||||
@ -174,7 +150,7 @@ func shortLinkProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
childNode.Attr = childNode.Attr[:2]
|
childNode.Attr = childNode.Attr[:2]
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
link, _ = ResolveLink(ctx, link, "")
|
link = ctx.RenderHelper.ResolveLink(link, LinkTypeDefault)
|
||||||
childNode.Type = html.TextNode
|
childNode.Type = html.TextNode
|
||||||
childNode.Data = name
|
childNode.Data = name
|
||||||
}
|
}
|
||||||
|
@ -25,15 +25,16 @@ func mentionProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
loc.Start += start
|
loc.Start += start
|
||||||
loc.End += start
|
loc.End += start
|
||||||
mention := node.Data[loc.Start:loc.End]
|
mention := node.Data[loc.Start:loc.End]
|
||||||
teams, ok := ctx.Metas["teams"]
|
teams, ok := ctx.RenderOptions.Metas["teams"]
|
||||||
// FIXME: util.URLJoin may not be necessary here:
|
// FIXME: util.URLJoin may not be necessary here:
|
||||||
// - setting.AppURL is defined to have a terminal '/' so unless mention[1:]
|
// - setting.AppURL is defined to have a terminal '/' so unless mention[1:]
|
||||||
// is an AppSubURL link we can probably fallback to concatenation.
|
// is an AppSubURL link we can probably fallback to concatenation.
|
||||||
// team mention should follow @orgName/teamName style
|
// team mention should follow @orgName/teamName style
|
||||||
if ok && strings.Contains(mention, "/") {
|
if ok && strings.Contains(mention, "/") {
|
||||||
mentionOrgAndTeam := strings.Split(mention, "/")
|
mentionOrgAndTeam := strings.Split(mention, "/")
|
||||||
if mentionOrgAndTeam[0][1:] == ctx.Metas["org"] && strings.Contains(teams, ","+strings.ToLower(mentionOrgAndTeam[1])+",") {
|
if mentionOrgAndTeam[0][1:] == ctx.RenderOptions.Metas["org"] && strings.Contains(teams, ","+strings.ToLower(mentionOrgAndTeam[1])+",") {
|
||||||
replaceContent(node, loc.Start, loc.End, createLink(ctx, util.URLJoin(ctx.Links.Prefix(), "org", ctx.Metas["org"], "teams", mentionOrgAndTeam[1]), mention, "" /*mention*/))
|
link := ctx.RenderHelper.ResolveLink(util.URLJoin("org", ctx.RenderOptions.Metas["org"], "teams", mentionOrgAndTeam[1]), LinkTypeApp)
|
||||||
|
replaceContent(node, loc.Start, loc.End, createLink(ctx, link, mention, "" /*mention*/))
|
||||||
node = node.NextSibling.NextSibling
|
node = node.NextSibling.NextSibling
|
||||||
start = 0
|
start = 0
|
||||||
continue
|
continue
|
||||||
@ -43,8 +44,9 @@ func mentionProcessor(ctx *RenderContext, node *html.Node) {
|
|||||||
}
|
}
|
||||||
mentionedUsername := mention[1:]
|
mentionedUsername := mention[1:]
|
||||||
|
|
||||||
if DefaultProcessorHelper.IsUsernameMentionable != nil && DefaultProcessorHelper.IsUsernameMentionable(ctx.Ctx, mentionedUsername) {
|
if DefaultRenderHelperFuncs != nil && DefaultRenderHelperFuncs.IsUsernameMentionable(ctx, mentionedUsername) {
|
||||||
replaceContent(node, loc.Start, loc.End, createLink(ctx, util.URLJoin(ctx.Links.Prefix(), mentionedUsername), mention, "" /*mention*/))
|
link := ctx.RenderHelper.ResolveLink(mentionedUsername, LinkTypeApp)
|
||||||
|
replaceContent(node, loc.Start, loc.End, createLink(ctx, link, mention, "" /*mention*/))
|
||||||
node = node.NextSibling.NextSibling
|
node = node.NextSibling.NextSibling
|
||||||
start = 0
|
start = 0
|
||||||
} else {
|
} else {
|
||||||
|
@ -4,8 +4,6 @@
|
|||||||
package markup
|
package markup
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"code.gitea.io/gitea/modules/util"
|
|
||||||
|
|
||||||
"golang.org/x/net/html"
|
"golang.org/x/net/html"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -17,7 +15,7 @@ func visitNodeImg(ctx *RenderContext, img *html.Node) (next *html.Node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if IsNonEmptyRelativePath(attr.Val) {
|
if IsNonEmptyRelativePath(attr.Val) {
|
||||||
attr.Val = util.URLJoin(ctx.Links.ResolveMediaLink(ctx.IsMarkupContentWiki()), attr.Val)
|
attr.Val = ctx.RenderHelper.ResolveLink(attr.Val, LinkTypeMedia)
|
||||||
|
|
||||||
// By default, the "<img>" tag should also be clickable,
|
// By default, the "<img>" tag should also be clickable,
|
||||||
// because frontend use `<img>` to paste the re-scaled image into the markdown,
|
// because frontend use `<img>` to paste the re-scaled image into the markdown,
|
||||||
@ -53,7 +51,7 @@ func visitNodeVideo(ctx *RenderContext, node *html.Node) (next *html.Node) {
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if IsNonEmptyRelativePath(attr.Val) {
|
if IsNonEmptyRelativePath(attr.Val) {
|
||||||
attr.Val = util.URLJoin(ctx.Links.ResolveMediaLink(ctx.IsMarkupContentWiki()), attr.Val)
|
attr.Val = ctx.RenderHelper.ResolveLink(attr.Val, LinkTypeMedia)
|
||||||
}
|
}
|
||||||
attr.Val = camoHandleLink(attr.Val)
|
attr.Val = camoHandleLink(attr.Val)
|
||||||
node.Attr[i] = attr
|
node.Attr[i] = attr
|
||||||
|
@ -9,8 +9,6 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/emoji"
|
"code.gitea.io/gitea/modules/emoji"
|
||||||
"code.gitea.io/gitea/modules/git"
|
|
||||||
"code.gitea.io/gitea/modules/gitrepo"
|
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
@ -23,50 +21,13 @@ import (
|
|||||||
var (
|
var (
|
||||||
testRepoOwnerName = "user13"
|
testRepoOwnerName = "user13"
|
||||||
testRepoName = "repo11"
|
testRepoName = "repo11"
|
||||||
localMetas = map[string]string{
|
localMetas = map[string]string{"user": testRepoOwnerName, "repo": testRepoName}
|
||||||
"user": testRepoOwnerName,
|
|
||||||
"repo": testRepoName,
|
|
||||||
}
|
|
||||||
localWikiMetas = map[string]string{
|
|
||||||
"user": testRepoOwnerName,
|
|
||||||
"repo": testRepoName,
|
|
||||||
"markupContentMode": "wiki",
|
|
||||||
}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type mockRepo struct {
|
|
||||||
OwnerName string
|
|
||||||
RepoName string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *mockRepo) GetOwnerName() string {
|
|
||||||
return m.OwnerName
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *mockRepo) GetName() string {
|
|
||||||
return m.RepoName
|
|
||||||
}
|
|
||||||
|
|
||||||
func newMockRepo(ownerName, repoName string) gitrepo.Repository {
|
|
||||||
return &mockRepo{
|
|
||||||
OwnerName: ownerName,
|
|
||||||
RepoName: repoName,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestRender_Commits(t *testing.T) {
|
func TestRender_Commits(t *testing.T) {
|
||||||
setting.AppURL = markup.TestAppURL
|
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
buffer, err := markup.RenderString(&markup.RenderContext{
|
rctx := markup.NewTestRenderContext(markup.TestAppURL, localMetas).WithRelativePath("a.md")
|
||||||
Ctx: git.DefaultContext,
|
buffer, err := markup.RenderString(rctx, input)
|
||||||
RelativePath: ".md",
|
|
||||||
Links: markup.Links{
|
|
||||||
AbsolutePrefix: true,
|
|
||||||
Base: markup.TestRepoURL,
|
|
||||||
},
|
|
||||||
Repo: newMockRepo(testRepoOwnerName, testRepoName),
|
|
||||||
Metas: localMetas,
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
||||||
}
|
}
|
||||||
@ -109,18 +70,10 @@ func TestRender_Commits(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestRender_CrossReferences(t *testing.T) {
|
func TestRender_CrossReferences(t *testing.T) {
|
||||||
setting.AppURL = markup.TestAppURL
|
defer testModule.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
defer testModule.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
buffer, err := markup.RenderString(&markup.RenderContext{
|
rctx := markup.NewTestRenderContext(markup.TestAppURL, localMetas).WithRelativePath("a.md")
|
||||||
Ctx: git.DefaultContext,
|
buffer, err := markup.RenderString(rctx, input)
|
||||||
RelativePath: "a.md",
|
|
||||||
Links: markup.Links{
|
|
||||||
AbsolutePrefix: true,
|
|
||||||
Base: setting.AppSubURL,
|
|
||||||
},
|
|
||||||
Metas: localMetas,
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
||||||
}
|
}
|
||||||
@ -152,15 +105,9 @@ func TestRender_CrossReferences(t *testing.T) {
|
|||||||
|
|
||||||
func TestRender_links(t *testing.T) {
|
func TestRender_links(t *testing.T) {
|
||||||
setting.AppURL = markup.TestAppURL
|
setting.AppURL = markup.TestAppURL
|
||||||
defer testModule.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer testModule.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
buffer, err := markup.RenderString(&markup.RenderContext{
|
buffer, err := markup.RenderString(markup.NewTestRenderContext().WithRelativePath("a.md"), input)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
RelativePath: "a.md",
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: markup.TestRepoURL,
|
|
||||||
},
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
||||||
}
|
}
|
||||||
@ -263,15 +210,9 @@ func TestRender_links(t *testing.T) {
|
|||||||
|
|
||||||
func TestRender_email(t *testing.T) {
|
func TestRender_email(t *testing.T) {
|
||||||
setting.AppURL = markup.TestAppURL
|
setting.AppURL = markup.TestAppURL
|
||||||
defer testModule.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer testModule.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
res, err := markup.RenderString(&markup.RenderContext{
|
res, err := markup.RenderString(markup.NewTestRenderContext().WithRelativePath("a.md"), input)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
RelativePath: "a.md",
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: markup.TestRepoURL,
|
|
||||||
},
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(res))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(res))
|
||||||
}
|
}
|
||||||
@ -338,13 +279,7 @@ func TestRender_emoji(t *testing.T) {
|
|||||||
|
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
expected = strings.ReplaceAll(expected, "&", "&")
|
expected = strings.ReplaceAll(expected, "&", "&")
|
||||||
buffer, err := markup.RenderString(&markup.RenderContext{
|
buffer, err := markup.RenderString(markup.NewTestRenderContext().WithRelativePath("a.md"), input)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
RelativePath: "a.md",
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: markup.TestRepoURL,
|
|
||||||
},
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
||||||
}
|
}
|
||||||
@ -403,217 +338,133 @@ func TestRender_ShortLinks(t *testing.T) {
|
|||||||
setting.AppURL = markup.TestAppURL
|
setting.AppURL = markup.TestAppURL
|
||||||
tree := util.URLJoin(markup.TestRepoURL, "src", "master")
|
tree := util.URLJoin(markup.TestRepoURL, "src", "master")
|
||||||
|
|
||||||
test := func(input, expected, expectedWiki string) {
|
test := func(input, expected string) {
|
||||||
buffer, err := markdown.RenderString(&markup.RenderContext{
|
buffer, err := markdown.RenderString(markup.NewTestRenderContext(tree), input)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: markup.TestRepoURL,
|
|
||||||
BranchPath: "master",
|
|
||||||
},
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(buffer)))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(buffer)))
|
||||||
buffer, err = markdown.RenderString(&markup.RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: markup.TestRepoURL,
|
|
||||||
},
|
|
||||||
Metas: localWikiMetas,
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Equal(t, strings.TrimSpace(expectedWiki), strings.TrimSpace(string(buffer)))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
mediatree := util.URLJoin(markup.TestRepoURL, "media", "master")
|
|
||||||
url := util.URLJoin(tree, "Link")
|
url := util.URLJoin(tree, "Link")
|
||||||
otherURL := util.URLJoin(tree, "Other-Link")
|
otherURL := util.URLJoin(tree, "Other-Link")
|
||||||
encodedURL := util.URLJoin(tree, "Link%3F")
|
encodedURL := util.URLJoin(tree, "Link%3F")
|
||||||
imgurl := util.URLJoin(mediatree, "Link.jpg")
|
imgurl := util.URLJoin(tree, "Link.jpg")
|
||||||
otherImgurl := util.URLJoin(mediatree, "Link+Other.jpg")
|
otherImgurl := util.URLJoin(tree, "Link+Other.jpg")
|
||||||
encodedImgurl := util.URLJoin(mediatree, "Link+%23.jpg")
|
encodedImgurl := util.URLJoin(tree, "Link+%23.jpg")
|
||||||
notencodedImgurl := util.URLJoin(mediatree, "some", "path", "Link+#.jpg")
|
notencodedImgurl := util.URLJoin(tree, "some", "path", "Link+#.jpg")
|
||||||
urlWiki := util.URLJoin(markup.TestRepoURL, "wiki", "Link")
|
|
||||||
otherURLWiki := util.URLJoin(markup.TestRepoURL, "wiki", "Other-Link")
|
|
||||||
encodedURLWiki := util.URLJoin(markup.TestRepoURL, "wiki", "Link%3F")
|
|
||||||
imgurlWiki := util.URLJoin(markup.TestRepoURL, "wiki", "raw", "Link.jpg")
|
|
||||||
otherImgurlWiki := util.URLJoin(markup.TestRepoURL, "wiki", "raw", "Link+Other.jpg")
|
|
||||||
encodedImgurlWiki := util.URLJoin(markup.TestRepoURL, "wiki", "raw", "Link+%23.jpg")
|
|
||||||
notencodedImgurlWiki := util.URLJoin(markup.TestRepoURL, "wiki", "raw", "some", "path", "Link+#.jpg")
|
|
||||||
renderableFileURL := util.URLJoin(tree, "markdown_file.md")
|
renderableFileURL := util.URLJoin(tree, "markdown_file.md")
|
||||||
renderableFileURLWiki := util.URLJoin(markup.TestRepoURL, "wiki", "markdown_file.md")
|
|
||||||
unrenderableFileURL := util.URLJoin(tree, "file.zip")
|
unrenderableFileURL := util.URLJoin(tree, "file.zip")
|
||||||
unrenderableFileURLWiki := util.URLJoin(markup.TestRepoURL, "wiki", "file.zip")
|
|
||||||
favicon := "http://google.com/favicon.ico"
|
favicon := "http://google.com/favicon.ico"
|
||||||
|
|
||||||
test(
|
test(
|
||||||
"[[Link]]",
|
"[[Link]]",
|
||||||
`<p><a href="`+url+`" rel="nofollow">Link</a></p>`,
|
`<p><a href="`+url+`" rel="nofollow">Link</a></p>`,
|
||||||
`<p><a href="`+urlWiki+`" rel="nofollow">Link</a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Link.-]]",
|
"[[Link.-]]",
|
||||||
`<p><a href="http://localhost:3000/test-owner/test-repo/src/master/Link.-" rel="nofollow">Link.-</a></p>`,
|
`<p><a href="http://localhost:3000/test-owner/test-repo/src/master/Link.-" rel="nofollow">Link.-</a></p>`,
|
||||||
`<p><a href="http://localhost:3000/test-owner/test-repo/wiki/Link.-" rel="nofollow">Link.-</a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Link.jpg]]",
|
"[[Link.jpg]]",
|
||||||
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="Link.jpg" alt="Link.jpg"/></a></p>`,
|
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="Link.jpg" alt="Link.jpg"/></a></p>`,
|
||||||
`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" title="Link.jpg" alt="Link.jpg"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[["+favicon+"]]",
|
"[["+favicon+"]]",
|
||||||
`<p><a href="`+favicon+`" rel="nofollow"><img src="`+favicon+`" title="favicon.ico" alt="`+favicon+`"/></a></p>`,
|
`<p><a href="`+favicon+`" rel="nofollow"><img src="`+favicon+`" title="favicon.ico" alt="`+favicon+`"/></a></p>`,
|
||||||
`<p><a href="`+favicon+`" rel="nofollow"><img src="`+favicon+`" title="favicon.ico" alt="`+favicon+`"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Name|Link]]",
|
"[[Name|Link]]",
|
||||||
`<p><a href="`+url+`" rel="nofollow">Name</a></p>`,
|
`<p><a href="`+url+`" rel="nofollow">Name</a></p>`,
|
||||||
`<p><a href="`+urlWiki+`" rel="nofollow">Name</a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Name|Link.jpg]]",
|
"[[Name|Link.jpg]]",
|
||||||
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="Name" alt="Name"/></a></p>`,
|
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="Name" alt="Name"/></a></p>`,
|
||||||
`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" title="Name" alt="Name"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Name|Link.jpg|alt=AltName]]",
|
"[[Name|Link.jpg|alt=AltName]]",
|
||||||
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="AltName" alt="AltName"/></a></p>`,
|
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="AltName" alt="AltName"/></a></p>`,
|
||||||
`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" title="AltName" alt="AltName"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Name|Link.jpg|title=Title]]",
|
"[[Name|Link.jpg|title=Title]]",
|
||||||
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="Title" alt="Title"/></a></p>`,
|
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="Title" alt="Title"/></a></p>`,
|
||||||
`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" title="Title" alt="Title"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Name|Link.jpg|alt=AltName|title=Title]]",
|
"[[Name|Link.jpg|alt=AltName|title=Title]]",
|
||||||
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="Title" alt="AltName"/></a></p>`,
|
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="Title" alt="AltName"/></a></p>`,
|
||||||
`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" title="Title" alt="AltName"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Name|Link.jpg|alt=\"AltName\"|title='Title']]",
|
"[[Name|Link.jpg|alt=\"AltName\"|title='Title']]",
|
||||||
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="Title" alt="AltName"/></a></p>`,
|
`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" title="Title" alt="AltName"/></a></p>`,
|
||||||
`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" title="Title" alt="AltName"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Name|Link Other.jpg|alt=\"AltName\"|title='Title']]",
|
"[[Name|Link Other.jpg|alt=\"AltName\"|title='Title']]",
|
||||||
`<p><a href="`+otherImgurl+`" rel="nofollow"><img src="`+otherImgurl+`" title="Title" alt="AltName"/></a></p>`,
|
`<p><a href="`+otherImgurl+`" rel="nofollow"><img src="`+otherImgurl+`" title="Title" alt="AltName"/></a></p>`,
|
||||||
`<p><a href="`+otherImgurlWiki+`" rel="nofollow"><img src="`+otherImgurlWiki+`" title="Title" alt="AltName"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Link]] [[Other Link]]",
|
"[[Link]] [[Other Link]]",
|
||||||
`<p><a href="`+url+`" rel="nofollow">Link</a> <a href="`+otherURL+`" rel="nofollow">Other Link</a></p>`,
|
`<p><a href="`+url+`" rel="nofollow">Link</a> <a href="`+otherURL+`" rel="nofollow">Other Link</a></p>`,
|
||||||
`<p><a href="`+urlWiki+`" rel="nofollow">Link</a> <a href="`+otherURLWiki+`" rel="nofollow">Other Link</a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Link?]]",
|
"[[Link?]]",
|
||||||
`<p><a href="`+encodedURL+`" rel="nofollow">Link?</a></p>`,
|
`<p><a href="`+encodedURL+`" rel="nofollow">Link?</a></p>`,
|
||||||
`<p><a href="`+encodedURLWiki+`" rel="nofollow">Link?</a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Link]] [[Other Link]] [[Link?]]",
|
"[[Link]] [[Other Link]] [[Link?]]",
|
||||||
`<p><a href="`+url+`" rel="nofollow">Link</a> <a href="`+otherURL+`" rel="nofollow">Other Link</a> <a href="`+encodedURL+`" rel="nofollow">Link?</a></p>`,
|
`<p><a href="`+url+`" rel="nofollow">Link</a> <a href="`+otherURL+`" rel="nofollow">Other Link</a> <a href="`+encodedURL+`" rel="nofollow">Link?</a></p>`,
|
||||||
`<p><a href="`+urlWiki+`" rel="nofollow">Link</a> <a href="`+otherURLWiki+`" rel="nofollow">Other Link</a> <a href="`+encodedURLWiki+`" rel="nofollow">Link?</a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[markdown_file.md]]",
|
"[[markdown_file.md]]",
|
||||||
`<p><a href="`+renderableFileURL+`" rel="nofollow">markdown_file.md</a></p>`,
|
`<p><a href="`+renderableFileURL+`" rel="nofollow">markdown_file.md</a></p>`,
|
||||||
`<p><a href="`+renderableFileURLWiki+`" rel="nofollow">markdown_file.md</a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[file.zip]]",
|
"[[file.zip]]",
|
||||||
`<p><a href="`+unrenderableFileURL+`" rel="nofollow">file.zip</a></p>`,
|
`<p><a href="`+unrenderableFileURL+`" rel="nofollow">file.zip</a></p>`,
|
||||||
`<p><a href="`+unrenderableFileURLWiki+`" rel="nofollow">file.zip</a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Link #.jpg]]",
|
"[[Link #.jpg]]",
|
||||||
`<p><a href="`+encodedImgurl+`" rel="nofollow"><img src="`+encodedImgurl+`" title="Link #.jpg" alt="Link #.jpg"/></a></p>`,
|
`<p><a href="`+encodedImgurl+`" rel="nofollow"><img src="`+encodedImgurl+`" title="Link #.jpg" alt="Link #.jpg"/></a></p>`,
|
||||||
`<p><a href="`+encodedImgurlWiki+`" rel="nofollow"><img src="`+encodedImgurlWiki+`" title="Link #.jpg" alt="Link #.jpg"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[Name|Link #.jpg|alt=\"AltName\"|title='Title']]",
|
"[[Name|Link #.jpg|alt=\"AltName\"|title='Title']]",
|
||||||
`<p><a href="`+encodedImgurl+`" rel="nofollow"><img src="`+encodedImgurl+`" title="Title" alt="AltName"/></a></p>`,
|
`<p><a href="`+encodedImgurl+`" rel="nofollow"><img src="`+encodedImgurl+`" title="Title" alt="AltName"/></a></p>`,
|
||||||
`<p><a href="`+encodedImgurlWiki+`" rel="nofollow"><img src="`+encodedImgurlWiki+`" title="Title" alt="AltName"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"[[some/path/Link #.jpg]]",
|
"[[some/path/Link #.jpg]]",
|
||||||
`<p><a href="`+notencodedImgurl+`" rel="nofollow"><img src="`+notencodedImgurl+`" title="Link #.jpg" alt="some/path/Link #.jpg"/></a></p>`,
|
`<p><a href="`+notencodedImgurl+`" rel="nofollow"><img src="`+notencodedImgurl+`" title="Link #.jpg" alt="some/path/Link #.jpg"/></a></p>`,
|
||||||
`<p><a href="`+notencodedImgurlWiki+`" rel="nofollow"><img src="`+notencodedImgurlWiki+`" title="Link #.jpg" alt="some/path/Link #.jpg"/></a></p>`)
|
)
|
||||||
test(
|
test(
|
||||||
"<p><a href=\"https://example.org\">[[foobar]]</a></p>",
|
"<p><a href=\"https://example.org\">[[foobar]]</a></p>",
|
||||||
`<p><a href="https://example.org" rel="nofollow">[[foobar]]</a></p>`,
|
`<p><a href="https://example.org" rel="nofollow">[[foobar]]</a></p>`,
|
||||||
`<p><a href="https://example.org" rel="nofollow">[[foobar]]</a></p>`)
|
)
|
||||||
}
|
|
||||||
|
|
||||||
func TestRender_RelativeMedias(t *testing.T) {
|
|
||||||
render := func(input string, isWiki bool, links markup.Links) string {
|
|
||||||
buffer, err := markdown.RenderString(&markup.RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: links,
|
|
||||||
Metas: util.Iif(isWiki, localWikiMetas, localMetas),
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
return strings.TrimSpace(string(buffer))
|
|
||||||
}
|
|
||||||
|
|
||||||
out := render(`<img src="LINK">`, false, markup.Links{Base: "/test-owner/test-repo"})
|
|
||||||
assert.Equal(t, `<a href="/test-owner/test-repo/LINK" target="_blank" rel="nofollow noopener"><img src="/test-owner/test-repo/LINK"/></a>`, out)
|
|
||||||
|
|
||||||
out = render(`<img src="LINK">`, true, markup.Links{Base: "/test-owner/test-repo"})
|
|
||||||
assert.Equal(t, `<a href="/test-owner/test-repo/wiki/raw/LINK" target="_blank" rel="nofollow noopener"><img src="/test-owner/test-repo/wiki/raw/LINK"/></a>`, out)
|
|
||||||
|
|
||||||
out = render(`<img src="LINK">`, false, markup.Links{Base: "/test-owner/test-repo", BranchPath: "test-branch"})
|
|
||||||
assert.Equal(t, `<a href="/test-owner/test-repo/media/test-branch/LINK" target="_blank" rel="nofollow noopener"><img src="/test-owner/test-repo/media/test-branch/LINK"/></a>`, out)
|
|
||||||
|
|
||||||
out = render(`<img src="LINK">`, true, markup.Links{Base: "/test-owner/test-repo", BranchPath: "test-branch"})
|
|
||||||
assert.Equal(t, `<a href="/test-owner/test-repo/wiki/raw/LINK" target="_blank" rel="nofollow noopener"><img src="/test-owner/test-repo/wiki/raw/LINK"/></a>`, out)
|
|
||||||
|
|
||||||
out = render(`<img src="/LINK">`, true, markup.Links{Base: "/test-owner/test-repo", BranchPath: "test-branch"})
|
|
||||||
assert.Equal(t, `<img src="/LINK"/>`, out)
|
|
||||||
|
|
||||||
out = render(`<video src="LINK">`, false, markup.Links{Base: "/test-owner/test-repo"})
|
|
||||||
assert.Equal(t, `<video src="/test-owner/test-repo/LINK"></video>`, out)
|
|
||||||
|
|
||||||
out = render(`<video src="LINK">`, true, markup.Links{Base: "/test-owner/test-repo"})
|
|
||||||
assert.Equal(t, `<video src="/test-owner/test-repo/wiki/raw/LINK"></video>`, out)
|
|
||||||
|
|
||||||
out = render(`<video src="/LINK">`, false, markup.Links{Base: "/test-owner/test-repo"})
|
|
||||||
assert.Equal(t, `<video src="/LINK"></video>`, out)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func Test_ParseClusterFuzz(t *testing.T) {
|
func Test_ParseClusterFuzz(t *testing.T) {
|
||||||
setting.AppURL = markup.TestAppURL
|
setting.AppURL = markup.TestAppURL
|
||||||
|
|
||||||
localMetas := map[string]string{
|
localMetas := map[string]string{"user": "go-gitea", "repo": "gitea"}
|
||||||
"user": "go-gitea",
|
|
||||||
"repo": "gitea",
|
|
||||||
}
|
|
||||||
|
|
||||||
data := "<A><maTH><tr><MN><bodY ÿ><temPlate></template><tH><tr></A><tH><d<bodY "
|
data := "<A><maTH><tr><MN><bodY ÿ><temPlate></template><tH><tr></A><tH><d<bodY "
|
||||||
|
|
||||||
var res strings.Builder
|
var res strings.Builder
|
||||||
err := markup.PostProcess(&markup.RenderContext{
|
err := markup.PostProcess(markup.NewTestRenderContext(localMetas), strings.NewReader(data), &res)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "https://example.com",
|
|
||||||
},
|
|
||||||
Metas: localMetas,
|
|
||||||
}, strings.NewReader(data), &res)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.NotContains(t, res.String(), "<html")
|
assert.NotContains(t, res.String(), "<html")
|
||||||
|
|
||||||
data = "<!DOCTYPE html>\n<A><maTH><tr><MN><bodY ÿ><temPlate></template><tH><tr></A><tH><d<bodY "
|
data = "<!DOCTYPE html>\n<A><maTH><tr><MN><bodY ÿ><temPlate></template><tH><tr></A><tH><d<bodY "
|
||||||
|
|
||||||
res.Reset()
|
res.Reset()
|
||||||
err = markup.PostProcess(&markup.RenderContext{
|
err = markup.PostProcess(markup.NewTestRenderContext(localMetas), strings.NewReader(data), &res)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "https://example.com",
|
|
||||||
},
|
|
||||||
Metas: localMetas,
|
|
||||||
}, strings.NewReader(data), &res)
|
|
||||||
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.NotContains(t, res.String(), "<html")
|
assert.NotContains(t, res.String(), "<html")
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestPostProcess_RenderDocument(t *testing.T) {
|
func TestPostProcess_RenderDocument(t *testing.T) {
|
||||||
setting.AppURL = markup.TestAppURL
|
|
||||||
setting.StaticURLPrefix = markup.TestAppURL // can't run standalone
|
setting.StaticURLPrefix = markup.TestAppURL // can't run standalone
|
||||||
defer testModule.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer testModule.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
|
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
var res strings.Builder
|
var res strings.Builder
|
||||||
err := markup.PostProcess(&markup.RenderContext{
|
err := markup.PostProcess(markup.NewTestRenderContext(markup.TestAppURL, map[string]string{"user": "go-gitea", "repo": "gitea"}), strings.NewReader(input), &res)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
AbsolutePrefix: true,
|
|
||||||
Base: "https://example.com",
|
|
||||||
},
|
|
||||||
Metas: map[string]string{"user": "go-gitea", "repo": "gitea"},
|
|
||||||
}, strings.NewReader(input), &res)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(res.String()))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(res.String()))
|
||||||
}
|
}
|
||||||
@ -650,10 +501,7 @@ func TestIssue16020(t *testing.T) {
|
|||||||
data := `<img src=""/>`
|
data := `<img src=""/>`
|
||||||
|
|
||||||
var res strings.Builder
|
var res strings.Builder
|
||||||
err := markup.PostProcess(&markup.RenderContext{
|
err := markup.PostProcess(markup.NewTestRenderContext(localMetas), strings.NewReader(data), &res)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: localMetas,
|
|
||||||
}, strings.NewReader(data), &res)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, data, res.String())
|
assert.Equal(t, data, res.String())
|
||||||
}
|
}
|
||||||
@ -666,29 +514,15 @@ func BenchmarkEmojiPostprocess(b *testing.B) {
|
|||||||
b.ResetTimer()
|
b.ResetTimer()
|
||||||
for i := 0; i < b.N; i++ {
|
for i := 0; i < b.N; i++ {
|
||||||
var res strings.Builder
|
var res strings.Builder
|
||||||
err := markup.PostProcess(&markup.RenderContext{
|
err := markup.PostProcess(markup.NewTestRenderContext(localMetas), strings.NewReader(data), &res)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: localMetas,
|
|
||||||
}, strings.NewReader(data), &res)
|
|
||||||
assert.NoError(b, err)
|
assert.NoError(b, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestFuzz(t *testing.T) {
|
func TestFuzz(t *testing.T) {
|
||||||
s := "t/l/issues/8#/../../a"
|
s := "t/l/issues/8#/../../a"
|
||||||
renderContext := markup.RenderContext{
|
renderContext := markup.NewTestRenderContext()
|
||||||
Ctx: git.DefaultContext,
|
err := markup.PostProcess(renderContext, strings.NewReader(s), io.Discard)
|
||||||
Links: markup.Links{
|
|
||||||
Base: "https://example.com/go-gitea/gitea",
|
|
||||||
},
|
|
||||||
Metas: map[string]string{
|
|
||||||
"user": "go-gitea",
|
|
||||||
"repo": "gitea",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
err := markup.PostProcess(&renderContext, strings.NewReader(s), io.Discard)
|
|
||||||
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -696,10 +530,7 @@ func TestIssue18471(t *testing.T) {
|
|||||||
data := `http://domain/org/repo/compare/783b039...da951ce`
|
data := `http://domain/org/repo/compare/783b039...da951ce`
|
||||||
|
|
||||||
var res strings.Builder
|
var res strings.Builder
|
||||||
err := markup.PostProcess(&markup.RenderContext{
|
err := markup.PostProcess(markup.NewTestRenderContext(localMetas), strings.NewReader(data), &res)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Metas: localMetas,
|
|
||||||
}, strings.NewReader(data), &res)
|
|
||||||
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, `<a href="http://domain/org/repo/compare/783b039...da951ce" class="compare"><code class="nohighlight">783b039...da951ce</code></a>`, res.String())
|
assert.Equal(t, `<a href="http://domain/org/repo/compare/783b039...da951ce" class="compare"><code class="nohighlight">783b039...da951ce</code></a>`, res.String())
|
||||||
|
@ -4,11 +4,15 @@
|
|||||||
package markup_test
|
package markup_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"code.gitea.io/gitea/models/unittest"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
"code.gitea.io/gitea/modules/setting"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestMain(m *testing.M) {
|
func TestMain(m *testing.M) {
|
||||||
unittest.MainTest(m)
|
setting.IsInTesting = true
|
||||||
|
markup.RenderBehaviorForTesting.DisableAdditionalAttributes = true
|
||||||
|
os.Exit(m.Run())
|
||||||
}
|
}
|
||||||
|
@ -37,8 +37,8 @@ func NewASTTransformer(renderInternal *internal.RenderInternal) *ASTTransformer
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (g *ASTTransformer) applyElementDir(n ast.Node) {
|
func (g *ASTTransformer) applyElementDir(n ast.Node) {
|
||||||
if markup.DefaultProcessorHelper.ElementDir != "" {
|
if !markup.RenderBehaviorForTesting.DisableAdditionalAttributes {
|
||||||
n.SetAttributeString("dir", []byte(markup.DefaultProcessorHelper.ElementDir))
|
n.SetAttributeString("dir", "auto")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -79,7 +79,7 @@ func (g *ASTTransformer) Transform(node *ast.Document, reader text.Reader, pc pa
|
|||||||
// TODO: this was a quite unclear part, old code: `if metas["mode"] != "document" { use comment link break setting }`
|
// TODO: this was a quite unclear part, old code: `if metas["mode"] != "document" { use comment link break setting }`
|
||||||
// many places render non-comment contents with no mode=document, then these contents also use comment's hard line break setting
|
// many places render non-comment contents with no mode=document, then these contents also use comment's hard line break setting
|
||||||
// especially in many tests.
|
// especially in many tests.
|
||||||
markdownLineBreakStyle := ctx.Metas["markdownLineBreakStyle"]
|
markdownLineBreakStyle := ctx.RenderOptions.Metas["markdownLineBreakStyle"]
|
||||||
if markup.RenderBehaviorForTesting.ForceHardLineBreak {
|
if markup.RenderBehaviorForTesting.ForceHardLineBreak {
|
||||||
v.SetHardLineBreak(true)
|
v.SetHardLineBreak(true)
|
||||||
} else if markdownLineBreakStyle == "comment" {
|
} else if markdownLineBreakStyle == "comment" {
|
||||||
|
@ -4,18 +4,15 @@
|
|||||||
package markdown
|
package markdown
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"code.gitea.io/gitea/models/unittest"
|
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
|
"code.gitea.io/gitea/modules/setting"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestMain(m *testing.M) {
|
func TestMain(m *testing.M) {
|
||||||
markup.Init(&markup.ProcessorHelper{
|
setting.IsInTesting = true
|
||||||
IsUsernameMentionable: func(ctx context.Context, username string) bool {
|
markup.RenderBehaviorForTesting.DisableAdditionalAttributes = true
|
||||||
return username == "r-lyeh"
|
os.Exit(m.Run())
|
||||||
},
|
|
||||||
})
|
|
||||||
unittest.MainTest(m)
|
|
||||||
}
|
}
|
||||||
|
@ -182,7 +182,7 @@ func render(ctx *markup.RenderContext, input io.Reader, output io.Writer) error
|
|||||||
bufWithMetadataLength := len(buf)
|
bufWithMetadataLength := len(buf)
|
||||||
|
|
||||||
rc := &RenderConfig{
|
rc := &RenderConfig{
|
||||||
Meta: renderMetaModeFromString(string(ctx.RenderMetaAs)),
|
Meta: markup.RenderMetaAsDetails,
|
||||||
Icon: "table",
|
Icon: "table",
|
||||||
Lang: "",
|
Lang: "",
|
||||||
}
|
}
|
||||||
@ -241,7 +241,7 @@ func (Renderer) Render(ctx *markup.RenderContext, input io.Reader, output io.Wri
|
|||||||
|
|
||||||
// Render renders Markdown to HTML with all specific handling stuff.
|
// Render renders Markdown to HTML with all specific handling stuff.
|
||||||
func Render(ctx *markup.RenderContext, input io.Reader, output io.Writer) error {
|
func Render(ctx *markup.RenderContext, input io.Reader, output io.Writer) error {
|
||||||
ctx.MarkupType = MarkupName
|
ctx.RenderOptions.MarkupType = MarkupName
|
||||||
return markup.Render(ctx, input, output)
|
return markup.Render(ctx, input, output)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,8 +9,6 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/git"
|
|
||||||
"code.gitea.io/gitea/modules/gitrepo"
|
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
@ -37,76 +35,23 @@ var localMetas = map[string]string{
|
|||||||
"repo": testRepoName,
|
"repo": testRepoName,
|
||||||
}
|
}
|
||||||
|
|
||||||
var localWikiMetas = map[string]string{
|
|
||||||
"user": testRepoOwnerName,
|
|
||||||
"repo": testRepoName,
|
|
||||||
"markupContentMode": "wiki",
|
|
||||||
}
|
|
||||||
|
|
||||||
type mockRepo struct {
|
|
||||||
OwnerName string
|
|
||||||
RepoName string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *mockRepo) GetOwnerName() string {
|
|
||||||
return m.OwnerName
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *mockRepo) GetName() string {
|
|
||||||
return m.RepoName
|
|
||||||
}
|
|
||||||
|
|
||||||
func newMockRepo(ownerName, repoName string) gitrepo.Repository {
|
|
||||||
return &mockRepo{
|
|
||||||
OwnerName: ownerName,
|
|
||||||
RepoName: repoName,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestRender_StandardLinks(t *testing.T) {
|
func TestRender_StandardLinks(t *testing.T) {
|
||||||
setting.AppURL = AppURL
|
test := func(input, expected string) {
|
||||||
|
buffer, err := markdown.RenderString(markup.NewTestRenderContext(), input)
|
||||||
test := func(input, expected, expectedWiki string) {
|
|
||||||
buffer, err := markdown.RenderString(&markup.RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: FullURL,
|
|
||||||
},
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(buffer)))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(buffer)))
|
||||||
|
|
||||||
buffer, err = markdown.RenderString(&markup.RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: FullURL,
|
|
||||||
},
|
|
||||||
Metas: localWikiMetas,
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Equal(t, strings.TrimSpace(expectedWiki), strings.TrimSpace(string(buffer)))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
googleRendered := `<p><a href="https://google.com/" rel="nofollow">https://google.com/</a></p>`
|
googleRendered := `<p><a href="https://google.com/" rel="nofollow">https://google.com/</a></p>`
|
||||||
test("<https://google.com/>", googleRendered, googleRendered)
|
test("<https://google.com/>", googleRendered)
|
||||||
|
test("[Link](Link)", `<p><a href="/Link" rel="nofollow">Link</a></p>`)
|
||||||
lnk := util.URLJoin(FullURL, "WikiPage")
|
|
||||||
lnkWiki := util.URLJoin(FullURL, "wiki", "WikiPage")
|
|
||||||
test("[WikiPage](WikiPage)",
|
|
||||||
`<p><a href="`+lnk+`" rel="nofollow">WikiPage</a></p>`,
|
|
||||||
`<p><a href="`+lnkWiki+`" rel="nofollow">WikiPage</a></p>`)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRender_Images(t *testing.T) {
|
func TestRender_Images(t *testing.T) {
|
||||||
setting.AppURL = AppURL
|
setting.AppURL = AppURL
|
||||||
|
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
buffer, err := markdown.RenderString(&markup.RenderContext{
|
buffer, err := markdown.RenderString(markup.NewTestRenderContext(FullURL), input)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: FullURL,
|
|
||||||
},
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(buffer)))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(buffer)))
|
||||||
}
|
}
|
||||||
@ -140,12 +85,12 @@ func TestRender_Images(t *testing.T) {
|
|||||||
`<p><a href="`+href+`" rel="nofollow"><img src="`+result+`" alt="`+title+`"/></a></p>`)
|
`<p><a href="`+href+`" rel="nofollow"><img src="`+result+`" alt="`+title+`"/></a></p>`)
|
||||||
}
|
}
|
||||||
|
|
||||||
func testAnswers(baseURLContent, baseURLImages string) []string {
|
func testAnswers(baseURL string) []string {
|
||||||
return []string{
|
return []string{
|
||||||
`<p>Wiki! Enjoy :)</p>
|
`<p>Wiki! Enjoy :)</p>
|
||||||
<ul>
|
<ul>
|
||||||
<li><a href="` + baseURLContent + `/Links" rel="nofollow">Links, Language bindings, Engine bindings</a></li>
|
<li><a href="` + baseURL + `/Links" rel="nofollow">Links, Language bindings, Engine bindings</a></li>
|
||||||
<li><a href="` + baseURLContent + `/Tips" rel="nofollow">Tips</a></li>
|
<li><a href="` + baseURL + `/Tips" rel="nofollow">Tips</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
<p>See commit <a href="/` + testRepoOwnerName + `/` + testRepoName + `/commit/65f1bf27bc" rel="nofollow"><code>65f1bf27bc</code></a></p>
|
<p>See commit <a href="/` + testRepoOwnerName + `/` + testRepoName + `/commit/65f1bf27bc" rel="nofollow"><code>65f1bf27bc</code></a></p>
|
||||||
<p>Ideas and codes</p>
|
<p>Ideas and codes</p>
|
||||||
@ -153,8 +98,8 @@ func testAnswers(baseURLContent, baseURLImages string) []string {
|
|||||||
<li>Bezier widget (by <a href="/r-lyeh" rel="nofollow">@r-lyeh</a>) <a href="http://localhost:3000/ocornut/imgui/issues/786" class="ref-issue" rel="nofollow">ocornut/imgui#786</a></li>
|
<li>Bezier widget (by <a href="/r-lyeh" rel="nofollow">@r-lyeh</a>) <a href="http://localhost:3000/ocornut/imgui/issues/786" class="ref-issue" rel="nofollow">ocornut/imgui#786</a></li>
|
||||||
<li>Bezier widget (by <a href="/r-lyeh" rel="nofollow">@r-lyeh</a>) <a href="` + FullURL + `issues/786" class="ref-issue" rel="nofollow">#786</a></li>
|
<li>Bezier widget (by <a href="/r-lyeh" rel="nofollow">@r-lyeh</a>) <a href="` + FullURL + `issues/786" class="ref-issue" rel="nofollow">#786</a></li>
|
||||||
<li>Node graph editors <a href="https://github.com/ocornut/imgui/issues/306" rel="nofollow">https://github.com/ocornut/imgui/issues/306</a></li>
|
<li>Node graph editors <a href="https://github.com/ocornut/imgui/issues/306" rel="nofollow">https://github.com/ocornut/imgui/issues/306</a></li>
|
||||||
<li><a href="` + baseURLContent + `/memory_editor_example" rel="nofollow">Memory Editor</a></li>
|
<li><a href="` + baseURL + `/memory_editor_example" rel="nofollow">Memory Editor</a></li>
|
||||||
<li><a href="` + baseURLContent + `/plot_var_example" rel="nofollow">Plot var helper</a></li>
|
<li><a href="` + baseURL + `/plot_var_example" rel="nofollow">Plot var helper</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
`,
|
`,
|
||||||
`<h2 id="user-content-what-is-wine-staging">What is Wine Staging?</h2>
|
`<h2 id="user-content-what-is-wine-staging">What is Wine Staging?</h2>
|
||||||
@ -164,14 +109,14 @@ func testAnswers(baseURLContent, baseURLImages string) []string {
|
|||||||
<table>
|
<table>
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th><a href="` + baseURLImages + `/images/icon-install.png" rel="nofollow"><img src="` + baseURLImages + `/images/icon-install.png" title="icon-install.png" alt="images/icon-install.png"/></a></th>
|
<th><a href="` + baseURL + `/images/icon-install.png" rel="nofollow"><img src="` + baseURL + `/images/icon-install.png" title="icon-install.png" alt="images/icon-install.png"/></a></th>
|
||||||
<th><a href="` + baseURLContent + `/Installation" rel="nofollow">Installation</a></th>
|
<th><a href="` + baseURL + `/Installation" rel="nofollow">Installation</a></th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<td><a href="` + baseURLImages + `/images/icon-usage.png" rel="nofollow"><img src="` + baseURLImages + `/images/icon-usage.png" title="icon-usage.png" alt="images/icon-usage.png"/></a></td>
|
<td><a href="` + baseURL + `/images/icon-usage.png" rel="nofollow"><img src="` + baseURL + `/images/icon-usage.png" title="icon-usage.png" alt="images/icon-usage.png"/></a></td>
|
||||||
<td><a href="` + baseURLContent + `/Usage" rel="nofollow">Usage</a></td>
|
<td><a href="` + baseURL + `/Usage" rel="nofollow">Usage</a></td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
@ -179,9 +124,9 @@ func testAnswers(baseURLContent, baseURLImages string) []string {
|
|||||||
`<p><a href="http://www.excelsiorjet.com/" rel="nofollow">Excelsior JET</a> allows you to create native executables for Windows, Linux and Mac OS X.</p>
|
`<p><a href="http://www.excelsiorjet.com/" rel="nofollow">Excelsior JET</a> allows you to create native executables for Windows, Linux and Mac OS X.</p>
|
||||||
<ol>
|
<ol>
|
||||||
<li><a href="https://github.com/libgdx/libgdx/wiki/Gradle-on-the-Commandline#packaging-for-the-desktop" rel="nofollow">Package your libGDX application</a><br/>
|
<li><a href="https://github.com/libgdx/libgdx/wiki/Gradle-on-the-Commandline#packaging-for-the-desktop" rel="nofollow">Package your libGDX application</a><br/>
|
||||||
<a href="` + baseURLImages + `/images/1.png" rel="nofollow"><img src="` + baseURLImages + `/images/1.png" title="1.png" alt="images/1.png"/></a></li>
|
<a href="` + baseURL + `/images/1.png" rel="nofollow"><img src="` + baseURL + `/images/1.png" title="1.png" alt="images/1.png"/></a></li>
|
||||||
<li>Perform a test run by hitting the Run! button.<br/>
|
<li>Perform a test run by hitting the Run! button.<br/>
|
||||||
<a href="` + baseURLImages + `/images/2.png" rel="nofollow"><img src="` + baseURLImages + `/images/2.png" title="2.png" alt="images/2.png"/></a></li>
|
<a href="` + baseURL + `/images/2.png" rel="nofollow"><img src="` + baseURL + `/images/2.png" title="2.png" alt="images/2.png"/></a></li>
|
||||||
</ol>
|
</ol>
|
||||||
<h2 id="user-content-custom-id">More tests</h2>
|
<h2 id="user-content-custom-id">More tests</h2>
|
||||||
<p>(from <a href="https://www.markdownguide.org/extended-syntax/" rel="nofollow">https://www.markdownguide.org/extended-syntax/</a>)</p>
|
<p>(from <a href="https://www.markdownguide.org/extended-syntax/" rel="nofollow">https://www.markdownguide.org/extended-syntax/</a>)</p>
|
||||||
@ -302,96 +247,35 @@ This PR has been generated by [Renovate Bot](https://github.com/renovatebot/reno
|
|||||||
<!-- test-comment -->`,
|
<!-- test-comment -->`,
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestTotal_RenderWiki(t *testing.T) {
|
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.ForceHardLineBreak, true)()
|
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
|
||||||
setting.AppURL = AppURL
|
|
||||||
answers := testAnswers(util.URLJoin(FullURL, "wiki"), util.URLJoin(FullURL, "wiki", "raw"))
|
|
||||||
for i := 0; i < len(sameCases); i++ {
|
|
||||||
line, err := markdown.RenderString(&markup.RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: FullURL,
|
|
||||||
},
|
|
||||||
Repo: newMockRepo(testRepoOwnerName, testRepoName),
|
|
||||||
Metas: localWikiMetas,
|
|
||||||
}, sameCases[i])
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Equal(t, answers[i], string(line))
|
|
||||||
}
|
|
||||||
|
|
||||||
testCases := []string{
|
|
||||||
// Guard wiki sidebar: special syntax
|
|
||||||
`[[Guardfile-DSL / Configuring-Guard|Guardfile-DSL---Configuring-Guard]]`,
|
|
||||||
// rendered
|
|
||||||
`<p><a href="` + FullURL + `wiki/Guardfile-DSL---Configuring-Guard" rel="nofollow">Guardfile-DSL / Configuring-Guard</a></p>
|
|
||||||
`,
|
|
||||||
// special syntax
|
|
||||||
`[[Name|Link]]`,
|
|
||||||
// rendered
|
|
||||||
`<p><a href="` + FullURL + `wiki/Link" rel="nofollow">Name</a></p>
|
|
||||||
`,
|
|
||||||
}
|
|
||||||
|
|
||||||
for i := 0; i < len(testCases); i += 2 {
|
|
||||||
line, err := markdown.RenderString(&markup.RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: FullURL,
|
|
||||||
},
|
|
||||||
Metas: localWikiMetas,
|
|
||||||
}, testCases[i])
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.EqualValues(t, testCases[i+1], string(line))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTotal_RenderString(t *testing.T) {
|
func TestTotal_RenderString(t *testing.T) {
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.ForceHardLineBreak, true)()
|
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.ForceHardLineBreak, true)()
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
setting.AppURL = AppURL
|
markup.Init(&markup.RenderHelperFuncs{
|
||||||
answers := testAnswers(util.URLJoin(FullURL, "src", "master"), util.URLJoin(FullURL, "media", "master"))
|
IsUsernameMentionable: func(ctx context.Context, username string) bool {
|
||||||
|
return username == "r-lyeh"
|
||||||
|
},
|
||||||
|
})
|
||||||
|
answers := testAnswers("")
|
||||||
for i := 0; i < len(sameCases); i++ {
|
for i := 0; i < len(sameCases); i++ {
|
||||||
line, err := markdown.RenderString(&markup.RenderContext{
|
line, err := markdown.RenderString(markup.NewTestRenderContext(localMetas), sameCases[i])
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: FullURL,
|
|
||||||
BranchPath: "master",
|
|
||||||
},
|
|
||||||
Repo: newMockRepo(testRepoOwnerName, testRepoName),
|
|
||||||
Metas: localMetas,
|
|
||||||
}, sameCases[i])
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, answers[i], string(line))
|
assert.Equal(t, answers[i], string(line))
|
||||||
}
|
}
|
||||||
|
|
||||||
testCases := []string{}
|
|
||||||
|
|
||||||
for i := 0; i < len(testCases); i += 2 {
|
|
||||||
line, err := markdown.RenderString(&markup.RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: FullURL,
|
|
||||||
},
|
|
||||||
}, testCases[i])
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Equal(t, template.HTML(testCases[i+1]), line)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRender_RenderParagraphs(t *testing.T) {
|
func TestRender_RenderParagraphs(t *testing.T) {
|
||||||
test := func(t *testing.T, str string, cnt int) {
|
test := func(t *testing.T, str string, cnt int) {
|
||||||
res, err := markdown.RenderRawString(&markup.RenderContext{Ctx: git.DefaultContext}, str)
|
res, err := markdown.RenderRawString(markup.NewTestRenderContext(), str)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, cnt, strings.Count(res, "<p"), "Rendered result for unix should have %d paragraph(s) but has %d:\n%s\n", cnt, strings.Count(res, "<p"), res)
|
assert.Equal(t, cnt, strings.Count(res, "<p"), "Rendered result for unix should have %d paragraph(s) but has %d:\n%s\n", cnt, strings.Count(res, "<p"), res)
|
||||||
|
|
||||||
mac := strings.ReplaceAll(str, "\n", "\r")
|
mac := strings.ReplaceAll(str, "\n", "\r")
|
||||||
res, err = markdown.RenderRawString(&markup.RenderContext{Ctx: git.DefaultContext}, mac)
|
res, err = markdown.RenderRawString(markup.NewTestRenderContext(), mac)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, cnt, strings.Count(res, "<p"), "Rendered result for mac should have %d paragraph(s) but has %d:\n%s\n", cnt, strings.Count(res, "<p"), res)
|
assert.Equal(t, cnt, strings.Count(res, "<p"), "Rendered result for mac should have %d paragraph(s) but has %d:\n%s\n", cnt, strings.Count(res, "<p"), res)
|
||||||
|
|
||||||
dos := strings.ReplaceAll(str, "\n", "\r\n")
|
dos := strings.ReplaceAll(str, "\n", "\r\n")
|
||||||
res, err = markdown.RenderRawString(&markup.RenderContext{Ctx: git.DefaultContext}, dos)
|
res, err = markdown.RenderRawString(markup.NewTestRenderContext(), dos)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, cnt, strings.Count(res, "<p"), "Rendered result for windows should have %d paragraph(s) but has %d:\n%s\n", cnt, strings.Count(res, "<p"), res)
|
assert.Equal(t, cnt, strings.Count(res, "<p"), "Rendered result for windows should have %d paragraph(s) but has %d:\n%s\n", cnt, strings.Count(res, "<p"), res)
|
||||||
}
|
}
|
||||||
@ -419,7 +303,7 @@ func TestMarkdownRenderRaw(t *testing.T) {
|
|||||||
|
|
||||||
for _, testcase := range testcases {
|
for _, testcase := range testcases {
|
||||||
log.Info("Test markdown render error with fuzzy data: %x, the following errors can be recovered", testcase)
|
log.Info("Test markdown render error with fuzzy data: %x, the following errors can be recovered", testcase)
|
||||||
_, err := markdown.RenderRawString(&markup.RenderContext{Ctx: git.DefaultContext}, string(testcase))
|
_, err := markdown.RenderRawString(markup.NewTestRenderContext(), string(testcase))
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -432,7 +316,7 @@ func TestRenderSiblingImages_Issue12925(t *testing.T) {
|
|||||||
<a href="/image2" target="_blank" rel="nofollow noopener"><img src="/image2" alt="image2"></a></p>
|
<a href="/image2" target="_blank" rel="nofollow noopener"><img src="/image2" alt="image2"></a></p>
|
||||||
`
|
`
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.ForceHardLineBreak, true)()
|
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.ForceHardLineBreak, true)()
|
||||||
res, err := markdown.RenderRawString(&markup.RenderContext{Ctx: git.DefaultContext}, testcase)
|
res, err := markdown.RenderRawString(markup.NewTestRenderContext(), testcase)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, expected, res)
|
assert.Equal(t, expected, res)
|
||||||
}
|
}
|
||||||
@ -441,7 +325,7 @@ func TestRenderEmojiInLinks_Issue12331(t *testing.T) {
|
|||||||
testcase := `[Link with emoji :moon: in text](https://gitea.io)`
|
testcase := `[Link with emoji :moon: in text](https://gitea.io)`
|
||||||
expected := `<p><a href="https://gitea.io" rel="nofollow">Link with emoji <span class="emoji" aria-label="waxing gibbous moon">🌔</span> in text</a></p>
|
expected := `<p><a href="https://gitea.io" rel="nofollow">Link with emoji <span class="emoji" aria-label="waxing gibbous moon">🌔</span> in text</a></p>
|
||||||
`
|
`
|
||||||
res, err := markdown.RenderString(&markup.RenderContext{Ctx: git.DefaultContext}, testcase)
|
res, err := markdown.RenderString(markup.NewTestRenderContext(), testcase)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, template.HTML(expected), res)
|
assert.Equal(t, template.HTML(expected), res)
|
||||||
}
|
}
|
||||||
@ -479,7 +363,7 @@ func TestColorPreview(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for _, test := range positiveTests {
|
for _, test := range positiveTests {
|
||||||
res, err := markdown.RenderString(&markup.RenderContext{Ctx: git.DefaultContext}, test.testcase)
|
res, err := markdown.RenderString(markup.NewTestRenderContext(), test.testcase)
|
||||||
assert.NoError(t, err, "Unexpected error in testcase: %q", test.testcase)
|
assert.NoError(t, err, "Unexpected error in testcase: %q", test.testcase)
|
||||||
assert.Equal(t, template.HTML(test.expected), res, "Unexpected result in testcase %q", test.testcase)
|
assert.Equal(t, template.HTML(test.expected), res, "Unexpected result in testcase %q", test.testcase)
|
||||||
}
|
}
|
||||||
@ -498,7 +382,7 @@ func TestColorPreview(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for _, test := range negativeTests {
|
for _, test := range negativeTests {
|
||||||
res, err := markdown.RenderString(&markup.RenderContext{Ctx: git.DefaultContext}, test)
|
res, err := markdown.RenderString(markup.NewTestRenderContext(), test)
|
||||||
assert.NoError(t, err, "Unexpected error in testcase: %q", test)
|
assert.NoError(t, err, "Unexpected error in testcase: %q", test)
|
||||||
assert.NotContains(t, res, `<span class="color-preview" style="background-color: `, "Unexpected result in testcase %q", test)
|
assert.NotContains(t, res, `<span class="color-preview" style="background-color: `, "Unexpected result in testcase %q", test)
|
||||||
}
|
}
|
||||||
@ -573,7 +457,7 @@ func TestMathBlock(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for _, test := range testcases {
|
for _, test := range testcases {
|
||||||
res, err := markdown.RenderString(&markup.RenderContext{Ctx: git.DefaultContext}, test.testcase)
|
res, err := markdown.RenderString(markup.NewTestRenderContext(), test.testcase)
|
||||||
assert.NoError(t, err, "Unexpected error in testcase: %q", test.testcase)
|
assert.NoError(t, err, "Unexpected error in testcase: %q", test.testcase)
|
||||||
assert.Equal(t, template.HTML(test.expected), res, "Unexpected result in testcase %q", test.testcase)
|
assert.Equal(t, template.HTML(test.expected), res, "Unexpected result in testcase %q", test.testcase)
|
||||||
}
|
}
|
||||||
@ -610,7 +494,7 @@ foo: bar
|
|||||||
}
|
}
|
||||||
|
|
||||||
for _, test := range testcases {
|
for _, test := range testcases {
|
||||||
res, err := markdown.RenderString(&markup.RenderContext{Ctx: git.DefaultContext}, test.testcase)
|
res, err := markdown.RenderString(markup.NewTestRenderContext(), test.testcase)
|
||||||
assert.NoError(t, err, "Unexpected error in testcase: %q", test.testcase)
|
assert.NoError(t, err, "Unexpected error in testcase: %q", test.testcase)
|
||||||
assert.Equal(t, template.HTML(test.expected), res, "Unexpected result in testcase %q", test.testcase)
|
assert.Equal(t, template.HTML(test.expected), res, "Unexpected result in testcase %q", test.testcase)
|
||||||
}
|
}
|
||||||
@ -642,13 +526,9 @@ mail@domain.com
|
|||||||
`
|
`
|
||||||
input = strings.ReplaceAll(input, "${SPACE}", " ") // replace ${SPACE} with " ", to avoid some editor's auto-trimming
|
input = strings.ReplaceAll(input, "${SPACE}", " ") // replace ${SPACE} with " ", to avoid some editor's auto-trimming
|
||||||
cases := []struct {
|
cases := []struct {
|
||||||
Links markup.Links
|
|
||||||
IsWiki bool
|
|
||||||
Expected string
|
Expected string
|
||||||
}{
|
}{
|
||||||
{ // 0
|
{
|
||||||
Links: markup.Links{},
|
|
||||||
IsWiki: false,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
Expected: `<p>space @mention-user<br/>
|
||||||
/just/a/path.bin<br/>
|
/just/a/path.bin<br/>
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
||||||
@ -671,343 +551,14 @@ com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|||||||
@mention-user test<br/>
|
@mention-user test<br/>
|
||||||
#123<br/>
|
#123<br/>
|
||||||
space</p>
|
space</p>
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 1
|
|
||||||
Links: markup.Links{},
|
|
||||||
IsWiki: true,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/wiki/raw/image.jpg" target="_blank" rel="nofollow noopener"><img src="/wiki/raw/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="/wiki/raw/image.jpg" rel="nofollow"><img src="/wiki/raw/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 2
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "https://gitea.io/",
|
|
||||||
},
|
|
||||||
IsWiki: false,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="https://gitea.io/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="https://gitea.io/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="https://gitea.io/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://gitea.io/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="https://gitea.io/path/file" target="_blank" rel="nofollow noopener"><img src="https://gitea.io/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://gitea.io/path/file" target="_blank" rel="nofollow noopener"><img src="https://gitea.io/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="https://gitea.io/image.jpg" rel="nofollow"><img src="https://gitea.io/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 3
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "https://gitea.io/",
|
|
||||||
},
|
|
||||||
IsWiki: true,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="https://gitea.io/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="https://gitea.io/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="https://gitea.io/wiki/raw/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://gitea.io/wiki/raw/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="https://gitea.io/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="https://gitea.io/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://gitea.io/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="https://gitea.io/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="https://gitea.io/wiki/raw/image.jpg" rel="nofollow"><img src="https://gitea.io/wiki/raw/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 4
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "/relative/path",
|
|
||||||
},
|
|
||||||
IsWiki: false,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="/relative/path/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/relative/path/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/relative/path/image.jpg" target="_blank" rel="nofollow noopener"><img src="/relative/path/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="/relative/path/path/file" target="_blank" rel="nofollow noopener"><img src="/relative/path/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="/relative/path/path/file" target="_blank" rel="nofollow noopener"><img src="/relative/path/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="/relative/path/image.jpg" rel="nofollow"><img src="/relative/path/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 5
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "/relative/path",
|
|
||||||
},
|
|
||||||
IsWiki: true,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="/relative/path/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/relative/path/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/image.jpg" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/image.jpg" rel="nofollow"><img src="/relative/path/wiki/raw/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 6
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "/user/repo",
|
|
||||||
BranchPath: "branch/main",
|
|
||||||
},
|
|
||||||
IsWiki: false,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="/user/repo/src/branch/main/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/user/repo/src/branch/main/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/user/repo/media/branch/main/image.jpg" target="_blank" rel="nofollow noopener"><img src="/user/repo/media/branch/main/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="/user/repo/media/branch/main/path/file" target="_blank" rel="nofollow noopener"><img src="/user/repo/media/branch/main/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="/user/repo/media/branch/main/path/file" target="_blank" rel="nofollow noopener"><img src="/user/repo/media/branch/main/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="/user/repo/media/branch/main/image.jpg" rel="nofollow"><img src="/user/repo/media/branch/main/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 7
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "/relative/path",
|
|
||||||
BranchPath: "branch/main",
|
|
||||||
},
|
|
||||||
IsWiki: true,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="/relative/path/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/relative/path/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/image.jpg" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/image.jpg" rel="nofollow"><img src="/relative/path/wiki/raw/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 8
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "/user/repo",
|
|
||||||
TreePath: "sub/folder",
|
|
||||||
},
|
|
||||||
IsWiki: false,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="/user/repo/src/sub/folder/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/user/repo/src/sub/folder/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/user/repo/image.jpg" target="_blank" rel="nofollow noopener"><img src="/user/repo/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="/user/repo/path/file" target="_blank" rel="nofollow noopener"><img src="/user/repo/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="/user/repo/path/file" target="_blank" rel="nofollow noopener"><img src="/user/repo/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="/user/repo/image.jpg" rel="nofollow"><img src="/user/repo/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 9
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "/relative/path",
|
|
||||||
TreePath: "sub/folder",
|
|
||||||
},
|
|
||||||
IsWiki: true,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="/relative/path/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/relative/path/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/image.jpg" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/image.jpg" rel="nofollow"><img src="/relative/path/wiki/raw/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 10
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "/user/repo",
|
|
||||||
BranchPath: "branch/main",
|
|
||||||
TreePath: "sub/folder",
|
|
||||||
},
|
|
||||||
IsWiki: false,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="/user/repo/src/branch/main/sub/folder/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/user/repo/src/branch/main/sub/folder/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/user/repo/media/branch/main/sub/folder/image.jpg" target="_blank" rel="nofollow noopener"><img src="/user/repo/media/branch/main/sub/folder/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="/user/repo/media/branch/main/sub/folder/path/file" target="_blank" rel="nofollow noopener"><img src="/user/repo/media/branch/main/sub/folder/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="/user/repo/media/branch/main/sub/folder/path/file" target="_blank" rel="nofollow noopener"><img src="/user/repo/media/branch/main/sub/folder/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="/user/repo/media/branch/main/sub/folder/image.jpg" rel="nofollow"><img src="/user/repo/media/branch/main/sub/folder/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{ // 11
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "/relative/path",
|
|
||||||
BranchPath: "branch/main",
|
|
||||||
TreePath: "sub/folder",
|
|
||||||
},
|
|
||||||
IsWiki: true,
|
|
||||||
Expected: `<p>space @mention-user<br/>
|
|
||||||
/just/a/path.bin<br/>
|
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a><br/>
|
|
||||||
<a href="/relative/path/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/relative/path/wiki/file.bin" rel="nofollow">local link</a><br/>
|
|
||||||
<a href="https://example.com" rel="nofollow">remote link</a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/image.jpg" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/image.jpg" alt="local image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/path/file" target="_blank" rel="nofollow noopener"><img src="/relative/path/wiki/raw/path/file" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" target="_blank" rel="nofollow noopener"><img src="https://example.com/image.jpg" alt="remote image"/></a><br/>
|
|
||||||
<a href="/relative/path/wiki/raw/image.jpg" rel="nofollow"><img src="/relative/path/wiki/raw/image.jpg" title="local image" alt="local image"/></a><br/>
|
|
||||||
<a href="https://example.com/image.jpg" rel="nofollow"><img src="https://example.com/image.jpg" title="remote link" alt="remote link"/></a><br/>
|
|
||||||
<a href="https://example.com/user/repo/compare/88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb#hash" rel="nofollow"><code>88fc37a3c0...12fc37a3c0 (hash)</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb...12fc37a3c0a4dda553bdcfc80c178a58247f42fb pare<br/>
|
|
||||||
<a href="https://example.com/user/repo/commit/88fc37a3c0a4dda553bdcfc80c178a58247f42fb" rel="nofollow"><code>88fc37a3c0</code></a><br/>
|
|
||||||
com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit<br/>
|
|
||||||
<span class="emoji" aria-label="thumbs up">👍</span><br/>
|
|
||||||
<a href="mailto:mail@domain.com" rel="nofollow">mail@domain.com</a><br/>
|
|
||||||
@mention-user test<br/>
|
|
||||||
#123<br/>
|
|
||||||
space</p>
|
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.ForceHardLineBreak, true)()
|
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.ForceHardLineBreak, true)()
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
for i, c := range cases {
|
for i, c := range cases {
|
||||||
result, err := markdown.RenderString(&markup.RenderContext{
|
result, err := markdown.RenderString(markup.NewTestRenderContext(localMetas), input)
|
||||||
Ctx: context.Background(),
|
|
||||||
Links: c.Links,
|
|
||||||
Metas: util.Iif(c.IsWiki, map[string]string{"markupContentMode": "wiki"}, map[string]string{}),
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err, "Unexpected error in testcase: %v", i)
|
assert.NoError(t, err, "Unexpected error in testcase: %v", i)
|
||||||
assert.Equal(t, c.Expected, string(result), "Unexpected result in testcase %v", i)
|
assert.Equal(t, c.Expected, string(result), "Unexpected result in testcase %v", i)
|
||||||
}
|
}
|
||||||
@ -1029,7 +580,7 @@ func TestAttention(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
result, err := markdown.RenderString(&markup.RenderContext{Ctx: context.Background()}, input)
|
result, err := markdown.RenderString(markup.NewTestRenderContext(), input)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(result)))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(result)))
|
||||||
}
|
}
|
||||||
@ -1062,6 +613,6 @@ func BenchmarkSpecializedMarkdown(b *testing.B) {
|
|||||||
func BenchmarkMarkdownRender(b *testing.B) {
|
func BenchmarkMarkdownRender(b *testing.B) {
|
||||||
// 23202 50840 ns/op
|
// 23202 50840 ns/op
|
||||||
for i := 0; i < b.N; i++ {
|
for i := 0; i < b.N; i++ {
|
||||||
_, _ = markdown.RenderString(&markup.RenderContext{Ctx: context.Background()}, "https://example.com\n- a\n- b\n")
|
_, _ = markdown.RenderString(markup.NewTestRenderContext(), "https://example.com\n- a\n- b\n")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4,10 +4,7 @@
|
|||||||
package markdown
|
package markdown
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"strings"
|
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
giteautil "code.gitea.io/gitea/modules/util"
|
|
||||||
|
|
||||||
"github.com/yuin/goldmark/ast"
|
"github.com/yuin/goldmark/ast"
|
||||||
)
|
)
|
||||||
@ -20,10 +17,7 @@ func (g *ASTTransformer) transformImage(ctx *markup.RenderContext, v *ast.Image)
|
|||||||
|
|
||||||
// Check if the destination is a real link
|
// Check if the destination is a real link
|
||||||
if len(v.Destination) > 0 && !markup.IsFullURLBytes(v.Destination) {
|
if len(v.Destination) > 0 && !markup.IsFullURLBytes(v.Destination) {
|
||||||
v.Destination = []byte(giteautil.URLJoin(
|
v.Destination = []byte(ctx.RenderHelper.ResolveLink(string(v.Destination), markup.LinkTypeMedia))
|
||||||
ctx.Links.ResolveMediaLink(ctx.IsMarkupContentWiki()),
|
|
||||||
strings.TrimLeft(string(v.Destination), "/"),
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
parent := v.Parent()
|
parent := v.Parent()
|
||||||
|
@ -9,8 +9,19 @@ import (
|
|||||||
"github.com/yuin/goldmark/ast"
|
"github.com/yuin/goldmark/ast"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func resolveLink(ctx *markup.RenderContext, link, userContentAnchorPrefix string) (result string, resolved bool) {
|
||||||
|
isAnchorFragment := link != "" && link[0] == '#'
|
||||||
|
if !isAnchorFragment && !markup.IsFullURLString(link) {
|
||||||
|
link, resolved = ctx.RenderHelper.ResolveLink(link, markup.LinkTypeDefault), true
|
||||||
|
}
|
||||||
|
if isAnchorFragment && userContentAnchorPrefix != "" {
|
||||||
|
link, resolved = userContentAnchorPrefix+link[1:], true
|
||||||
|
}
|
||||||
|
return link, resolved
|
||||||
|
}
|
||||||
|
|
||||||
func (g *ASTTransformer) transformLink(ctx *markup.RenderContext, v *ast.Link) {
|
func (g *ASTTransformer) transformLink(ctx *markup.RenderContext, v *ast.Link) {
|
||||||
if link, resolved := markup.ResolveLink(ctx, string(v.Destination), "#user-content-"); resolved {
|
if link, resolved := resolveLink(ctx, string(v.Destination), "#user-content-"); resolved {
|
||||||
v.Destination = []byte(link)
|
v.Destination = []byte(link)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -13,7 +13,6 @@ import (
|
|||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
"code.gitea.io/gitea/modules/util"
|
|
||||||
|
|
||||||
"github.com/alecthomas/chroma/v2"
|
"github.com/alecthomas/chroma/v2"
|
||||||
"github.com/alecthomas/chroma/v2/lexers"
|
"github.com/alecthomas/chroma/v2/lexers"
|
||||||
@ -142,19 +141,11 @@ func (r *Writer) resolveLink(kind, link string) string {
|
|||||||
// so we need to try to guess the link kind again here
|
// so we need to try to guess the link kind again here
|
||||||
kind = org.RegularLink{URL: link}.Kind()
|
kind = org.RegularLink{URL: link}.Kind()
|
||||||
}
|
}
|
||||||
|
|
||||||
base := r.Ctx.Links.Base
|
|
||||||
if r.Ctx.IsMarkupContentWiki() {
|
|
||||||
base = r.Ctx.Links.WikiLink()
|
|
||||||
} else if r.Ctx.Links.HasBranchInfo() {
|
|
||||||
base = r.Ctx.Links.SrcLink()
|
|
||||||
}
|
|
||||||
|
|
||||||
if kind == "image" || kind == "video" {
|
if kind == "image" || kind == "video" {
|
||||||
base = r.Ctx.Links.ResolveMediaLink(r.Ctx.IsMarkupContentWiki())
|
link = r.Ctx.RenderHelper.ResolveLink(link, markup.LinkTypeMedia)
|
||||||
|
} else {
|
||||||
|
link = r.Ctx.RenderHelper.ResolveLink(link, markup.LinkTypeDefault)
|
||||||
}
|
}
|
||||||
|
|
||||||
link = util.URLJoin(base, link)
|
|
||||||
}
|
}
|
||||||
return link
|
return link
|
||||||
}
|
}
|
||||||
|
@ -4,54 +4,38 @@
|
|||||||
package markup
|
package markup
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/git"
|
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
"code.gitea.io/gitea/modules/util"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
)
|
)
|
||||||
|
|
||||||
const AppURL = "http://localhost:3000/"
|
func TestMain(m *testing.M) {
|
||||||
|
setting.AppURL = "http://localhost:3000/"
|
||||||
|
setting.IsInTesting = true
|
||||||
|
os.Exit(m.Run())
|
||||||
|
}
|
||||||
|
|
||||||
func TestRender_StandardLinks(t *testing.T) {
|
func TestRender_StandardLinks(t *testing.T) {
|
||||||
setting.AppURL = AppURL
|
test := func(input, expected string) {
|
||||||
|
buffer, err := RenderString(markup.NewTestRenderContext("/relative-path/media/branch/main/"), input)
|
||||||
test := func(input, expected string, isWiki bool) {
|
|
||||||
buffer, err := RenderString(&markup.RenderContext{
|
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "/relative-path",
|
|
||||||
BranchPath: "branch/main",
|
|
||||||
},
|
|
||||||
Metas: map[string]string{"markupContentMode": util.Iif(isWiki, "wiki", "")},
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
||||||
}
|
}
|
||||||
|
|
||||||
test("[[https://google.com/]]",
|
test("[[https://google.com/]]",
|
||||||
`<p><a href="https://google.com/">https://google.com/</a></p>`, false)
|
`<p><a href="https://google.com/">https://google.com/</a></p>`)
|
||||||
test("[[WikiPage][The WikiPage Desc]]",
|
|
||||||
`<p><a href="/relative-path/wiki/WikiPage">The WikiPage Desc</a></p>`, true)
|
|
||||||
test("[[ImageLink.svg][The Image Desc]]",
|
test("[[ImageLink.svg][The Image Desc]]",
|
||||||
`<p><a href="/relative-path/media/branch/main/ImageLink.svg">The Image Desc</a></p>`, false)
|
`<p><a href="/relative-path/media/branch/main/ImageLink.svg">The Image Desc</a></p>`)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRender_InternalLinks(t *testing.T) {
|
func TestRender_InternalLinks(t *testing.T) {
|
||||||
setting.AppURL = AppURL
|
|
||||||
|
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
buffer, err := RenderString(&markup.RenderContext{
|
buffer, err := RenderString(markup.NewTestRenderContext("/relative-path/src/branch/main"), input)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "/relative-path",
|
|
||||||
BranchPath: "branch/main",
|
|
||||||
},
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
||||||
}
|
}
|
||||||
@ -67,15 +51,8 @@ func TestRender_InternalLinks(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestRender_Media(t *testing.T) {
|
func TestRender_Media(t *testing.T) {
|
||||||
setting.AppURL = AppURL
|
|
||||||
|
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
buffer, err := RenderString(&markup.RenderContext{
|
buffer, err := RenderString(markup.NewTestRenderContext("./relative-path"), input)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: "./relative-path",
|
|
||||||
},
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
||||||
}
|
}
|
||||||
@ -113,12 +90,8 @@ func TestRender_Media(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestRender_Source(t *testing.T) {
|
func TestRender_Source(t *testing.T) {
|
||||||
setting.AppURL = AppURL
|
|
||||||
|
|
||||||
test := func(input, expected string) {
|
test := func(input, expected string) {
|
||||||
buffer, err := RenderString(&markup.RenderContext{
|
buffer, err := RenderString(markup.NewTestRenderContext(), input)
|
||||||
Ctx: git.DefaultContext,
|
|
||||||
}, input)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
|
||||||
}
|
}
|
||||||
|
@ -9,9 +9,8 @@ import (
|
|||||||
"io"
|
"io"
|
||||||
"net/url"
|
"net/url"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/git"
|
|
||||||
"code.gitea.io/gitea/modules/gitrepo"
|
|
||||||
"code.gitea.io/gitea/modules/markup/internal"
|
"code.gitea.io/gitea/modules/markup/internal"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
"code.gitea.io/gitea/modules/util"
|
"code.gitea.io/gitea/modules/util"
|
||||||
@ -37,88 +36,111 @@ var RenderBehaviorForTesting struct {
|
|||||||
// * However, many places render the content without setting "mode" in Metas, all these places used comment line break setting incorrectly
|
// * However, many places render the content without setting "mode" in Metas, all these places used comment line break setting incorrectly
|
||||||
ForceHardLineBreak bool
|
ForceHardLineBreak bool
|
||||||
|
|
||||||
// Gitea will emit some internal attributes for various purposes, these attributes don't affect rendering.
|
// Gitea will emit some additional attributes for various purposes, these attributes don't affect rendering.
|
||||||
// But there are too many hard-coded test cases, to avoid changing all of them again and again, we can disable emitting these internal attributes.
|
// But there are too many hard-coded test cases, to avoid changing all of them again and again, we can disable emitting these internal attributes.
|
||||||
DisableInternalAttributes bool
|
DisableAdditionalAttributes bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// RenderContext represents a render context
|
type RenderOptions struct {
|
||||||
type RenderContext struct {
|
UseAbsoluteLink bool
|
||||||
Ctx context.Context
|
|
||||||
RelativePath string // relative path from tree root of the branch
|
// relative path from tree root of the branch
|
||||||
|
RelativePath string
|
||||||
|
|
||||||
// eg: "orgmode", "asciicast", "console"
|
// eg: "orgmode", "asciicast", "console"
|
||||||
// for file mode, it could be left as empty, and will be detected by file extension in RelativePath
|
// for file mode, it could be left as empty, and will be detected by file extension in RelativePath
|
||||||
MarkupType string
|
MarkupType string
|
||||||
|
|
||||||
Links Links // special link references for rendering, especially when there is a branch/tree path
|
|
||||||
|
|
||||||
// user&repo, format&style®exp (for external issue pattern), teams&org (for mention)
|
// user&repo, format&style®exp (for external issue pattern), teams&org (for mention)
|
||||||
// BranchNameSubURL (for iframe&asciicast)
|
// BranchNameSubURL (for iframe&asciicast)
|
||||||
// markupAllowShortIssuePattern, markupContentMode (wiki)
|
// markupAllowShortIssuePattern
|
||||||
// markdownLineBreakStyle (comment, document)
|
// markdownLineBreakStyle (comment, document)
|
||||||
Metas map[string]string
|
Metas map[string]string
|
||||||
|
|
||||||
GitRepo *git.Repository
|
// used by external render. the router "/org/repo/render/..." will output the rendered content in a standalone page
|
||||||
Repo gitrepo.Repository
|
InStandalonePage bool
|
||||||
ShaExistCache map[string]bool
|
}
|
||||||
cancelFn func()
|
|
||||||
SidebarTocNode ast.Node
|
|
||||||
RenderMetaAs RenderMetaMode
|
|
||||||
InStandalonePage bool // used by external render. the router "/org/repo/render/..." will output the rendered content in a standalone page
|
|
||||||
|
|
||||||
|
// RenderContext represents a render context
|
||||||
|
type RenderContext struct {
|
||||||
|
ctx context.Context
|
||||||
|
|
||||||
|
SidebarTocNode ast.Node
|
||||||
|
|
||||||
|
RenderHelper RenderHelper
|
||||||
|
RenderOptions RenderOptions
|
||||||
RenderInternal internal.RenderInternal
|
RenderInternal internal.RenderInternal
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cancel runs any cleanup functions that have been registered for this Ctx
|
func (ctx *RenderContext) Deadline() (deadline time.Time, ok bool) {
|
||||||
func (ctx *RenderContext) Cancel() {
|
return ctx.ctx.Deadline()
|
||||||
if ctx == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ctx.ShaExistCache = map[string]bool{}
|
|
||||||
if ctx.cancelFn == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ctx.cancelFn()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// AddCancel adds the provided fn as a Cleanup for this Ctx
|
func (ctx *RenderContext) Done() <-chan struct{} {
|
||||||
func (ctx *RenderContext) AddCancel(fn func()) {
|
return ctx.ctx.Done()
|
||||||
if ctx == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
oldCancelFn := ctx.cancelFn
|
|
||||||
if oldCancelFn == nil {
|
|
||||||
ctx.cancelFn = fn
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ctx.cancelFn = func() {
|
|
||||||
defer oldCancelFn()
|
|
||||||
fn()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ctx *RenderContext) IsMarkupContentWiki() bool {
|
func (ctx *RenderContext) Err() error {
|
||||||
return ctx.Metas != nil && ctx.Metas["markupContentMode"] == "wiki"
|
return ctx.ctx.Err()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctx *RenderContext) Value(key any) any {
|
||||||
|
return ctx.ctx.Value(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ context.Context = (*RenderContext)(nil)
|
||||||
|
|
||||||
|
func NewRenderContext(ctx context.Context) *RenderContext {
|
||||||
|
return &RenderContext{ctx: ctx, RenderHelper: &SimpleRenderHelper{}}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctx *RenderContext) WithMarkupType(typ string) *RenderContext {
|
||||||
|
ctx.RenderOptions.MarkupType = typ
|
||||||
|
return ctx
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctx *RenderContext) WithRelativePath(path string) *RenderContext {
|
||||||
|
ctx.RenderOptions.RelativePath = path
|
||||||
|
return ctx
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctx *RenderContext) WithMetas(metas map[string]string) *RenderContext {
|
||||||
|
ctx.RenderOptions.Metas = metas
|
||||||
|
return ctx
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctx *RenderContext) WithInStandalonePage(v bool) *RenderContext {
|
||||||
|
ctx.RenderOptions.InStandalonePage = v
|
||||||
|
return ctx
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctx *RenderContext) WithUseAbsoluteLink(v bool) *RenderContext {
|
||||||
|
ctx.RenderOptions.UseAbsoluteLink = v
|
||||||
|
return ctx
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctx *RenderContext) WithHelper(helper RenderHelper) *RenderContext {
|
||||||
|
ctx.RenderHelper = helper
|
||||||
|
return ctx
|
||||||
}
|
}
|
||||||
|
|
||||||
// Render renders markup file to HTML with all specific handling stuff.
|
// Render renders markup file to HTML with all specific handling stuff.
|
||||||
func Render(ctx *RenderContext, input io.Reader, output io.Writer) error {
|
func Render(ctx *RenderContext, input io.Reader, output io.Writer) error {
|
||||||
if ctx.MarkupType == "" && ctx.RelativePath != "" {
|
if ctx.RenderOptions.MarkupType == "" && ctx.RenderOptions.RelativePath != "" {
|
||||||
ctx.MarkupType = DetectMarkupTypeByFileName(ctx.RelativePath)
|
ctx.RenderOptions.MarkupType = DetectMarkupTypeByFileName(ctx.RenderOptions.RelativePath)
|
||||||
if ctx.MarkupType == "" {
|
if ctx.RenderOptions.MarkupType == "" {
|
||||||
return util.NewInvalidArgumentErrorf("unsupported file to render: %q", ctx.RelativePath)
|
return util.NewInvalidArgumentErrorf("unsupported file to render: %q", ctx.RenderOptions.RelativePath)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
renderer := renderers[ctx.MarkupType]
|
renderer := renderers[ctx.RenderOptions.MarkupType]
|
||||||
if renderer == nil {
|
if renderer == nil {
|
||||||
return util.NewInvalidArgumentErrorf("unsupported markup type: %q", ctx.MarkupType)
|
return util.NewInvalidArgumentErrorf("unsupported markup type: %q", ctx.RenderOptions.MarkupType)
|
||||||
}
|
}
|
||||||
|
|
||||||
if ctx.RelativePath != "" {
|
if ctx.RenderOptions.RelativePath != "" {
|
||||||
if externalRender, ok := renderer.(ExternalRenderer); ok && externalRender.DisplayInIFrame() {
|
if externalRender, ok := renderer.(ExternalRenderer); ok && externalRender.DisplayInIFrame() {
|
||||||
if !ctx.InStandalonePage {
|
if !ctx.RenderOptions.InStandalonePage {
|
||||||
// for an external "DisplayInIFrame" render, it could only output its content in a standalone page
|
// for an external "DisplayInIFrame" render, it could only output its content in a standalone page
|
||||||
// otherwise, a <iframe> should be outputted to embed the external rendered page
|
// otherwise, a <iframe> should be outputted to embed the external rendered page
|
||||||
return renderIFrame(ctx, output)
|
return renderIFrame(ctx, output)
|
||||||
@ -151,10 +173,10 @@ width="100%%" height="0" scrolling="no" frameborder="0" style="overflow: hidden"
|
|||||||
sandbox="allow-scripts"
|
sandbox="allow-scripts"
|
||||||
></iframe>`,
|
></iframe>`,
|
||||||
setting.AppSubURL,
|
setting.AppSubURL,
|
||||||
url.PathEscape(ctx.Metas["user"]),
|
url.PathEscape(ctx.RenderOptions.Metas["user"]),
|
||||||
url.PathEscape(ctx.Metas["repo"]),
|
url.PathEscape(ctx.RenderOptions.Metas["repo"]),
|
||||||
ctx.Metas["BranchNameSubURL"],
|
ctx.RenderOptions.Metas["BranchNameSubURL"],
|
||||||
url.PathEscape(ctx.RelativePath),
|
url.PathEscape(ctx.RenderOptions.RelativePath),
|
||||||
))
|
))
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -168,6 +190,10 @@ func pipes() (io.ReadCloser, io.WriteCloser, func()) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func render(ctx *RenderContext, renderer Renderer, input io.Reader, output io.Writer) error {
|
func render(ctx *RenderContext, renderer Renderer, input io.Reader, output io.Writer) error {
|
||||||
|
if ctx.RenderHelper != nil {
|
||||||
|
defer ctx.RenderHelper.CleanUp()
|
||||||
|
}
|
||||||
|
|
||||||
finalProcessor := ctx.RenderInternal.Init(output)
|
finalProcessor := ctx.RenderInternal.Init(output)
|
||||||
defer finalProcessor.Close()
|
defer finalProcessor.Close()
|
||||||
|
|
||||||
@ -176,7 +202,7 @@ func render(ctx *RenderContext, renderer Renderer, input io.Reader, output io.Wr
|
|||||||
pr1, pw1, close1 := pipes()
|
pr1, pw1, close1 := pipes()
|
||||||
defer close1()
|
defer close1()
|
||||||
|
|
||||||
eg, _ := errgroup.WithContext(ctx.Ctx)
|
eg, _ := errgroup.WithContext(ctx)
|
||||||
var pw2 io.WriteCloser = util.NopCloser{Writer: finalProcessor}
|
var pw2 io.WriteCloser = util.NopCloser{Writer: finalProcessor}
|
||||||
|
|
||||||
if r, ok := renderer.(ExternalRenderer); !ok || !r.SanitizerDisabled() {
|
if r, ok := renderer.(ExternalRenderer); !ok || !r.SanitizerDisabled() {
|
||||||
@ -209,11 +235,8 @@ func render(ctx *RenderContext, renderer Renderer, input io.Reader, output io.Wr
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Init initializes the render global variables
|
// Init initializes the render global variables
|
||||||
func Init(ph *ProcessorHelper) {
|
func Init(renderHelpFuncs *RenderHelperFuncs) {
|
||||||
if ph != nil {
|
DefaultRenderHelperFuncs = renderHelpFuncs
|
||||||
DefaultProcessorHelper = *ph
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(setting.Markdown.CustomURLSchemes) > 0 {
|
if len(setting.Markdown.CustomURLSchemes) > 0 {
|
||||||
CustomLinkURLSchemes(setting.Markdown.CustomURLSchemes)
|
CustomLinkURLSchemes(setting.Markdown.CustomURLSchemes)
|
||||||
}
|
}
|
||||||
@ -230,3 +253,42 @@ func Init(ph *ProcessorHelper) {
|
|||||||
func ComposeSimpleDocumentMetas() map[string]string {
|
func ComposeSimpleDocumentMetas() map[string]string {
|
||||||
return map[string]string{"markdownLineBreakStyle": "document"}
|
return map[string]string{"markdownLineBreakStyle": "document"}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type TestRenderHelper struct {
|
||||||
|
ctx *RenderContext
|
||||||
|
BaseLink string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *TestRenderHelper) CleanUp() {}
|
||||||
|
|
||||||
|
func (r *TestRenderHelper) IsCommitIDExisting(commitID string) bool {
|
||||||
|
return strings.HasPrefix(commitID, "65f1bf2") //|| strings.HasPrefix(commitID, "88fc37a")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *TestRenderHelper) ResolveLink(link string, likeType LinkType) string {
|
||||||
|
return r.ctx.ResolveLinkRelative(r.BaseLink, "", link)
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ RenderHelper = (*TestRenderHelper)(nil)
|
||||||
|
|
||||||
|
// NewTestRenderContext is a helper function to create a RenderContext for testing purpose
|
||||||
|
// It accepts string (BaseLink), map[string]string (Metas)
|
||||||
|
func NewTestRenderContext(baseLinkOrMetas ...any) *RenderContext {
|
||||||
|
if !setting.IsInTesting {
|
||||||
|
panic("NewTestRenderContext should only be used in testing")
|
||||||
|
}
|
||||||
|
helper := &TestRenderHelper{}
|
||||||
|
ctx := NewRenderContext(context.Background()).WithHelper(helper)
|
||||||
|
helper.ctx = ctx
|
||||||
|
for _, v := range baseLinkOrMetas {
|
||||||
|
switch v := v.(type) {
|
||||||
|
case string:
|
||||||
|
helper.BaseLink = v
|
||||||
|
case map[string]string:
|
||||||
|
ctx = ctx.WithMetas(v)
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf("unknown type %T", v))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ctx
|
||||||
|
}
|
||||||
|
@ -6,16 +6,52 @@ package markup
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"html/template"
|
"html/template"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/modules/setting"
|
||||||
)
|
)
|
||||||
|
|
||||||
// ProcessorHelper is a helper for the rendering processors (it could be renamed to RenderHelper in the future).
|
type LinkType string
|
||||||
// The main purpose of this helper is to decouple some functions which are not directly available in this package.
|
|
||||||
type ProcessorHelper struct {
|
|
||||||
IsUsernameMentionable func(ctx context.Context, username string) bool
|
|
||||||
|
|
||||||
ElementDir string // the direction of the elements, eg: "ltr", "rtl", "auto", default to no direction attribute
|
const (
|
||||||
|
LinkTypeApp LinkType = "app" // the link is relative to the AppSubURL
|
||||||
|
LinkTypeDefault LinkType = "default" // the link is relative to the default base (eg: repo link, or current ref tree path)
|
||||||
|
LinkTypeMedia LinkType = "media" // the link should be used to access media files (images, videos)
|
||||||
|
LinkTypeRaw LinkType = "raw" // not really useful, mainly for environment GITEA_PREFIX_RAW for external renders
|
||||||
|
)
|
||||||
|
|
||||||
|
type RenderHelper interface {
|
||||||
|
CleanUp()
|
||||||
|
|
||||||
|
// TODO: such dependency is not ideal. We should decouple the processors step by step.
|
||||||
|
// It should make the render choose different processors for different purposes,
|
||||||
|
// but not make processors to guess "is it rendering a comment or a wiki?" or "does it need to check commit ID?"
|
||||||
|
|
||||||
|
IsCommitIDExisting(commitID string) bool
|
||||||
|
ResolveLink(link string, likeType LinkType) string
|
||||||
|
}
|
||||||
|
|
||||||
|
// RenderHelperFuncs is used to decouple cycle-import
|
||||||
|
// At the moment there are different packages:
|
||||||
|
// modules/markup: basic markup rendering
|
||||||
|
// models/renderhelper: need to access models and git repo, and models/issues needs it
|
||||||
|
// services/markup: some real helper functions could only be provided here because it needs to access various services & templates
|
||||||
|
type RenderHelperFuncs struct {
|
||||||
|
IsUsernameMentionable func(ctx context.Context, username string) bool
|
||||||
RenderRepoFileCodePreview func(ctx context.Context, options RenderCodePreviewOptions) (template.HTML, error)
|
RenderRepoFileCodePreview func(ctx context.Context, options RenderCodePreviewOptions) (template.HTML, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
var DefaultProcessorHelper ProcessorHelper
|
var DefaultRenderHelperFuncs *RenderHelperFuncs
|
||||||
|
|
||||||
|
type SimpleRenderHelper struct{}
|
||||||
|
|
||||||
|
func (r *SimpleRenderHelper) CleanUp() {}
|
||||||
|
|
||||||
|
func (r *SimpleRenderHelper) IsCommitIDExisting(commitID string) bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *SimpleRenderHelper) ResolveLink(link string, likeType LinkType) string {
|
||||||
|
return resolveLinkRelative(context.Background(), setting.AppSubURL+"/", "", link, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ RenderHelper = (*SimpleRenderHelper)(nil)
|
||||||
|
42
modules/markup/render_link.go
Normal file
42
modules/markup/render_link.go
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package markup
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/modules/httplib"
|
||||||
|
"code.gitea.io/gitea/modules/setting"
|
||||||
|
"code.gitea.io/gitea/modules/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
func resolveLinkRelative(ctx context.Context, base, cur, link string, absolute bool) (finalLink string) {
|
||||||
|
if IsFullURLString(link) {
|
||||||
|
return link
|
||||||
|
}
|
||||||
|
if strings.HasPrefix(link, "/") {
|
||||||
|
if strings.HasPrefix(link, base) && strings.Count(base, "/") >= 4 {
|
||||||
|
// a trick to tolerate that some users were using absolut paths (the old gitea's behavior)
|
||||||
|
finalLink = link
|
||||||
|
} else {
|
||||||
|
finalLink = util.URLJoin(base, "./", link)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
finalLink = util.URLJoin(base, "./", cur, link)
|
||||||
|
}
|
||||||
|
finalLink = strings.TrimSuffix(finalLink, "/")
|
||||||
|
if absolute {
|
||||||
|
finalLink = httplib.MakeAbsoluteURL(ctx, finalLink)
|
||||||
|
}
|
||||||
|
return finalLink
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctx *RenderContext) ResolveLinkRelative(base, cur, link string) (finalLink string) {
|
||||||
|
return resolveLinkRelative(ctx, base, cur, link, ctx.RenderOptions.UseAbsoluteLink)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctx *RenderContext) ResolveLinkApp(link string) string {
|
||||||
|
return ctx.ResolveLinkRelative(setting.AppSubURL+"/", "", link)
|
||||||
|
}
|
27
modules/markup/render_link_test.go
Normal file
27
modules/markup/render_link_test.go
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package markup
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/modules/setting"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestResolveLinkRelative(t *testing.T) {
|
||||||
|
ctx := context.Background()
|
||||||
|
setting.AppURL = "http://localhost:3000"
|
||||||
|
assert.Equal(t, "/a", resolveLinkRelative(ctx, "/a", "", "", false))
|
||||||
|
assert.Equal(t, "/a/b", resolveLinkRelative(ctx, "/a", "b", "", false))
|
||||||
|
assert.Equal(t, "/a/b/c", resolveLinkRelative(ctx, "/a", "b", "c", false))
|
||||||
|
assert.Equal(t, "/a/c", resolveLinkRelative(ctx, "/a", "b", "/c", false))
|
||||||
|
assert.Equal(t, "http://localhost:3000/a", resolveLinkRelative(ctx, "/a", "", "", true))
|
||||||
|
|
||||||
|
// some users might have used absolute paths a lot, so if the prefix overlaps and has enough slashes, we should tolerate it
|
||||||
|
assert.Equal(t, "/owner/repo/foo/owner/repo/foo/bar/xxx", resolveLinkRelative(ctx, "/owner/repo/foo", "", "/owner/repo/foo/bar/xxx", false))
|
||||||
|
assert.Equal(t, "/owner/repo/foo/bar/xxx", resolveLinkRelative(ctx, "/owner/repo/foo/bar", "", "/owner/repo/foo/bar/xxx", false))
|
||||||
|
}
|
@ -1,56 +0,0 @@
|
|||||||
// Copyright 2024 The Gitea Authors. All rights reserved.
|
|
||||||
// SPDX-License-Identifier: MIT
|
|
||||||
|
|
||||||
package markup
|
|
||||||
|
|
||||||
import (
|
|
||||||
"code.gitea.io/gitea/modules/setting"
|
|
||||||
"code.gitea.io/gitea/modules/util"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Links struct {
|
|
||||||
AbsolutePrefix bool // add absolute URL prefix to auto-resolved links like "#issue", but not for pre-provided links and medias
|
|
||||||
Base string // base prefix for pre-provided links and medias (images, videos), usually it is the path to the repo
|
|
||||||
BranchPath string // actually it is the ref path, eg: "branch/features/feat-12", "tag/v1.0"
|
|
||||||
TreePath string // the dir of the file, eg: "doc" if the file "doc/CHANGE.md" is being rendered
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *Links) Prefix() string {
|
|
||||||
if l.AbsolutePrefix {
|
|
||||||
return setting.AppURL
|
|
||||||
}
|
|
||||||
return setting.AppSubURL
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *Links) HasBranchInfo() bool {
|
|
||||||
return l.BranchPath != ""
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *Links) SrcLink() string {
|
|
||||||
return util.URLJoin(l.Base, "src", l.BranchPath, l.TreePath)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *Links) MediaLink() string {
|
|
||||||
return util.URLJoin(l.Base, "media", l.BranchPath, l.TreePath)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *Links) RawLink() string {
|
|
||||||
return util.URLJoin(l.Base, "raw", l.BranchPath, l.TreePath)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *Links) WikiLink() string {
|
|
||||||
return util.URLJoin(l.Base, "wiki")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *Links) WikiRawLink() string {
|
|
||||||
return util.URLJoin(l.Base, "wiki/raw")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *Links) ResolveMediaLink(isWiki bool) string {
|
|
||||||
if isWiki {
|
|
||||||
return l.WikiRawLink()
|
|
||||||
} else if l.HasBranchInfo() {
|
|
||||||
return l.MediaLink()
|
|
||||||
}
|
|
||||||
return l.Base
|
|
||||||
}
|
|
@ -6,7 +6,7 @@ package markup
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"io"
|
"io"
|
||||||
"path/filepath"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
@ -55,7 +55,7 @@ func RegisterRenderer(renderer Renderer) {
|
|||||||
|
|
||||||
// GetRendererByFileName get renderer by filename
|
// GetRendererByFileName get renderer by filename
|
||||||
func GetRendererByFileName(filename string) Renderer {
|
func GetRendererByFileName(filename string) Renderer {
|
||||||
extension := strings.ToLower(filepath.Ext(filename))
|
extension := strings.ToLower(path.Ext(filename))
|
||||||
return extRenderers[extension]
|
return extRenderers[extension]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -26,6 +26,9 @@ func (st *Sanitizer) createDefaultPolicy() *bluemonday.Policy {
|
|||||||
policy.AllowAttrs("type").Matching(regexp.MustCompile(`^checkbox$`)).OnElements("input")
|
policy.AllowAttrs("type").Matching(regexp.MustCompile(`^checkbox$`)).OnElements("input")
|
||||||
policy.AllowAttrs("checked", "disabled", "data-source-position").OnElements("input")
|
policy.AllowAttrs("checked", "disabled", "data-source-position").OnElements("input")
|
||||||
|
|
||||||
|
// Chroma always uses 1-2 letters for style names, we could tolerate it at the moment
|
||||||
|
policy.AllowAttrs("class").Matching(regexp.MustCompile(`^\w{0,2}$`)).OnElements("span")
|
||||||
|
|
||||||
// Custom URL-Schemes
|
// Custom URL-Schemes
|
||||||
if len(setting.Markdown.CustomURLSchemes) > 0 {
|
if len(setting.Markdown.CustomURLSchemes) > 0 {
|
||||||
policy.AllowURLSchemes(setting.Markdown.CustomURLSchemes...)
|
policy.AllowURLSchemes(setting.Markdown.CustomURLSchemes...)
|
||||||
|
@ -19,6 +19,7 @@ func TestSanitizer(t *testing.T) {
|
|||||||
// Code highlighting class
|
// Code highlighting class
|
||||||
`<code class="random string"></code>`, `<code></code>`,
|
`<code class="random string"></code>`, `<code></code>`,
|
||||||
`<code class="language-random ui tab active menu attached animating sidebar following bar center"></code>`, `<code></code>`,
|
`<code class="language-random ui tab active menu attached animating sidebar following bar center"></code>`, `<code></code>`,
|
||||||
|
`<span class="k"></span><span class="nb"></span>`, `<span class="k"></span><span class="nb"></span>`,
|
||||||
|
|
||||||
// Input checkbox
|
// Input checkbox
|
||||||
`<input type="hidden">`, ``,
|
`<input type="hidden">`, ``,
|
||||||
|
@ -43,6 +43,7 @@ type MinioStorageConfig struct {
|
|||||||
Endpoint string `ini:"MINIO_ENDPOINT" json:",omitempty"`
|
Endpoint string `ini:"MINIO_ENDPOINT" json:",omitempty"`
|
||||||
AccessKeyID string `ini:"MINIO_ACCESS_KEY_ID" json:",omitempty"`
|
AccessKeyID string `ini:"MINIO_ACCESS_KEY_ID" json:",omitempty"`
|
||||||
SecretAccessKey string `ini:"MINIO_SECRET_ACCESS_KEY" json:",omitempty"`
|
SecretAccessKey string `ini:"MINIO_SECRET_ACCESS_KEY" json:",omitempty"`
|
||||||
|
IamEndpoint string `ini:"MINIO_IAM_ENDPOINT" json:",omitempty"`
|
||||||
Bucket string `ini:"MINIO_BUCKET" json:",omitempty"`
|
Bucket string `ini:"MINIO_BUCKET" json:",omitempty"`
|
||||||
Location string `ini:"MINIO_LOCATION" json:",omitempty"`
|
Location string `ini:"MINIO_LOCATION" json:",omitempty"`
|
||||||
BasePath string `ini:"MINIO_BASE_PATH" json:",omitempty"`
|
BasePath string `ini:"MINIO_BASE_PATH" json:",omitempty"`
|
||||||
|
@ -470,6 +470,19 @@ MINIO_BASE_PATH = /prefix
|
|||||||
cfg, err = NewConfigProviderFromData(`
|
cfg, err = NewConfigProviderFromData(`
|
||||||
[storage]
|
[storage]
|
||||||
STORAGE_TYPE = minio
|
STORAGE_TYPE = minio
|
||||||
|
MINIO_IAM_ENDPOINT = 127.0.0.1
|
||||||
|
MINIO_USE_SSL = true
|
||||||
|
MINIO_BASE_PATH = /prefix
|
||||||
|
`)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.NoError(t, loadRepoArchiveFrom(cfg))
|
||||||
|
assert.EqualValues(t, "127.0.0.1", RepoArchive.Storage.MinioConfig.IamEndpoint)
|
||||||
|
assert.EqualValues(t, true, RepoArchive.Storage.MinioConfig.UseSSL)
|
||||||
|
assert.EqualValues(t, "/prefix/repo-archive/", RepoArchive.Storage.MinioConfig.BasePath)
|
||||||
|
|
||||||
|
cfg, err = NewConfigProviderFromData(`
|
||||||
|
[storage]
|
||||||
|
STORAGE_TYPE = minio
|
||||||
MINIO_ACCESS_KEY_ID = my_access_key
|
MINIO_ACCESS_KEY_ID = my_access_key
|
||||||
MINIO_SECRET_ACCESS_KEY = my_secret_key
|
MINIO_SECRET_ACCESS_KEY = my_secret_key
|
||||||
MINIO_USE_SSL = true
|
MINIO_USE_SSL = true
|
||||||
|
@ -97,7 +97,7 @@ func NewMinioStorage(ctx context.Context, cfg *setting.Storage) (ObjectStorage,
|
|||||||
}
|
}
|
||||||
|
|
||||||
minioClient, err := minio.New(config.Endpoint, &minio.Options{
|
minioClient, err := minio.New(config.Endpoint, &minio.Options{
|
||||||
Creds: buildMinioCredentials(config, credentials.DefaultIAMRoleEndpoint),
|
Creds: buildMinioCredentials(config),
|
||||||
Secure: config.UseSSL,
|
Secure: config.UseSSL,
|
||||||
Transport: &http.Transport{TLSClientConfig: &tls.Config{InsecureSkipVerify: config.InsecureSkipVerify}},
|
Transport: &http.Transport{TLSClientConfig: &tls.Config{InsecureSkipVerify: config.InsecureSkipVerify}},
|
||||||
Region: config.Location,
|
Region: config.Location,
|
||||||
@ -164,7 +164,7 @@ func (m *MinioStorage) buildMinioDirPrefix(p string) string {
|
|||||||
return p
|
return p
|
||||||
}
|
}
|
||||||
|
|
||||||
func buildMinioCredentials(config setting.MinioStorageConfig, iamEndpoint string) *credentials.Credentials {
|
func buildMinioCredentials(config setting.MinioStorageConfig) *credentials.Credentials {
|
||||||
// If static credentials are provided, use those
|
// If static credentials are provided, use those
|
||||||
if config.AccessKeyID != "" {
|
if config.AccessKeyID != "" {
|
||||||
return credentials.NewStaticV4(config.AccessKeyID, config.SecretAccessKey, "")
|
return credentials.NewStaticV4(config.AccessKeyID, config.SecretAccessKey, "")
|
||||||
@ -184,7 +184,9 @@ func buildMinioCredentials(config setting.MinioStorageConfig, iamEndpoint string
|
|||||||
&credentials.FileAWSCredentials{},
|
&credentials.FileAWSCredentials{},
|
||||||
// read IAM role from EC2 metadata endpoint if available
|
// read IAM role from EC2 metadata endpoint if available
|
||||||
&credentials.IAM{
|
&credentials.IAM{
|
||||||
Endpoint: iamEndpoint,
|
// passing in an empty Endpoint lets the IAM Provider
|
||||||
|
// decide which endpoint to resolve internally
|
||||||
|
Endpoint: config.IamEndpoint,
|
||||||
Client: &http.Client{
|
Client: &http.Client{
|
||||||
Transport: http.DefaultTransport,
|
Transport: http.DefaultTransport,
|
||||||
},
|
},
|
||||||
|
@ -107,8 +107,9 @@ func TestMinioCredentials(t *testing.T) {
|
|||||||
cfg := setting.MinioStorageConfig{
|
cfg := setting.MinioStorageConfig{
|
||||||
AccessKeyID: ExpectedAccessKey,
|
AccessKeyID: ExpectedAccessKey,
|
||||||
SecretAccessKey: ExpectedSecretAccessKey,
|
SecretAccessKey: ExpectedSecretAccessKey,
|
||||||
|
IamEndpoint: FakeEndpoint,
|
||||||
}
|
}
|
||||||
creds := buildMinioCredentials(cfg, FakeEndpoint)
|
creds := buildMinioCredentials(cfg)
|
||||||
v, err := creds.Get()
|
v, err := creds.Get()
|
||||||
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
@ -117,13 +118,15 @@ func TestMinioCredentials(t *testing.T) {
|
|||||||
})
|
})
|
||||||
|
|
||||||
t.Run("Chain", func(t *testing.T) {
|
t.Run("Chain", func(t *testing.T) {
|
||||||
cfg := setting.MinioStorageConfig{}
|
cfg := setting.MinioStorageConfig{
|
||||||
|
IamEndpoint: FakeEndpoint,
|
||||||
|
}
|
||||||
|
|
||||||
t.Run("EnvMinio", func(t *testing.T) {
|
t.Run("EnvMinio", func(t *testing.T) {
|
||||||
t.Setenv("MINIO_ACCESS_KEY", ExpectedAccessKey+"Minio")
|
t.Setenv("MINIO_ACCESS_KEY", ExpectedAccessKey+"Minio")
|
||||||
t.Setenv("MINIO_SECRET_KEY", ExpectedSecretAccessKey+"Minio")
|
t.Setenv("MINIO_SECRET_KEY", ExpectedSecretAccessKey+"Minio")
|
||||||
|
|
||||||
creds := buildMinioCredentials(cfg, FakeEndpoint)
|
creds := buildMinioCredentials(cfg)
|
||||||
v, err := creds.Get()
|
v, err := creds.Get()
|
||||||
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
@ -135,7 +138,7 @@ func TestMinioCredentials(t *testing.T) {
|
|||||||
t.Setenv("AWS_ACCESS_KEY", ExpectedAccessKey+"AWS")
|
t.Setenv("AWS_ACCESS_KEY", ExpectedAccessKey+"AWS")
|
||||||
t.Setenv("AWS_SECRET_KEY", ExpectedSecretAccessKey+"AWS")
|
t.Setenv("AWS_SECRET_KEY", ExpectedSecretAccessKey+"AWS")
|
||||||
|
|
||||||
creds := buildMinioCredentials(cfg, FakeEndpoint)
|
creds := buildMinioCredentials(cfg)
|
||||||
v, err := creds.Get()
|
v, err := creds.Get()
|
||||||
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
@ -144,11 +147,11 @@ func TestMinioCredentials(t *testing.T) {
|
|||||||
})
|
})
|
||||||
|
|
||||||
t.Run("FileMinio", func(t *testing.T) {
|
t.Run("FileMinio", func(t *testing.T) {
|
||||||
t.Setenv("MINIO_SHARED_CREDENTIALS_FILE", "testdata/minio.json")
|
|
||||||
// prevent loading any actual credentials files from the user
|
// prevent loading any actual credentials files from the user
|
||||||
|
t.Setenv("MINIO_SHARED_CREDENTIALS_FILE", "testdata/minio.json")
|
||||||
t.Setenv("AWS_SHARED_CREDENTIALS_FILE", "testdata/fake")
|
t.Setenv("AWS_SHARED_CREDENTIALS_FILE", "testdata/fake")
|
||||||
|
|
||||||
creds := buildMinioCredentials(cfg, FakeEndpoint)
|
creds := buildMinioCredentials(cfg)
|
||||||
v, err := creds.Get()
|
v, err := creds.Get()
|
||||||
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
@ -161,7 +164,7 @@ func TestMinioCredentials(t *testing.T) {
|
|||||||
t.Setenv("MINIO_SHARED_CREDENTIALS_FILE", "testdata/fake.json")
|
t.Setenv("MINIO_SHARED_CREDENTIALS_FILE", "testdata/fake.json")
|
||||||
t.Setenv("AWS_SHARED_CREDENTIALS_FILE", "testdata/aws_credentials")
|
t.Setenv("AWS_SHARED_CREDENTIALS_FILE", "testdata/aws_credentials")
|
||||||
|
|
||||||
creds := buildMinioCredentials(cfg, FakeEndpoint)
|
creds := buildMinioCredentials(cfg)
|
||||||
v, err := creds.Get()
|
v, err := creds.Get()
|
||||||
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
@ -187,7 +190,9 @@ func TestMinioCredentials(t *testing.T) {
|
|||||||
defer server.Close()
|
defer server.Close()
|
||||||
|
|
||||||
// Use the provided EC2 Instance Metadata server
|
// Use the provided EC2 Instance Metadata server
|
||||||
creds := buildMinioCredentials(cfg, server.URL)
|
creds := buildMinioCredentials(setting.MinioStorageConfig{
|
||||||
|
IamEndpoint: server.URL,
|
||||||
|
})
|
||||||
v, err := creds.Get()
|
v, err := creds.Get()
|
||||||
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
|
@ -38,10 +38,7 @@ func (ut *RenderUtils) RenderCommitMessage(msg string, metas map[string]string)
|
|||||||
cleanMsg := template.HTMLEscapeString(msg)
|
cleanMsg := template.HTMLEscapeString(msg)
|
||||||
// we can safely assume that it will not return any error, since there
|
// we can safely assume that it will not return any error, since there
|
||||||
// shouldn't be any special HTML.
|
// shouldn't be any special HTML.
|
||||||
fullMessage, err := markup.RenderCommitMessage(&markup.RenderContext{
|
fullMessage, err := markup.RenderCommitMessage(markup.NewRenderContext(ut.ctx).WithMetas(metas), cleanMsg)
|
||||||
Ctx: ut.ctx,
|
|
||||||
Metas: metas,
|
|
||||||
}, cleanMsg)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("RenderCommitMessage: %v", err)
|
log.Error("RenderCommitMessage: %v", err)
|
||||||
return ""
|
return ""
|
||||||
@ -68,10 +65,7 @@ func (ut *RenderUtils) RenderCommitMessageLinkSubject(msg, urlDefault string, me
|
|||||||
|
|
||||||
// we can safely assume that it will not return any error, since there
|
// we can safely assume that it will not return any error, since there
|
||||||
// shouldn't be any special HTML.
|
// shouldn't be any special HTML.
|
||||||
renderedMessage, err := markup.RenderCommitMessageSubject(&markup.RenderContext{
|
renderedMessage, err := markup.RenderCommitMessageSubject(markup.NewRenderContext(ut.ctx).WithMetas(metas), urlDefault, template.HTMLEscapeString(msgLine))
|
||||||
Ctx: ut.ctx,
|
|
||||||
Metas: metas,
|
|
||||||
}, urlDefault, template.HTMLEscapeString(msgLine))
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("RenderCommitMessageSubject: %v", err)
|
log.Error("RenderCommitMessageSubject: %v", err)
|
||||||
return ""
|
return ""
|
||||||
@ -93,10 +87,7 @@ func (ut *RenderUtils) RenderCommitBody(msg string, metas map[string]string) tem
|
|||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
renderedMessage, err := markup.RenderCommitMessage(&markup.RenderContext{
|
renderedMessage, err := markup.RenderCommitMessage(markup.NewRenderContext(ut.ctx).WithMetas(metas), template.HTMLEscapeString(msgLine))
|
||||||
Ctx: ut.ctx,
|
|
||||||
Metas: metas,
|
|
||||||
}, template.HTMLEscapeString(msgLine))
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("RenderCommitMessage: %v", err)
|
log.Error("RenderCommitMessage: %v", err)
|
||||||
return ""
|
return ""
|
||||||
@ -115,10 +106,7 @@ func renderCodeBlock(htmlEscapedTextToRender template.HTML) template.HTML {
|
|||||||
|
|
||||||
// RenderIssueTitle renders issue/pull title with defined post processors
|
// RenderIssueTitle renders issue/pull title with defined post processors
|
||||||
func (ut *RenderUtils) RenderIssueTitle(text string, metas map[string]string) template.HTML {
|
func (ut *RenderUtils) RenderIssueTitle(text string, metas map[string]string) template.HTML {
|
||||||
renderedText, err := markup.RenderIssueTitle(&markup.RenderContext{
|
renderedText, err := markup.RenderIssueTitle(markup.NewRenderContext(ut.ctx).WithMetas(metas), template.HTMLEscapeString(text))
|
||||||
Ctx: ut.ctx,
|
|
||||||
Metas: metas,
|
|
||||||
}, template.HTMLEscapeString(text))
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("RenderIssueTitle: %v", err)
|
log.Error("RenderIssueTitle: %v", err)
|
||||||
return ""
|
return ""
|
||||||
@ -186,7 +174,7 @@ func (ut *RenderUtils) RenderLabel(label *issues_model.Label) template.HTML {
|
|||||||
|
|
||||||
// RenderEmoji renders html text with emoji post processors
|
// RenderEmoji renders html text with emoji post processors
|
||||||
func (ut *RenderUtils) RenderEmoji(text string) template.HTML {
|
func (ut *RenderUtils) RenderEmoji(text string) template.HTML {
|
||||||
renderedText, err := markup.RenderEmoji(&markup.RenderContext{Ctx: ut.ctx}, template.HTMLEscapeString(text))
|
renderedText, err := markup.RenderEmoji(markup.NewRenderContext(ut.ctx), template.HTMLEscapeString(text))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("RenderEmoji: %v", err)
|
log.Error("RenderEmoji: %v", err)
|
||||||
return ""
|
return ""
|
||||||
@ -208,10 +196,7 @@ func reactionToEmoji(reaction string) template.HTML {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (ut *RenderUtils) MarkdownToHtml(input string) template.HTML { //nolint:revive
|
func (ut *RenderUtils) MarkdownToHtml(input string) template.HTML { //nolint:revive
|
||||||
output, err := markdown.RenderString(&markup.RenderContext{
|
output, err := markdown.RenderString(markup.NewRenderContext(ut.ctx).WithMetas(markup.ComposeSimpleDocumentMetas()), input)
|
||||||
Ctx: ut.ctx,
|
|
||||||
Metas: markup.ComposeSimpleDocumentMetas(),
|
|
||||||
}, input)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("RenderString: %v", err)
|
log.Error("RenderString: %v", err)
|
||||||
}
|
}
|
||||||
|
@ -59,7 +59,7 @@ func TestMain(m *testing.M) {
|
|||||||
if err := git.InitSimple(context.Background()); err != nil {
|
if err := git.InitSimple(context.Background()); err != nil {
|
||||||
log.Fatal("git init failed, err: %v", err)
|
log.Fatal("git init failed, err: %v", err)
|
||||||
}
|
}
|
||||||
markup.Init(&markup.ProcessorHelper{
|
markup.Init(&markup.RenderHelperFuncs{
|
||||||
IsUsernameMentionable: func(ctx context.Context, username string) bool {
|
IsUsernameMentionable: func(ctx context.Context, username string) bool {
|
||||||
return username == "mention-user"
|
return username == "mention-user"
|
||||||
},
|
},
|
||||||
@ -74,7 +74,7 @@ func newTestRenderUtils() *RenderUtils {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestRenderCommitBody(t *testing.T) {
|
func TestRenderCommitBody(t *testing.T) {
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
type args struct {
|
type args struct {
|
||||||
msg string
|
msg string
|
||||||
}
|
}
|
||||||
@ -145,7 +145,7 @@ func TestRenderCommitMessageLinkSubject(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestRenderIssueTitle(t *testing.T) {
|
func TestRenderIssueTitle(t *testing.T) {
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
expected := ` space @mention-user<SPACE><SPACE>
|
expected := ` space @mention-user<SPACE><SPACE>
|
||||||
/just/a/path.bin
|
/just/a/path.bin
|
||||||
https://example.com/file.bin
|
https://example.com/file.bin
|
||||||
@ -172,7 +172,7 @@ mail@domain.com
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestRenderMarkdownToHtml(t *testing.T) {
|
func TestRenderMarkdownToHtml(t *testing.T) {
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
expected := `<p>space <a href="/mention-user" rel="nofollow">@mention-user</a><br/>
|
expected := `<p>space <a href="/mention-user" rel="nofollow">@mention-user</a><br/>
|
||||||
/just/a/path.bin
|
/just/a/path.bin
|
||||||
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a>
|
<a href="https://example.com/file.bin" rel="nofollow">https://example.com/file.bin</a>
|
||||||
@ -211,6 +211,7 @@ func TestRenderLabels(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestUserMention(t *testing.T) {
|
func TestUserMention(t *testing.T) {
|
||||||
|
markup.RenderBehaviorForTesting.DisableAdditionalAttributes = true
|
||||||
rendered := newTestRenderUtils().MarkdownToHtml("@no-such-user @mention-user @mention-user")
|
rendered := newTestRenderUtils().MarkdownToHtml("@no-such-user @mention-user @mention-user")
|
||||||
assert.EqualValues(t, `<p>@no-such-user <a href="/mention-user" data-markdown-generated-content="" rel="nofollow">@mention-user</a> <a href="/mention-user" data-markdown-generated-content="" rel="nofollow">@mention-user</a></p>`, strings.TrimSpace(string(rendered)))
|
assert.EqualValues(t, `<p>@no-such-user <a href="/mention-user" rel="nofollow">@mention-user</a> <a href="/mention-user" rel="nofollow">@mention-user</a></p>`, strings.TrimSpace(string(rendered)))
|
||||||
}
|
}
|
||||||
|
@ -459,6 +459,7 @@ authorize_application = Authorize Application
|
|||||||
authorize_redirect_notice = You will be redirected to %s if you authorize this application.
|
authorize_redirect_notice = You will be redirected to %s if you authorize this application.
|
||||||
authorize_application_created_by = This application was created by %s.
|
authorize_application_created_by = This application was created by %s.
|
||||||
authorize_application_description = If you grant the access, it will be able to access and write to all your account information, including private repos and organisations.
|
authorize_application_description = If you grant the access, it will be able to access and write to all your account information, including private repos and organisations.
|
||||||
|
authorize_application_with_scopes = With scopes: %s
|
||||||
authorize_title = Authorize "%s" to access your account?
|
authorize_title = Authorize "%s" to access your account?
|
||||||
authorization_failed = Authorization failed
|
authorization_failed = Authorization failed
|
||||||
authorization_failed_desc = The authorization failed because we detected an invalid request. Please contact the maintainer of the app you have tried to authorize.
|
authorization_failed_desc = The authorization failed because we detected an invalid request. Please contact the maintainer of the app you have tried to authorize.
|
||||||
|
140
package-lock.json
generated
140
package-lock.json
generated
@ -110,7 +110,8 @@
|
|||||||
"type-fest": "4.26.1",
|
"type-fest": "4.26.1",
|
||||||
"updates": "16.4.0",
|
"updates": "16.4.0",
|
||||||
"vite-string-plugin": "1.3.4",
|
"vite-string-plugin": "1.3.4",
|
||||||
"vitest": "2.1.4"
|
"vitest": "2.1.4",
|
||||||
|
"vue-tsc": "2.1.10"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">= 18.0.0"
|
"node": ">= 18.0.0"
|
||||||
@ -5390,6 +5391,35 @@
|
|||||||
"url": "https://opencollective.com/vitest"
|
"url": "https://opencollective.com/vitest"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@volar/language-core": {
|
||||||
|
"version": "2.4.10",
|
||||||
|
"resolved": "https://registry.npmjs.org/@volar/language-core/-/language-core-2.4.10.tgz",
|
||||||
|
"integrity": "sha512-hG3Z13+nJmGaT+fnQzAkS0hjJRa2FCeqZt6Bd+oGNhUkQ+mTFsDETg5rqUTxyzIh5pSOGY7FHCWUS8G82AzLCA==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@volar/source-map": "2.4.10"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@volar/source-map": {
|
||||||
|
"version": "2.4.10",
|
||||||
|
"resolved": "https://registry.npmjs.org/@volar/source-map/-/source-map-2.4.10.tgz",
|
||||||
|
"integrity": "sha512-OCV+b5ihV0RF3A7vEvNyHPi4G4kFa6ukPmyVocmqm5QzOd8r5yAtiNvaPEjl8dNvgC/lj4JPryeeHLdXd62rWA==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/@volar/typescript": {
|
||||||
|
"version": "2.4.10",
|
||||||
|
"resolved": "https://registry.npmjs.org/@volar/typescript/-/typescript-2.4.10.tgz",
|
||||||
|
"integrity": "sha512-F8ZtBMhSXyYKuBfGpYwqA5rsONnOwAVvjyE7KPYJ7wgZqo2roASqNWUnianOomJX5u1cxeRooHV59N0PhvEOgw==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@volar/language-core": "2.4.10",
|
||||||
|
"path-browserify": "^1.0.1",
|
||||||
|
"vscode-uri": "^3.0.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@vue/compiler-core": {
|
"node_modules/@vue/compiler-core": {
|
||||||
"version": "3.5.12",
|
"version": "3.5.12",
|
||||||
"resolved": "https://registry.npmjs.org/@vue/compiler-core/-/compiler-core-3.5.12.tgz",
|
"resolved": "https://registry.npmjs.org/@vue/compiler-core/-/compiler-core-3.5.12.tgz",
|
||||||
@ -5449,6 +5479,58 @@
|
|||||||
"@vue/shared": "3.5.12"
|
"@vue/shared": "3.5.12"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@vue/compiler-vue2": {
|
||||||
|
"version": "2.7.16",
|
||||||
|
"resolved": "https://registry.npmjs.org/@vue/compiler-vue2/-/compiler-vue2-2.7.16.tgz",
|
||||||
|
"integrity": "sha512-qYC3Psj9S/mfu9uVi5WvNZIzq+xnXMhOwbTFKKDD7b1lhpnn71jXSFdTQ+WsIEk0ONCd7VV2IMm7ONl6tbQ86A==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"de-indent": "^1.0.2",
|
||||||
|
"he": "^1.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@vue/language-core": {
|
||||||
|
"version": "2.1.10",
|
||||||
|
"resolved": "https://registry.npmjs.org/@vue/language-core/-/language-core-2.1.10.tgz",
|
||||||
|
"integrity": "sha512-DAI289d0K3AB5TUG3xDp9OuQ71CnrujQwJrQnfuZDwo6eGNf0UoRlPuaVNO+Zrn65PC3j0oB2i7mNmVPggeGeQ==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@volar/language-core": "~2.4.8",
|
||||||
|
"@vue/compiler-dom": "^3.5.0",
|
||||||
|
"@vue/compiler-vue2": "^2.7.16",
|
||||||
|
"@vue/shared": "^3.5.0",
|
||||||
|
"alien-signals": "^0.2.0",
|
||||||
|
"minimatch": "^9.0.3",
|
||||||
|
"muggle-string": "^0.4.1",
|
||||||
|
"path-browserify": "^1.0.1"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"typescript": "*"
|
||||||
|
},
|
||||||
|
"peerDependenciesMeta": {
|
||||||
|
"typescript": {
|
||||||
|
"optional": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@vue/language-core/node_modules/minimatch": {
|
||||||
|
"version": "9.0.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz",
|
||||||
|
"integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "ISC",
|
||||||
|
"dependencies": {
|
||||||
|
"brace-expansion": "^2.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=16 || 14 >=14.17"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/isaacs"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@vue/reactivity": {
|
"node_modules/@vue/reactivity": {
|
||||||
"version": "3.5.12",
|
"version": "3.5.12",
|
||||||
"resolved": "https://registry.npmjs.org/@vue/reactivity/-/reactivity-3.5.12.tgz",
|
"resolved": "https://registry.npmjs.org/@vue/reactivity/-/reactivity-3.5.12.tgz",
|
||||||
@ -5821,6 +5903,13 @@
|
|||||||
"ajv": "^8.8.2"
|
"ajv": "^8.8.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/alien-signals": {
|
||||||
|
"version": "0.2.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/alien-signals/-/alien-signals-0.2.2.tgz",
|
||||||
|
"integrity": "sha512-cZIRkbERILsBOXTQmMrxc9hgpxglstn69zm+F1ARf4aPAzdAFYd6sBq87ErO0Fj3DV94tglcyHG5kQz9nDC/8A==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/ansi_up": {
|
"node_modules/ansi_up": {
|
||||||
"version": "6.0.2",
|
"version": "6.0.2",
|
||||||
"resolved": "https://registry.npmjs.org/ansi_up/-/ansi_up-6.0.2.tgz",
|
"resolved": "https://registry.npmjs.org/ansi_up/-/ansi_up-6.0.2.tgz",
|
||||||
@ -7484,6 +7573,13 @@
|
|||||||
"integrity": "sha512-oaMBel6gjolK862uaPQOVTA7q3TZhuSvuMQAAglQDOWYO9A91IrAOUJEyKVlqJlHE0vq5p5UXxzdPfMH/x6xNg==",
|
"integrity": "sha512-oaMBel6gjolK862uaPQOVTA7q3TZhuSvuMQAAglQDOWYO9A91IrAOUJEyKVlqJlHE0vq5p5UXxzdPfMH/x6xNg==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/de-indent": {
|
||||||
|
"version": "1.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/de-indent/-/de-indent-1.0.2.tgz",
|
||||||
|
"integrity": "sha512-e/1zu3xH5MQryN2zdVaF0OrdNLUbvWxzMbi+iNA6Bky7l1RoP8a2fIbRocyHclXt/arDrrR6lL3TqFD9pMQTsg==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/debug": {
|
"node_modules/debug": {
|
||||||
"version": "4.3.7",
|
"version": "4.3.7",
|
||||||
"resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz",
|
"resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz",
|
||||||
@ -10337,6 +10433,16 @@
|
|||||||
"node": ">=12.4.0"
|
"node": ">=12.4.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/he": {
|
||||||
|
"version": "1.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz",
|
||||||
|
"integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"bin": {
|
||||||
|
"he": "bin/he"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/hosted-git-info": {
|
"node_modules/hosted-git-info": {
|
||||||
"version": "2.8.9",
|
"version": "2.8.9",
|
||||||
"resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz",
|
"resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz",
|
||||||
@ -11793,6 +11899,13 @@
|
|||||||
"integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
|
"integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/muggle-string": {
|
||||||
|
"version": "0.4.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/muggle-string/-/muggle-string-0.4.1.tgz",
|
||||||
|
"integrity": "sha512-VNTrAak/KhO2i8dqqnqnAHOa3cYBwXEZe9h+D5h/1ZqFSTEFHdM65lR7RoIqq3tBBYavsOXV84NoHXZ0AkPyqQ==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/mz": {
|
"node_modules/mz": {
|
||||||
"version": "2.7.0",
|
"version": "2.7.0",
|
||||||
"resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz",
|
"resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz",
|
||||||
@ -12168,6 +12281,13 @@
|
|||||||
"url": "https://github.com/sponsors/sindresorhus"
|
"url": "https://github.com/sponsors/sindresorhus"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/path-browserify": {
|
||||||
|
"version": "1.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/path-browserify/-/path-browserify-1.0.1.tgz",
|
||||||
|
"integrity": "sha512-b7uo2UCUOYZcnF/3ID0lulOJi/bafxa1xPe7ZPsammBSpjSWQkjNxlt635YGS2MiR9GjvuXCtz2emr3jbsz98g==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/path-data-parser": {
|
"node_modules/path-data-parser": {
|
||||||
"version": "0.1.0",
|
"version": "0.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/path-data-parser/-/path-data-parser-0.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/path-data-parser/-/path-data-parser-0.1.0.tgz",
|
||||||
@ -15521,6 +15641,24 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/vue-tsc": {
|
||||||
|
"version": "2.1.10",
|
||||||
|
"resolved": "https://registry.npmjs.org/vue-tsc/-/vue-tsc-2.1.10.tgz",
|
||||||
|
"integrity": "sha512-RBNSfaaRHcN5uqVqJSZh++Gy/YUzryuv9u1aFWhsammDJXNtUiJMNoJ747lZcQ68wUQFx6E73y4FY3D8E7FGMA==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@volar/typescript": "~2.4.8",
|
||||||
|
"@vue/language-core": "2.1.10",
|
||||||
|
"semver": "^7.5.4"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"vue-tsc": "bin/vue-tsc.js"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"typescript": ">=5.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/watchpack": {
|
"node_modules/watchpack": {
|
||||||
"version": "2.4.2",
|
"version": "2.4.2",
|
||||||
"resolved": "https://registry.npmjs.org/watchpack/-/watchpack-2.4.2.tgz",
|
"resolved": "https://registry.npmjs.org/watchpack/-/watchpack-2.4.2.tgz",
|
||||||
|
@ -109,7 +109,8 @@
|
|||||||
"type-fest": "4.26.1",
|
"type-fest": "4.26.1",
|
||||||
"updates": "16.4.0",
|
"updates": "16.4.0",
|
||||||
"vite-string-plugin": "1.3.4",
|
"vite-string-plugin": "1.3.4",
|
||||||
"vitest": "2.1.4"
|
"vitest": "2.1.4",
|
||||||
|
"vue-tsc": "2.1.10"
|
||||||
},
|
},
|
||||||
"browserslist": [
|
"browserslist": [
|
||||||
"defaults"
|
"defaults"
|
||||||
|
@ -99,9 +99,7 @@ func MarkdownRaw(ctx *context.APIContext) {
|
|||||||
// "422":
|
// "422":
|
||||||
// "$ref": "#/responses/validationError"
|
// "$ref": "#/responses/validationError"
|
||||||
defer ctx.Req.Body.Close()
|
defer ctx.Req.Body.Close()
|
||||||
if err := markdown.RenderRaw(&markup.RenderContext{
|
if err := markdown.RenderRaw(markup.NewRenderContext(ctx), ctx.Req.Body, ctx.Resp); err != nil {
|
||||||
Ctx: ctx,
|
|
||||||
}, ctx.Req.Body, ctx.Resp); err != nil {
|
|
||||||
ctx.InternalServerError(err)
|
ctx.InternalServerError(err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -25,7 +25,7 @@ const AppURL = "http://localhost:3000/"
|
|||||||
|
|
||||||
func testRenderMarkup(t *testing.T, mode string, wiki bool, filePath, text, expectedBody string, expectedCode int) {
|
func testRenderMarkup(t *testing.T, mode string, wiki bool, filePath, text, expectedBody string, expectedCode int) {
|
||||||
setting.AppURL = AppURL
|
setting.AppURL = AppURL
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
context := "/gogits/gogs"
|
context := "/gogits/gogs"
|
||||||
if !wiki {
|
if !wiki {
|
||||||
context += path.Join("/src/branch/main", path.Dir(filePath))
|
context += path.Join("/src/branch/main", path.Dir(filePath))
|
||||||
@ -46,7 +46,7 @@ func testRenderMarkup(t *testing.T, mode string, wiki bool, filePath, text, expe
|
|||||||
}
|
}
|
||||||
|
|
||||||
func testRenderMarkdown(t *testing.T, mode string, wiki bool, text, responseBody string, responseCode int) {
|
func testRenderMarkdown(t *testing.T, mode string, wiki bool, text, responseBody string, responseCode int) {
|
||||||
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableInternalAttributes, true)()
|
defer test.MockVariableValue(&markup.RenderBehaviorForTesting.DisableAdditionalAttributes, true)()
|
||||||
setting.AppURL = AppURL
|
setting.AppURL = AppURL
|
||||||
context := "/gogits/gogs"
|
context := "/gogits/gogs"
|
||||||
if !wiki {
|
if !wiki {
|
||||||
@ -67,7 +67,7 @@ func testRenderMarkdown(t *testing.T, mode string, wiki bool, text, responseBody
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestAPI_RenderGFM(t *testing.T) {
|
func TestAPI_RenderGFM(t *testing.T) {
|
||||||
markup.Init(&markup.ProcessorHelper{
|
markup.Init(&markup.RenderHelperFuncs{
|
||||||
IsUsernameMentionable: func(ctx go_context.Context, username string) bool {
|
IsUsernameMentionable: func(ctx go_context.Context, username string) bool {
|
||||||
return username == "r-lyeh"
|
return username == "r-lyeh"
|
||||||
},
|
},
|
||||||
@ -182,6 +182,7 @@ var simpleCases = []string{
|
|||||||
|
|
||||||
func TestAPI_RenderSimple(t *testing.T) {
|
func TestAPI_RenderSimple(t *testing.T) {
|
||||||
setting.AppURL = AppURL
|
setting.AppURL = AppURL
|
||||||
|
markup.RenderBehaviorForTesting.DisableAdditionalAttributes = true
|
||||||
options := api.MarkdownOption{
|
options := api.MarkdownOption{
|
||||||
Mode: "markdown",
|
Mode: "markdown",
|
||||||
Text: "",
|
Text: "",
|
||||||
@ -199,6 +200,7 @@ func TestAPI_RenderSimple(t *testing.T) {
|
|||||||
|
|
||||||
func TestAPI_RenderRaw(t *testing.T) {
|
func TestAPI_RenderRaw(t *testing.T) {
|
||||||
setting.AppURL = AppURL
|
setting.AppURL = AppURL
|
||||||
|
markup.RenderBehaviorForTesting.DisableAdditionalAttributes = true
|
||||||
ctx, resp := contexttest.MockAPIContext(t, "POST /api/v1/markdown")
|
ctx, resp := contexttest.MockAPIContext(t, "POST /api/v1/markdown")
|
||||||
for i := 0; i < len(simpleCases); i += 2 {
|
for i := 0; i < len(simpleCases); i += 2 {
|
||||||
ctx.Req.Body = io.NopCloser(strings.NewReader(simpleCases[i]))
|
ctx.Req.Body = io.NopCloser(strings.NewReader(simpleCases[i]))
|
||||||
|
@ -17,6 +17,8 @@ import (
|
|||||||
"code.gitea.io/gitea/routers/api/v1/utils"
|
"code.gitea.io/gitea/routers/api/v1/utils"
|
||||||
"code.gitea.io/gitea/services/context"
|
"code.gitea.io/gitea/services/context"
|
||||||
"code.gitea.io/gitea/services/convert"
|
"code.gitea.io/gitea/services/convert"
|
||||||
|
issue_service "code.gitea.io/gitea/services/issue"
|
||||||
|
pull_service "code.gitea.io/gitea/services/pull"
|
||||||
repo_service "code.gitea.io/gitea/services/repository"
|
repo_service "code.gitea.io/gitea/services/repository"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -320,7 +322,13 @@ func GetReviewers(ctx *context.APIContext) {
|
|||||||
// "404":
|
// "404":
|
||||||
// "$ref": "#/responses/notFound"
|
// "$ref": "#/responses/notFound"
|
||||||
|
|
||||||
reviewers, err := repo_model.GetReviewers(ctx, ctx.Repo.Repository, ctx.Doer.ID, 0)
|
canChooseReviewer := issue_service.CanDoerChangeReviewRequests(ctx, ctx.Doer, ctx.Repo.Repository, 0)
|
||||||
|
if !canChooseReviewer {
|
||||||
|
ctx.Error(http.StatusForbidden, "GetReviewers", errors.New("doer has no permission to get reviewers"))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
reviewers, err := pull_service.GetReviewers(ctx, ctx.Repo.Repository, ctx.Doer.ID, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.Error(http.StatusInternalServerError, "ListCollaborators", err)
|
ctx.Error(http.StatusInternalServerError, "ListCollaborators", err)
|
||||||
return
|
return
|
||||||
|
@ -11,6 +11,8 @@ import (
|
|||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
|
"code.gitea.io/gitea/models/repo"
|
||||||
"code.gitea.io/gitea/modules/httplib"
|
"code.gitea.io/gitea/modules/httplib"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/modules/markup"
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
@ -20,7 +22,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
// RenderMarkup renders markup text for the /markup and /markdown endpoints
|
// RenderMarkup renders markup text for the /markup and /markdown endpoints
|
||||||
func RenderMarkup(ctx *context.Base, repo *context.Repository, mode, text, urlPathContext, filePath string) {
|
func RenderMarkup(ctx *context.Base, ctxRepo *context.Repository, mode, text, urlPathContext, filePath string) {
|
||||||
// urlPathContext format is "/subpath/{user}/{repo}/src/{branch, commit, tag}/{identifier/path}/{file/dir}"
|
// urlPathContext format is "/subpath/{user}/{repo}/src/{branch, commit, tag}/{identifier/path}/{file/dir}"
|
||||||
// filePath is the path of the file to render if the end user is trying to preview a repo file (mode == "file")
|
// filePath is the path of the file to render if the end user is trying to preview a repo file (mode == "file")
|
||||||
// filePath will be used as RenderContext.RelativePath
|
// filePath will be used as RenderContext.RelativePath
|
||||||
@ -28,62 +30,67 @@ func RenderMarkup(ctx *context.Base, repo *context.Repository, mode, text, urlPa
|
|||||||
// for example, when previewing file "/gitea/owner/repo/src/branch/features/feat-123/doc/CHANGE.md", then filePath is "doc/CHANGE.md"
|
// for example, when previewing file "/gitea/owner/repo/src/branch/features/feat-123/doc/CHANGE.md", then filePath is "doc/CHANGE.md"
|
||||||
// and the urlPathContext is "/gitea/owner/repo/src/branch/features/feat-123/doc"
|
// and the urlPathContext is "/gitea/owner/repo/src/branch/features/feat-123/doc"
|
||||||
|
|
||||||
renderCtx := &markup.RenderContext{
|
|
||||||
Ctx: ctx,
|
|
||||||
Links: markup.Links{AbsolutePrefix: true},
|
|
||||||
MarkupType: markdown.MarkupName,
|
|
||||||
}
|
|
||||||
if urlPathContext != "" {
|
|
||||||
renderCtx.Links.Base = fmt.Sprintf("%s%s", httplib.GuessCurrentHostURL(ctx), urlPathContext)
|
|
||||||
}
|
|
||||||
|
|
||||||
if mode == "" || mode == "markdown" {
|
if mode == "" || mode == "markdown" {
|
||||||
// raw markdown doesn't need any special handling
|
// raw markdown doesn't need any special handling
|
||||||
if err := markdown.RenderRaw(renderCtx, strings.NewReader(text), ctx.Resp); err != nil {
|
baseLink := urlPathContext
|
||||||
|
if baseLink == "" {
|
||||||
|
baseLink = fmt.Sprintf("%s%s", httplib.GuessCurrentHostURL(ctx), urlPathContext)
|
||||||
|
}
|
||||||
|
rctx := renderhelper.NewRenderContextSimpleDocument(ctx, baseLink).WithUseAbsoluteLink(true).
|
||||||
|
WithMarkupType(markdown.MarkupName)
|
||||||
|
if err := markdown.RenderRaw(rctx, strings.NewReader(text), ctx.Resp); err != nil {
|
||||||
ctx.Error(http.StatusInternalServerError, err.Error())
|
ctx.Error(http.StatusInternalServerError, err.Error())
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Ideally, this handler should be called with RepoAssigment and get the related repo from context "/owner/repo/markup"
|
||||||
|
// then render could use the repo to do various things (the permission check has passed)
|
||||||
|
//
|
||||||
|
// However, this handler is also exposed as "/markup" without any repo context,
|
||||||
|
// then since there is no permission check, so we can't use the repo from "context" parameter,
|
||||||
|
// in this case, only the "path" information could be used which doesn't cause security problems.
|
||||||
|
var repoModel *repo.Repository
|
||||||
|
if ctxRepo != nil {
|
||||||
|
repoModel = ctxRepo.Repository
|
||||||
|
}
|
||||||
|
var repoOwnerName, repoName, refPath, treePath string
|
||||||
|
repoLinkPath := strings.TrimPrefix(urlPathContext, setting.AppSubURL+"/")
|
||||||
|
fields := strings.SplitN(repoLinkPath, "/", 5)
|
||||||
|
if len(fields) == 5 && fields[2] == "src" && (fields[3] == "branch" || fields[3] == "commit" || fields[3] == "tag") {
|
||||||
|
// absolute base prefix is something like "https://host/subpath/{user}/{repo}"
|
||||||
|
repoOwnerName, repoName = fields[0], fields[1]
|
||||||
|
treePath = path.Dir(filePath) // it is "doc" if filePath is "doc/CHANGE.md"
|
||||||
|
refPath = strings.Join(fields[3:], "/") // it is "branch/features/feat-12/doc"
|
||||||
|
refPath = strings.TrimSuffix(refPath, "/"+treePath) // now we get the correct branch path: "branch/features/feat-12"
|
||||||
|
} else if fields = strings.SplitN(repoLinkPath, "/", 3); len(fields) == 2 {
|
||||||
|
repoOwnerName, repoName = fields[0], fields[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
var rctx *markup.RenderContext
|
||||||
switch mode {
|
switch mode {
|
||||||
case "gfm": // legacy mode, do nothing
|
case "gfm": // legacy mode
|
||||||
|
rctx = renderhelper.NewRenderContextRepoFile(ctx, repoModel, renderhelper.RepoFileOptions{
|
||||||
|
DeprecatedOwnerName: repoOwnerName, DeprecatedRepoName: repoName,
|
||||||
|
CurrentRefPath: refPath, CurrentTreePath: treePath,
|
||||||
|
})
|
||||||
|
rctx = rctx.WithMarkupType(markdown.MarkupName)
|
||||||
case "comment":
|
case "comment":
|
||||||
renderCtx.Metas = map[string]string{"markdownLineBreakStyle": "comment"}
|
rctx = renderhelper.NewRenderContextRepoComment(ctx, repoModel, renderhelper.RepoCommentOptions{DeprecatedOwnerName: repoOwnerName, DeprecatedRepoName: repoName})
|
||||||
case "wiki":
|
case "wiki":
|
||||||
renderCtx.Metas = map[string]string{"markdownLineBreakStyle": "document", "markupContentMode": "wiki"}
|
rctx = renderhelper.NewRenderContextRepoWiki(ctx, repoModel, renderhelper.RepoWikiOptions{DeprecatedOwnerName: repoOwnerName, DeprecatedRepoName: repoName})
|
||||||
case "file":
|
case "file":
|
||||||
// render the repo file content by its extension
|
rctx = renderhelper.NewRenderContextRepoFile(ctx, repoModel, renderhelper.RepoFileOptions{
|
||||||
renderCtx.Metas = map[string]string{"markdownLineBreakStyle": "document"}
|
DeprecatedOwnerName: repoOwnerName, DeprecatedRepoName: repoName,
|
||||||
renderCtx.MarkupType = ""
|
CurrentRefPath: refPath, CurrentTreePath: treePath,
|
||||||
renderCtx.RelativePath = filePath
|
})
|
||||||
renderCtx.InStandalonePage = true
|
rctx = rctx.WithMarkupType("").WithRelativePath(filePath) // render the repo file content by its extension
|
||||||
default:
|
default:
|
||||||
ctx.Error(http.StatusUnprocessableEntity, fmt.Sprintf("Unknown mode: %s", mode))
|
ctx.Error(http.StatusUnprocessableEntity, fmt.Sprintf("Unknown mode: %s", mode))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
rctx = rctx.WithUseAbsoluteLink(true)
|
||||||
fields := strings.SplitN(strings.TrimPrefix(urlPathContext, setting.AppSubURL+"/"), "/", 5)
|
if err := markup.Render(rctx, strings.NewReader(text), ctx.Resp); err != nil {
|
||||||
if len(fields) == 5 && fields[2] == "src" && (fields[3] == "branch" || fields[3] == "commit" || fields[3] == "tag") {
|
|
||||||
// absolute base prefix is something like "https://host/subpath/{user}/{repo}"
|
|
||||||
absoluteBasePrefix := fmt.Sprintf("%s%s/%s", httplib.GuessCurrentAppURL(ctx), fields[0], fields[1])
|
|
||||||
|
|
||||||
fileDir := path.Dir(filePath) // it is "doc" if filePath is "doc/CHANGE.md"
|
|
||||||
refPath := strings.Join(fields[3:], "/") // it is "branch/features/feat-12/doc"
|
|
||||||
refPath = strings.TrimSuffix(refPath, "/"+fileDir) // now we get the correct branch path: "branch/features/feat-12"
|
|
||||||
|
|
||||||
renderCtx.Links = markup.Links{AbsolutePrefix: true, Base: absoluteBasePrefix, BranchPath: refPath, TreePath: fileDir}
|
|
||||||
}
|
|
||||||
|
|
||||||
if repo != nil && repo.Repository != nil {
|
|
||||||
renderCtx.Repo = repo.Repository
|
|
||||||
if mode == "file" {
|
|
||||||
renderCtx.Metas = repo.Repository.ComposeDocumentMetas(ctx)
|
|
||||||
} else if mode == "wiki" {
|
|
||||||
renderCtx.Metas = repo.Repository.ComposeWikiMetas(ctx)
|
|
||||||
} else if mode == "comment" {
|
|
||||||
renderCtx.Metas = repo.Repository.ComposeMetas(ctx)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if err := markup.Render(renderCtx, strings.NewReader(text), ctx.Resp); err != nil {
|
|
||||||
if errors.Is(err, util.ErrInvalidArgument) {
|
if errors.Is(err, util.ErrInvalidArgument) {
|
||||||
ctx.Error(http.StatusUnprocessableEntity, err.Error())
|
ctx.Error(http.StatusUnprocessableEntity, err.Error())
|
||||||
} else {
|
} else {
|
||||||
|
@ -104,7 +104,18 @@ func InfoOAuth(ctx *context.Context) {
|
|||||||
Picture: ctx.Doer.AvatarLink(ctx),
|
Picture: ctx.Doer.AvatarLink(ctx),
|
||||||
}
|
}
|
||||||
|
|
||||||
groups, err := oauth2_provider.GetOAuthGroupsForUser(ctx, ctx.Doer)
|
var accessTokenScope auth.AccessTokenScope
|
||||||
|
if auHead := ctx.Req.Header.Get("Authorization"); auHead != "" {
|
||||||
|
auths := strings.Fields(auHead)
|
||||||
|
if len(auths) == 2 && (auths[0] == "token" || strings.ToLower(auths[0]) == "bearer") {
|
||||||
|
accessTokenScope, _ = auth_service.GetOAuthAccessTokenScopeAndUserID(ctx, auths[1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// since version 1.22 does not verify if groups should be public-only,
|
||||||
|
// onlyPublicGroups will be set only if 'public-only' is included in a valid scope
|
||||||
|
onlyPublicGroups, _ := accessTokenScope.PublicOnly()
|
||||||
|
groups, err := oauth2_provider.GetOAuthGroupsForUser(ctx, ctx.Doer, onlyPublicGroups)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("Oauth groups for user", err)
|
ctx.ServerError("Oauth groups for user", err)
|
||||||
return
|
return
|
||||||
@ -304,6 +315,9 @@ func AuthorizeOAuth(ctx *context.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// check if additional scopes
|
||||||
|
ctx.Data["AdditionalScopes"] = oauth2_provider.GrantAdditionalScopes(form.Scope) != auth.AccessTokenScopeAll
|
||||||
|
|
||||||
// show authorize page to grant access
|
// show authorize page to grant access
|
||||||
ctx.Data["Application"] = app
|
ctx.Data["Application"] = app
|
||||||
ctx.Data["RedirectURI"] = form.RedirectURI
|
ctx.Data["RedirectURI"] = form.RedirectURI
|
||||||
|
@ -13,8 +13,8 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
activities_model "code.gitea.io/gitea/models/activities"
|
activities_model "code.gitea.io/gitea/models/activities"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
"code.gitea.io/gitea/modules/templates"
|
"code.gitea.io/gitea/modules/templates"
|
||||||
@ -48,24 +48,18 @@ func toReleaseLink(ctx *context.Context, act *activities_model.Action) string {
|
|||||||
return act.GetRepoAbsoluteLink(ctx) + "/releases/tag/" + util.PathEscapeSegments(act.GetBranch())
|
return act.GetRepoAbsoluteLink(ctx) + "/releases/tag/" + util.PathEscapeSegments(act.GetBranch())
|
||||||
}
|
}
|
||||||
|
|
||||||
// renderMarkdown creates a minimal markdown render context from an action.
|
// renderCommentMarkdown renders the comment markdown to html
|
||||||
// If rendering fails, the original markdown text is returned
|
func renderCommentMarkdown(ctx *context.Context, act *activities_model.Action, content string) template.HTML {
|
||||||
func renderMarkdown(ctx *context.Context, act *activities_model.Action, content string) template.HTML {
|
act.LoadRepo(ctx)
|
||||||
markdownCtx := &markup.RenderContext{
|
if act.Repo == nil {
|
||||||
Ctx: ctx,
|
return ""
|
||||||
Links: markup.Links{
|
|
||||||
Base: act.GetRepoLink(ctx),
|
|
||||||
},
|
|
||||||
Metas: map[string]string{ // FIXME: not right here, it should use issue to compose the metas
|
|
||||||
"user": act.GetRepoUserName(ctx),
|
|
||||||
"repo": act.GetRepoName(ctx),
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
markdown, err := markdown.RenderString(markdownCtx, content)
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, act.Repo).WithUseAbsoluteLink(true)
|
||||||
|
rendered, err := markdown.RenderString(rctx, content)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return templates.SanitizeHTML(content) // old code did so: use SanitizeHTML to render in tmpl
|
return ""
|
||||||
}
|
}
|
||||||
return markdown
|
return rendered
|
||||||
}
|
}
|
||||||
|
|
||||||
// feedActionsToFeedItems convert gitea's Action feed to feeds Item
|
// feedActionsToFeedItems convert gitea's Action feed to feeds Item
|
||||||
@ -227,12 +221,12 @@ func feedActionsToFeedItems(ctx *context.Context, actions activities_model.Actio
|
|||||||
|
|
||||||
case activities_model.ActionCreateIssue, activities_model.ActionCreatePullRequest:
|
case activities_model.ActionCreateIssue, activities_model.ActionCreatePullRequest:
|
||||||
desc = strings.Join(act.GetIssueInfos(), "#")
|
desc = strings.Join(act.GetIssueInfos(), "#")
|
||||||
content = renderMarkdown(ctx, act, act.GetIssueContent(ctx))
|
content = renderCommentMarkdown(ctx, act, act.GetIssueContent(ctx))
|
||||||
case activities_model.ActionCommentIssue, activities_model.ActionApprovePullRequest, activities_model.ActionRejectPullRequest, activities_model.ActionCommentPull:
|
case activities_model.ActionCommentIssue, activities_model.ActionApprovePullRequest, activities_model.ActionRejectPullRequest, activities_model.ActionCommentPull:
|
||||||
desc = act.GetIssueTitle(ctx)
|
desc = act.GetIssueTitle(ctx)
|
||||||
comment := act.GetIssueInfos()[1]
|
comment := act.GetIssueInfos()[1]
|
||||||
if len(comment) != 0 {
|
if len(comment) != 0 {
|
||||||
desc += "\n\n" + string(renderMarkdown(ctx, act, comment))
|
desc += "\n\n" + string(renderCommentMarkdown(ctx, act, comment))
|
||||||
}
|
}
|
||||||
case activities_model.ActionMergePullRequest, activities_model.ActionAutoMergePullRequest:
|
case activities_model.ActionMergePullRequest, activities_model.ActionAutoMergePullRequest:
|
||||||
desc = act.GetIssueInfos()[1]
|
desc = act.GetIssueInfos()[1]
|
||||||
@ -296,14 +290,9 @@ func releasesToFeedItems(ctx *context.Context, releases []*repo_model.Release) (
|
|||||||
}
|
}
|
||||||
|
|
||||||
link := &feeds.Link{Href: rel.HTMLURL()}
|
link := &feeds.Link{Href: rel.HTMLURL()}
|
||||||
content, err = markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, rel.Repo).WithUseAbsoluteLink(true)
|
||||||
Ctx: ctx,
|
content, err = markdown.RenderString(rctx,
|
||||||
Repo: rel.Repo,
|
rel.Note)
|
||||||
Links: markup.Links{
|
|
||||||
Base: rel.Repo.Link(),
|
|
||||||
},
|
|
||||||
Metas: rel.Repo.ComposeMetas(ctx),
|
|
||||||
}, rel.Note)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -7,7 +7,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
activities_model "code.gitea.io/gitea/models/activities"
|
activities_model "code.gitea.io/gitea/models/activities"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
"code.gitea.io/gitea/services/context"
|
"code.gitea.io/gitea/services/context"
|
||||||
|
|
||||||
@ -41,13 +41,9 @@ func showUserFeed(ctx *context.Context, formatType string) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
ctxUserDescription, err := markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextSimpleDocument(ctx, ctx.ContextUser.HTMLURL())
|
||||||
Ctx: ctx,
|
ctxUserDescription, err := markdown.RenderString(rctx,
|
||||||
Links: markup.Links{
|
ctx.ContextUser.Description)
|
||||||
Base: ctx.ContextUser.HTMLURL(),
|
|
||||||
},
|
|
||||||
Metas: markup.ComposeSimpleDocumentMetas(),
|
|
||||||
}, ctx.ContextUser.Description)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
|
@ -11,10 +11,10 @@ import (
|
|||||||
|
|
||||||
"code.gitea.io/gitea/models/db"
|
"code.gitea.io/gitea/models/db"
|
||||||
"code.gitea.io/gitea/models/organization"
|
"code.gitea.io/gitea/models/organization"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
"code.gitea.io/gitea/modules/base"
|
"code.gitea.io/gitea/modules/base"
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
"code.gitea.io/gitea/modules/util"
|
"code.gitea.io/gitea/modules/util"
|
||||||
@ -180,17 +180,10 @@ func prepareOrgProfileReadme(ctx *context.Context, viewRepositories bool) bool {
|
|||||||
if bytes, err := profileReadme.GetBlobContent(setting.UI.MaxDisplayFileSize); err != nil {
|
if bytes, err := profileReadme.GetBlobContent(setting.UI.MaxDisplayFileSize); err != nil {
|
||||||
log.Error("failed to GetBlobContent: %v", err)
|
log.Error("failed to GetBlobContent: %v", err)
|
||||||
} else {
|
} else {
|
||||||
if profileContent, err := markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoFile(ctx, profileDbRepo, renderhelper.RepoFileOptions{
|
||||||
Ctx: ctx,
|
CurrentRefPath: path.Join("branch", util.PathEscapeSegments(profileDbRepo.DefaultBranch)),
|
||||||
GitRepo: profileGitRepo,
|
})
|
||||||
Links: markup.Links{
|
if profileContent, err := markdown.RenderString(rctx, bytes); err != nil {
|
||||||
// Pass repo link to markdown render for the full link of media elements.
|
|
||||||
// The profile of default branch would be shown.
|
|
||||||
Base: profileDbRepo.Link(),
|
|
||||||
BranchPath: path.Join("branch", util.PathEscapeSegments(profileDbRepo.DefaultBranch)),
|
|
||||||
},
|
|
||||||
Metas: markup.ComposeSimpleDocumentMetas(),
|
|
||||||
}, bytes); err != nil {
|
|
||||||
log.Error("failed to RenderString: %v", err)
|
log.Error("failed to RenderString: %v", err)
|
||||||
} else {
|
} else {
|
||||||
ctx.Data["ProfileReadme"] = profileContent
|
ctx.Data["ProfileReadme"] = profileContent
|
||||||
|
@ -15,6 +15,7 @@ import (
|
|||||||
asymkey_model "code.gitea.io/gitea/models/asymkey"
|
asymkey_model "code.gitea.io/gitea/models/asymkey"
|
||||||
"code.gitea.io/gitea/models/db"
|
"code.gitea.io/gitea/models/db"
|
||||||
git_model "code.gitea.io/gitea/models/git"
|
git_model "code.gitea.io/gitea/models/git"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
unit_model "code.gitea.io/gitea/models/unit"
|
unit_model "code.gitea.io/gitea/models/unit"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
@ -392,16 +393,8 @@ func Diff(ctx *context.Context) {
|
|||||||
if err == nil {
|
if err == nil {
|
||||||
ctx.Data["NoteCommit"] = note.Commit
|
ctx.Data["NoteCommit"] = note.Commit
|
||||||
ctx.Data["NoteAuthor"] = user_model.ValidateCommitWithEmail(ctx, note.Commit)
|
ctx.Data["NoteAuthor"] = user_model.ValidateCommitWithEmail(ctx, note.Commit)
|
||||||
ctx.Data["NoteRendered"], err = markup.RenderCommitMessage(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, ctx.Repo.Repository, renderhelper.RepoCommentOptions{CurrentRefPath: path.Join("commit", util.PathEscapeSegments(commitID))})
|
||||||
Links: markup.Links{
|
ctx.Data["NoteRendered"], err = markup.RenderCommitMessage(rctx, template.HTMLEscapeString(string(charset.ToUTF8WithFallback(note.Message, charset.ConvertOpts{}))))
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
BranchPath: path.Join("commit", util.PathEscapeSegments(commitID)),
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, template.HTMLEscapeString(string(charset.ToUTF8WithFallback(note.Message, charset.ConvertOpts{}))))
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderCommitMessage", err)
|
ctx.ServerError("RenderCommitMessage", err)
|
||||||
return
|
return
|
||||||
|
@ -149,7 +149,7 @@ func setCsvCompareContext(ctx *context.Context) {
|
|||||||
return csvReader, reader, err
|
return csvReader, reader, err
|
||||||
}
|
}
|
||||||
|
|
||||||
baseReader, baseBlobCloser, err := csvReaderFromCommit(&markup.RenderContext{Ctx: ctx, RelativePath: diffFile.OldName}, baseBlob)
|
baseReader, baseBlobCloser, err := csvReaderFromCommit(markup.NewRenderContext(ctx).WithRelativePath(diffFile.OldName), baseBlob)
|
||||||
if baseBlobCloser != nil {
|
if baseBlobCloser != nil {
|
||||||
defer baseBlobCloser.Close()
|
defer baseBlobCloser.Close()
|
||||||
}
|
}
|
||||||
@ -161,7 +161,7 @@ func setCsvCompareContext(ctx *context.Context) {
|
|||||||
return CsvDiffResult{nil, "unable to load file"}
|
return CsvDiffResult{nil, "unable to load file"}
|
||||||
}
|
}
|
||||||
|
|
||||||
headReader, headBlobCloser, err := csvReaderFromCommit(&markup.RenderContext{Ctx: ctx, RelativePath: diffFile.Name}, headBlob)
|
headReader, headBlobCloser, err := csvReaderFromCommit(markup.NewRenderContext(ctx).WithRelativePath(diffFile.Name), headBlob)
|
||||||
if headBlobCloser != nil {
|
if headBlobCloser != nil {
|
||||||
defer headBlobCloser.Close()
|
defer headBlobCloser.Close()
|
||||||
}
|
}
|
||||||
|
@ -18,12 +18,12 @@ import (
|
|||||||
"code.gitea.io/gitea/models/organization"
|
"code.gitea.io/gitea/models/organization"
|
||||||
access_model "code.gitea.io/gitea/models/perm/access"
|
access_model "code.gitea.io/gitea/models/perm/access"
|
||||||
project_model "code.gitea.io/gitea/models/project"
|
project_model "code.gitea.io/gitea/models/project"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
"code.gitea.io/gitea/models/unit"
|
"code.gitea.io/gitea/models/unit"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
"code.gitea.io/gitea/modules/base"
|
"code.gitea.io/gitea/modules/base"
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
"code.gitea.io/gitea/modules/optional"
|
"code.gitea.io/gitea/modules/optional"
|
||||||
api "code.gitea.io/gitea/modules/structs"
|
api "code.gitea.io/gitea/modules/structs"
|
||||||
@ -366,15 +366,8 @@ func UpdateIssueContent(ctx *context.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
content, err := markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, ctx.Repo.Repository)
|
||||||
Links: markup.Links{
|
content, err := markdown.RenderString(rctx, issue.Content)
|
||||||
Base: ctx.FormString("context"), // FIXME: <- IS THIS SAFE ?
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, issue.Content)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
|
@ -10,10 +10,10 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
issues_model "code.gitea.io/gitea/models/issues"
|
issues_model "code.gitea.io/gitea/models/issues"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
"code.gitea.io/gitea/modules/git"
|
"code.gitea.io/gitea/modules/git"
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
repo_module "code.gitea.io/gitea/modules/repository"
|
repo_module "code.gitea.io/gitea/modules/repository"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
@ -267,15 +267,8 @@ func UpdateCommentContent(ctx *context.Context) {
|
|||||||
|
|
||||||
var renderedContent template.HTML
|
var renderedContent template.HTML
|
||||||
if comment.Content != "" {
|
if comment.Content != "" {
|
||||||
renderedContent, err = markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, ctx.Repo.Repository)
|
||||||
Links: markup.Links{
|
renderedContent, err = markdown.RenderString(rctx, comment.Content)
|
||||||
Base: ctx.FormString("context"), // FIXME: <- IS THIS SAFE ?
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, comment.Content)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
|
@ -19,7 +19,7 @@ import (
|
|||||||
shared_user "code.gitea.io/gitea/routers/web/shared/user"
|
shared_user "code.gitea.io/gitea/routers/web/shared/user"
|
||||||
"code.gitea.io/gitea/services/context"
|
"code.gitea.io/gitea/services/context"
|
||||||
issue_service "code.gitea.io/gitea/services/issue"
|
issue_service "code.gitea.io/gitea/services/issue"
|
||||||
repo_service "code.gitea.io/gitea/services/repository"
|
pull_service "code.gitea.io/gitea/services/pull"
|
||||||
)
|
)
|
||||||
|
|
||||||
type issueSidebarMilestoneData struct {
|
type issueSidebarMilestoneData struct {
|
||||||
@ -186,7 +186,7 @@ func (d *IssuePageMetaData) retrieveReviewersData(ctx *context.Context) {
|
|||||||
if d.Issue == nil {
|
if d.Issue == nil {
|
||||||
data.CanChooseReviewer = true
|
data.CanChooseReviewer = true
|
||||||
} else {
|
} else {
|
||||||
data.CanChooseReviewer = issue_service.CanDoerChangeReviewRequests(ctx, ctx.Doer, repo, d.Issue)
|
data.CanChooseReviewer = issue_service.CanDoerChangeReviewRequests(ctx, ctx.Doer, repo, d.Issue.PosterID)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -231,13 +231,13 @@ func (d *IssuePageMetaData) retrieveReviewersData(ctx *context.Context) {
|
|||||||
|
|
||||||
if data.CanChooseReviewer {
|
if data.CanChooseReviewer {
|
||||||
var err error
|
var err error
|
||||||
reviewers, err = repo_model.GetReviewers(ctx, repo, ctx.Doer.ID, posterID)
|
reviewers, err = pull_service.GetReviewers(ctx, repo, ctx.Doer.ID, posterID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("GetReviewers", err)
|
ctx.ServerError("GetReviewers", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
teamReviewers, err = repo_service.GetReviewerTeams(ctx, repo)
|
teamReviewers, err = pull_service.GetReviewerTeams(ctx, repo)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("GetReviewerTeams", err)
|
ctx.ServerError("GetReviewerTeams", err)
|
||||||
return
|
return
|
||||||
|
@ -19,6 +19,7 @@ import (
|
|||||||
access_model "code.gitea.io/gitea/models/perm/access"
|
access_model "code.gitea.io/gitea/models/perm/access"
|
||||||
project_model "code.gitea.io/gitea/models/project"
|
project_model "code.gitea.io/gitea/models/project"
|
||||||
pull_model "code.gitea.io/gitea/models/pull"
|
pull_model "code.gitea.io/gitea/models/pull"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
"code.gitea.io/gitea/models/unit"
|
"code.gitea.io/gitea/models/unit"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
@ -359,15 +360,8 @@ func ViewIssue(ctx *context.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
ctx.Data["IssueWatch"] = iw
|
ctx.Data["IssueWatch"] = iw
|
||||||
issue.RenderedContent, err = markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, ctx.Repo.Repository)
|
||||||
Links: markup.Links{
|
issue.RenderedContent, err = markdown.RenderString(rctx, issue.Content)
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, issue.Content)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
@ -467,15 +461,8 @@ func ViewIssue(ctx *context.Context) {
|
|||||||
comment.Issue = issue
|
comment.Issue = issue
|
||||||
|
|
||||||
if comment.Type == issues_model.CommentTypeComment || comment.Type == issues_model.CommentTypeReview {
|
if comment.Type == issues_model.CommentTypeComment || comment.Type == issues_model.CommentTypeReview {
|
||||||
comment.RenderedContent, err = markdown.RenderString(&markup.RenderContext{
|
rctx = renderhelper.NewRenderContextRepoComment(ctx, repo)
|
||||||
Links: markup.Links{
|
comment.RenderedContent, err = markdown.RenderString(rctx, comment.Content)
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, comment.Content)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
@ -550,15 +537,8 @@ func ViewIssue(ctx *context.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if comment.Type.HasContentSupport() {
|
} else if comment.Type.HasContentSupport() {
|
||||||
comment.RenderedContent, err = markdown.RenderString(&markup.RenderContext{
|
rctx = renderhelper.NewRenderContextRepoComment(ctx, repo)
|
||||||
Links: markup.Links{
|
comment.RenderedContent, err = markdown.RenderString(rctx, comment.Content)
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, comment.Content)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
|
@ -10,8 +10,8 @@ import (
|
|||||||
|
|
||||||
"code.gitea.io/gitea/models/db"
|
"code.gitea.io/gitea/models/db"
|
||||||
issues_model "code.gitea.io/gitea/models/issues"
|
issues_model "code.gitea.io/gitea/models/issues"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
"code.gitea.io/gitea/modules/base"
|
"code.gitea.io/gitea/modules/base"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
"code.gitea.io/gitea/modules/optional"
|
"code.gitea.io/gitea/modules/optional"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
@ -79,15 +79,8 @@ func Milestones(ctx *context.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
for _, m := range miles {
|
for _, m := range miles {
|
||||||
m.RenderedContent, err = markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, ctx.Repo.Repository)
|
||||||
Links: markup.Links{
|
m.RenderedContent, err = markdown.RenderString(rctx, m.Content)
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, m.Content)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
@ -268,15 +261,8 @@ func MilestoneIssuesAndPulls(ctx *context.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
milestone.RenderedContent, err = markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, ctx.Repo.Repository)
|
||||||
Links: markup.Links{
|
milestone.RenderedContent, err = markdown.RenderString(rctx, milestone.Content)
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, milestone.Content)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
|
@ -13,11 +13,11 @@ import (
|
|||||||
issues_model "code.gitea.io/gitea/models/issues"
|
issues_model "code.gitea.io/gitea/models/issues"
|
||||||
"code.gitea.io/gitea/models/perm"
|
"code.gitea.io/gitea/models/perm"
|
||||||
project_model "code.gitea.io/gitea/models/project"
|
project_model "code.gitea.io/gitea/models/project"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
"code.gitea.io/gitea/models/unit"
|
"code.gitea.io/gitea/models/unit"
|
||||||
"code.gitea.io/gitea/modules/base"
|
"code.gitea.io/gitea/modules/base"
|
||||||
"code.gitea.io/gitea/modules/json"
|
"code.gitea.io/gitea/modules/json"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
"code.gitea.io/gitea/modules/optional"
|
"code.gitea.io/gitea/modules/optional"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
@ -92,15 +92,8 @@ func Projects(ctx *context.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for i := range projects {
|
for i := range projects {
|
||||||
projects[i].RenderedContent, err = markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, repo)
|
||||||
Links: markup.Links{
|
projects[i].RenderedContent, err = markdown.RenderString(rctx, projects[i].Description)
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, projects[i].Description)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
@ -425,15 +418,8 @@ func ViewProject(ctx *context.Context) {
|
|||||||
ctx.Data["SelectLabels"] = selectLabels
|
ctx.Data["SelectLabels"] = selectLabels
|
||||||
ctx.Data["AssigneeID"] = assigneeID
|
ctx.Data["AssigneeID"] = assigneeID
|
||||||
|
|
||||||
project.RenderedContent, err = markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, ctx.Repo.Repository)
|
||||||
Links: markup.Links{
|
project.RenderedContent, err = markdown.RenderString(rctx, project.Description)
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, project.Description)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
|
@ -13,13 +13,13 @@ import (
|
|||||||
"code.gitea.io/gitea/models"
|
"code.gitea.io/gitea/models"
|
||||||
"code.gitea.io/gitea/models/db"
|
"code.gitea.io/gitea/models/db"
|
||||||
git_model "code.gitea.io/gitea/models/git"
|
git_model "code.gitea.io/gitea/models/git"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
"code.gitea.io/gitea/models/unit"
|
"code.gitea.io/gitea/models/unit"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
"code.gitea.io/gitea/modules/base"
|
"code.gitea.io/gitea/modules/base"
|
||||||
"code.gitea.io/gitea/modules/git"
|
"code.gitea.io/gitea/modules/git"
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
"code.gitea.io/gitea/modules/optional"
|
"code.gitea.io/gitea/modules/optional"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
@ -114,15 +114,8 @@ func getReleaseInfos(ctx *context.Context, opts *repo_model.FindReleasesOptions)
|
|||||||
cacheUsers[r.PublisherID] = r.Publisher
|
cacheUsers[r.PublisherID] = r.Publisher
|
||||||
}
|
}
|
||||||
|
|
||||||
r.RenderedNote, err = markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, r.Repo)
|
||||||
Links: markup.Links{
|
r.RenderedNote, err = markdown.RenderString(rctx, r.Note)
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
Repo: ctx.Repo.Repository,
|
|
||||||
Ctx: ctx,
|
|
||||||
}, r.Note)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,7 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
"path"
|
"path"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
"code.gitea.io/gitea/modules/charset"
|
"code.gitea.io/gitea/modules/charset"
|
||||||
"code.gitea.io/gitea/modules/git"
|
"code.gitea.io/gitea/modules/git"
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
@ -56,18 +57,12 @@ func RenderFile(ctx *context.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
err = markup.Render(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoFile(ctx, ctx.Repo.Repository, renderhelper.RepoFileOptions{
|
||||||
Ctx: ctx,
|
CurrentRefPath: ctx.Repo.BranchNameSubURL(),
|
||||||
RelativePath: ctx.Repo.TreePath,
|
CurrentTreePath: path.Dir(ctx.Repo.TreePath),
|
||||||
Links: markup.Links{
|
}).WithRelativePath(ctx.Repo.TreePath).WithInStandalonePage(true)
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
BranchPath: ctx.Repo.BranchNameSubURL(),
|
err = markup.Render(rctx, rd, ctx.Resp)
|
||||||
TreePath: path.Dir(ctx.Repo.TreePath),
|
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeDocumentMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
InStandalonePage: true,
|
|
||||||
}, rd, ctx.Resp)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("Failed to render file %q: %v", ctx.Repo.TreePath, err)
|
log.Error("Failed to render file %q: %v", ctx.Repo.TreePath, err)
|
||||||
http.Error(ctx.Resp, "Failed to render file", http.StatusInternalServerError)
|
http.Error(ctx.Resp, "Failed to render file", http.StatusInternalServerError)
|
||||||
|
@ -352,6 +352,9 @@ func Action(ctx *context.Context) {
|
|||||||
ctx.Data["IsStaringRepo"] = repo_model.IsStaring(ctx, ctx.Doer.ID, ctx.Repo.Repository.ID)
|
ctx.Data["IsStaringRepo"] = repo_model.IsStaring(ctx, ctx.Doer.ID, ctx.Repo.Repository.ID)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// see the `hx-trigger="refreshUserCards ..."` comments in tmpl
|
||||||
|
ctx.RespHeader().Add("hx-trigger", "refreshUserCards")
|
||||||
|
|
||||||
switch ctx.PathParam(":action") {
|
switch ctx.PathParam(":action") {
|
||||||
case "watch", "unwatch", "star", "unstar":
|
case "watch", "unwatch", "star", "unstar":
|
||||||
// we have to reload the repository because NumStars or NumWatching (used in the templates) has just changed
|
// we have to reload the repository because NumStars or NumWatching (used in the templates) has just changed
|
||||||
|
@ -31,6 +31,7 @@ import (
|
|||||||
git_model "code.gitea.io/gitea/models/git"
|
git_model "code.gitea.io/gitea/models/git"
|
||||||
issue_model "code.gitea.io/gitea/models/issues"
|
issue_model "code.gitea.io/gitea/models/issues"
|
||||||
access_model "code.gitea.io/gitea/models/perm/access"
|
access_model "code.gitea.io/gitea/models/perm/access"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
unit_model "code.gitea.io/gitea/models/unit"
|
unit_model "code.gitea.io/gitea/models/unit"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
@ -310,18 +311,14 @@ func renderReadmeFile(ctx *context.Context, subfolder string, readmeFile *git.Tr
|
|||||||
ctx.Data["IsMarkup"] = true
|
ctx.Data["IsMarkup"] = true
|
||||||
ctx.Data["MarkupType"] = markupType
|
ctx.Data["MarkupType"] = markupType
|
||||||
|
|
||||||
ctx.Data["EscapeStatus"], ctx.Data["FileContent"], err = markupRender(ctx, &markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoFile(ctx, ctx.Repo.Repository, renderhelper.RepoFileOptions{
|
||||||
Ctx: ctx,
|
CurrentRefPath: ctx.Repo.BranchNameSubURL(),
|
||||||
MarkupType: markupType,
|
CurrentTreePath: path.Join(ctx.Repo.TreePath, subfolder),
|
||||||
RelativePath: path.Join(ctx.Repo.TreePath, readmeFile.Name()), // ctx.Repo.TreePath is the directory not the Readme so we must append the Readme filename (and path).
|
}).
|
||||||
Links: markup.Links{
|
WithMarkupType(markupType).
|
||||||
Base: ctx.Repo.RepoLink,
|
WithRelativePath(path.Join(ctx.Repo.TreePath, subfolder, readmeFile.Name())) // ctx.Repo.TreePath is the directory not the Readme so we must append the Readme filename (and path).
|
||||||
BranchPath: ctx.Repo.BranchNameSubURL(),
|
|
||||||
TreePath: path.Join(ctx.Repo.TreePath, subfolder),
|
ctx.Data["EscapeStatus"], ctx.Data["FileContent"], err = markupRender(ctx, rctx, rd)
|
||||||
},
|
|
||||||
Metas: ctx.Repo.Repository.ComposeDocumentMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
}, rd)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("Render failed for %s in %-v: %v Falling back to rendering source", readmeFile.Name(), ctx.Repo.Repository, err)
|
log.Error("Render failed for %s in %-v: %v Falling back to rendering source", readmeFile.Name(), ctx.Repo.Repository, err)
|
||||||
delete(ctx.Data, "IsMarkup")
|
delete(ctx.Data, "IsMarkup")
|
||||||
@ -514,18 +511,15 @@ func renderFile(ctx *context.Context, entry *git.TreeEntry) {
|
|||||||
ctx.Data["MarkupType"] = markupType
|
ctx.Data["MarkupType"] = markupType
|
||||||
metas := ctx.Repo.Repository.ComposeDocumentMetas(ctx)
|
metas := ctx.Repo.Repository.ComposeDocumentMetas(ctx)
|
||||||
metas["BranchNameSubURL"] = ctx.Repo.BranchNameSubURL()
|
metas["BranchNameSubURL"] = ctx.Repo.BranchNameSubURL()
|
||||||
ctx.Data["EscapeStatus"], ctx.Data["FileContent"], err = markupRender(ctx, &markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoFile(ctx, ctx.Repo.Repository, renderhelper.RepoFileOptions{
|
||||||
Ctx: ctx,
|
CurrentRefPath: ctx.Repo.BranchNameSubURL(),
|
||||||
MarkupType: markupType,
|
CurrentTreePath: path.Dir(ctx.Repo.TreePath),
|
||||||
RelativePath: ctx.Repo.TreePath,
|
}).
|
||||||
Links: markup.Links{
|
WithMarkupType(markupType).
|
||||||
Base: ctx.Repo.RepoLink,
|
WithRelativePath(ctx.Repo.TreePath).
|
||||||
BranchPath: ctx.Repo.BranchNameSubURL(),
|
WithMetas(metas)
|
||||||
TreePath: path.Dir(ctx.Repo.TreePath),
|
|
||||||
},
|
ctx.Data["EscapeStatus"], ctx.Data["FileContent"], err = markupRender(ctx, rctx, rd)
|
||||||
Metas: metas,
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
}, rd)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("Render", err)
|
ctx.ServerError("Render", err)
|
||||||
return
|
return
|
||||||
@ -606,18 +600,15 @@ func renderFile(ctx *context.Context, entry *git.TreeEntry) {
|
|||||||
rd := io.MultiReader(bytes.NewReader(buf), dataRc)
|
rd := io.MultiReader(bytes.NewReader(buf), dataRc)
|
||||||
ctx.Data["IsMarkup"] = true
|
ctx.Data["IsMarkup"] = true
|
||||||
ctx.Data["MarkupType"] = markupType
|
ctx.Data["MarkupType"] = markupType
|
||||||
ctx.Data["EscapeStatus"], ctx.Data["FileContent"], err = markupRender(ctx, &markup.RenderContext{
|
|
||||||
Ctx: ctx,
|
rctx := renderhelper.NewRenderContextRepoFile(ctx, ctx.Repo.Repository, renderhelper.RepoFileOptions{
|
||||||
MarkupType: markupType,
|
CurrentRefPath: ctx.Repo.BranchNameSubURL(),
|
||||||
RelativePath: ctx.Repo.TreePath,
|
CurrentTreePath: path.Dir(ctx.Repo.TreePath),
|
||||||
Links: markup.Links{
|
}).
|
||||||
Base: ctx.Repo.RepoLink,
|
WithMarkupType(markupType).
|
||||||
BranchPath: ctx.Repo.BranchNameSubURL(),
|
WithRelativePath(ctx.Repo.TreePath)
|
||||||
TreePath: path.Dir(ctx.Repo.TreePath),
|
|
||||||
},
|
ctx.Data["EscapeStatus"], ctx.Data["FileContent"], err = markupRender(ctx, rctx, rd)
|
||||||
Metas: ctx.Repo.Repository.ComposeDocumentMetas(ctx),
|
|
||||||
GitRepo: ctx.Repo.GitRepo,
|
|
||||||
}, rd)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("Render", err)
|
ctx.ServerError("Render", err)
|
||||||
return
|
return
|
||||||
@ -1126,8 +1117,6 @@ func RenderUserCards(ctx *context.Context, total int, getter func(opts db.ListOp
|
|||||||
func Watchers(ctx *context.Context) {
|
func Watchers(ctx *context.Context) {
|
||||||
ctx.Data["Title"] = ctx.Tr("repo.watchers")
|
ctx.Data["Title"] = ctx.Tr("repo.watchers")
|
||||||
ctx.Data["CardsTitle"] = ctx.Tr("repo.watchers")
|
ctx.Data["CardsTitle"] = ctx.Tr("repo.watchers")
|
||||||
ctx.Data["PageIsWatchers"] = true
|
|
||||||
|
|
||||||
RenderUserCards(ctx, ctx.Repo.Repository.NumWatches, func(opts db.ListOptions) ([]*user_model.User, error) {
|
RenderUserCards(ctx, ctx.Repo.Repository.NumWatches, func(opts db.ListOptions) ([]*user_model.User, error) {
|
||||||
return repo_model.GetRepoWatchers(ctx, ctx.Repo.Repository.ID, opts)
|
return repo_model.GetRepoWatchers(ctx, ctx.Repo.Repository.ID, opts)
|
||||||
}, tplWatchers)
|
}, tplWatchers)
|
||||||
@ -1137,7 +1126,6 @@ func Watchers(ctx *context.Context) {
|
|||||||
func Stars(ctx *context.Context) {
|
func Stars(ctx *context.Context) {
|
||||||
ctx.Data["Title"] = ctx.Tr("repo.stargazers")
|
ctx.Data["Title"] = ctx.Tr("repo.stargazers")
|
||||||
ctx.Data["CardsTitle"] = ctx.Tr("repo.stargazers")
|
ctx.Data["CardsTitle"] = ctx.Tr("repo.stargazers")
|
||||||
ctx.Data["PageIsStargazers"] = true
|
|
||||||
RenderUserCards(ctx, ctx.Repo.Repository.NumStars, func(opts db.ListOptions) ([]*user_model.User, error) {
|
RenderUserCards(ctx, ctx.Repo.Repository.NumStars, func(opts db.ListOptions) ([]*user_model.User, error) {
|
||||||
return repo_model.GetStargazers(ctx, ctx.Repo.Repository, opts)
|
return repo_model.GetStargazers(ctx, ctx.Repo.Repository, opts)
|
||||||
}, tplWatchers)
|
}, tplWatchers)
|
||||||
|
@ -14,6 +14,7 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
git_model "code.gitea.io/gitea/models/git"
|
git_model "code.gitea.io/gitea/models/git"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
"code.gitea.io/gitea/models/unit"
|
"code.gitea.io/gitea/models/unit"
|
||||||
"code.gitea.io/gitea/modules/base"
|
"code.gitea.io/gitea/modules/base"
|
||||||
@ -288,15 +289,9 @@ func renderViewPage(ctx *context.Context) (*git.Repository, *git.TreeEntry) {
|
|||||||
footerContent = data
|
footerContent = data
|
||||||
}
|
}
|
||||||
|
|
||||||
rctx := &markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoWiki(ctx, ctx.Repo.Repository)
|
||||||
Ctx: ctx,
|
|
||||||
Metas: ctx.Repo.Repository.ComposeWikiMetas(ctx),
|
|
||||||
Links: markup.Links{
|
|
||||||
Base: ctx.Repo.RepoLink,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
buf := &strings.Builder{}
|
|
||||||
|
|
||||||
|
buf := &strings.Builder{}
|
||||||
renderFn := func(data []byte) (escaped *charset.EscapeStatus, output string, err error) {
|
renderFn := func(data []byte) (escaped *charset.EscapeStatus, output string, err error) {
|
||||||
markupRd, markupWr := io.Pipe()
|
markupRd, markupWr := io.Pipe()
|
||||||
defer markupWr.Close()
|
defer markupWr.Close()
|
||||||
|
@ -49,10 +49,7 @@ func PrepareContextForProfileBigAvatar(ctx *context.Context) {
|
|||||||
}
|
}
|
||||||
ctx.Data["OpenIDs"] = openIDs
|
ctx.Data["OpenIDs"] = openIDs
|
||||||
if len(ctx.ContextUser.Description) != 0 {
|
if len(ctx.ContextUser.Description) != 0 {
|
||||||
content, err := markdown.RenderString(&markup.RenderContext{
|
content, err := markdown.RenderString(markup.NewRenderContext(ctx).WithMetas(markup.ComposeSimpleDocumentMetas()), ctx.ContextUser.Description)
|
||||||
Metas: markup.ComposeSimpleDocumentMetas(),
|
|
||||||
Ctx: ctx,
|
|
||||||
}, ctx.ContextUser.Description)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
|
@ -20,6 +20,7 @@ import (
|
|||||||
git_model "code.gitea.io/gitea/models/git"
|
git_model "code.gitea.io/gitea/models/git"
|
||||||
issues_model "code.gitea.io/gitea/models/issues"
|
issues_model "code.gitea.io/gitea/models/issues"
|
||||||
"code.gitea.io/gitea/models/organization"
|
"code.gitea.io/gitea/models/organization"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
"code.gitea.io/gitea/models/unit"
|
"code.gitea.io/gitea/models/unit"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
@ -27,7 +28,6 @@ import (
|
|||||||
"code.gitea.io/gitea/modules/container"
|
"code.gitea.io/gitea/modules/container"
|
||||||
issue_indexer "code.gitea.io/gitea/modules/indexer/issues"
|
issue_indexer "code.gitea.io/gitea/modules/indexer/issues"
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
"code.gitea.io/gitea/modules/optional"
|
"code.gitea.io/gitea/modules/optional"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
@ -257,14 +257,8 @@ func Milestones(ctx *context.Context) {
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
milestones[i].RenderedContent, err = markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoComment(ctx, milestones[i].Repo)
|
||||||
Links: markup.Links{
|
milestones[i].RenderedContent, err = markdown.RenderString(rctx, milestones[i].Content)
|
||||||
Base: milestones[i].Repo.Link(),
|
|
||||||
},
|
|
||||||
Metas: milestones[i].Repo.ComposeMetas(ctx),
|
|
||||||
Ctx: ctx,
|
|
||||||
Repo: milestones[i].Repo,
|
|
||||||
}, milestones[i].Content)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("RenderString", err)
|
ctx.ServerError("RenderString", err)
|
||||||
return
|
return
|
||||||
|
@ -12,12 +12,12 @@ import (
|
|||||||
|
|
||||||
activities_model "code.gitea.io/gitea/models/activities"
|
activities_model "code.gitea.io/gitea/models/activities"
|
||||||
"code.gitea.io/gitea/models/db"
|
"code.gitea.io/gitea/models/db"
|
||||||
|
"code.gitea.io/gitea/models/renderhelper"
|
||||||
repo_model "code.gitea.io/gitea/models/repo"
|
repo_model "code.gitea.io/gitea/models/repo"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
"code.gitea.io/gitea/modules/base"
|
"code.gitea.io/gitea/modules/base"
|
||||||
"code.gitea.io/gitea/modules/git"
|
"code.gitea.io/gitea/modules/git"
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/markup"
|
|
||||||
"code.gitea.io/gitea/modules/markup/markdown"
|
"code.gitea.io/gitea/modules/markup/markdown"
|
||||||
"code.gitea.io/gitea/modules/optional"
|
"code.gitea.io/gitea/modules/optional"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
@ -72,17 +72,17 @@ func userProfile(ctx *context.Context) {
|
|||||||
ctx.Data["HeatmapTotalContributions"] = activities_model.GetTotalContributionsInHeatmap(data)
|
ctx.Data["HeatmapTotalContributions"] = activities_model.GetTotalContributionsInHeatmap(data)
|
||||||
}
|
}
|
||||||
|
|
||||||
profileDbRepo, profileGitRepo, profileReadmeBlob, profileClose := shared_user.FindUserProfileReadme(ctx, ctx.Doer)
|
profileDbRepo, _ /*profileGitRepo*/, profileReadmeBlob, profileClose := shared_user.FindUserProfileReadme(ctx, ctx.Doer)
|
||||||
defer profileClose()
|
defer profileClose()
|
||||||
|
|
||||||
showPrivate := ctx.IsSigned && (ctx.Doer.IsAdmin || ctx.Doer.ID == ctx.ContextUser.ID)
|
showPrivate := ctx.IsSigned && (ctx.Doer.IsAdmin || ctx.Doer.ID == ctx.ContextUser.ID)
|
||||||
prepareUserProfileTabData(ctx, showPrivate, profileDbRepo, profileGitRepo, profileReadmeBlob)
|
prepareUserProfileTabData(ctx, showPrivate, profileDbRepo, profileReadmeBlob)
|
||||||
// call PrepareContextForProfileBigAvatar later to avoid re-querying the NumFollowers & NumFollowing
|
// call PrepareContextForProfileBigAvatar later to avoid re-querying the NumFollowers & NumFollowing
|
||||||
shared_user.PrepareContextForProfileBigAvatar(ctx)
|
shared_user.PrepareContextForProfileBigAvatar(ctx)
|
||||||
ctx.HTML(http.StatusOK, tplProfile)
|
ctx.HTML(http.StatusOK, tplProfile)
|
||||||
}
|
}
|
||||||
|
|
||||||
func prepareUserProfileTabData(ctx *context.Context, showPrivate bool, profileDbRepo *repo_model.Repository, profileGitRepo *git.Repository, profileReadme *git.Blob) {
|
func prepareUserProfileTabData(ctx *context.Context, showPrivate bool, profileDbRepo *repo_model.Repository, profileReadme *git.Blob) {
|
||||||
// if there is a profile readme, default to "overview" page, otherwise, default to "repositories" page
|
// if there is a profile readme, default to "overview" page, otherwise, default to "repositories" page
|
||||||
// if there is not a profile readme, the overview tab should be treated as the repositories tab
|
// if there is not a profile readme, the overview tab should be treated as the repositories tab
|
||||||
tab := ctx.FormString("tab")
|
tab := ctx.FormString("tab")
|
||||||
@ -246,19 +246,10 @@ func prepareUserProfileTabData(ctx *context.Context, showPrivate bool, profileDb
|
|||||||
if bytes, err := profileReadme.GetBlobContent(setting.UI.MaxDisplayFileSize); err != nil {
|
if bytes, err := profileReadme.GetBlobContent(setting.UI.MaxDisplayFileSize); err != nil {
|
||||||
log.Error("failed to GetBlobContent: %v", err)
|
log.Error("failed to GetBlobContent: %v", err)
|
||||||
} else {
|
} else {
|
||||||
if profileContent, err := markdown.RenderString(&markup.RenderContext{
|
rctx := renderhelper.NewRenderContextRepoFile(ctx, profileDbRepo, renderhelper.RepoFileOptions{
|
||||||
Ctx: ctx,
|
CurrentRefPath: path.Join("branch", util.PathEscapeSegments(profileDbRepo.DefaultBranch)),
|
||||||
GitRepo: profileGitRepo,
|
})
|
||||||
Links: markup.Links{
|
if profileContent, err := markdown.RenderString(rctx, bytes); err != nil {
|
||||||
// Give the repo link to the markdown render for the full link of media element.
|
|
||||||
// the media link usually be like /[user]/[repoName]/media/branch/[branchName],
|
|
||||||
// Eg. /Tom/.profile/media/branch/main
|
|
||||||
// The branch shown on the profile page is the default branch, this need to be in sync with doc, see:
|
|
||||||
// https://docs.gitea.com/usage/profile-readme
|
|
||||||
Base: profileDbRepo.Link(),
|
|
||||||
BranchPath: path.Join("branch", util.PathEscapeSegments(profileDbRepo.DefaultBranch)),
|
|
||||||
},
|
|
||||||
}, bytes); err != nil {
|
|
||||||
log.Error("failed to RenderString: %v", err)
|
log.Error("failed to RenderString: %v", err)
|
||||||
} else {
|
} else {
|
||||||
ctx.Data["ProfileReadme"] = profileContent
|
ctx.Data["ProfileReadme"] = profileContent
|
||||||
|
@ -77,8 +77,8 @@ func (b *Basic) Verify(req *http.Request, w http.ResponseWriter, store DataStore
|
|||||||
log.Trace("Basic Authorization: Attempting login with username as token")
|
log.Trace("Basic Authorization: Attempting login with username as token")
|
||||||
}
|
}
|
||||||
|
|
||||||
// check oauth2 token
|
// get oauth2 token's user's ID
|
||||||
uid := CheckOAuthAccessToken(req.Context(), authToken)
|
_, uid := GetOAuthAccessTokenScopeAndUserID(req.Context(), authToken)
|
||||||
if uid != 0 {
|
if uid != 0 {
|
||||||
log.Trace("Basic Authorization: Valid OAuthAccessToken for user[%d]", uid)
|
log.Trace("Basic Authorization: Valid OAuthAccessToken for user[%d]", uid)
|
||||||
|
|
||||||
|
@ -26,33 +26,35 @@ var (
|
|||||||
_ Method = &OAuth2{}
|
_ Method = &OAuth2{}
|
||||||
)
|
)
|
||||||
|
|
||||||
// CheckOAuthAccessToken returns uid of user from oauth token
|
// GetOAuthAccessTokenScopeAndUserID returns access token scope and user id
|
||||||
func CheckOAuthAccessToken(ctx context.Context, accessToken string) int64 {
|
func GetOAuthAccessTokenScopeAndUserID(ctx context.Context, accessToken string) (auth_model.AccessTokenScope, int64) {
|
||||||
|
var accessTokenScope auth_model.AccessTokenScope
|
||||||
if !setting.OAuth2.Enabled {
|
if !setting.OAuth2.Enabled {
|
||||||
return 0
|
return accessTokenScope, 0
|
||||||
}
|
}
|
||||||
|
|
||||||
// JWT tokens require a ".", if the token isn't like that, return early
|
// JWT tokens require a ".", if the token isn't like that, return early
|
||||||
if !strings.Contains(accessToken, ".") {
|
if !strings.Contains(accessToken, ".") {
|
||||||
return 0
|
return accessTokenScope, 0
|
||||||
}
|
}
|
||||||
|
|
||||||
token, err := oauth2_provider.ParseToken(accessToken, oauth2_provider.DefaultSigningKey)
|
token, err := oauth2_provider.ParseToken(accessToken, oauth2_provider.DefaultSigningKey)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Trace("oauth2.ParseToken: %v", err)
|
log.Trace("oauth2.ParseToken: %v", err)
|
||||||
return 0
|
return accessTokenScope, 0
|
||||||
}
|
}
|
||||||
var grant *auth_model.OAuth2Grant
|
var grant *auth_model.OAuth2Grant
|
||||||
if grant, err = auth_model.GetOAuth2GrantByID(ctx, token.GrantID); err != nil || grant == nil {
|
if grant, err = auth_model.GetOAuth2GrantByID(ctx, token.GrantID); err != nil || grant == nil {
|
||||||
return 0
|
return accessTokenScope, 0
|
||||||
}
|
}
|
||||||
if token.Kind != oauth2_provider.KindAccessToken {
|
if token.Kind != oauth2_provider.KindAccessToken {
|
||||||
return 0
|
return accessTokenScope, 0
|
||||||
}
|
}
|
||||||
if token.ExpiresAt.Before(time.Now()) || token.IssuedAt.After(time.Now()) {
|
if token.ExpiresAt.Before(time.Now()) || token.IssuedAt.After(time.Now()) {
|
||||||
return 0
|
return accessTokenScope, 0
|
||||||
}
|
}
|
||||||
return grant.UserID
|
accessTokenScope = oauth2_provider.GrantAdditionalScopes(grant.Scope)
|
||||||
|
return accessTokenScope, grant.UserID
|
||||||
}
|
}
|
||||||
|
|
||||||
// CheckTaskIsRunning verifies that the TaskID corresponds to a running task
|
// CheckTaskIsRunning verifies that the TaskID corresponds to a running task
|
||||||
@ -120,10 +122,10 @@ func (o *OAuth2) userIDFromToken(ctx context.Context, tokenSHA string, store Dat
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, check if this is an OAuth access token
|
// Otherwise, check if this is an OAuth access token
|
||||||
uid := CheckOAuthAccessToken(ctx, tokenSHA)
|
accessTokenScope, uid := GetOAuthAccessTokenScopeAndUserID(ctx, tokenSHA)
|
||||||
if uid != 0 {
|
if uid != 0 {
|
||||||
store.GetData()["IsApiToken"] = true
|
store.GetData()["IsApiToken"] = true
|
||||||
store.GetData()["ApiTokenScope"] = auth_model.AccessTokenScopeAll // fallback to all
|
store.GetData()["ApiTokenScope"] = accessTokenScope
|
||||||
}
|
}
|
||||||
return uid
|
return uid
|
||||||
}
|
}
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user