isn/src/isnd/Services/PackageManager.cs

532 lines
22 KiB
C#

3 years ago
using System;
using System.Collections.Generic;
2 years ago
using System.IO;
using System.IO.Compression;
3 years ago
using System.Linq;
2 years ago
using System.Text;
3 years ago
using System.Threading.Tasks;
2 years ago
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Options;
3 years ago
using isn.abst;
using isn.Abstract;
using isnd.Data;
using isnd.Data.Catalog;
using isnd.Data.Packages;
using isnd.Entities;
2 years ago
using isnd.Helpers;
3 years ago
using isnd.Interfaces;
using isnd.ViewModels;
2 years ago
using NuGet.Packaging.Core;
3 years ago
using NuGet.Versioning;
2 years ago
using System.Xml;
using System.Xml.Linq;
using System.Threading;
2 years ago
using NuGet.Protocol;
3 years ago
namespace isnd.Services
{
public class PackageManager : IPackageManager
{
public const string BASE_API_LEVEL = "3.0.0";
readonly ApplicationDbContext dbContext;
public PackageManager(ApplicationDbContext dbContext,
IOptions<IsndSettings> siteConfigOptionsOptions)
{
this.dbContext = dbContext;
isndSettings = siteConfigOptionsOptions.Value;
3 years ago
apiBase = isndSettings.ExternalUrl + Constants.ApiVersionPrefix;
3 years ago
}
public IEnumerable<Resource> GetResources()
3 years ago
{
2 years ago
3 years ago
var res = new List<Resource>
{
new Resource(apiBase + ApiConfig.Package,
3 years ago
"PackagePublish/2.0.0")
{
Comment = "Package Publish service"
},
// under dev, only leash in release mode
new Resource(apiBase + ApiConfig.Package + "/{id}/{version}",
3 years ago
"PackageDetailsUriTemplate/5.1.0")
{
Comment = "URI template used by NuGet Client to construct details URL for packages"
},
2 years ago
3 years ago
new Resource(apiBase + ApiConfig.Content,
3 years ago
"PackageBaseAddress/3.0.0")
{
Comment = "Package Base Address service - " +
3 years ago
"Base URL of where NuGet packages are stored, in the format "
// "https://<host>/nupkg/{id-lower}/{version-lower}/{id-lower}.{version-lower}.nupkg"
3 years ago
},
new Resource(apiBase + ApiConfig.AutoComplete,
3 years ago
"SearchAutocompleteService/" + BASE_API_LEVEL)
{
Comment = "Auto complete service"
},
2 years ago
new Resource(apiBase + ApiConfig.Search,
"SearchQueryService")
{
Comment = "Query endpoint of NuGet Search service (primary) used by RC clients"
},
new Resource(apiBase + ApiConfig.Search,
"SearchQueryService/3.0.0-beta")
{
Comment = "Query endpoint of NuGet Search service (primary) used by RC clients"
},
3 years ago
new Resource(apiBase + ApiConfig.Search,
2 years ago
"SearchQueryService/3.0.0-rc")
3 years ago
{
2 years ago
Comment = "Query endpoint of NuGet Search service (primary) used by RC clients"
3 years ago
},
2 years ago
new Resource(apiBase + ApiConfig.Search,
"SearchQueryService/3.5.0")
{
Comment = "Query endpoint of NuGet Search service (primary) used by RC clients"
},
3 years ago
new Resource(apiBase + ApiConfig.Registration,
"RegistrationsBaseUrl/Versioned")
{
Comment = "Base URL of storage where package registration info is stored. " +
"This base URL includes SemVer 2.0.0 packages."
}
};
return res;
}
public AutoCompleteResult AutoComplete(string id,
int skip, int take, bool prerelease = false,
string packageType = null)
{
var scope = dbContext.PackageVersions.Where(
v => v.PackageId.StartsWith(id)
3 years ago
&& (prerelease || !v.IsPrerelease)
&& (packageType == null || v.Type == packageType)
)
.OrderBy(v => v.FullString);
return new AutoCompleteResult
{
totalHits = scope.Count(),
data = scope.Select(v => v.FullString)
.Skip(skip).Take(take).ToArray()
};
}
public string[] GetVersions(
string id,
3 years ago
NuGetVersion parsedVersion = null,
3 years ago
bool prerelease = false,
string packageType = null,
int skip = 0,
int take = 25)
{
return dbContext.PackageVersions.Where(
v => v.PackageId == id
&& (prerelease || !v.IsPrerelease)
&& (packageType == null || v.Type == packageType)
2 years ago
&& (parsedVersion == null || parsedVersion.CompareTo
3 years ago
(new SemanticVersion(v.Major, v.Minor, v.Patch)) < 0)
)
.OrderBy(v => v.NugetVersion)
.Select(v => v.FullString)
.Skip(skip).Take(take).ToArray();
}
public string CatalogBaseUrl => apiBase;
private IsndSettings isndSettings;
private readonly string apiBase;
3 years ago
public virtual async Task<PackageRegistration> GetCatalogIndexAsync()
3 years ago
{
2 years ago
return await UpdateCatalogForAsync(null);
3 years ago
}
2 years ago
public async Task<PackageRegistration> UpdateCatalogForAsync
3 years ago
(Commit reason = null)
{
int i = 0;
2 years ago
string baseId = apiBase + ApiConfig.Catalog;
string baseRegistrationId = $"{apiBase}{ApiConfig.Registration}";
PackageRegistration index = new PackageRegistration(baseId);
3 years ago
var scope = await dbContext.Commits.OrderBy(c => c.TimeStamp).ToArrayAsync();
i = isndSettings.CatalogPageLen;
foreach (var commit in scope)
{
var validPkgs = (await dbContext.Packages
.Include(po => po.Owner)
.Include(pkg => pkg.Versions)
.Include(pkg => pkg.LatestCommit)
3 years ago
.ToArrayAsync())
.GroupBy((q) => q.Id);
2 years ago
foreach (var pkgIdGroup in validPkgs)
3 years ago
{
3 years ago
CatalogPage page = index.Items.FirstOrDefault
2 years ago
(p => p.GetPackageId() == pkgIdGroup.Key);
3 years ago
if (page == null)
{
2 years ago
page = new CatalogPage(pkgIdGroup.Key, apiBase);
3 years ago
index.Items.Add(page);
}
2 years ago
foreach (var package in pkgIdGroup)
3 years ago
{
2 years ago
page.AddVersionRange(package.Versions);
3 years ago
}
}
reason = commit;
i++;
}
return index;
}
public async Task<PackageDeletionReport> DeletePackageAsync
2 years ago
(string pkgId, string version, string type)
3 years ago
{
// TODO deletion on disk
var commit = new Commit
{
Action = PackageAction.DeletePackage,
TimeStamp = DateTimeOffset.Now.ToUniversalTime()
3 years ago
};
dbContext.Commits.Add(commit);
var pkg = await dbContext.PackageVersions.SingleOrDefaultAsync(
2 years ago
v => v.PackageId == pkgId &&
3 years ago
v.FullString == version &&
v.Type == type
);
if (pkg == null)
{
return new PackageDeletionReport { Deleted = false };
}
dbContext.PackageVersions.Remove(pkg);
await dbContext.SaveChangesAsync();
2 years ago
await UpdateCatalogForAsync(commit);
3 years ago
return new PackageDeletionReport { Deleted = true, DeletedVersion = pkg };
}
public async Task<PackageVersion> GetPackageAsync
2 years ago
(string pkgId, string version, string type)
3 years ago
{
return await dbContext.PackageVersions.SingleOrDefaultAsync(
2 years ago
v => v.PackageId == pkgId &&
3 years ago
v.FullString == version &&
v.Type == type
);
}
2 years ago
public async Task<Data.Catalog.RegistrationLeave> GetCatalogEntryAsync
(string pkgId, string semver, string pkgType = null)
3 years ago
{
2 years ago
var version = await dbContext.PackageVersions
.Include(v => v.Package)
.Include(v => v.Package.LatestCommit)
.Include(v => v.Package.Owner)
2 years ago
.Include(v => v.DependencyGroups)
.Include(v => v.LatestCommit)
2 years ago
.Where(v => v.PackageId == pkgId
&& v.FullString == semver
&& v.LatestCommit != null
2 years ago
).SingleOrDefaultAsync();
foreach (var g in version.DependencyGroups)
{
g.Dependencies = dbContext.Dependencies.Where(d => d.DependencyGroupId == g.Id).ToList();
}
return version.ToPackage(apiBase);
3 years ago
}
public async Task<PackageDeletionReport> UserAskForPackageDeletionAsync
(string uid, string id, string lower, string type)
{
PackageVersion packageVersion = await dbContext.PackageVersions
.Include(pv => pv.Package)
.FirstOrDefaultAsync(m => m.PackageId == id
&& m.FullString == lower && m.Type == type);
if (packageVersion == null) return null;
if (packageVersion.Package.OwnerId != uid) return null;
return new PackageDeletionReport { Deleted = true, DeletedVersion = packageVersion };
}
2 years ago
public IEnumerable<Data.Catalog.RegistrationLeave> SearchCatalogEntriesById
(string pkgId, string semver, string pkgType, bool preRelease)
3 years ago
{
2 years ago
// PackageDependency
3 years ago
return dbContext.PackageVersions
.Include(v => v.Package)
.Include(v => v.Package.Owner)
.Include(v => v.Package.LatestCommit)
3 years ago
.Include(v => v.LatestCommit)
2 years ago
.Include(v => v.DependencyGroups)
3 years ago
.Where(v => v.PackageId == pkgId && semver == v.FullString
&& (pkgType == null || pkgType == v.Type)
&& (preRelease || !v.IsPrerelease))
3 years ago
.OrderByDescending(p => p.CommitNId)
3 years ago
.Select(p => p.ToPackage(apiBase))
3 years ago
;
}
public PackageVersion GetPackage(string pkgId, string semver, string pkgType)
{
return dbContext.PackageVersions
.Include(v => v.Package)
.Include(v => v.LatestCommit)
2 years ago
.Include(v => v.DependencyGroups.Last().Dependencies)
3 years ago
.Single(v => v.PackageId == pkgId && semver == v.FullString
&& (pkgType == null || pkgType == v.Type));
}
3 years ago
public async Task<PackageRegistration> GetPackageRegistrationIndexAsync
(PackageRegistrationQuery query)
3 years ago
{
// RegistrationPageIndexAndQuery
if (string.IsNullOrWhiteSpace(query.Query)) return null;
query.Query = query.Query.ToLower();
2 years ago
var scope = await dbContext.PackageVersions
.Include(p => p.Package)
.Include(p => p.Package.Owner)
.Include(p => p.LatestCommit)
2 years ago
.Where(p => p.PackageId.ToLower() == query.Query).ToArrayAsync();
2 years ago
if (scope == null) return null;
2 years ago
if (scope.Length == 0) return null;
3 years ago
string bid = $"{apiBase}{ApiConfig.Registration}";
2 years ago
foreach (var version in scope)
{
version.DependencyGroups = dbContext.PackageDependencyGroups.Include(d => d.Dependencies)
.Where(d => d.PackageId == version.PackageId && d.PackageVersionFullString == version.FullString)
.ToList();
2 years ago
version.LatestCommit = dbContext.Commits.Single(c => c.Id == version.CommitNId);
2 years ago
}
3 years ago
return
2 years ago
new PackageRegistration(apiBase, query.Query, scope);
3 years ago
}
2 years ago
2 years ago
public async Task<PackageSearchResult> SearchPackageAsync(PackageRegistrationQuery query)
3 years ago
{
3 years ago
string bid = $"{apiBase}{ApiConfig.Registration}";
2 years ago
if (string.IsNullOrWhiteSpace(query.Query))
query.Query = "";
2 years ago
var packages = await dbContext.Packages
.Include(g => g.Versions)
.Where(d => d.Id.StartsWith(query.Query)
&& (query.Prerelease || d.Versions.Any(v => !v.IsPrerelease)))
.Skip(query.Skip).Take(query.Take).ToArrayAsync();
foreach (var package in packages)
foreach (var version in package.Versions)
{
version.DependencyGroups = dbContext.PackageDependencyGroups.Include(d => d.Dependencies)
.Where(d => d.PackageVersionFullString == version.FullString && d.PackageId == version.PackageId)
.ToList();
}
return new PackageSearchResult(packages, apiBase, packages.Count());
3 years ago
}
2 years ago
public async Task<PackageVersion> PutPackageAsync(Stream packageStream, string ownerId)
{
PackageVersion version = null;
using (packageStream)
{
using (var archive = new ZipArchive(packageStream))
{
var spec = archive.Entries.FirstOrDefault(e => e.FullName.EndsWith("." + Constants.SpecFileExtension));
if (spec == null) throw new InvalidPackageException("no " + Constants.SpecFileExtension + " from archive");
string pkgPath;
NuGetVersion nugetVersion;
string pkgId;
string fullPath;
using var specificationStream = spec.Open();
2 years ago
using XmlReader xmlReader = XmlReader.Create(specificationStream);
var xMeta = XElement.Load(xmlReader, LoadOptions.None).Descendants().First();
string packageDescription = xMeta.Descendants().FirstOrDefault(x => x.Name.LocalName == "description")?.Value;
var dependencies = xMeta
.Descendants().FirstOrDefault(x => x.Name.LocalName == "dependencies");
var frameworkReferences= xMeta
.Descendants().FirstOrDefault(x => x.Name.LocalName == "frameworkReferences");
var frameWorks = (dependencies ?? frameworkReferences)
2 years ago
.Descendants().Where(x => x.Name.LocalName == "group")
.Select(x => NewFrameworkDependencyGroup(x)).ToArray();
var types = "Package";
pkgId = xMeta.Descendants().FirstOrDefault(x => x.Name.LocalName == "id")?.Value;
string pkgVersion = xMeta.Descendants().FirstOrDefault(x => x.Name.LocalName == "version")?.Value;
if (!NuGetVersion.TryParse(pkgVersion, out nugetVersion))
throw new InvalidPackageException("metadata/version");
string packageIdPath = Path.Combine(isndSettings.PackagesRootDir,
pkgId);
pkgPath = Path.Combine(packageIdPath, nugetVersion.ToFullString());
string name = $"{pkgId}-{nugetVersion}." + Constants.PacketFileExtension;
fullPath = Path.Combine(pkgPath, name);
var packageIdPathInfo = new DirectoryInfo(packageIdPath);
Data.Packages.Package pkg = dbContext.Packages.SingleOrDefault(p => p.Id == pkgId);
Commit commit = new Commit
{
Action = PackageAction.PublishPackage,
TimeStamp = DateTimeOffset.Now.ToUniversalTime()
};
if (pkg != null)
{
// Update
pkg.Description = packageDescription;
pkg.LatestCommit = commit;
}
else
{
// First version
pkg = new Data.Packages.Package
2 years ago
{
2 years ago
Id = pkgId,
Description = packageDescription,
OwnerId = ownerId,
LatestCommit = commit
2 years ago
};
2 years ago
dbContext.Packages.Add(pkg);
}
2 years ago
2 years ago
// here, the package is or new, or owned by the key owner
if (!packageIdPathInfo.Exists) packageIdPathInfo.Create();
2 years ago
2 years ago
var dest = new FileInfo(fullPath);
var destDir = new DirectoryInfo(dest.DirectoryName);
if (dest.Exists) dest.Delete();
if (!destDir.Exists) destDir.Create();
2 years ago
2 years ago
packageStream.Seek(0, SeekOrigin.Begin);
using (var fileStream = File.Create(fullPath))
{
await packageStream.CopyToAsync(fileStream);
}
2 years ago
2 years ago
string fullStringVersion = nugetVersion.ToFullString();
var pkgVersions = dbContext.PackageVersions.Where
(v => v.PackageId == pkg.Id && v.FullString == fullStringVersion);
if (pkgVersions.Count() > 0)
{
foreach (var v in pkgVersions.ToArray())
dbContext.PackageVersions.Remove(v);
}
string versionFullString = nugetVersion.ToFullString();
2 years ago
2 years ago
// FIXME default package type or null
dbContext.PackageVersions.Add
(version = new PackageVersion
{
Package = pkg,
Major = nugetVersion.Major,
Minor = nugetVersion.Minor,
Patch = nugetVersion.Patch,
Revision = nugetVersion.Revision,
IsPrerelease = nugetVersion.IsPrerelease,
FullString = versionFullString,
Type = types,
LatestCommit = commit
});
dbContext.Commits.Add(commit);
foreach (var group in dbContext.PackageDependencyGroups.Include(g => g.PackageVersion)
.Where(x => x.PackageId == pkgId && x.PackageVersionFullString == versionFullString)
.ToList())
{
dbContext.PackageDependencyGroups.Remove(group);
}
version.DependencyGroups = new List<PackageDependencyGroup>();
foreach (var framework in frameWorks)
{
var group = new PackageDependencyGroup
{
TargetFramework = framework.FrameworkName,
PackageId = pkgId,
PackageVersionFullString = versionFullString,
Dependencies = framework.Dependencies.Select(
d => new Dependency
{
PackageId = d.PackageId,
2 years ago
Version = d.PackageVersion,
}).ToList()
};
2 years ago
version.DependencyGroups.Add(group);
dbContext.PackageDependencyGroups.Add(group);
2 years ago
2 years ago
}
await dbContext.SaveChangesAsync();
await UpdateCatalogForAsync(commit);
2 years ago
using (var shaCrypto = System.Security.Cryptography.SHA512.Create())
{
using (var stream = System.IO.File.OpenRead(fullPath))
{
var hash = shaCrypto.ComputeHash(stream);
var shaFullName = fullPath + ".sha512";
var hashText = Convert.ToBase64String(hash);
var hashTextBytes = Encoding.ASCII.GetBytes(hashText);
using (var shaFile = System.IO.File.OpenWrite(shaFullName))
{
shaFile.Write(hashTextBytes, 0, hashTextBytes.Length);
}
}
}
string nugetSpecificationFullPath = Path.Combine(pkgPath, pkgId + "." + Constants.SpecFileExtension);
FileInfo nugetSpecificationFullPathInfo = new(nugetSpecificationFullPath);
if (nugetSpecificationFullPathInfo.Exists)
nugetSpecificationFullPathInfo.Delete();
spec.ExtractToFile(nugetSpecificationFullPath);
}
}
return version;
}
2 years ago
private FrameworkDependencyGroup NewFrameworkDependencyGroup(XElement x)
{
var view = x.ToJson();
var frameworkReferences = x.Descendants();
var framework = x.Attribute("targetFramework").Value;
var deps = frameworkReferences.Select(r => new ShortDependencyInfo
{
PackageId = (r.Attribute("name") ?? r.Attribute("id")).Value,
PackageVersion = r.Attribute("version")?.Value
});
2 years ago
return new FrameworkDependencyGroup
{
FrameworkName = framework,
Dependencies = deps.ToList()
2 years ago
};
}
3 years ago
}
}