mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-01-26 15:19:47 +01:00
Use Project-URL
metadata field to get a PyPI package's homepage URL (#33089)
Resolves #33085. (cherry picked from commit 188e0ee8e40ad0b32f9db33a0a217043cfdf3610) Conflicts: tests/integration/api_packages_pypi_test.go trivial context conflict
This commit is contained in:
parent
e507fa30df
commit
76a85d26c8
3 changed files with 116 additions and 10 deletions
|
@ -10,6 +10,7 @@ import (
|
||||||
"regexp"
|
"regexp"
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
|
"unicode"
|
||||||
|
|
||||||
packages_model "code.gitea.io/gitea/models/packages"
|
packages_model "code.gitea.io/gitea/models/packages"
|
||||||
packages_module "code.gitea.io/gitea/modules/packages"
|
packages_module "code.gitea.io/gitea/modules/packages"
|
||||||
|
@ -139,9 +140,30 @@ func UploadPackageFile(ctx *context.Context) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
projectURL := ctx.Req.FormValue("home_page")
|
// Ensure ctx.Req.Form exists.
|
||||||
if !validation.IsValidURL(projectURL) {
|
_ = ctx.Req.ParseForm()
|
||||||
projectURL = ""
|
|
||||||
|
var homepageURL string
|
||||||
|
projectURLs := ctx.Req.Form["project_urls"]
|
||||||
|
for _, purl := range projectURLs {
|
||||||
|
label, url, found := strings.Cut(purl, ",")
|
||||||
|
if !found {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if normalizeLabel(label) != "homepage" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
homepageURL = strings.TrimSpace(url)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(homepageURL) == 0 {
|
||||||
|
// TODO: Home-page is a deprecated metadata field. Remove this branch once it's no longer apart of the spec.
|
||||||
|
homepageURL = ctx.Req.FormValue("home_page")
|
||||||
|
}
|
||||||
|
|
||||||
|
if !validation.IsValidURL(homepageURL) {
|
||||||
|
homepageURL = ""
|
||||||
}
|
}
|
||||||
|
|
||||||
_, _, err = packages_service.CreatePackageOrAddFileToExisting(
|
_, _, err = packages_service.CreatePackageOrAddFileToExisting(
|
||||||
|
@ -160,7 +182,7 @@ func UploadPackageFile(ctx *context.Context) {
|
||||||
Description: ctx.Req.FormValue("description"),
|
Description: ctx.Req.FormValue("description"),
|
||||||
LongDescription: ctx.Req.FormValue("long_description"),
|
LongDescription: ctx.Req.FormValue("long_description"),
|
||||||
Summary: ctx.Req.FormValue("summary"),
|
Summary: ctx.Req.FormValue("summary"),
|
||||||
ProjectURL: projectURL,
|
ProjectURL: homepageURL,
|
||||||
License: ctx.Req.FormValue("license"),
|
License: ctx.Req.FormValue("license"),
|
||||||
RequiresPython: ctx.Req.FormValue("requires_python"),
|
RequiresPython: ctx.Req.FormValue("requires_python"),
|
||||||
},
|
},
|
||||||
|
@ -189,6 +211,23 @@ func UploadPackageFile(ctx *context.Context) {
|
||||||
ctx.Status(http.StatusCreated)
|
ctx.Status(http.StatusCreated)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Normalizes a Project-URL label.
|
||||||
|
// See https://packaging.python.org/en/latest/specifications/well-known-project-urls/#label-normalization.
|
||||||
|
func normalizeLabel(label string) string {
|
||||||
|
var builder strings.Builder
|
||||||
|
|
||||||
|
// "A label is normalized by deleting all ASCII punctuation and whitespace, and then converting the result
|
||||||
|
// to lowercase."
|
||||||
|
for _, r := range label {
|
||||||
|
if unicode.IsPunct(r) || unicode.IsSpace(r) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
builder.WriteRune(unicode.ToLower(r))
|
||||||
|
}
|
||||||
|
|
||||||
|
return builder.String()
|
||||||
|
}
|
||||||
|
|
||||||
func isValidNameAndVersion(packageName, packageVersion string) bool {
|
func isValidNameAndVersion(packageName, packageVersion string) bool {
|
||||||
return nameMatcher.MatchString(packageName) && versionMatcher.MatchString(packageVersion)
|
return nameMatcher.MatchString(packageName) && versionMatcher.MatchString(packageVersion)
|
||||||
}
|
}
|
||||||
|
|
|
@ -36,3 +36,13 @@ func TestIsValidNameAndVersion(t *testing.T) {
|
||||||
assert.False(t, isValidNameAndVersion("test-name", "1.0.1aa"))
|
assert.False(t, isValidNameAndVersion("test-name", "1.0.1aa"))
|
||||||
assert.False(t, isValidNameAndVersion("test-name", "1.0.0-alpha.beta"))
|
assert.False(t, isValidNameAndVersion("test-name", "1.0.0-alpha.beta"))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestNormalizeLabel(t *testing.T) {
|
||||||
|
// Cases fetched from https://packaging.python.org/en/latest/specifications/well-known-project-urls/#label-normalization.
|
||||||
|
assert.Equal(t, "homepage", normalizeLabel("Homepage"))
|
||||||
|
assert.Equal(t, "homepage", normalizeLabel("Home-page"))
|
||||||
|
assert.Equal(t, "homepage", normalizeLabel("Home page"))
|
||||||
|
assert.Equal(t, "changelog", normalizeLabel("Change_Log"))
|
||||||
|
assert.Equal(t, "whatsnew", normalizeLabel("What's New?"))
|
||||||
|
assert.Equal(t, "github", normalizeLabel("github"))
|
||||||
|
}
|
||||||
|
|
|
@ -33,15 +33,16 @@ func TestPackagePyPI(t *testing.T) {
|
||||||
packageVersion := "1!1.0.1+r1234"
|
packageVersion := "1!1.0.1+r1234"
|
||||||
packageAuthor := "KN4CK3R"
|
packageAuthor := "KN4CK3R"
|
||||||
packageDescription := "Test Description"
|
packageDescription := "Test Description"
|
||||||
|
projectURL := "https://example.com"
|
||||||
|
|
||||||
content := "test"
|
content := "test"
|
||||||
hashSHA256 := "9f86d081884c7d659a2feaa0c55ad015a3bf4f1b2b0b822cd15d6c15b0f00a08"
|
hashSHA256 := "9f86d081884c7d659a2feaa0c55ad015a3bf4f1b2b0b822cd15d6c15b0f00a08"
|
||||||
|
|
||||||
root := fmt.Sprintf("/api/packages/%s/pypi", user.Name)
|
root := fmt.Sprintf("/api/packages/%s/pypi", user.Name)
|
||||||
|
|
||||||
uploadFile := func(t *testing.T, filename, content string, expectedStatus int) {
|
createBasicMultipartFile := func(filename, packageName, content string) (body *bytes.Buffer, writer *multipart.Writer, closer func() error) {
|
||||||
body := &bytes.Buffer{}
|
body = &bytes.Buffer{}
|
||||||
writer := multipart.NewWriter(body)
|
writer = multipart.NewWriter(body)
|
||||||
part, _ := writer.CreateFormFile("content", filename)
|
part, _ := writer.CreateFormFile("content", filename)
|
||||||
_, _ = io.Copy(part, strings.NewReader(content))
|
_, _ = io.Copy(part, strings.NewReader(content))
|
||||||
|
|
||||||
|
@ -53,14 +54,27 @@ func TestPackagePyPI(t *testing.T) {
|
||||||
writer.WriteField("sha256_digest", hashSHA256)
|
writer.WriteField("sha256_digest", hashSHA256)
|
||||||
writer.WriteField("requires_python", "3.6")
|
writer.WriteField("requires_python", "3.6")
|
||||||
|
|
||||||
_ = writer.Close()
|
return body, writer, writer.Close
|
||||||
|
}
|
||||||
|
|
||||||
|
uploadHelper := func(t *testing.T, body *bytes.Buffer, contentType string, expectedStatus int) {
|
||||||
req := NewRequestWithBody(t, "POST", root, body).
|
req := NewRequestWithBody(t, "POST", root, body).
|
||||||
SetHeader("Content-Type", writer.FormDataContentType()).
|
SetHeader("Content-Type", contentType).
|
||||||
AddBasicAuth(user.Name)
|
AddBasicAuth(user.Name)
|
||||||
MakeRequest(t, req, expectedStatus)
|
MakeRequest(t, req, expectedStatus)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uploadFile := func(t *testing.T, filename, content string, expectedStatus int) {
|
||||||
|
body, writer, closeFunc := createBasicMultipartFile(filename, packageName, content)
|
||||||
|
|
||||||
|
writer.WriteField("project_urls", "DOCUMENTATION , https://readthedocs.org")
|
||||||
|
writer.WriteField("project_urls", fmt.Sprintf("Home-page, %s", projectURL))
|
||||||
|
|
||||||
|
_ = closeFunc()
|
||||||
|
|
||||||
|
uploadHelper(t, body, writer.FormDataContentType(), expectedStatus)
|
||||||
|
}
|
||||||
|
|
||||||
t.Run("Upload", func(t *testing.T) {
|
t.Run("Upload", func(t *testing.T) {
|
||||||
defer tests.PrintCurrentTest(t)()
|
defer tests.PrintCurrentTest(t)()
|
||||||
|
|
||||||
|
@ -75,6 +89,7 @@ func TestPackagePyPI(t *testing.T) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Nil(t, pd.SemVer)
|
assert.Nil(t, pd.SemVer)
|
||||||
assert.IsType(t, &pypi.Metadata{}, pd.Metadata)
|
assert.IsType(t, &pypi.Metadata{}, pd.Metadata)
|
||||||
|
assert.Equal(t, projectURL, pd.Metadata.(*pypi.Metadata).ProjectURL)
|
||||||
assert.Equal(t, packageName, pd.Package.Name)
|
assert.Equal(t, packageName, pd.Package.Name)
|
||||||
assert.Equal(t, packageVersion, pd.Version.Version)
|
assert.Equal(t, packageVersion, pd.Version.Version)
|
||||||
|
|
||||||
|
@ -134,6 +149,48 @@ func TestPackagePyPI(t *testing.T) {
|
||||||
uploadFile(t, "test.tar.gz", content, http.StatusConflict)
|
uploadFile(t, "test.tar.gz", content, http.StatusConflict)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.Run("UploadUsingDeprecatedHomepageMetadata", func(t *testing.T) {
|
||||||
|
defer tests.PrintCurrentTest(t)()
|
||||||
|
|
||||||
|
pkgName := "homepage-package"
|
||||||
|
body, writer, closeFunc := createBasicMultipartFile("test.whl", pkgName, content)
|
||||||
|
|
||||||
|
writer.WriteField("home_page", projectURL)
|
||||||
|
|
||||||
|
_ = closeFunc()
|
||||||
|
|
||||||
|
uploadHelper(t, body, writer.FormDataContentType(), http.StatusCreated)
|
||||||
|
|
||||||
|
pvs, err := packages.GetVersionsByPackageName(db.DefaultContext, user.ID, packages.TypePyPI, pkgName)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Len(t, pvs, 1)
|
||||||
|
|
||||||
|
pd, err := packages.GetPackageDescriptor(db.DefaultContext, pvs[0])
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.IsType(t, &pypi.Metadata{}, pd.Metadata)
|
||||||
|
assert.Equal(t, projectURL, pd.Metadata.(*pypi.Metadata).ProjectURL)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("UploadWithoutAnyHomepageURLMetadata", func(t *testing.T) {
|
||||||
|
defer tests.PrintCurrentTest(t)()
|
||||||
|
|
||||||
|
pkgName := "no-project-url-or-homepage-package"
|
||||||
|
body, writer, closeFunc := createBasicMultipartFile("test.whl", pkgName, content)
|
||||||
|
|
||||||
|
_ = closeFunc()
|
||||||
|
|
||||||
|
uploadHelper(t, body, writer.FormDataContentType(), http.StatusCreated)
|
||||||
|
|
||||||
|
pvs, err := packages.GetVersionsByPackageName(db.DefaultContext, user.ID, packages.TypePyPI, pkgName)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Len(t, pvs, 1)
|
||||||
|
|
||||||
|
pd, err := packages.GetPackageDescriptor(db.DefaultContext, pvs[0])
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.IsType(t, &pypi.Metadata{}, pd.Metadata)
|
||||||
|
assert.Empty(t, pd.Metadata.(*pypi.Metadata).ProjectURL)
|
||||||
|
})
|
||||||
|
|
||||||
t.Run("Download", func(t *testing.T) {
|
t.Run("Download", func(t *testing.T) {
|
||||||
defer tests.PrintCurrentTest(t)()
|
defer tests.PrintCurrentTest(t)()
|
||||||
|
|
||||||
|
@ -148,7 +205,7 @@ func TestPackagePyPI(t *testing.T) {
|
||||||
downloadFile("test.whl")
|
downloadFile("test.whl")
|
||||||
downloadFile("test.tar.gz")
|
downloadFile("test.tar.gz")
|
||||||
|
|
||||||
pvs, err := packages.GetVersionsByPackageType(db.DefaultContext, user.ID, packages.TypePyPI)
|
pvs, err := packages.GetVersionsByPackageName(db.DefaultContext, user.ID, packages.TypePyPI, packageName)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Len(t, pvs, 1)
|
assert.Len(t, pvs, 1)
|
||||||
assert.Equal(t, int64(2), pvs[0].DownloadCount)
|
assert.Equal(t, int64(2), pvs[0].DownloadCount)
|
||||||
|
|
Loading…
Reference in a new issue