mirror of
https://codeberg.org/forgejo/forgejo
synced 2024-11-24 10:46:10 +01:00
86ace4b5c2
Fixes #22178 After this change upload versions with different semver metadata are treated as the same version and trigger a duplicated version error. Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com>
393 lines
12 KiB
Go
393 lines
12 KiB
Go
// Copyright 2022 The Gitea Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package nuget
|
|
|
|
import (
|
|
"encoding/xml"
|
|
"strings"
|
|
"time"
|
|
|
|
packages_model "code.gitea.io/gitea/models/packages"
|
|
nuget_module "code.gitea.io/gitea/modules/packages/nuget"
|
|
)
|
|
|
|
type AtomTitle struct {
|
|
Type string `xml:"type,attr"`
|
|
Text string `xml:",chardata"`
|
|
}
|
|
|
|
type ServiceCollection struct {
|
|
Href string `xml:"href,attr"`
|
|
Title AtomTitle `xml:"atom:title"`
|
|
}
|
|
|
|
type ServiceWorkspace struct {
|
|
Title AtomTitle `xml:"atom:title"`
|
|
Collection ServiceCollection `xml:"collection"`
|
|
}
|
|
|
|
type ServiceIndexResponseV2 struct {
|
|
XMLName xml.Name `xml:"service"`
|
|
Base string `xml:"base,attr"`
|
|
Xmlns string `xml:"xmlns,attr"`
|
|
XmlnsAtom string `xml:"xmlns:atom,attr"`
|
|
Workspace ServiceWorkspace `xml:"workspace"`
|
|
}
|
|
|
|
type EdmxPropertyRef struct {
|
|
Name string `xml:"Name,attr"`
|
|
}
|
|
|
|
type EdmxProperty struct {
|
|
Name string `xml:"Name,attr"`
|
|
Type string `xml:"Type,attr"`
|
|
Nullable bool `xml:"Nullable,attr"`
|
|
}
|
|
|
|
type EdmxEntityType struct {
|
|
Name string `xml:"Name,attr"`
|
|
HasStream bool `xml:"m:HasStream,attr"`
|
|
Keys []EdmxPropertyRef `xml:"Key>PropertyRef"`
|
|
Properties []EdmxProperty `xml:"Property"`
|
|
}
|
|
|
|
type EdmxFunctionParameter struct {
|
|
Name string `xml:"Name,attr"`
|
|
Type string `xml:"Type,attr"`
|
|
}
|
|
|
|
type EdmxFunctionImport struct {
|
|
Name string `xml:"Name,attr"`
|
|
ReturnType string `xml:"ReturnType,attr"`
|
|
EntitySet string `xml:"EntitySet,attr"`
|
|
Parameter []EdmxFunctionParameter `xml:"Parameter"`
|
|
}
|
|
|
|
type EdmxEntitySet struct {
|
|
Name string `xml:"Name,attr"`
|
|
EntityType string `xml:"EntityType,attr"`
|
|
}
|
|
|
|
type EdmxEntityContainer struct {
|
|
Name string `xml:"Name,attr"`
|
|
IsDefaultEntityContainer bool `xml:"m:IsDefaultEntityContainer,attr"`
|
|
EntitySet EdmxEntitySet `xml:"EntitySet"`
|
|
FunctionImports []EdmxFunctionImport `xml:"FunctionImport"`
|
|
}
|
|
|
|
type EdmxSchema struct {
|
|
Xmlns string `xml:"xmlns,attr"`
|
|
Namespace string `xml:"Namespace,attr"`
|
|
EntityType *EdmxEntityType `xml:"EntityType,omitempty"`
|
|
EntityContainer *EdmxEntityContainer `xml:"EntityContainer,omitempty"`
|
|
}
|
|
|
|
type EdmxDataServices struct {
|
|
XmlnsM string `xml:"xmlns:m,attr"`
|
|
DataServiceVersion string `xml:"m:DataServiceVersion,attr"`
|
|
MaxDataServiceVersion string `xml:"m:MaxDataServiceVersion,attr"`
|
|
Schema []EdmxSchema `xml:"Schema"`
|
|
}
|
|
|
|
type EdmxMetadata struct {
|
|
XMLName xml.Name `xml:"edmx:Edmx"`
|
|
XmlnsEdmx string `xml:"xmlns:edmx,attr"`
|
|
Version string `xml:"Version,attr"`
|
|
DataServices EdmxDataServices `xml:"edmx:DataServices"`
|
|
}
|
|
|
|
var Metadata = &EdmxMetadata{
|
|
XmlnsEdmx: "http://schemas.microsoft.com/ado/2007/06/edmx",
|
|
Version: "1.0",
|
|
DataServices: EdmxDataServices{
|
|
XmlnsM: "http://schemas.microsoft.com/ado/2007/08/dataservices/metadata",
|
|
DataServiceVersion: "2.0",
|
|
MaxDataServiceVersion: "2.0",
|
|
Schema: []EdmxSchema{
|
|
{
|
|
Xmlns: "http://schemas.microsoft.com/ado/2006/04/edm",
|
|
Namespace: "NuGetGallery.OData",
|
|
EntityType: &EdmxEntityType{
|
|
Name: "V2FeedPackage",
|
|
HasStream: true,
|
|
Keys: []EdmxPropertyRef{
|
|
{Name: "Id"},
|
|
{Name: "Version"},
|
|
},
|
|
Properties: []EdmxProperty{
|
|
{
|
|
Name: "Id",
|
|
Type: "Edm.String",
|
|
},
|
|
{
|
|
Name: "Version",
|
|
Type: "Edm.String",
|
|
},
|
|
{
|
|
Name: "NormalizedVersion",
|
|
Type: "Edm.String",
|
|
Nullable: true,
|
|
},
|
|
{
|
|
Name: "Authors",
|
|
Type: "Edm.String",
|
|
Nullable: true,
|
|
},
|
|
{
|
|
Name: "Created",
|
|
Type: "Edm.DateTime",
|
|
},
|
|
{
|
|
Name: "Dependencies",
|
|
Type: "Edm.String",
|
|
},
|
|
{
|
|
Name: "Description",
|
|
Type: "Edm.String",
|
|
},
|
|
{
|
|
Name: "DownloadCount",
|
|
Type: "Edm.Int64",
|
|
},
|
|
{
|
|
Name: "LastUpdated",
|
|
Type: "Edm.DateTime",
|
|
},
|
|
{
|
|
Name: "Published",
|
|
Type: "Edm.DateTime",
|
|
},
|
|
{
|
|
Name: "PackageSize",
|
|
Type: "Edm.Int64",
|
|
},
|
|
{
|
|
Name: "ProjectUrl",
|
|
Type: "Edm.String",
|
|
Nullable: true,
|
|
},
|
|
{
|
|
Name: "ReleaseNotes",
|
|
Type: "Edm.String",
|
|
Nullable: true,
|
|
},
|
|
{
|
|
Name: "RequireLicenseAcceptance",
|
|
Type: "Edm.Boolean",
|
|
Nullable: false,
|
|
},
|
|
{
|
|
Name: "Title",
|
|
Type: "Edm.String",
|
|
Nullable: true,
|
|
},
|
|
{
|
|
Name: "VersionDownloadCount",
|
|
Type: "Edm.Int64",
|
|
Nullable: false,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Xmlns: "http://schemas.microsoft.com/ado/2006/04/edm",
|
|
Namespace: "NuGetGallery",
|
|
EntityContainer: &EdmxEntityContainer{
|
|
Name: "V2FeedContext",
|
|
IsDefaultEntityContainer: true,
|
|
EntitySet: EdmxEntitySet{
|
|
Name: "Packages",
|
|
EntityType: "NuGetGallery.OData.V2FeedPackage",
|
|
},
|
|
FunctionImports: []EdmxFunctionImport{
|
|
{
|
|
Name: "Search",
|
|
ReturnType: "Collection(NuGetGallery.OData.V2FeedPackage)",
|
|
EntitySet: "Packages",
|
|
Parameter: []EdmxFunctionParameter{
|
|
{
|
|
Name: "searchTerm",
|
|
Type: "Edm.String",
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Name: "FindPackagesById",
|
|
ReturnType: "Collection(NuGetGallery.OData.V2FeedPackage)",
|
|
EntitySet: "Packages",
|
|
Parameter: []EdmxFunctionParameter{
|
|
{
|
|
Name: "id",
|
|
Type: "Edm.String",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
type FeedEntryCategory struct {
|
|
Term string `xml:"term,attr"`
|
|
Scheme string `xml:"scheme,attr"`
|
|
}
|
|
|
|
type FeedEntryLink struct {
|
|
Rel string `xml:"rel,attr"`
|
|
Href string `xml:"href,attr"`
|
|
}
|
|
|
|
type TypedValue[T any] struct {
|
|
Type string `xml:"type,attr,omitempty"`
|
|
Value T `xml:",chardata"`
|
|
}
|
|
|
|
type FeedEntryProperties struct {
|
|
Version string `xml:"d:Version"`
|
|
NormalizedVersion string `xml:"d:NormalizedVersion"`
|
|
Authors string `xml:"d:Authors"`
|
|
Dependencies string `xml:"d:Dependencies"`
|
|
Description string `xml:"d:Description"`
|
|
VersionDownloadCount TypedValue[int64] `xml:"d:VersionDownloadCount"`
|
|
DownloadCount TypedValue[int64] `xml:"d:DownloadCount"`
|
|
PackageSize TypedValue[int64] `xml:"d:PackageSize"`
|
|
Created TypedValue[time.Time] `xml:"d:Created"`
|
|
LastUpdated TypedValue[time.Time] `xml:"d:LastUpdated"`
|
|
Published TypedValue[time.Time] `xml:"d:Published"`
|
|
ProjectURL string `xml:"d:ProjectUrl,omitempty"`
|
|
ReleaseNotes string `xml:"d:ReleaseNotes,omitempty"`
|
|
RequireLicenseAcceptance TypedValue[bool] `xml:"d:RequireLicenseAcceptance"`
|
|
Title string `xml:"d:Title"`
|
|
}
|
|
|
|
type FeedEntry struct {
|
|
XMLName xml.Name `xml:"entry"`
|
|
Xmlns string `xml:"xmlns,attr,omitempty"`
|
|
XmlnsD string `xml:"xmlns:d,attr,omitempty"`
|
|
XmlnsM string `xml:"xmlns:m,attr,omitempty"`
|
|
Base string `xml:"xml:base,attr,omitempty"`
|
|
ID string `xml:"id"`
|
|
Category FeedEntryCategory `xml:"category"`
|
|
Links []FeedEntryLink `xml:"link"`
|
|
Title TypedValue[string] `xml:"title"`
|
|
Updated time.Time `xml:"updated"`
|
|
Author string `xml:"author>name"`
|
|
Summary string `xml:"summary"`
|
|
Properties *FeedEntryProperties `xml:"m:properties"`
|
|
Content string `xml:",innerxml"`
|
|
}
|
|
|
|
type FeedResponse struct {
|
|
XMLName xml.Name `xml:"feed"`
|
|
Xmlns string `xml:"xmlns,attr,omitempty"`
|
|
XmlnsD string `xml:"xmlns:d,attr,omitempty"`
|
|
XmlnsM string `xml:"xmlns:m,attr,omitempty"`
|
|
Base string `xml:"xml:base,attr,omitempty"`
|
|
ID string `xml:"id"`
|
|
Title TypedValue[string] `xml:"title"`
|
|
Updated time.Time `xml:"updated"`
|
|
Link FeedEntryLink `xml:"link"`
|
|
Entries []*FeedEntry `xml:"entry"`
|
|
Count int64 `xml:"m:count"`
|
|
}
|
|
|
|
func createFeedResponse(l *linkBuilder, totalEntries int64, pds []*packages_model.PackageDescriptor) *FeedResponse {
|
|
entries := make([]*FeedEntry, 0, len(pds))
|
|
for _, pd := range pds {
|
|
entries = append(entries, createEntry(l, pd, false))
|
|
}
|
|
|
|
return &FeedResponse{
|
|
Xmlns: "http://www.w3.org/2005/Atom",
|
|
Base: l.Base,
|
|
XmlnsD: "http://schemas.microsoft.com/ado/2007/08/dataservices",
|
|
XmlnsM: "http://schemas.microsoft.com/ado/2007/08/dataservices/metadata",
|
|
ID: "http://schemas.datacontract.org/2004/07/",
|
|
Updated: time.Now(),
|
|
Link: FeedEntryLink{Rel: "self", Href: l.Base},
|
|
Count: totalEntries,
|
|
Entries: entries,
|
|
}
|
|
}
|
|
|
|
func createEntryResponse(l *linkBuilder, pd *packages_model.PackageDescriptor) *FeedEntry {
|
|
return createEntry(l, pd, true)
|
|
}
|
|
|
|
func createEntry(l *linkBuilder, pd *packages_model.PackageDescriptor, withNamespace bool) *FeedEntry {
|
|
metadata := pd.Metadata.(*nuget_module.Metadata)
|
|
|
|
id := l.GetPackageMetadataURL(pd.Package.Name, pd.Version.Version)
|
|
|
|
// Workaround to force a self-closing tag to satisfy XmlReader.IsEmptyElement used by the NuGet client.
|
|
// https://learn.microsoft.com/en-us/dotnet/api/system.xml.xmlreader.isemptyelement
|
|
content := `<content type="application/zip" src="` + l.GetPackageDownloadURL(pd.Package.Name, pd.Version.Version) + `"/>`
|
|
|
|
createdValue := TypedValue[time.Time]{
|
|
Type: "Edm.DateTime",
|
|
Value: pd.Version.CreatedUnix.AsLocalTime(),
|
|
}
|
|
|
|
entry := &FeedEntry{
|
|
ID: id,
|
|
Category: FeedEntryCategory{Term: "NuGetGallery.OData.V2FeedPackage", Scheme: "http://schemas.microsoft.com/ado/2007/08/dataservices/scheme"},
|
|
Links: []FeedEntryLink{
|
|
{Rel: "self", Href: id},
|
|
{Rel: "edit", Href: id},
|
|
},
|
|
Title: TypedValue[string]{Type: "text", Value: pd.Package.Name},
|
|
Updated: pd.Version.CreatedUnix.AsLocalTime(),
|
|
Author: metadata.Authors,
|
|
Content: content,
|
|
Properties: &FeedEntryProperties{
|
|
Version: pd.Version.Version,
|
|
NormalizedVersion: pd.Version.Version,
|
|
Authors: metadata.Authors,
|
|
Dependencies: buildDependencyString(metadata),
|
|
Description: metadata.Description,
|
|
VersionDownloadCount: TypedValue[int64]{Type: "Edm.Int64", Value: pd.Version.DownloadCount},
|
|
DownloadCount: TypedValue[int64]{Type: "Edm.Int64", Value: pd.Version.DownloadCount},
|
|
PackageSize: TypedValue[int64]{Type: "Edm.Int64", Value: pd.CalculateBlobSize()},
|
|
Created: createdValue,
|
|
LastUpdated: createdValue,
|
|
Published: createdValue,
|
|
ProjectURL: metadata.ProjectURL,
|
|
ReleaseNotes: metadata.ReleaseNotes,
|
|
RequireLicenseAcceptance: TypedValue[bool]{Type: "Edm.Boolean", Value: metadata.RequireLicenseAcceptance},
|
|
Title: pd.Package.Name,
|
|
},
|
|
}
|
|
|
|
if withNamespace {
|
|
entry.Xmlns = "http://www.w3.org/2005/Atom"
|
|
entry.Base = l.Base
|
|
entry.XmlnsD = "http://schemas.microsoft.com/ado/2007/08/dataservices"
|
|
entry.XmlnsM = "http://schemas.microsoft.com/ado/2007/08/dataservices/metadata"
|
|
}
|
|
|
|
return entry
|
|
}
|
|
|
|
func buildDependencyString(metadata *nuget_module.Metadata) string {
|
|
var b strings.Builder
|
|
first := true
|
|
for group, deps := range metadata.Dependencies {
|
|
for _, dep := range deps {
|
|
if !first {
|
|
b.WriteByte('|')
|
|
}
|
|
first = false
|
|
|
|
b.WriteString(dep.ID)
|
|
b.WriteByte(':')
|
|
b.WriteString(dep.Version)
|
|
b.WriteByte(':')
|
|
b.WriteString(group)
|
|
}
|
|
}
|
|
return b.String()
|
|
}
|