Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: include packages from other vpm repository #23

Open
wants to merge 7 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
81 changes: 81 additions & 0 deletions PackageBuilder/Build.cs
Original file line number Diff line number Diff line change
Expand Up @@ -13,9 +13,12 @@
using Nuke.Common.CI.GitHubActions;
using Nuke.Common.IO;
using Octokit;
using VRC.PackageManagement.Automation.Multi;
using VRC.PackageManagement.Core;
using VRC.PackageManagement.Core.Types.Packages;
using ProductHeaderValue = Octokit.ProductHeaderValue;
using ListingSource = VRC.PackageManagement.Automation.Multi.ListingSource;
using Version = SemanticVersioning.Version;

namespace VRC.PackageManagement.Automation
{
Expand Down Expand Up @@ -189,6 +192,10 @@ ListingSource MakeListingSourceFromManifest(VRCPackageManifest manifest)
packages.Add(manifest);
}

// download package manifest from other vpm repository
if (listSource.vpmPackages != null)
await LoadPackageInfo(listSource.vpmPackages, packages);

// Copy listing-source.json to new Json Object
Serilog.Log.Information($"All packages prepared, generating Listing.");
var repoList = new VRCRepoList(packages)
Expand Down Expand Up @@ -275,6 +282,80 @@ ListingSource MakeListingSourceFromManifest(VRCPackageManifest manifest)
Serilog.Log.Information($"Saved Listing to {savePath}.");
});

async Task LoadPackageInfo(Dictionary<string, VpmPackageInfo> listSourceVpmPackages, List<VRCPackageManifest> packages)
{
var packagesByRepository = new Dictionary<string, List<(string id, VpmPackageInfo info)>>();

// collect packages by repository to reduce request per repository
foreach (var (packageId, package) in listSourceVpmPackages)
{
if (package.sources == null || package.sources.Length == 0)
{
Serilog.Log.Warning($"Source repositories for {packageId} is not defined! This package will be ignored.");
continue;
}

foreach (var sourceUrl in package.sources)
anatawa12 marked this conversation as resolved.
Show resolved Hide resolved
{
if (!packagesByRepository.TryGetValue(sourceUrl, out var packageInfos))
packagesByRepository.Add(sourceUrl, packageInfos = new List<(string id, VpmPackageInfo info)>());
packageInfos.Add((packageId, package));
}
}

// fetch packages
var repositories = await Task.WhenAll(packagesByRepository.Select(p =>
DownloadPackageManifestFromVpmRepository(p.Key, p.Value)));

// add to packages
foreach (var manifests in repositories)
packages.AddRange(manifests);
}

async Task<IEnumerable<VRCPackageManifest>> DownloadPackageManifestFromVpmRepository(string url,
List<(string id, VpmPackageInfo info)> packages)
{
Serilog.Log.Information($"Downloading from vpm repository {url}");
var response = await Http.GetAsync(url);
var jsonText = await response.Content.ReadAsStringAsync();
var repository = JsonConvert.DeserializeObject<VRCRepoList>(jsonText, Settings.JsonReadOptions);

var result = new List<VRCPackageManifest>();

foreach (var (packageId, packageInfo) in packages)
{
if (!repository.Versions.TryGetValue(packageId, out var versions))
{
Serilog.Log.Warning($"{packageId} is not defined in {url}!");
continue;
}

foreach (var (versionString, package) in versions.Versions)
{
if (!Version.TryParse(versionString, out var version))
{
Serilog.Log.Warning($"We found invalid version of {packageId} in {url}: {versionString}");
continue;
}

if (!packageInfo.includePrerelease && version.IsPreRelease) continue;
// TODO: add option to specify include version range

// In Settings.JsonReadOptions, we have JsonConverter that deserializes IVRCPackage with VRCPackageManifest
// so this cast should not be failed.
if (package is not VRCPackageManifest vpmPackageInfo)
{
Serilog.Log.Error($"logic failure: deserialized IVRCPackage is not VRCPackageManifest!");
continue;
}

result.Add(vpmPackageInfo);
}
}

return result;
}

GitHubClient _client;
GitHubClient Client
{
Expand Down
10 changes: 10 additions & 0 deletions PackageBuilder/ListingSource.cs
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ public class ListingSource
public string bannerUrl { get; set; }
public List<PackageInfo> packages { get; set; }
public List<string> githubRepos { get; set; }
public Dictionary<string, VpmPackageInfo> vpmPackages { get; set; }
}

public class InfoLink {
Expand All @@ -32,4 +33,13 @@ public class PackageInfo
public string id { get; set; }
public List<string> releases { get; set; }
}

public class VpmPackageInfo
{
/// <summary>URL of source vpm repository</summary>
public string[] sources { get; set; }

/// <summary>True if you want to include prerelease of this package to your repository.</summary>
public bool includePrerelease { get; set; }
}
}
Loading