mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-31 06:21:11 +00:00 
			
		
		
		
	Merge pull request '[GITEA] Improve HTML title on repositories' (#1283) from Gusted/forgejo:forgejo-html-title-repo-fullname into forgejo-dependency
Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/1283
This commit is contained in:
		
				commit
				
					
						c21c463d12
					
				
			
		
					 25 changed files with 206 additions and 10 deletions
				
			
		|  | @ -136,3 +136,17 @@ | |||
|   is_prerelease: false | ||||
|   is_tag: false | ||||
|   created_unix: 946684803 | ||||
| 
 | ||||
| - id: 11 | ||||
|   repo_id: 59 | ||||
|   publisher_id: 2 | ||||
|   tag_name: "v1.0" | ||||
|   lower_tag_name: "v1.0" | ||||
|   target: "main" | ||||
|   title: "v1.0" | ||||
|   sha1: "d8f53dfb33f6ccf4169c34970b5e747511c18beb" | ||||
|   num_commits: 1 | ||||
|   is_draft: false | ||||
|   is_prerelease: false | ||||
|   is_tag: false | ||||
|   created_unix: 946684803 | ||||
|  |  | |||
|  | @ -622,6 +622,38 @@ | |||
|   created_unix: 946684810 | ||||
| # END Forgejo [GITEA] Fix generated source URL on rendered files | ||||
| 
 | ||||
| # BEGIN Forgejo [GITEA] Improve HTML title on repositories | ||||
| - | ||||
|   id: 1093 | ||||
|   repo_id: 59 | ||||
|   type: 1 | ||||
|   created_unix: 946684810 | ||||
| 
 | ||||
| - | ||||
|   id: 1094 | ||||
|   repo_id: 59 | ||||
|   type: 2 | ||||
|   created_unix: 946684810 | ||||
| 
 | ||||
| - | ||||
|   id: 1095 | ||||
|   repo_id: 59 | ||||
|   type: 3 | ||||
|   created_unix: 946684810 | ||||
| 
 | ||||
| - | ||||
|   id: 1096 | ||||
|   repo_id: 59 | ||||
|   type: 4 | ||||
|   created_unix: 946684810 | ||||
| 
 | ||||
| - | ||||
|   id: 1097 | ||||
|   repo_id: 59 | ||||
|   type: 5 | ||||
|   created_unix: 946684810 | ||||
| # END Forgejo [GITEA] Improve HTML title on repositories | ||||
| 
 | ||||
| - | ||||
|   id: 91 | ||||
|   repo_id: 58 | ||||
|  |  | |||
|  | @ -1467,6 +1467,7 @@ | |||
|   owner_name: user27 | ||||
|   lower_name: repo49 | ||||
|   name: repo49 | ||||
|   description: A wonderful repository with more than just a README.md | ||||
|   default_branch: master | ||||
|   num_watches: 0 | ||||
|   num_stars: 0 | ||||
|  | @ -1693,3 +1694,16 @@ | |||
|   size: 0 | ||||
|   is_fsck_enabled: true | ||||
|   close_issues_via_commit_in_any_branch: false | ||||
| 
 | ||||
| - | ||||
|   id: 59 | ||||
|   owner_id: 2 | ||||
|   owner_name: user2 | ||||
|   lower_name: repo59 | ||||
|   name: repo59 | ||||
|   default_branch: master | ||||
|   is_empty: false | ||||
|   is_archived: false | ||||
|   is_private: false | ||||
|   status: 0 | ||||
|   num_issues: 0 | ||||
|  |  | |||
|  | @ -66,7 +66,7 @@ | |||
|   num_followers: 2 | ||||
|   num_following: 1 | ||||
|   num_stars: 2 | ||||
|   num_repos: 14 | ||||
|   num_repos: 15 | ||||
|   num_teams: 0 | ||||
|   num_members: 0 | ||||
|   visibility: 0 | ||||
|  |  | |||
|  | @ -138,12 +138,12 @@ func getTestCases() []struct { | |||
| 		{ | ||||
| 			name:  "AllPublic/PublicRepositoriesOfUserIncludingCollaborative", | ||||
| 			opts:  &repo_model.SearchRepoOptions{ListOptions: db.ListOptions{Page: 1, PageSize: 10}, OwnerID: 15, AllPublic: true, Template: util.OptionalBoolFalse}, | ||||
| 			count: 31, | ||||
| 			count: 32, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:  "AllPublic/PublicAndPrivateRepositoriesOfUserIncludingCollaborative", | ||||
| 			opts:  &repo_model.SearchRepoOptions{ListOptions: db.ListOptions{Page: 1, PageSize: 10}, OwnerID: 15, Private: true, AllPublic: true, AllLimited: true, Template: util.OptionalBoolFalse}, | ||||
| 			count: 36, | ||||
| 			count: 37, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:  "AllPublic/PublicAndPrivateRepositoriesOfUserIncludingCollaborativeByName", | ||||
|  | @ -158,7 +158,7 @@ func getTestCases() []struct { | |||
| 		{ | ||||
| 			name:  "AllPublic/PublicRepositoriesOfOrganization", | ||||
| 			opts:  &repo_model.SearchRepoOptions{ListOptions: db.ListOptions{Page: 1, PageSize: 10}, OwnerID: 17, AllPublic: true, Collaborate: util.OptionalBoolFalse, Template: util.OptionalBoolFalse}, | ||||
| 			count: 31, | ||||
| 			count: 32, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:  "AllTemplates", | ||||
|  |  | |||
|  | @ -164,7 +164,7 @@ func renderDirectory(ctx *context.Context, treeLink string) { | |||
| 
 | ||||
| 	if ctx.Repo.TreePath != "" { | ||||
| 		ctx.Data["HideRepoInfo"] = true | ||||
| 		ctx.Data["Title"] = ctx.Tr("repo.file.title", ctx.Repo.Repository.Name+"/"+path.Base(ctx.Repo.TreePath), ctx.Repo.RefName) | ||||
| 		ctx.Data["Title"] = ctx.Tr("repo.file.title", ctx.Repo.Repository.Name+"/"+util.PathEscapeSegments(ctx.Repo.TreePath), ctx.Repo.RefName) | ||||
| 	} | ||||
| 
 | ||||
| 	subfolder, readmeFile, err := findReadmeFileInEntries(ctx, entries, true) | ||||
|  | @ -343,7 +343,7 @@ func renderFile(ctx *context.Context, entry *git.TreeEntry, treeLink, rawLink st | |||
| 	} | ||||
| 	defer dataRc.Close() | ||||
| 
 | ||||
| 	ctx.Data["Title"] = ctx.Tr("repo.file.title", ctx.Repo.Repository.Name+"/"+path.Base(ctx.Repo.TreePath), ctx.Repo.RefName) | ||||
| 	ctx.Data["Title"] = ctx.Tr("repo.file.title", ctx.Repo.Repository.Name+"/"+util.PathEscapeSegments(ctx.Repo.TreePath), ctx.Repo.RefName) | ||||
| 	ctx.Data["FileIsSymlink"] = entry.IsLink() | ||||
| 	ctx.Data["FileName"] = blob.Name() | ||||
| 	ctx.Data["RawFileLink"] = rawLink + "/" + util.PathEscapeSegments(ctx.Repo.TreePath) | ||||
|  |  | |||
|  | @ -2,7 +2,8 @@ | |||
| <html lang="{{ctx.Locale.Lang}}" class="theme-{{if .SignedUser.Theme}}{{.SignedUser.Theme}}{{else}}{{DefaultTheme}}{{end}}"> | ||||
| <head> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=1"> | ||||
| 	<title>{{if .Title}}{{.Title | RenderEmojiPlain}} - {{end}}{{if .Repository.Name}}{{.Repository.Name}} - {{end}}{{AppName}}</title> | ||||
| 	{{/* Display `- .Repsository.FullName` only if `.Title` does not already start with that. */}} | ||||
| 	<title>{{if .Title}}{{.Title | RenderEmojiPlain}} - {{end}}{{if and (.Repository.Name) (not (StringUtils.HasPrefix .Title .Repository.FullName))}}{{.Repository.FullName}} - {{end}}{{AppName}}</title> | ||||
| 	{{if .ManifestData}}<link rel="manifest" href="data:{{.ManifestData}}">{{end}} | ||||
| 	<meta name="author" content="{{if .Repository}}{{.Owner.Name}}{{else}}{{MetaAuthor}}{{end}}"> | ||||
| 	<meta name="description" content="{{if .Repository}}{{.Repository.Name}}{{if .Repository.Description}} - {{.Repository.Description}}{{end}}{{else}}{{MetaDescription}}{{end}}"> | ||||
|  |  | |||
							
								
								
									
										1
									
								
								tests/gitea-repositories-meta/user2/repo59.git/HEAD
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								tests/gitea-repositories-meta/user2/repo59.git/HEAD
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | |||
| ref: refs/heads/master | ||||
							
								
								
									
										4
									
								
								tests/gitea-repositories-meta/user2/repo59.git/config
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								tests/gitea-repositories-meta/user2/repo59.git/config
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,4 @@ | |||
| [core] | ||||
| 	repositoryformatversion = 0 | ||||
| 	filemode = true | ||||
| 	bare = true | ||||
|  | @ -0,0 +1 @@ | |||
| Unnamed repository; edit this file 'description' to name the repository. | ||||
|  | @ -0,0 +1,6 @@ | |||
| # git ls-files --others --exclude-from=.git/info/exclude | ||||
| # Lines that start with '#' are comments. | ||||
| # For a project mostly in C, the following would be a good set of | ||||
| # exclude patterns (uncomment them if you want to use them): | ||||
| # *.[oa] | ||||
| # *~ | ||||
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							|  | @ -0,0 +1,3 @@ | |||
| # pack-refs with: peeled fully-peeled sorted  | ||||
| d8f53dfb33f6ccf4169c34970b5e747511c18beb refs/heads/master | ||||
| d8f53dfb33f6ccf4169c34970b5e747511c18beb refs/tags/v1.0 | ||||
|  | @ -0,0 +1 @@ | |||
| d8f53dfb33f6ccf4169c34970b5e747511c18beb | ||||
|  | @ -93,9 +93,9 @@ func TestAPISearchRepo(t *testing.T) { | |||
| 	}{ | ||||
| 		{ | ||||
| 			name: "RepositoriesMax50", requestURL: "/api/v1/repos/search?limit=50&private=false", expectedResults: expectedResults{ | ||||
| 				nil:   {count: 33}, | ||||
| 				user:  {count: 33}, | ||||
| 				user2: {count: 33}, | ||||
| 				nil:   {count: 34}, | ||||
| 				user:  {count: 34}, | ||||
| 				user2: {count: 34}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
|  |  | |||
|  | @ -531,3 +531,18 @@ func GetCSRF(t testing.TB, session *TestSession, urlStr string) string { | |||
| 	doc := NewHTMLParser(t, resp.Body) | ||||
| 	return doc.GetCSRF() | ||||
| } | ||||
| 
 | ||||
| func GetHTMLTitle(t testing.TB, session *TestSession, urlStr string) string { | ||||
| 	t.Helper() | ||||
| 
 | ||||
| 	req := NewRequest(t, "GET", urlStr) | ||||
| 	var resp *httptest.ResponseRecorder | ||||
| 	if session == nil { | ||||
| 		resp = MakeRequest(t, req, http.StatusOK) | ||||
| 	} else { | ||||
| 		resp = session.MakeRequest(t, req, http.StatusOK) | ||||
| 	} | ||||
| 
 | ||||
| 	doc := NewHTMLParser(t, resp.Body) | ||||
| 	return doc.Find("head title").Text() | ||||
| } | ||||
|  |  | |||
|  | @ -444,3 +444,107 @@ func TestGeneratedSourceLink(t *testing.T) { | |||
| 		assert.Equal(t, "/user27/repo49/src/commit/aacbdfe9e1c4b47f60abe81849045fa4e96f1d75/test/test.txt", dataURL) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func TestRepoHTMLTitle(t *testing.T) { | ||||
| 	defer tests.PrepareTestEnv(t)() | ||||
| 
 | ||||
| 	t.Run("Repository homepage", func(t *testing.T) { | ||||
| 		t.Run("Without description", func(t *testing.T) { | ||||
| 			defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 			htmlTitle := GetHTMLTitle(t, nil, "/user2/repo1") | ||||
| 			assert.EqualValues(t, "user2/repo1 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 		}) | ||||
| 		t.Run("With description", func(t *testing.T) { | ||||
| 			defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 			htmlTitle := GetHTMLTitle(t, nil, "/user27/repo49") | ||||
| 			assert.EqualValues(t, "user27/repo49: A wonderful repository with more than just a README.md - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 		}) | ||||
| 	}) | ||||
| 
 | ||||
| 	t.Run("Code view", func(t *testing.T) { | ||||
| 		t.Run("Directory", func(t *testing.T) { | ||||
| 			t.Run("Default branch", func(t *testing.T) { | ||||
| 				defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 				htmlTitle := GetHTMLTitle(t, nil, "/user2/repo59/src/branch/master/deep/nesting") | ||||
| 				assert.EqualValues(t, "repo59/deep/nesting at master - user2/repo59 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 			}) | ||||
| 			t.Run("Non-default branch", func(t *testing.T) { | ||||
| 				defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 				htmlTitle := GetHTMLTitle(t, nil, "/user2/repo59/src/branch/cake-recipe/deep/nesting") | ||||
| 				assert.EqualValues(t, "repo59/deep/nesting at cake-recipe - user2/repo59 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 			}) | ||||
| 			t.Run("Commit", func(t *testing.T) { | ||||
| 				defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 				htmlTitle := GetHTMLTitle(t, nil, "/user2/repo59/src/commit/d8f53dfb33f6ccf4169c34970b5e747511c18beb/deep/nesting/") | ||||
| 				assert.EqualValues(t, "repo59/deep/nesting at d8f53dfb33f6ccf4169c34970b5e747511c18beb - user2/repo59 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 			}) | ||||
| 			t.Run("Tag", func(t *testing.T) { | ||||
| 				defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 				htmlTitle := GetHTMLTitle(t, nil, "/user2/repo59/src/tag/v1.0/deep/nesting/") | ||||
| 				assert.EqualValues(t, "repo59/deep/nesting at v1.0 - user2/repo59 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 			}) | ||||
| 		}) | ||||
| 		t.Run("File", func(t *testing.T) { | ||||
| 			t.Run("Default branch", func(t *testing.T) { | ||||
| 				defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 				htmlTitle := GetHTMLTitle(t, nil, "/user2/repo59/src/branch/master/deep/nesting/folder/secret_sauce_recipe.txt") | ||||
| 				assert.EqualValues(t, "repo59/deep/nesting/folder/secret_sauce_recipe.txt at master - user2/repo59 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 			}) | ||||
| 			t.Run("Non-default branch", func(t *testing.T) { | ||||
| 				defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 				htmlTitle := GetHTMLTitle(t, nil, "/user2/repo59/src/branch/cake-recipe/deep/nesting/folder/secret_sauce_recipe.txt") | ||||
| 				assert.EqualValues(t, "repo59/deep/nesting/folder/secret_sauce_recipe.txt at cake-recipe - user2/repo59 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 			}) | ||||
| 			t.Run("Commit", func(t *testing.T) { | ||||
| 				defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 				htmlTitle := GetHTMLTitle(t, nil, "/user2/repo59/src/commit/d8f53dfb33f6ccf4169c34970b5e747511c18beb/deep/nesting/folder/secret_sauce_recipe.txt") | ||||
| 				assert.EqualValues(t, "repo59/deep/nesting/folder/secret_sauce_recipe.txt at d8f53dfb33f6ccf4169c34970b5e747511c18beb - user2/repo59 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 			}) | ||||
| 			t.Run("Tag", func(t *testing.T) { | ||||
| 				defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 				htmlTitle := GetHTMLTitle(t, nil, "/user2/repo59/src/tag/v1.0/deep/nesting/folder/secret_sauce_recipe.txt") | ||||
| 				assert.EqualValues(t, "repo59/deep/nesting/folder/secret_sauce_recipe.txt at v1.0 - user2/repo59 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 			}) | ||||
| 		}) | ||||
| 	}) | ||||
| 
 | ||||
| 	t.Run("Issues view", func(t *testing.T) { | ||||
| 		t.Run("Overview page", func(t *testing.T) { | ||||
| 			defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 			htmlTitle := GetHTMLTitle(t, nil, "/user2/repo1/issues") | ||||
| 			assert.EqualValues(t, "Issues - user2/repo1 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 		}) | ||||
| 		t.Run("View issue page", func(t *testing.T) { | ||||
| 			defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 			htmlTitle := GetHTMLTitle(t, nil, "/user2/repo1/issues/1") | ||||
| 			assert.EqualValues(t, "#1 - issue1 - user2/repo1 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 		}) | ||||
| 	}) | ||||
| 
 | ||||
| 	t.Run("Pull requests view", func(t *testing.T) { | ||||
| 		t.Run("Overview page", func(t *testing.T) { | ||||
| 			defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 			htmlTitle := GetHTMLTitle(t, nil, "/user2/repo1/pulls") | ||||
| 			assert.EqualValues(t, "Pull Requests - user2/repo1 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 		}) | ||||
| 		t.Run("View pull request", func(t *testing.T) { | ||||
| 			defer tests.PrintCurrentTest(t)() | ||||
| 
 | ||||
| 			htmlTitle := GetHTMLTitle(t, nil, "/user2/repo1/pulls/2") | ||||
| 			assert.EqualValues(t, "#2 - issue2 - user2/repo1 - Gitea: Git with a cup of tea", htmlTitle) | ||||
| 		}) | ||||
| 	}) | ||||
| } | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue